commit
6b55205c8c
@ -0,0 +1,178 @@ |
||||
<?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)); |
||||
} |
||||
|
||||
} |
||||
|
||||
?> |
@ -0,0 +1,557 @@ |
||||
<?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. |
||||
* |
||||
* |
||||
* @copyright Copyright (c) 2004, 2009 Jan Krakora, WAGO Kontakttechnik GmbH & Co. KG (http://www.wago.com) |
||||
* @license PhpModbus license |
||||
* @category Phpmodbus |
||||
* @tutorial Phpmodbus.pkg |
||||
* @package Phpmodbus |
||||
* @version $id$ |
||||
* |
||||
*/ |
||||
|
||||
require_once dirname(__FILE__) . '/IecType.php'; |
||||
require_once dirname(__FILE__) . '/PhpType.php'; |
||||
|
||||
/** |
||||
* ModbusMasterUdp |
||||
* |
||||
* This class deals with the MODBUS master using UDP stack. |
||||
* |
||||
* Implemented MODBUS functions: |
||||
* - FC 3: read multiple registers |
||||
* - FC 16: write multiple registers |
||||
* - FC 23: read write registers |
||||
* |
||||
* @author Jan Krakora |
||||
* @copyright Copyright (c) 2004, 2009 Jan Krakora, WAGO Kontakttechnik GmbH & Co. KG (http://www.wago.com) |
||||
* @package Phpmodbus |
||||
* |
||||
*/ |
||||
class ModbusMasterUdp { |
||||
var $sock; |
||||
var $port = "502"; |
||||
var $host = "192.168.1.1"; |
||||
var $errstr; |
||||
var $status; |
||||
var $timeout_sec = 5; // 5 sec |
||||
var $endianess = 0; // defines endian codding (little endian == 0, big endian == 1) |
||||
|
||||
/** |
||||
* Modbus |
||||
* |
||||
* This is the constructor that defines {@link $host} IP address of the object. |
||||
* |
||||
* @param String $host An IP address of a Modbus TCP device. E.g. "192.168.1.1". |
||||
*/ |
||||
function ModbusMasterUdp($host){ |
||||
$this->host = $host; |
||||
} |
||||
|
||||
/** |
||||
* connect |
||||
* |
||||
* Connect the socket |
||||
* |
||||
* @return bool |
||||
*/ |
||||
private function connect(){ |
||||
// UDP socket |
||||
$this->sock = socket_create(AF_INET, SOCK_DGRAM, SOL_UDP); |
||||
// connect |
||||
$result = @socket_connect($this->sock, $this->host, $this->port); |
||||
if ($result === false) { |
||||
$this->errstr .= "socket_connect() failed.</br>Reason: ($result) " . |
||||
socket_strerror(socket_last_error($this->sock)); |
||||
return false; |
||||
} else { |
||||
$this->status .= "Connected</br>"; |
||||
return true; |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* disconnect |
||||
* |
||||
* Disconnect the socket |
||||
*/ |
||||
private function disconnect(){ |
||||
socket_close($this->sock); |
||||
$this->status .= "Disconnected</br>"; |
||||
} |
||||
|
||||
/** |
||||
* send |
||||
* |
||||
* Send the packet via Modbus |
||||
* |
||||
* @param string $packet |
||||
*/ |
||||
private function send($packet){ |
||||
socket_write($this->sock, $packet, strlen($packet)); |
||||
$this->status .= "Send</br>"; |
||||
} |
||||
|
||||
/** |
||||
* rec |
||||
* |
||||
* Receive data from the socket |
||||
* |
||||
* @return bool |
||||
*/ |
||||
private function rec(){ |
||||
socket_set_nonblock($this->sock); |
||||
$readsocks[] = $this->sock; |
||||
$writesocks = NULL; |
||||
$exceptsocks = NULL; |
||||
$rec = ""; |
||||
$lastAccess = time(); |
||||
while (socket_select($readsocks, |
||||
$writesocks, |
||||
$exceptsocks, |
||||
0, |
||||
300000) !== FALSE) { |
||||
$this->status .= "Wait received data</br>"; |
||||
if (in_array($this->sock, $readsocks)) { |
||||
while (@socket_recv($this->sock, $rec, 2000, 0)) { |
||||
$this->status .= "Received</br>"; |
||||
return $rec; |
||||
} |
||||
$lastAccess = time(); |
||||
} else { |
||||
if (time()-$lastAccess >= $this->timeout_sec) { |
||||
$this->errstr .= "Watchdog time expired [ " . |
||||
$this->timeout_sec . " sec]!!! Connection to " . |
||||
$this->host . " is not established."; |
||||
return false; |
||||
} |
||||
} |
||||
$readsocks[] = $this->sock; |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* responseCode |
||||
* |
||||
* Check the Modbus response code |
||||
* |
||||
* @param string $packet |
||||
* @return bool |
||||
*/ |
||||
private function responseCode($packet){ |
||||
if(($packet[7] & 0x80) > 0) { |
||||
$this->errstr .= "Modbus response error code:" . ord($packet[8]); |
||||
return false; |
||||
} |
||||
else |
||||
{ |
||||
$this->status .= "Modbus response error code: NOERROR</br>"; |
||||
return true; |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* readMultipleRegisters |
||||
* |
||||
* Modbus function FC 3(0x03) - Read Multiple Registers. |
||||
* |
||||
* This function reads {@link $quantity} of Words (2 bytes) from reference |
||||
* {@link $referenceRead} of a memory of a Modbus device given by |
||||
* {@link $unitId}. |
||||
* |
||||
* |
||||
* @param int $unitId usually ID of Modbus device |
||||
* @param int $reference Reference in the device memory to read data (e.g. in device WAGO 750-841, memory MW0 starts at address 12288). |
||||
* @param int $quantity Amounth of the data to be read from device. |
||||
* @return false|Array Success flag or array of received data. |
||||
*/ |
||||
function readMultipleRegisters($unitId, $reference, $quantity){ |
||||
$this->errstr = ""; |
||||
$this->status = "readMultipleRegisters: START</br>"; |
||||
// connect |
||||
if(!$this->connect()) |
||||
return false; |
||||
// send FC 3 |
||||
$packet = $this->readMultipleRegistersPacketBuilder($unitId, $reference, $quantity); |
||||
$this->status .= $this->printPacket($packet); |
||||
$this->send($packet); |
||||
// receive response |
||||
$rpacket = $this->rec(); |
||||
if(!$rpacket) |
||||
return false; |
||||
$this->status .= $this->printPacket($rpacket); |
||||
// parse packet |
||||
$receivedData = $this->readMultipleRegistersParser($rpacket); |
||||
if(!$receivedData) |
||||
return false; |
||||
// disconnect |
||||
$this->disconnect(); |
||||
$this->status .= "readMultipleRegisters: DONE</br>"; |
||||
// return |
||||
return $receivedData; |
||||
} |
||||
|
||||
/** |
||||
* fc3 |
||||
* |
||||
* Alias to {@link readMultipleRegisters} method. |
||||
* |
||||
* @param int $unitId |
||||
* @param int $reference |
||||
* @param int $quantity |
||||
* @return false|Array |
||||
*/ |
||||
function fc3($unitId, $reference, $quantity){ |
||||
return $this->readMultipleRegisters($unitId, $reference, $quantity); |
||||
} |
||||
|
||||
/** |
||||
* readMultipleRegistersPacketBuilder |
||||
* |
||||
* Packet FC 3 builder - read multiple registers |
||||
* |
||||
* @param int $unitId |
||||
* @param int $reference |
||||
* @param int $quantity |
||||
* @return string |
||||
*/ |
||||
private function readMultipleRegistersPacketBuilder($unitId, $reference, $quantity){ |
||||
$dataLen = 0; |
||||
// build body |
||||
$buffer2 = ""; |
||||
$buffer2 .= iecType::iecBYTE(3); // FC 3 = 3(0x03) |
||||
// build body - read section |
||||
$buffer2 .= iecType::iecINT($reference); // refnumber = 12288 |
||||
$buffer2 .= iecType::iecINT($quantity); // quantity |
||||
$dataLen += 5; |
||||
// build header |
||||
$buffer3 = ''; |
||||
$buffer3 .= iecType::iecINT(rand(0,65000)); // transaction ID |
||||
$buffer3 .= iecType::iecINT(0); // protocol ID |
||||
$buffer3 .= iecType::iecINT($dataLen + 1); // lenght |
||||
$buffer3 .= iecType::iecBYTE($unitId); //unit ID |
||||
// return packet string |
||||
return $buffer3. $buffer2. $buffer1; |
||||
} |
||||
|
||||
/** |
||||
* readMultipleRegistersParser |
||||
* |
||||
* FC 3 response parser |
||||
* |
||||
* @param string $packet |
||||
* @return array |
||||
*/ |
||||
private function readMultipleRegistersParser($packet){ |
||||
$data = array(); |
||||
// if not exception |
||||
if(!$this->responseCode($packet)) |
||||
return false; |
||||
// get data |
||||
for($i=0;$i<ord($packet[8]);$i++){ |
||||
$data[$i] = ord($packet[9+$i]); |
||||
} |
||||
return $data; |
||||
} |
||||
|
||||
/** |
||||
* writeMultipleRegister |
||||
* |
||||
* Modbus function FC16(0x10) - Write Multiple Register. |
||||
* |
||||
* This function writes {@link $data} array at {@link $reference} position of |
||||
* memory of a Modbus device given by {@link $unitId}. |
||||
* |
||||
* |
||||
* @param int $unitId usually ID of Modbus device |
||||
* @param int $reference Reference in the device memory (e.g. in device WAGO 750-841, memory MW0 starts at address 12288) |
||||
* @param array $data Array of values to be written. |
||||
* @param array $dataTypes Array of types of values to be written. The array should consists of string "INT", "DINT" and "REAL". |
||||
* @return bool Success flag |
||||
*/ |
||||
function writeMultipleRegister($unitId, $reference, $data, $dataTypes){ |
||||
$this->errstr = ""; |
||||
$this->status = "writeMultipleRegister: START</br>"; |
||||
// connect |
||||
if(!$this->connect()) |
||||
return false; |
||||
// send FC16 |
||||
$packet = $this->writeMultipleRegisterPacketBuilder($unitId, $reference, $data, $dataTypes); |
||||
$this->status .= $this->printPacket($packet); |
||||
$this->send($packet); |
||||
// receive response |
||||
$rpacket = $this->rec(); |
||||
if(!$rpacket) |
||||
return false; |
||||
$this->status .= $this->printPacket($rpacket); |
||||
// parse packet |
||||
if(!$this->writeMultipleRegisterParser($rpacket)) |
||||
return false; |
||||
// disconnect |
||||
$this->disconnect(); |
||||
$this->status .= "writeMultipleRegister: DONE</br>"; |
||||
return true; |
||||
} |
||||
|
||||
|
||||
/** |
||||
* fc16 |
||||
* |
||||
* Alias to {@link writeMultipleRegister} method |
||||
* |
||||
* @param int $unitId |
||||
* @param int $reference |
||||
* @param array $data |
||||
* @param array $dataTypes |
||||
* @return bool |
||||
*/ |
||||
function fc16($unitId, $reference, $data, $dataTypes){ |
||||
return $this->writeMultipleRegister($unitId, $reference, $data, $dataTypes); |
||||
} |
||||
|
||||
|
||||
/** |
||||
* writeMultipleRegisterPacketBuilder |
||||
* |
||||
* Packet builder FC16 - WRITE multiple register |
||||
* e.g.: 4dd90000000d0010300000030603e807d00bb8 |
||||
* |
||||
* @param int $unitId |
||||
* @param int $reference |
||||
* @param array $data |
||||
* @param array $dataTypes |
||||
* @return string |
||||
*/ |
||||
private function writeMultipleRegisterPacketBuilder($unitId, $reference, $data, $dataTypes){ |
||||
$dataLen = 0; |
||||
// build data section |
||||
$buffer1 = ""; |
||||
foreach($data as $key=>$dataitem) { |
||||
if($dataTypes[$key]=="INT"){ |
||||
$buffer1 .= iecType::iecINT($dataitem); // register values x |
||||
$dataLen += 2; |
||||
} |
||||
elseif($dataTypes[$key]=="DINT"){ |
||||
$buffer1 .= iecType::iecDINT($dataitem, $endianess); // register values x |
||||
$dataLen += 4; |
||||
} |
||||
elseif($dataTypes[$key]=="REAL") { |
||||
$buffer1 .= iecType::iecREAL($dataitem, $endianess); // register values x |
||||
$dataLen += 4; |
||||
} |
||||
else{ |
||||
$buffer1 .= iecType::iecINT($dataitem); // register values x |
||||
$dataLen += 2; |
||||
} |
||||
} |
||||
// build body |
||||
$buffer2 = ""; |
||||
$buffer2 .= iecType::iecBYTE(16); // FC 16 = 16(0x10) |
||||
$buffer2 .= iecType::iecINT($reference); // refnumber = 12288 |
||||
$buffer2 .= iecType::iecINT($dataLen/2); // word count |
||||
$buffer2 .= iecType::iecBYTE($dataLen); // byte count |
||||
$dataLen += 6; |
||||
// build header |
||||
$buffer3 = ''; |
||||
$buffer3 .= iecType::iecINT(rand(0,65000)); // transaction ID |
||||
$buffer3 .= iecType::iecINT(0); // protocol ID |
||||
$buffer3 .= iecType::iecINT($dataLen + 1); // lenght |
||||
$buffer3 .= iecType::iecBYTE($unitId); //unit ID |
||||
|
||||
// return packet string |
||||
return $buffer3. $buffer2. $buffer1; |
||||
} |
||||
|
||||
/** |
||||
* writeMultipleRegisterParser |
||||
* |
||||
* FC16 response parser |
||||
* |
||||
* @param string $packet |
||||
* @return bool |
||||
*/ |
||||
private function writeMultipleRegisterParser($packet){ |
||||
if(!$this->responseCode($rpacket)) |
||||
return false; |
||||
return true; |
||||
} |
||||
|
||||
/** |
||||
* readWriteRegisters |
||||
* |
||||
* Modbus function FC23(0x17) - Read Write Registers. |
||||
* |
||||
* This function writes {@link $data} array at reference {@link $referenceWrite} |
||||
* position of memory of a Modbus device given by {@link $unitId}. Simultanously, |
||||
* it returns {@link $quantity} of Words (2 bytes) from reference {@link $referenceRead}. |
||||
* |
||||
* |
||||
* @param int $unitId usually ID of Modbus device |
||||
* @param int $referenceRead Reference in the device memory to read data (e.g. in device WAGO 750-841, memory MW0 starts at address 12288). |
||||
* @param int $quantity Amounth of the data to be read from device. |
||||
* @param int $referenceWrite Reference in the device memory to write data. |
||||
* @param array $data Array of values to be written. |
||||
* @param array $dataTypes Array of types of values to be written. The array should consists of string "INT", "DINT" and "REAL". |
||||
* @return false|Array Success flag or array of data. |
||||
*/ |
||||
function readWriteRegisters($unitId, $referenceRead, $quantity, $referenceWrite, $data, $dataTypes){ |
||||
$this->errstr = ""; |
||||
$this->status = "readWriteRegisters: START</br>"; |
||||
// connect |
||||
if(!$this->connect()) |
||||
return false; |
||||
// send FC23 |
||||
$packet = $this->readWriteRegistersPacketBuilder($unitId, $referenceRead, $quantity, $referenceWrite, $data, $dataTypes); |
||||
$this->status .= $this->printPacket($packet); |
||||
$this->send($packet); |
||||
// receive response |
||||
$rpacket = $this->rec(); |
||||
if(!$rpacket) |
||||
return false; |
||||
$this->status .= $this->printPacket($rpacket); |
||||
// parse packet |
||||
$receivedData = $this->readWriteRegistersParser($rpacket); |
||||
if(!$receivedData) |
||||
return false; |
||||
// disconnect |
||||
$this->disconnect(); |
||||
$this->status .= "writeMultipleRegister: DONE</br>"; |
||||
// return |
||||
return $receivedData; |
||||
} |
||||
|
||||
/** |
||||
* fc23 |
||||
* |
||||
* Alias to {@link readWriteRegisters} method. |
||||
* |
||||
* @param int $unitId |
||||
* @param int $referenceRead |
||||
* @param int $quantity |
||||
* @param int $referenceWrite |
||||
* @param array $data |
||||
* @param array $dataTypes |
||||
* @return false|Array |
||||
*/ |
||||
function fc23($unitId, $referenceRead, $quantity, $referenceWrite, $data, $dataTypes){ |
||||
return $this->readWriteRegisters($unitId, $referenceRead, $quantity, $referenceWrite, $data, $dataTypes); |
||||
} |
||||
|
||||
/** |
||||
* readWriteRegistersPacketBuilder |
||||
* |
||||
* Packet FC23 builder - READ WRITE registers |
||||
* |
||||
* |
||||
* @param int $unitId |
||||
* @param int $referenceRead |
||||
* @param int $quantity |
||||
* @param int $referenceWrite |
||||
* @param array $data |
||||
* @param array $dataTypes |
||||
* @return string |
||||
*/ |
||||
private function readWriteRegistersPacketBuilder($unitId, $referenceRead, $quantity, $referenceWrite, $data, $dataTypes){ |
||||
$dataLen = 0; |
||||
// build data section |
||||
$buffer1 = ""; |
||||
foreach($data as $key => $dataitem) { |
||||
if($dataTypes[$key]=="INT"){ |
||||
$buffer1 .= iecType::iecINT($dataitem); // register values x |
||||
$dataLen += 2; |
||||
} |
||||
elseif($dataTypes[$key]=="DINT"){ |
||||
$buffer1 .= iecType::iecDINT($dataitem, $endianess); // register values x |
||||
$dataLen += 4; |
||||
} |
||||
elseif($dataTypes[$key]=="REAL") { |
||||
$buffer1 .= iecType::iecREAL($dataitem, $endianess); // register values x |
||||
$dataLen += 4; |
||||
} |
||||
else{ |
||||
$buffer1 .= iecType::iecINT($dataitem); // register values x |
||||
$dataLen += 2; |
||||
} |
||||
} |
||||
// build body |
||||
$buffer2 = ""; |
||||
$buffer2 .= iecType::iecBYTE(23); // FC 23 = 23(0x17) |
||||
// build body - read section |
||||
$buffer2 .= iecType::iecINT($referenceRead); // refnumber = 12288 |
||||
$buffer2 .= iecType::iecINT($quantity); // quantity |
||||
// build body - write section |
||||
$buffer2 .= iecType::iecINT($referenceWrite); // refnumber = 12288 |
||||
$buffer2 .= iecType::iecINT($dataLen/2); // word count |
||||
$buffer2 .= iecType::iecBYTE($dataLen); // byte count |
||||
$dataLen += 10; |
||||
// build header |
||||
$buffer3 = ''; |
||||
$buffer3 .= iecType::iecINT(rand(0,65000)); // transaction ID |
||||
$buffer3 .= iecType::iecINT(0); // protocol ID |
||||
$buffer3 .= iecType::iecINT($dataLen + 1); // lenght |
||||
$buffer3 .= iecType::iecBYTE($unitId); //unit ID |
||||
|
||||
// return packet string |
||||
return $buffer3. $buffer2. $buffer1; |
||||
} |
||||
|
||||
|
||||
/** |
||||
* readWriteRegistersParser |
||||
* |
||||
* FC23 response parser |
||||
* |
||||
* @param string $packet |
||||
* @return array |
||||
*/ |
||||
private function readWriteRegistersParser($packet){ |
||||
$data = array(); |
||||
// if not exception |
||||
if(!$this->responseCode($packet)) |
||||
return false; |
||||
// get data |
||||
for($i=0;$i<ord($packet[8]);$i++){ |
||||
$data[$i] = ord($packet[9+$i]); |
||||
} |
||||
return $data; |
||||
} |
||||
|
||||
/** |
||||
* byte2hex |
||||
* |
||||
* Parse data and get it to the Hex form |
||||
* |
||||
* @param char $value |
||||
* @return string |
||||
*/ |
||||
private function byte2hex($value){ |
||||
$h = dechex(($value >> 4) & 0x0F); |
||||
$l = dechex($value & 0x0F); |
||||
return "$h$l"; |
||||
} |
||||
|
||||
/** |
||||
* printPacket |
||||
* |
||||
* Print whole packet in the hex form |
||||
* |
||||
* @param string $packet |
||||
* @return string |
||||
*/ |
||||
private function printPacket($packet){ |
||||
$str = ""; |
||||
$str .= "Packet: "; |
||||
for($i=0;$i<strlen($packet);$i++){ |
||||
$str .= $this->byte2hex(ord($packet[$i])); |
||||
} |
||||
$str .= "</br>"; |
||||
return $str; |
||||
} |
||||
} |
||||
|
||||
?> |
@ -0,0 +1,219 @@ |
||||
<? |
||||
/** |
||||
* 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; |
||||
} |
||||
} |
||||
?> |
@ -0,0 +1,17 @@ |
||||
{ |
||||
"name": "adduc/phpmodbus", |
||||
"description": "Composer version of PhpModBus", |
||||
"license": "LGPL", |
||||
"authors": [ |
||||
{ |
||||
"name": "Honza Krakora", |
||||
"email": "krakora.jan@googlemail.com" |
||||
} |
||||
], |
||||
"autoload": { |
||||
"classmap": [ "Phpmodbus" ] |
||||
}, |
||||
"require": { |
||||
"ext-sockets": "*" |
||||
} |
||||
} |
@ -0,0 +1,73 @@ |
||||
<?php |
||||
|
||||
require_once dirname(__FILE__) . './Phpmodbus/ModbusMasterUdp.php'; |
||||
|
||||
// Create Modbus object |
||||
$ip = "192.168.1.99"; |
||||
$modbus = new ModbusMasterUdp($ip); |
||||
|
||||
// FC 3 |
||||
$moduleId = 0; |
||||
$reference = 12288; |
||||
$mw0address = 12288; |
||||
$quantity = 6; |
||||
$recData = $modbus->readMultipleRegisters($moduleId, $reference, $quantity); |
||||
|
||||
?> |
||||
<html> |
||||
<head> |
||||
<meta http-equiv="content-type" content="text/html; charset=windows-1250"> |
||||
<meta name="generator" content="PSPad editor, www.pspad.com"> |
||||
<title>WAGO 750-841 M-memory dump</title> |
||||
</head> |
||||
<body> |
||||
<h1>Dump of M-memory from WAGO 750-84x series coupler.</h1> |
||||
<ul> |
||||
<li>PLC: 750-84x series</li> |
||||
<li>IP: <?=$ip?></li>
|
||||
<li>Modbus module ID: <?=$moduleId?></li>
|
||||
<li>Modbus memory reference: <?=$reference?></li>
|
||||
<li>Modbus memory quantity: <?=$quantity?></li>
|
||||
</ul> |
||||
|
||||
<h2>M-memory dump</h2> |
||||
|
||||
<?php |
||||
if(!$recData) { |
||||
// Print error information if any |
||||
echo "</br>Error:</br>" . $modbus->errstr . "</br>"; |
||||
} |
||||
else |
||||
{ |
||||
?> |
||||
|
||||
<table border="1px" width="400px"> |
||||
<tr> |
||||
<td>Modbus address</td> |
||||
<td>MWx</td> |
||||
<td>value</td> |
||||
</tr> |
||||
<?php |
||||
for($i=0;$i<count($recData);$i+=2){ |
||||
?> |
||||
<tr> |
||||
<td><?=$i+$reference?></td>
|
||||
<td>MW<?=($i + $reference - $mw0address)/2?></td>
|
||||
<td><?=($recData[$i] << 8)+ $recData[$i+1]?></td>
|
||||
</tr> |
||||
<?php |
||||
} |
||||
?> |
||||
</table> |
||||
|
||||
<?php |
||||
} |
||||
?> |
||||
|
||||
<h2>Modbus class status</h2> |
||||
<?php |
||||
echo $modbus->status; |
||||
?> |
||||
|
||||
</body> |
||||
</html> |
@ -0,0 +1,55 @@ |
||||
<?php |
||||
|
||||
require_once dirname(__FILE__) . './Phpmodbus/ModbusMasterUdp.php'; |
||||
|
||||
// Create Modbus object |
||||
$modbus = new ModbusMasterUdp("192.168.1.99"); |
||||
|
||||
// FC 3 |
||||
// read 10 words (20 bytes) from device ID=0, address=12288 |
||||
$recData = $modbus->readMultipleRegisters(0, 12288, 10); |
||||
|
||||
if(!$recData) { |
||||
// Print error information if any |
||||
echo "</br>Error:</br>" . $modbus->errstr . "</br>"; |
||||
exit; |
||||
} |
||||
|
||||
// Received data |
||||
echo "<h1>Received Data</h1>"; |
||||
print_r($recData); |
||||
|
||||
// Conversion |
||||
echo "<h2>32 bits types</h2>"; |
||||
// Chunk the data array to set of 4 bytes |
||||
$values = array_chunk($recData, 4); |
||||
|
||||
// Get float from REAL interpretation |
||||
echo "<h3>REAL to Float</h3>"; |
||||
foreach($values as $bytes) |
||||
echo PhpType::bytes2float($bytes) . "</br>"; |
||||
|
||||
// Get integer from DINT interpretation |
||||
echo "<h3>DINT to integer </h3>"; |
||||
foreach($values as $bytes) |
||||
echo PhpType::bytes2signedInt($bytes) . "</br>"; |
||||
|
||||
// Get integer of float from DINT interpretation |
||||
echo "<h3>DWORD to integer (or float) </h3>"; |
||||
foreach($values as $bytes) |
||||
echo PhpType::bytes2unsignedInt($bytes) . "</br>"; |
||||
|
||||
echo "<h2>16 bit types</h2>"; |
||||
// Chunk the data array to set of 4 bytes |
||||
$values = array_chunk($recData, 2); |
||||
|
||||
// Get signed integer from INT interpretation |
||||
echo "<h3>INT to integer </h3>"; |
||||
foreach($values as $bytes) |
||||
echo PhpType::bytes2signedInt($bytes) . "</br>"; |
||||
|
||||
// Get unsigned integer from WORD interpretation |
||||
echo "<h3>WORD to integer </h3>"; |
||||
foreach($values as $bytes) |
||||
echo PhpType::bytes2unsignedInt($bytes) . "</br>"; |
||||
?> |
@ -0,0 +1,21 @@ |
||||
<?php |
||||
|
||||
require_once dirname(__FILE__) . './Phpmodbus/ModbusMasterUdp.php'; |
||||
|
||||
// Create Modbus object |
||||
$modbus = new ModbusMasterUdp("192.168.1.99"); |
||||
|
||||
// Data to be writen |
||||
$data = array(1000, 2000, 3.0); |
||||
$dataTypes = array("INT", "DINT", "REAL"); |
||||
|
||||
// FC16 |
||||
if(!$modbus->writeMultipleRegister(0, 12288, $data, $dataTypes)) { |
||||
// Print error information if any |
||||
echo "</br>Error:</br>" . $modbus->errstr . "</br>"; |
||||
} |
||||
|
||||
// Print status information |
||||
echo "</br>Status:</br>" . $modbus->status . "</br>"; |
||||
|
||||
?> |
@ -0,0 +1,26 @@ |
||||
<?php |
||||
|
||||
require_once dirname(__FILE__) . './Phpmodbus/ModbusMasterUdp.php'; |
||||
|
||||
// Create Modbus object |
||||
$modbus = new ModbusMasterUdp("192.168.1.99"); |
||||
|
||||
// Data to be writen |
||||
$data = array(1000, 2000, 3.0); |
||||
$dataTypes = array("INT", "DINT", "REAL"); |
||||
|
||||
// FC23 |
||||
$recData = $modbus->readWriteRegisters(0, 12288, 6, 12288, $data, $dataTypes); |
||||
|
||||
if(!$recData) { |
||||
// Print error information if any |
||||
echo "</br>Error:</br>" . $modbus->errstr . "</br>"; |
||||
} |
||||
|
||||
// Print status information |
||||
echo "</br>Status:</br>" . $modbus->status . "</br>"; |
||||
|
||||
// Print read data |
||||
echo "</br>Data:</br>"; print_r($recData); echo "</br>"; |
||||
|
||||
?> |
@ -0,0 +1,21 @@ |
||||
<?php |
||||
|
||||
require_once dirname(__FILE__) . './Phpmodbus/ModbusMasterUdp.php'; |
||||
|
||||
// Create Modbus object |
||||
$modbus = new ModbusMasterUdp("192.168.1.99"); |
||||
|
||||
// FC 3 |
||||
$recData = $modbus->readMultipleRegisters(0, 12288, 6); |
||||
|
||||
if(!$recData) { |
||||
// Print error information if any |
||||
echo "</br>Error:</br>" . $modbus->errstr . "</br>"; |
||||
} |
||||
|
||||
// Print status information |
||||
echo "</br>Status:</br>" . $modbus->status . "</br>"; |
||||
|
||||
// Print read data |
||||
echo "</br>Data:</br>"; print_r($recData); echo "</br>"; |
||||
?> |
@ -0,0 +1,62 @@ |
||||
The Phpmodbus License, Version 1 |
||||
============================ |
||||
|
||||
Copyright (c) 2004, 2008 Jan Krakora, Wago (http://www.wago.com) |
||||
All rights reserved. |
||||
|
||||
This license is a legal agreement between you and Jan Krakora, Wago (the "Author") |
||||
for the use of Phpmodbus (the "Software"). By obtaining, using and/or |
||||
copying the Software, you agree that you have read, understood, and will |
||||
comply with the terms and conditions of this license. |
||||
|
||||
|
||||
PERMITTED USE |
||||
------------- |
||||
|
||||
You are permitted to use, copy, modify, and distribute the Software and its |
||||
documentation, with or without modification, for any purpose, provided that |
||||
the following conditions are met: |
||||
|
||||
1. A copy of this license agreement must be included with the distribution. |
||||
|
||||
2. Redistributions of source code must retain the above copyright notice in |
||||
all source code files. |
||||
|
||||
3. Redistributions in binary form must reproduce the above copyright notice |
||||
in the documentation and/or other materials provided with the distribution. |
||||
|
||||
4. Products derived from the Software must include an acknowledgment that |
||||
they are derived from "Phpmodbus" in their documentation and/or other |
||||
materials provided with the distribution. |
||||
|
||||
5. The name "Phpmodbus" must not be used to endorse or promote products |
||||
derived from the Software without prior written permission from Author. |
||||
|
||||
6. Products derived from the Software may not be called "Phpmodbus", |
||||
nor may "Phpmodbus" appear in their name, without prior written |
||||
permission from Author. |
||||
|
||||
|
||||
INDEMNITY |
||||
--------- |
||||
|
||||
You agree to indemnify and hold harmless the Author and any contributors |
||||
for any direct, indirect, incidental, or consequential third-party claims, |
||||
actions or suits, as well as any related expenses, liabilities, damages, |
||||
settlements or fees arising from your use or misuse of the Software, |
||||
or a violation of any terms of this license. |
||||
|
||||
|
||||
DISCLAIMER OF WARRANTY |
||||
---------------------- |
||||
|
||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER "AS IS" AND |
||||
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED |
||||
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE |
||||
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE FOR |
||||
ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES |
||||
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; |
||||
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON |
||||
ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS |
||||
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
@ -0,0 +1,19 @@ |
||||
Copyright (c) 2004, 2009 Jan Krakora, Wago (http://www.wago.com) |
||||
All rights reserved. |
||||
|
||||
Phpmodbus library |
||||
#################### |
||||
|
||||
Phpmodbus for PHP is a small and easy-to-use Modbus UDP master library. For more |
||||
see project at http://phpmodbus.googlecode.com |
||||
|
||||
Release notes |
||||
=============== |
||||
|
||||
0.1 -> 0.2.r20 |
||||
--------------- |
||||
+ Added new class for conversion from received bytes to PHP data types (PhpType class) |
||||
+ Added new data conversion using PhpType example |
||||
+ Added new alias methods fc3, fc16 and fc23 (ModbusMasterUdp class) |
||||
* Fixed problems with the endianess when data written (IecType class) |
||||
* Improved commentaries for documentation |
@ -0,0 +1,11 @@ |
||||
@echo off |
||||
call ../config.bat |
||||
|
||||
for %%f in (test.*.php) do %php% -q "%%f" > "output/%%f.html" |
||||
|
||||
cd output |
||||
for %%f in (*.html) do %diff% "%%f" ../ref/"%%f" |
||||
cd .. |
||||
pause |
||||
|
||||
@echo on |
@ -0,0 +1 @@ |
||||
125<br>98<br>0<br>0<br>0<br>0<br>0<br>0<br>0<br>0<br>0<br>0<br>255<br>255<br>255<br>255<br>158<br>88<br>97<br>168<br> |
@ -0,0 +1 @@ |
||||
Endianing off <hr>0 --> Packet: 0000_0000_</br>1 --> Packet: 0001_0000_</br>-1 --> Packet: ffff_ffff_</br>2147483647 --> Packet: ffff_7fff_</br>-2147483648 --> Packet: 0000_8000_</br>Endianing on <hr>0 --> Packet: 0000_0000_</br>1 --> Packet: 0000_0001_</br>-1 --> Packet: ffff_ffff_</br>2147483647 --> Packet: 7fff_ffff_</br>-2147483648 --> Packet: 8000_0000_</br> |
@ -0,0 +1 @@ |
||||
Endianing off <hr>0 --> Packet: 0000_</br>1 --> Packet: 0001_</br>-1 --> Packet: ffff_</br>32767 --> Packet: 7fff_</br>-32768 --> Packet: 8000_</br>Endianing on <hr>0 --> Packet: 0000_</br>1 --> Packet: 0001_</br>-1 --> Packet: ffff_</br>32767 --> Packet: 7fff_</br>-32768 --> Packet: 8000_</br> |
@ -0,0 +1 @@ |
||||
Endianing off <hr>0 --> Packet: 0000_0000_</br>1 --> Packet: 0000_3f80_</br>-2 --> Packet: 0000_c000_</br>0.333333333333 --> Packet: aaaa_3eaa_</br>25 --> Packet: 0000_41c8_</br>Endianing on <hr>0 --> Packet: 0000_0000_</br>1 --> Packet: 3f80_0000_</br>-2 --> Packet: c000_0000_</br>0.333333333333 --> Packet: 3eaa_aaaa_</br>25 --> Packet: 41c8_0000_</br> |
@ -0,0 +1,33 @@ |
||||
<?php |
||||
|
||||
require_once dirname(__FILE__) . '/../../Phpmodbus/ModbusMasterUdp.php'; |
||||
|
||||
// Received bytes interpreting Mixed values |
||||
$data = Array ( |
||||
"0" => 125, // 32098 (DINT) |
||||
"1" => 98, |
||||
"2" => 0, |
||||
"3" => 0, |
||||
"4" => 0, // 0 (DINT) |
||||
"5" => 0, |
||||
"6" => 0, |
||||
"7" => 0, |
||||
"8" => 0, // 0 (DINT) |
||||
"9" => 0, |
||||
"10" => 0, |
||||
"11" => 0, |
||||
"12" => 255, // -1 (DINT) |
||||
"13" => 255, |
||||
"14" => 255, |
||||
"15" => 255, |
||||
"16" => 158, // -25000 (INT) |
||||
"17" => 88, |
||||
"18" => 97, // 25000 (INT) |
||||
"19" => 168 |
||||
); |
||||
|
||||
// Print mixed values |
||||
foreach($data as $d) |
||||
echo ord(IecType::iecBYTE($d)) . "<br>"; |
||||
|
||||
?> |
@ -0,0 +1,50 @@ |
||||
<?php |
||||
|
||||
require_once dirname(__FILE__) . '/../../Phpmodbus/ModbusMasterUdp.php'; |
||||
|
||||
// Received bytes interpreting Mixed values |
||||
$data = Array ( |
||||
"0" => 0, |
||||
"1" => 1, |
||||
"2" => -1, |
||||
"3" => pow(2,31)-1, |
||||
"4" => -pow(2,31) |
||||
); |
||||
|
||||
function byte2hex($value){ |
||||
$h = dechex(($value >> 4) & 0x0F); |
||||
$l = dechex($value & 0x0F); |
||||
return "$h$l"; |
||||
} |
||||
|
||||
function printPacket($packet){ |
||||
$str = ""; |
||||
$str .= "Packet: "; |
||||
for($i=0;$i<strlen($packet);$i++){ |
||||
$str .= byte2hex(ord($packet[$i])); |
||||
if($i % 2) |
||||
$str .= "_"; |
||||
} |
||||
$str .= "</br>"; |
||||
return $str; |
||||
} |
||||
|
||||
echo "Endianing off <hr>"; |
||||
// Print mixed values |
||||
for($i=0;$i<count($data);$i++) { |
||||
echo $data[$i] . " --> "; |
||||
$v = IecType::iecDINT($data[$i], 0); |
||||
echo printPacket($v); |
||||
"<br>"; |
||||
} |
||||
|
||||
echo "Endianing on <hr>"; |
||||
// Print mixed values |
||||
for($i=0;$i<count($data);$i++) { |
||||
echo $data[$i] . " --> "; |
||||
$v = IecType::iecDINT($data[$i], 1); |
||||
echo printPacket($v); |
||||
"<br>"; |
||||
} |
||||
|
||||
?> |
@ -0,0 +1,50 @@ |
||||
<?php |
||||
|
||||
require_once dirname(__FILE__) . '/../../Phpmodbus/ModbusMasterUdp.php'; |
||||
|
||||
// Received bytes interpreting Mixed values |
||||
$data = Array ( |
||||
"0" => 0, |
||||
"1" => 1, |
||||
"2" => -1, |
||||
"3" => pow(2,15)-1, |
||||
"4" => -pow(2,15) |
||||
); |
||||
|
||||
function byte2hex($value){ |
||||
$h = dechex(($value >> 4) & 0x0F); |
||||
$l = dechex($value & 0x0F); |
||||
return "$h$l"; |
||||
} |
||||
|
||||
function printPacket($packet){ |
||||
$str = ""; |
||||
$str .= "Packet: "; |
||||
for($i=0;$i<strlen($packet);$i++){ |
||||
$str .= byte2hex(ord($packet[$i])); |
||||
if($i % 2) |
||||
$str .= "_"; |
||||
} |
||||
$str .= "</br>"; |
||||
return $str; |
||||
} |
||||
|
||||
echo "Endianing off <hr>"; |
||||
// Print mixed values |
||||
for($i=0;$i<count($data);$i++) { |
||||
echo $data[$i] . " --> "; |
||||
$v = IecType::iecINT($data[$i], 0); |
||||
echo printPacket($v); |
||||
"<br>"; |
||||
} |
||||
|
||||
echo "Endianing on <hr>"; |
||||
// Print mixed values |
||||
for($i=0;$i<count($data);$i++) { |
||||
echo $data[$i] . " --> "; |
||||
$v = IecType::iecINT($data[$i], 1); |
||||
echo printPacket($v); |
||||
"<br>"; |
||||
} |
||||
|
||||
?> |
@ -0,0 +1,50 @@ |
||||
<?php |
||||
|
||||
require_once dirname(__FILE__) . '/../../Phpmodbus/ModbusMasterUdp.php'; |
||||
|
||||
// http://en.wikipedia.org/wiki/Single_precision |
||||
$data = Array ( |
||||
"0" => 0, // -> 0000 0000 |
||||
"1" => 1, // -> 3f80 0000 |
||||
"2" => -2, // -> c000 0000 |
||||
"3" => 1/3, // -> 3eaa aaab |
||||
"4" => 25 // -> 41c8 0000 |
||||
); |
||||
|
||||
function byte2hex($value){ |
||||
$h = dechex(($value >> 4) & 0x0F); |
||||
$l = dechex($value & 0x0F); |
||||
return "$h$l"; |
||||
} |
||||
|
||||
function printPacket($packet){ |
||||
$str = ""; |
||||
$str .= "Packet: "; |
||||
for($i=0;$i<strlen($packet);$i++){ |
||||
$str .= byte2hex(ord($packet[$i])); |
||||
if($i % 2) |
||||
$str .= "_"; |
||||
} |
||||
$str .= "</br>"; |
||||
return $str; |
||||
} |
||||
|
||||
echo "Endianing off <hr>"; |
||||
// Print mixed values |
||||
for($i=0;$i<count($data);$i++) { |
||||
echo $data[$i] . " --> "; |
||||
$v = IecType::iecREAL($data[$i], 0); |
||||
echo printPacket($v); |
||||
"<br>"; |
||||
} |
||||
|
||||
echo "Endianing on <hr>"; |
||||
// Print mixed values |
||||
for($i=0;$i<count($data);$i++) { |
||||
echo $data[$i] . " --> "; |
||||
$v = IecType::iecREAL($data[$i], 1); |
||||
echo printPacket($v); |
||||
"<br>"; |
||||
} |
||||
|
||||
?> |
@ -0,0 +1,11 @@ |
||||
@echo off |
||||
call ../config.bat |
||||
|
||||
for %%f in (test.*.php) do %php% -q "%%f" > "output/%%f.html" |
||||
|
||||
cd output |
||||
for %%f in (*.html) do %diff% "%%f" ../ref/"%%f" |
||||
cd .. |
||||
pause |
||||
|
||||
@echo on |
@ -0,0 +1,72 @@ |
||||
Array |
||||
( |
||||
[0] => 0 |
||||
[1] => 0 |
||||
[2] => 0 |
||||
[3] => 1 |
||||
[4] => 0 |
||||
[5] => 1 |
||||
[6] => 0 |
||||
[7] => 255 |
||||
[8] => 0 |
||||
[9] => 255 |
||||
) |
||||
Array |
||||
( |
||||
[0] => 0 |
||||
[1] => 0 |
||||
[2] => 0 |
||||
[3] => 1 |
||||
[4] => 255 |
||||
[5] => 255 |
||||
[6] => 127 |
||||
[7] => 255 |
||||
[8] => 128 |
||||
[9] => 0 |
||||
) |
||||
Array |
||||
( |
||||
[0] => 0 |
||||
[1] => 0 |
||||
[2] => 0 |
||||
[3] => 0 |
||||
[4] => 0 |
||||
[5] => 1 |
||||
[6] => 0 |
||||
[7] => 0 |
||||
[8] => 255 |
||||
[9] => 255 |
||||
[10] => 255 |
||||
[11] => 255 |
||||
[12] => 255 |
||||
[13] => 255 |
||||
[14] => 127 |
||||
[15] => 255 |
||||
[16] => 0 |
||||
[17] => 0 |
||||
[18] => 128 |
||||
[19] => 0 |
||||
) |
||||
Array |
||||
( |
||||
[0] => 0 |
||||
[1] => 0 |
||||
[2] => 0 |
||||
[3] => 0 |
||||
[4] => 0 |
||||
[5] => 0 |
||||
[6] => 63 |
||||
[7] => 128 |
||||
[8] => 0 |
||||
[9] => 0 |
||||
[10] => 192 |
||||
[11] => 0 |
||||
[12] => 170 |
||||
[13] => 170 |
||||
[14] => 62 |
||||
[15] => 170 |
||||
[16] => 0 |
||||
[17] => 0 |
||||
[18] => 65 |
||||
[19] => 200 |
||||
) |
@ -0,0 +1 @@ |
||||
writeMultipleRegister (FC26): DONE |
@ -0,0 +1,44 @@ |
||||
<?php |
||||
require_once dirname(__FILE__) . '/../../Phpmodbus/ModbusMasterUdp.php'; |
||||
require_once dirname(__FILE__) . '/../config.php'; |
||||
|
||||
// Create Modbus object |
||||
$modbus = new ModbusMasterUdp($testip); |
||||
|
||||
// Data to be writen - BYTE |
||||
$data = array(0, 1, 1, pow(2,8)-1, pow(2,8)-1); |
||||
$dataTypes = array("BYTE", "BYTE", "BYTE", "BYTE", "BYTE"); |
||||
// Write data - FC 16 |
||||
$modbus->writeMultipleRegister(0, 12288, $data, $dataTypes); |
||||
// Read data - FC3 |
||||
$recData = $modbus->readMultipleRegisters(0, 12288, 5); |
||||
print_r($recData); |
||||
|
||||
// Data to be writen - INT |
||||
$data = array(0, 1, -1, pow(2,15)-1, -pow(2,15)); |
||||
$dataTypes = array("INT", "INT", "INT", "INT", "INT"); |
||||
// Write data - FC 16 |
||||
$modbus->writeMultipleRegister(0, 12288, $data, $dataTypes); |
||||
// Read data - FC3 |
||||
$recData = $modbus->readMultipleRegisters(0, 12288, 5); |
||||
print_r($recData); |
||||
|
||||
// Data to be writen - DINT |
||||
$data = array(0, 1, -1, pow(2,31)-1, -pow(2,31)); |
||||
$dataTypes = array("DINT", "DINT", "DINT", "DINT", "DINT"); |
||||
// Write data - FC 16 |
||||
$modbus->writeMultipleRegister(0, 12288, $data, $dataTypes); |
||||
// Read data - FC3 |
||||
$recData = $modbus->readMultipleRegisters(0, 12288, 10); |
||||
print_r($recData); |
||||
|
||||
// Data to be writen - REAL |
||||
$data = array(0, 1, -2, 1/3, 25); |
||||
$dataTypes = array("REAL", "REAL", "REAL", "REAL", "REAL"); |
||||
// Write data - FC 16 |
||||
$modbus->writeMultipleRegister(0, 12288, $data, $dataTypes); |
||||
// Read data - FC3 |
||||
$recData = $modbus->readMultipleRegisters(0, 12288, 10); |
||||
print_r($recData); |
||||
|
||||
?> |
@ -0,0 +1,24 @@ |
||||
<?php |
||||
require_once dirname(__FILE__) . '/../../Phpmodbus/ModbusMasterUdp.php'; |
||||
require_once dirname(__FILE__) . '/../config.php'; |
||||
|
||||
// Create Modbus object |
||||
$modbus = new ModbusMasterUdp($testip); |
||||
|
||||
// Data to be writen |
||||
$data = array(1000, 2000, 1.250, 1.250); |
||||
$dataTypes = array("REAL", "REAL", "REAL", "REAL"); |
||||
|
||||
// FC23 |
||||
$recData = $modbus->readWriteRegisters(0, 12288, 6, 12288, $data, $dataTypes); |
||||
|
||||
if(!$recData) { |
||||
// Print error information if any |
||||
echo "</br>Error:</br>" . $modbus->errstr . "</br>"; |
||||
// |
||||
exit(); |
||||
} |
||||
|
||||
// Print status information |
||||
echo "writeMultipleRegister (FC26): DONE"; |
||||
?> |
@ -0,0 +1,11 @@ |
||||
@echo off |
||||
call ../config.bat |
||||
|
||||
for %%f in (test.*.php) do %php% -q "%%f" > "output/%%f.html" |
||||
|
||||
cd output |
||||
for %%f in (*.html) do %diff% "%%f" ../ref/"%%f" |
||||
cd .. |
||||
pause |
||||
|
||||
@echo on |
@ -0,0 +1 @@ |
||||
32098<br>0<br>0<br>-1<br>-25000<br>25000<br> |
@ -0,0 +1 @@ |
||||
1000<br>2000<br>1.25<br> |
@ -0,0 +1 @@ |
||||
-1<br>0<br>1<br>-2147483648<br>2147483647<br> |
@ -0,0 +1,6 @@ |
||||
float(4294967295) |
||||
<br>int(0) |
||||
<br>int(1) |
||||
<br>float(2147483648) |
||||
<br>int(2147483647) |
||||
<br> |
@ -0,0 +1,37 @@ |
||||
<?php |
||||
|
||||
require_once dirname(__FILE__) . '/../../Phpmodbus/ModbusMasterUdp.php'; |
||||
|
||||
// Received bytes interpreting Mixed values |
||||
$data = Array ( |
||||
"0" => 125, // 32098 (DINT) |
||||
"1" => 98, |
||||
"2" => 0, |
||||
"3" => 0, |
||||
"4" => 0, // 0 (DINT) |
||||
"5" => 0, |
||||
"6" => 0, |
||||
"7" => 0, |
||||
"8" => 0, // 0 (DINT) |
||||
"9" => 0, |
||||
"10" => 0, |
||||
"11" => 0, |
||||
"12" => 255, // -1 (DINT) |
||||
"13" => 255, |
||||
"14" => 255, |
||||
"15" => 255, |
||||
"16" => 158, // -25000 (INT) |
||||
"17" => 88, |
||||
"18" => 97, // 25000 (INT) |
||||
"19" => 168 |
||||
); |
||||
|
||||
// Print mixed values |
||||
echo PhpType::bytes2unsignedInt(array_slice($data, 0, 4)) . "<br>"; |
||||
echo PhpType::bytes2signedInt(array_slice($data, 4, 4)) . "<br>"; |
||||
echo PhpType::bytes2signedInt(array_slice($data, 8, 4)) . "<br>"; |
||||
echo PhpType::bytes2signedInt(array_slice($data, 12, 4)) . "<br>"; |
||||
echo PhpType::bytes2signedInt(array_slice($data, 16, 2)) . "<br>"; |
||||
echo PhpType::bytes2signedInt(array_slice($data, 18, 2)) . "<br>"; |
||||
|
||||
?> |
@ -0,0 +1,27 @@ |
||||
<?php |
||||
|
||||
require_once dirname(__FILE__) . '/../../Phpmodbus/ModbusMasterUdp.php'; |
||||
|
||||
// Received bytes interpreting 3 REAL values (6 words) |
||||
$data = array( // 1000 |
||||
0 => 0, |
||||
1 => 0, |
||||
2 => 68, |
||||
3 => 122, |
||||
4 => 0, |
||||
5 => 0, |
||||
6 => 68, |
||||
7 => 250, |
||||
8 => 0, |
||||
9 => 0, |
||||
10 => 63, |
||||
11 => 160, |
||||
); |
||||
|
||||
$dword = array_chunk($data, 4); |
||||
|
||||
// Print float interpretation of the real value |
||||
echo PhpType::bytes2float($dword[0]) . "<br>"; |
||||
echo PhpType::bytes2float($dword[1]) . "<br>"; |
||||
echo PhpType::bytes2float($dword[2]) . "<br>"; |
||||
?> |
@ -0,0 +1,35 @@ |
||||
<?php |
||||
|
||||
require_once dirname(__FILE__) . '/../../Phpmodbus/ModbusMasterUdp.php'; |
||||
|
||||
// Received bytes interpreting DINT values |
||||
$data = array( |
||||
0xFF, // -1 |
||||
0xFF, |
||||
0xFF, |
||||
0xFF, |
||||
0, // 0 |
||||
0, |
||||
0, |
||||
0, |
||||
0, // 1 |
||||
0x01, |
||||
0, |
||||
0, |
||||
0, // minus max |
||||
0, |
||||
0x80, |
||||
0x0, |
||||
0xFF, // plus max |
||||
0xFF, |
||||
0x7F, |
||||
0xFF, |
||||
); |
||||
|
||||
$dword = array_chunk($data, 4); |
||||
|
||||
// Print float interpretation of the real value |
||||
foreach($dword as $value) { |
||||
echo PhpType::bytes2signedInt($value) . "<br>"; |
||||
} |
||||
?> |
@ -0,0 +1,36 @@ |
||||
<?php |
||||
|
||||
require_once dirname(__FILE__) . '/../../Phpmodbus/ModbusMasterUdp.php'; |
||||
|
||||
// Received bytes interpreting DINT values |
||||
$data = array( |
||||
0xFF, // -1 |
||||
0xFF, |
||||
0xFF, |
||||
0xFF, |
||||
0, // 0 |
||||
0, |
||||
0, |
||||
0, |
||||
0, // 1 |
||||
0x01, |
||||
0, |
||||
0, |
||||
0, // minus max |
||||
0, |
||||
0x80, |
||||
0x0, |
||||
0xFF, // plus max |
||||
0xFF, |
||||
0x7F, |
||||
0xFF, |
||||
); |
||||
|
||||
$dword = array_chunk($data, 4); |
||||
|
||||
// Print float interpretation of the real value |
||||
foreach($dword as $value) { |
||||
var_dump(PhpType::bytes2unsignedInt($value)); |
||||
echo "<br>"; |
||||
} |
||||
?> |
@ -0,0 +1,5 @@ |
||||
set php=C:\Programme\xampplite\php\php.exe |
||||
rem set php=C:\PHP\versions\php-5.2.6-Win32\php.exe -d auto_prepend_file=C:\PHP\locale.php |
||||
set diff="C:\Program Files\Octave\msys\bin\diff.exe" |
||||
set testUri=http://localHost/nette/_trunk/tests |
||||
set wget=wget.exe |
@ -0,0 +1,3 @@ |
||||
<?php |
||||
$testip = "192.168.1.15"; |
||||
?> |
@ -0,0 +1,121 @@ |
||||
<refentry id="{@id}"> |
||||
<refnamediv> |
||||
<refname>Phpmodbus user's guide</refname> |
||||
<refpurpose>Phpmodbus How-to</refpurpose> |
||||
</refnamediv> |
||||
<refsynopsisdiv> |
||||
<author> |
||||
Jan Krakora |
||||
<authorblurb> |
||||
{@link mailto:jak.krakora@wago.com email} |
||||
</authorblurb> |
||||
</author> |
||||
</refsynopsisdiv> |
||||
{@toc} |
||||
<refsect1 id="{@id intro}"> |
||||
<title>Introduction</title> |
||||
<para> |
||||
Phpmodbus is a PHP library for the Modbus protocol. The library implements |
||||
Modbus UDP master class with subset of the most used Modbus commands. |
||||
</para> |
||||
<para> |
||||
The library implements: |
||||
<itemizedlist> |
||||
<listitem><para>FC 3: read multiple registers</para></listitem> |
||||
<listitem><para>FC 16: write multiple registers</para></listitem> |
||||
<listitem><para>FC 23: read write registers</para></listitem> |
||||
</itemizedlist> |
||||
</para> |
||||
<para> |
||||
For more about Modbus protocol see [{@link http://www.modbus.org}] or |
||||
[{@link http://en.wikipedia.org/wiki/Modbus Wiki}] |
||||
</para> |
||||
<note> |
||||
Developed with support of <graphic fileref="wago_logo.png"/> {@link http://www.wago.com}. |
||||
</note> |
||||
</refsect1> |
||||
<refsect1 id="{@id intro}"> |
||||
<title>Installation</title> |
||||
<para> |
||||
At the first, it is supposed an PHP solution has been already installed on |
||||
your server (LAMP, WAMP, MSS+CGI etc.). |
||||
</para> |
||||
<para> |
||||
Copy the Phpmodbus library to your PHP project folder. |
||||
</para> |
||||
<para> |
||||
Create a PHP script and assign the library using |
||||
<programlisting role="ini"> |
||||
<![CDATA[ require_once dirname(__FILE__) . './Phpmodbus/ModbusMasterUdp.php'; ]]> |
||||
</programlisting> |
||||
</para> |
||||
<para> |
||||
To create the Modbus master object, that will communicate with a Modbus slave |
||||
at IP address 192.168.1.1, write |
||||
<programlisting role="ini"> |
||||
<![CDATA[ $modbus = new ModbusMasterUdp("192.168.1.1"); ]]> |
||||
</programlisting> |
||||
</para> |
||||
<para> |
||||
To read 5 words (10 bytes) located at the memory beginning at address = 12288 of |
||||
the devide with Modbus ID=0 use |
||||
<programlisting role="ini"> |
||||
<![CDATA[ $recData = $modbus->readMultipleRegisters(0, 12288, 5); ]]> |
||||
</programlisting> |
||||
</para> |
||||
<para> |
||||
For other examples see sections bellow. |
||||
</para> |
||||
</refsect1> |
||||
<refsect1 id="{@id examples}"> |
||||
<title>Examples</title> |
||||
<para> |
||||
This section presents library features and examples |
||||
</para> |
||||
<refsect2 id="{@id example_fc3}"> |
||||
<title>FC3 - read mutliple registers</title> |
||||
<para> |
||||
FC3 functionality example |
||||
</para> |
||||
<para> |
||||
{@example example_fc3.php} |
||||
</para> |
||||
</refsect2> |
||||
<refsect2 id="{@id example_fc16}"> |
||||
<title>FC16 - write mutliple registers</title> |
||||
<para> |
||||
FC16 functionality example |
||||
</para> |
||||
<para> |
||||
{@example example_fc16.php} |
||||
</para> |
||||
</refsect2> |
||||
<refsect2 id="{@id example_fc23}"> |
||||
<title>FC23 - read write registers</title> |
||||
<para> |
||||
FC23 functionality example |
||||
</para> |
||||
<para> |
||||
{@example example_fc23.php} |
||||
</para> |
||||
</refsect2> |
||||
<refsect2 id="{@id wago_example}"> |
||||
<title>Dump of M-memory from WAGO 750-84x series coupler.</title> |
||||
<para> |
||||
Dump of M-memory from WAGO 750-84x series coupler. |
||||
</para> |
||||
<para> |
||||
{@example example_750841_Mmemory.php} |
||||
</para> |
||||
</refsect2> |
||||
<refsect2 id="{@id datatype}"> |
||||
<title>Data conversion to PHP types.</title> |
||||
<para> |
||||
Conversion of the data bytes, received from Modbus, to a PHP type. |
||||
</para> |
||||
<para> |
||||
{@example example_datatype.php} |
||||
</para> |
||||
</refsect2> |
||||
</refsect1> |
||||
</refentry> |
After Width: | Height: | Size: 1.2 KiB |
Loading…
Reference in new issue