ModBusTCP with some fixes

Revision:
2:fcd20e2cd110
Parent:
1:b724fdb741e7
Child:
3:ebea8e061ae6
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/ModBus-TCP.cpp	Thu Jun 29 10:17:29 2017 +0000
@@ -0,0 +1,137 @@
+/* Copyright (C) 2013 Hiroshi Suga, MIT License
+ *
+ * Permission is hereby granted, free of charge, to any person obtaining a copy of this software
+ * and associated documentation files (the "Software"), to deal in the Software without restriction,
+ * including without limitation the rights to use, copy, modify, merge, publish, distribute,
+ * sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is
+ * furnished to do so, subject to the following conditions:
+ *
+ * The above copyright notice and this permission notice shall be included in all copies or
+ * substantial portions of the Software.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING
+ * BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+ * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
+ * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+ */
+
+#include "ModBus-TCP.h"
+
+Modbus * Modbus::_inst;
+
+Modbus::Modbus () {
+    _inst = this;
+    memset(_state, 0, sizeof(_state));
+    _handler_count = 0;
+}
+
+int Modbus::start (NetworkStack *ns, int port) {
+    int i;
+
+    m_ns = ns;
+    
+    for (i = 0; i < MODBUS_MAX_CLIENTS; i ++) {
+        _state[i].buf = new CircBuffer<char>(MODBUS_BUF_SIZE);
+        _state[i].thread = new Thread(osPriorityNormal, MODBUS_STACK_SIZE);
+        _state[i].client = new TCPSocket();
+        _state[i].thread->start(callback(child, (void*)i));
+    }
+    
+#ifdef MODBUS_ENABLE_CLOSER
+    _state[MODBUS_MAX_CLIENTS].thread = new Thread(closer, (void*)MODBUS_MAX_CLIENTS, osPriorityNormal, 128);
+    _state[MODBUS_MAX_CLIENTS].client = new TCPSocket(m_ns);
+#endif
+
+    _server.open(m_ns);
+    _server.bind(port);
+    _server.set_blocking(true);
+    _server.listen();
+    _daemon = new Thread(osPriorityNormal, MODBUS_STACK_SIZE);
+    _daemon->start(Modbus::daemon);
+    return 0;
+}
+
+void Modbus::daemon () {
+    Modbus *modbus = Modbus::getInstance();
+    int i, t = 0;
+
+    INFO("Wait for new connection...\r\n");
+    for (;;) {
+        if (t >= 0) {
+            if (modbus->_server.accept(modbus->_state[t].client) == 0) {
+                INFO("accept %d\r\n", t);
+                modbus->_state[t].thread->signal_set(1);
+            }
+        } else {
+#ifdef HTTPD_ENABLE_CLOSER
+            if (modbus->_server.accept(modbus->_state[MODBUS_MAX_CLIENTS].client) == 0) {
+                INFO("accept x\r\n");
+                modbus->_state[MODBUS_MAX_CLIENTS].thread->signal_set(1);
+            }
+#endif
+        }
+
+        t = -1;
+        for (i = 0; i < MODBUS_MAX_CLIENTS; i ++) {
+            if (modbus->_state[i].thread->get_state() == Thread::WaitingAnd) {
+                if (t < 0) t = i; // next empty thread
+            }
+        }
+    }
+}
+
+void Modbus::child (void const *arg) {
+    Modbus *modbus = ModBus::getInstance();
+    int id = (int)arg;
+    int i, n;
+    char buf[ModBus_BUF_SIZE];
+
+    for (;;) {
+        Thread::signal_wait(1);
+        modbus->_state[id].mode = MODE_REQUEST;
+        modbus->_state[id].buf->flush();
+        modbus->_state[id].keepalive = 0;
+        INFO("Connection from client\r\n");
+//      INFO("Connection from %s\r\n", httpd->_state[id].client->get_ip_address());
+
+        modbus->_state[id].client->set_blocking(false);
+        modbus->_state[id].client->set_timeout(HTTPD_TIMEOUT);
+        
+        for (;;) {
+            //if (! httpd->_state[id].client->is_connected()) break;
+
+            n = modbus->_state[id].client->recv(buf, sizeof(buf));
+            
+            if (n < 0 ) {
+                printf("HTTPD::child breaking n = %d\r\n", n);
+                break;
+            }
+            buf[n] = 0;
+            //DBG("Recv %d ", n);
+            DBG("Recv %d '%s'", n, buf);
+
+            for (i = 0; i < n; i ++) {
+                modbus->recvData(id, buf[i]);
+            }
+        }
+
+        modbus->_state[id].client->close(); // Needs to bere moved 
+        INFO("Closed client connection\r\n");
+        //INFO("Close %s\r\n", httpd->_state[id].client->get_ip_address());
+    }
+}
+
+void Modbus::closer (void const *arg) {
+    Modbus *modbus = Modbus::getInstance();
+    int id = (int)arg;
+
+    for (;;) {
+        Thread::signal_wait(1);
+
+        modbus->_state[id].client->close();
+        INFO("Closed client connection\r\n");
+        //INFO("Close %s\r\n", httpd->_state[id].client->get_ip_address());
+    }
+}
+