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