You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
178 lines
5.4 KiB
178 lines
5.4 KiB
<?php
|
|
/**
|
|
* Phpmodbus Copyright (c) 2004, 2009 Jan Krakora, WAGO Kontakttechnik GmbH & Co. KG (http://www.wago.com)
|
|
*
|
|
* This source file is subject to the "PhpModbus license" that is bundled
|
|
* with this package in the file license.txt.
|
|
*
|
|
* @author Jan Krakora
|
|
* @copyright Copyright (c) 2004, 2009 Jan Krakora, WAGO Kontakttechnik GmbH & Co. KG (http://www.wago.com)
|
|
* @license PhpModbus license
|
|
* @category Phpmodbus
|
|
* @package Phpmodbus
|
|
* @version $id$
|
|
*/
|
|
|
|
/**
|
|
* IecType
|
|
*
|
|
* The class includes set of IEC-1131 data type functions that converts a PHP
|
|
* data types to a IEC data type.
|
|
*
|
|
* @author Jan Krakora
|
|
* @copyright Copyright (c) 2004, 2009 Jan Krakora, WAGO Kontakttechnik GmbH & Co. KG (http://www.wago.com)
|
|
* @package Phpmodbus
|
|
*/
|
|
class IecType {
|
|
|
|
/**
|
|
* iecBYTE
|
|
*
|
|
* Converts a value to IEC-1131 BYTE data type
|
|
*
|
|
* @param value value from 0 to 255
|
|
* @return value IEC BYTE data type
|
|
*
|
|
*/
|
|
function iecBYTE($value){
|
|
return chr($value & 0xFF);
|
|
}
|
|
|
|
/**
|
|
* iecINT
|
|
*
|
|
* Converts a value to IEC-1131 INT data type
|
|
*
|
|
* @param value value to be converted
|
|
* @return value IEC-1131 INT data type
|
|
*
|
|
*/
|
|
function iecINT($value){
|
|
return self::iecBYTE(($value >> 8) & 0x00FF) .
|
|
self::iecBYTE(($value & 0x00FF));
|
|
}
|
|
|
|
/**
|
|
* iecDINT
|
|
*
|
|
* Converts a value to IEC-1131 DINT data type
|
|
*
|
|
* @param value value to be converted
|
|
* @param value endianness defines endian codding (little endian == 0, big endian == 1)
|
|
* @return value IEC-1131 INT data type
|
|
*
|
|
*/
|
|
function iecDINT($value, $endianness = 0){
|
|
// result with right endianness
|
|
return self::endianness($value, $endianness);
|
|
}
|
|
|
|
/**
|
|
* iecREAL
|
|
*
|
|
* Converts a value to IEC-1131 REAL data type. The function uses function @use float2iecReal.
|
|
*
|
|
* @param value value to be converted
|
|
* @param value endianness defines endian codding (little endian == 0, big endian == 1)
|
|
* @return value IEC-1131 REAL data type
|
|
*/
|
|
function iecREAL($value, $endianness = 0){
|
|
// iecREAL representation
|
|
$real = self::float2iecReal($value);
|
|
// result with right endianness
|
|
return self::endianness($real, $endianness);
|
|
}
|
|
|
|
/**
|
|
* float2iecReal
|
|
*
|
|
* This function converts float value to IEC-1131 REAL single precision form.
|
|
*
|
|
* For more see [{@link http://en.wikipedia.org/wiki/Single_precision Single precision on Wiki}] or
|
|
* [{@link http://de.php.net/manual/en/function.base-convert.php PHP base_convert function commentary}, Todd Stokes @ Georgia Tech 21-Nov-2007]*
|
|
*
|
|
* @param float value to be converted
|
|
* @return value IEC REAL data type
|
|
*/
|
|
private function float2iecReal($value){
|
|
$bias = 128;
|
|
$cnst = 281; // 1 (carry bit) + 127 + 1 + 126 + 24 + 2 (round bits)
|
|
$two_power_x = array(1, 2, 4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048,
|
|
4096, 8192, 16384, 32768, 65536, 131072, 262144, 524288, 1048576,
|
|
2097152, 4194304);
|
|
//convert and seperate input to integer and decimal parts
|
|
$val = abs($value);
|
|
$intpart = floor($val);
|
|
$decpart = $val - $intpart;
|
|
//convert integer part
|
|
for ($i=0;$i<$cnst;$i++) $real_significand_bin[$i] = 0;
|
|
$i = $bias;
|
|
while ((($intpart / 2) != 0) && ($i >= 0))
|
|
{
|
|
$real_significand_bin[$i] = $intpart % 2;
|
|
if (($intpart % 2) == 0) $intpart = $intpart / 2;
|
|
else $intpart = $intpart / 2 - 0.5;
|
|
$i -= 1;
|
|
}
|
|
//convert decimal part
|
|
$i = $bias+1;
|
|
while (($decpart > 0) && ($i < $cnst))
|
|
{
|
|
$decpart *= 2;
|
|
if ($decpart >= 1) {
|
|
$real_significand_bin[$i] = 1;
|
|
$decpart --;
|
|
$i++;
|
|
}
|
|
else
|
|
{
|
|
$real_significand_bin[i] = 0;
|
|
$i++;
|
|
}
|
|
}
|
|
//obtain exponent value
|
|
$i = 0;
|
|
//find most significant bit of significand
|
|
while (($i < $cnst) && ($real_significand_bin[$i] != 1)) $i++;
|
|
//
|
|
$index_exp = $i;
|
|
$real_exponent = 128 - $index_exp;
|
|
if ($real_exponent < -126) return 0;
|
|
if (($real_exponent > 127)&&($real_float>0)) return 0x7F7FFFFF;
|
|
if (($real_exponent > 127)&&($real_float<0)) return 0xFF7FFFFF;
|
|
for ($i=0; $i<23; $i++)
|
|
$real_significand = $real_significand + $real_significand_bin[$index_exp+1+$i] * $two_power_x[22-$i];
|
|
// return
|
|
if ($value<0) $w = 0x80000000 + ($real_significand & 0x7FFFFF) + ((($real_exponent+127)<<23) & 0x7F800000);
|
|
else $w = ($real_significand & 0x7FFFFF) + ((($real_exponent+127)<<23) & 0x7F800000);
|
|
return $w;
|
|
}
|
|
|
|
/**
|
|
* endianness
|
|
*
|
|
* Make endianess as required.
|
|
* For more see http://en.wikipedia.org/wiki/Endianness
|
|
*
|
|
* @param int $value
|
|
* @param bool $endianness
|
|
* @return int
|
|
*/
|
|
private function endianness($value, $endianness = 0){
|
|
if ($endianness == 0)
|
|
return
|
|
self::iecBYTE(($value >> 8) & 0x000000FF) .
|
|
self::iecBYTE(($value & 0x000000FF)) .
|
|
self::iecBYTE(($value >> 24) & 0x000000FF) .
|
|
self::iecBYTE(($value >> 16) & 0x000000FF);
|
|
else
|
|
return
|
|
self::iecBYTE(($value >> 24) & 0x000000FF) .
|
|
self::iecBYTE(($value >> 16) & 0x000000FF) .
|
|
self::iecBYTE(($value >> 8) & 0x000000FF) .
|
|
self::iecBYTE(($value & 0x000000FF));
|
|
}
|
|
|
|
}
|
|
|
|
?>
|