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