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.
219 lines
6.3 KiB
219 lines
6.3 KiB
<?
|
|
/**
|
|
* 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$
|
|
*
|
|
*/
|
|
|
|
/**
|
|
* PhpType
|
|
*
|
|
* The class includes set of methods that convert the received Modbus data
|
|
* (array of bytes) to the PHP data type, i.e. signed int, unsigned int and float.
|
|
*
|
|
* @author Jan Krakora
|
|
* @copyright Copyright (c) 2004, 2009 Jan Krakora, WAGO Kontakttechnik GmbH & Co. KG (http://www.wago.com)
|
|
* @package Phpmodbus
|
|
*
|
|
*/
|
|
class PhpType {
|
|
|
|
/**
|
|
* bytes2float
|
|
*
|
|
* The function converts array of 4 bytes to float. The return value
|
|
* depends on order of the input bytes (endianning).
|
|
*
|
|
* @param array $values
|
|
* @param bool $endianness
|
|
* @return float
|
|
*/
|
|
public static function bytes2float($values, $endianness = 0){
|
|
$data = array();
|
|
$real = 0;
|
|
|
|
// Set the array to correct form
|
|
$data = self::checkData($values);
|
|
// Combine bytes
|
|
$real = self::combineBytes($data, $endianness);
|
|
// Convert the real value to float
|
|
return (float) self::real2float($real);
|
|
}
|
|
|
|
/**
|
|
* bytes2signedInt
|
|
*
|
|
* The function converts array of 2 or 4 bytes to signed integer.
|
|
* The return value depends on order of the input bytes (endianning).
|
|
*
|
|
* @param array $values
|
|
* @param bool $endianness
|
|
* @return int
|
|
*/
|
|
public static function bytes2signedInt($values, $endianness = 0){
|
|
$data = array();
|
|
$int = 0;
|
|
// Set the array to correct form
|
|
$data = self::checkData($values);
|
|
// Combine bytes
|
|
$int = self::combineBytes($data, $endianness);
|
|
// In the case of signed 2 byte value convert it to 4 byte one
|
|
if ((count($values) == 2) && ((0x8000 & $int) > 0)){
|
|
$int = 0xFFFF8000 | $int;
|
|
}
|
|
// Convert the value
|
|
return (int) self::dword2signedInt($int);
|
|
}
|
|
|
|
/**
|
|
* bytes2unsignedInt
|
|
*
|
|
* The function converts array of 2 or 4 bytes to unsigned integer.
|
|
* The return value depends on order of the input bytes (endianning).
|
|
*
|
|
* @param array $values
|
|
* @param bool $endianness
|
|
* @return int|float
|
|
*/
|
|
public static function bytes2unsignedInt($values, $endianness = 0){
|
|
$data = array();
|
|
$int = 0;
|
|
// Set the array to correct form
|
|
$data = self::checkData($values);
|
|
// Combine bytes
|
|
$int = self::combineBytes($data, $endianness);
|
|
// Convert the value
|
|
return self::dword2unsignedInt($int);
|
|
}
|
|
|
|
/**
|
|
* real2float
|
|
*
|
|
* This function converts a value in IEC-1131 REAL single precision form to float.
|
|
*
|
|
* 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 value value in IEC REAL data type to be converted
|
|
* @return float float value
|
|
*/
|
|
private static function real2float($value){
|
|
$two_pow_minus_x = array(
|
|
1, 0.5, 0.25, 0.125, 0.0625, 0.03125, 0.015625,
|
|
0.0078125, 0.00390625, 0.001953125, 0.0009765625,
|
|
0.00048828125, 0.000244140625, 0.0001220703125,
|
|
0.00006103515625, 0.000030517578125, 0.0000152587890625,
|
|
0.00000762939453125, 0.000003814697265625, 0.0000019073486328125,
|
|
0.00000095367431640625, 0.000000476837158203125,
|
|
0.0000002384185791015625, 0.00000011920928955078125);
|
|
// get sign, mantisa, exponent
|
|
$real_mantisa = $value & 0x7FFFFF | 0x800000;
|
|
$real_exponent = ($value>>23) & 0xFF;
|
|
$real_sign = ($value>>31) & 0x01;
|
|
$bin_exponent = $real_exponent - 127;
|
|
// decode value
|
|
if (( $bin_exponent >= -126) && ($bin_exponent <= 127)) {
|
|
// Mantissa decoding
|
|
for ($i=0; $i<24; $i++) {
|
|
if ($real_mantisa & 0x01)
|
|
$val += $two_pow_minus_x[23-$i];
|
|
$real_mantisa = $real_mantisa >> 1;
|
|
}
|
|
// Base
|
|
$val = $val * pow(2,$bin_exponent);
|
|
if (($real_sign == 1)) $val = -$val;
|
|
}
|
|
return (float)$val;
|
|
}
|
|
|
|
/**
|
|
* dword2signedInt
|
|
*
|
|
* Switch double word to signed integer
|
|
*
|
|
* @param int $value
|
|
* @return int
|
|
*/
|
|
private static function dword2signedInt($value){
|
|
if ((0x80000000 & $value) != 0) {
|
|
return -(0x7FFFFFFF & ~$value)-1;
|
|
} else {
|
|
return (0x7FFFFFFF & $value);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* dword2signedInt
|
|
*
|
|
* Switch double word to unsigned integer
|
|
*
|
|
* @param int $value
|
|
* @return int|float
|
|
*/
|
|
private static function dword2unsignedInt($value){
|
|
if ((0x80000000 & $value) != 0) {
|
|
return ((float) (0x7FFFFFFF & $value)) + 2147483648;
|
|
} else {
|
|
return (int) (0x7FFFFFFF & $value);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* checkData
|
|
*
|
|
* Check if the data variable is array, and check if the values are numeric
|
|
*
|
|
* @param int $data
|
|
* @return int
|
|
*/
|
|
private static function checkData($data){
|
|
// Check the data
|
|
if (!is_array($data)) {
|
|
throw new Exception('The input data should be an array of bytes.');
|
|
}
|
|
// Check the values to be number - must be
|
|
if (!is_numeric($data[0]) || !is_numeric($data[1])) {
|
|
throw new Exception('Data are not numeric.');
|
|
}
|
|
if (!is_numeric($data[2])) $data[2] = 0;
|
|
if (!is_numeric($data[3])) $data[3] = 0;
|
|
|
|
return $data;
|
|
}
|
|
|
|
/**
|
|
* combineBytes
|
|
*
|
|
* Combine bytes together
|
|
*
|
|
* @param int $data
|
|
* @param bool $endianness
|
|
* @return int
|
|
*/
|
|
private static function combineBytes($data, $endianness){
|
|
$value = 0;
|
|
// Combine bytes
|
|
if ($endianness == 0)
|
|
$value = (($data[3] & 0xFF)<<16) |
|
|
(($data[2] & 0xFF)<<24) |
|
|
(($data[1] & 0xFF)) |
|
|
(($data[0] & 0xFF)<<8);
|
|
else
|
|
$value = (($data[3] & 0xFF)<<24) |
|
|
(($data[2] & 0xFF)<<16) |
|
|
(($data[1] & 0xFF)<<8) |
|
|
(($data[0] & 0xFF));
|
|
|
|
return $value;
|
|
}
|
|
}
|
|
?>
|