unit-test-server.c 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229
  1. /*
  2. * Copyright © Stéphane Raimbault <stephane.raimbault@gmail.com>
  3. *
  4. * SPDX-License-Identifier: BSD-3-Clause
  5. */
  6. #include <errno.h>
  7. #include <modbus.h>
  8. #include <stdio.h>
  9. #include <stdlib.h>
  10. #include <string.h>
  11. #include <unistd.h>
  12. // clang-format off
  13. #ifdef _WIN32
  14. # include <winsock2.h>
  15. #else
  16. # include <sys/socket.h>
  17. #endif
  18. /* For MinGW */
  19. #ifndef MSG_NOSIGNAL
  20. # define MSG_NOSIGNAL 0
  21. #endif
  22. // clang-format on
  23. #include "unit-test.h"
  24. enum {
  25. TCP,
  26. TCP_PI,
  27. RTU
  28. };
  29. int main(int argc, char *argv[])
  30. {
  31. int s = -1;
  32. modbus_t *ctx;
  33. modbus_mapping_t *mb_mapping;
  34. int rc;
  35. int i;
  36. int use_backend;
  37. uint8_t *query;
  38. int header_length;
  39. char *ip_or_device;
  40. if (argc > 1) {
  41. if (strcmp(argv[1], "tcp") == 0) {
  42. use_backend = TCP;
  43. } else if (strcmp(argv[1], "tcppi") == 0) {
  44. use_backend = TCP_PI;
  45. } else if (strcmp(argv[1], "rtu") == 0) {
  46. use_backend = RTU;
  47. } else {
  48. printf("Modbus server for unit testing.\n");
  49. printf("Usage:\n %s [tcp|tcppi|rtu] [<ip or device>]\n", argv[0]);
  50. printf("Eg. tcp 127.0.0.1 or rtu /dev/ttyUSB0\n\n");
  51. return -1;
  52. }
  53. } else {
  54. /* By default */
  55. use_backend = TCP;
  56. }
  57. if (argc > 2) {
  58. ip_or_device = argv[2];
  59. } else {
  60. switch (use_backend) {
  61. case TCP:
  62. ip_or_device = "127.0.0.1";
  63. break;
  64. case TCP_PI:
  65. ip_or_device = "::1";
  66. break;
  67. case RTU:
  68. ip_or_device = "/dev/ttyUSB0";
  69. break;
  70. default:
  71. break;
  72. }
  73. }
  74. if (use_backend == TCP) {
  75. ctx = modbus_new_tcp(ip_or_device, 1502);
  76. query = malloc(MODBUS_TCP_MAX_ADU_LENGTH);
  77. } else if (use_backend == TCP_PI) {
  78. ctx = modbus_new_tcp_pi(ip_or_device, "1502");
  79. query = malloc(MODBUS_TCP_MAX_ADU_LENGTH);
  80. } else {
  81. ctx = modbus_new_rtu(ip_or_device, 115200, 'N', 8, 1);
  82. modbus_set_slave(ctx, SERVER_ID);
  83. query = malloc(MODBUS_RTU_MAX_ADU_LENGTH);
  84. }
  85. header_length = modbus_get_header_length(ctx);
  86. modbus_set_debug(ctx, TRUE);
  87. mb_mapping = modbus_mapping_new_start_address(UT_BITS_ADDRESS,
  88. UT_BITS_NB,
  89. UT_INPUT_BITS_ADDRESS,
  90. UT_INPUT_BITS_NB,
  91. UT_REGISTERS_ADDRESS,
  92. UT_REGISTERS_NB_MAX,
  93. UT_INPUT_REGISTERS_ADDRESS,
  94. UT_INPUT_REGISTERS_NB);
  95. if (mb_mapping == NULL) {
  96. fprintf(stderr, "Failed to allocate the mapping: %s\n", modbus_strerror(errno));
  97. modbus_free(ctx);
  98. return -1;
  99. }
  100. /* Examples from PI_MODBUS_300.pdf.
  101. Only the read-only input values are assigned. */
  102. /* Initialize input values that's can be only done server side. */
  103. modbus_set_bits_from_bytes(
  104. mb_mapping->tab_input_bits, 0, UT_INPUT_BITS_NB, UT_INPUT_BITS_TAB);
  105. /* Initialize values of INPUT REGISTERS */
  106. for (i = 0; i < UT_INPUT_REGISTERS_NB; i++) {
  107. mb_mapping->tab_input_registers[i] = UT_INPUT_REGISTERS_TAB[i];
  108. }
  109. if (use_backend == TCP) {
  110. s = modbus_tcp_listen(ctx, 1);
  111. modbus_tcp_accept(ctx, &s);
  112. } else if (use_backend == TCP_PI) {
  113. s = modbus_tcp_pi_listen(ctx, 1);
  114. modbus_tcp_pi_accept(ctx, &s);
  115. } else {
  116. rc = modbus_connect(ctx);
  117. if (rc == -1) {
  118. fprintf(stderr, "Unable to connect %s\n", modbus_strerror(errno));
  119. modbus_free(ctx);
  120. return -1;
  121. }
  122. }
  123. for (;;) {
  124. do {
  125. rc = modbus_receive(ctx, query);
  126. /* Filtered queries return 0 */
  127. } while (rc == 0);
  128. /* The connection is not closed on errors which require on reply such as
  129. bad CRC in RTU. */
  130. if (rc == -1 && errno != EMBBADCRC) {
  131. /* Quit */
  132. break;
  133. }
  134. /* Special server behavior to test client */
  135. if (query[header_length] == 0x03) {
  136. /* Read holding registers */
  137. if (MODBUS_GET_INT16_FROM_INT8(query, header_length + 3) ==
  138. UT_REGISTERS_NB_SPECIAL) {
  139. printf("Set an incorrect number of values\n");
  140. MODBUS_SET_INT16_TO_INT8(
  141. query, header_length + 3, UT_REGISTERS_NB_SPECIAL - 1);
  142. } else if (MODBUS_GET_INT16_FROM_INT8(query, header_length + 1) ==
  143. UT_REGISTERS_ADDRESS_SPECIAL) {
  144. printf("Reply to this special register address by an exception\n");
  145. modbus_reply_exception(ctx, query, MODBUS_EXCEPTION_SLAVE_OR_SERVER_BUSY);
  146. continue;
  147. } else if (MODBUS_GET_INT16_FROM_INT8(query, header_length + 1) ==
  148. UT_REGISTERS_ADDRESS_INVALID_TID_OR_SLAVE) {
  149. const int RAW_REQ_LENGTH = 5;
  150. uint8_t raw_req[] = {(use_backend == RTU) ? INVALID_SERVER_ID : 0xFF,
  151. 0x03,
  152. 0x02,
  153. 0x00,
  154. 0x00};
  155. printf("Reply with an invalid TID or slave\n");
  156. modbus_send_raw_request(ctx, raw_req, RAW_REQ_LENGTH * sizeof(uint8_t));
  157. continue;
  158. } else if (MODBUS_GET_INT16_FROM_INT8(query, header_length + 1) ==
  159. UT_REGISTERS_ADDRESS_SLEEP_500_MS) {
  160. printf("Sleep 0.5 s before replying\n");
  161. usleep(500000);
  162. } else if (MODBUS_GET_INT16_FROM_INT8(query, header_length + 1) ==
  163. UT_REGISTERS_ADDRESS_BYTE_SLEEP_5_MS) {
  164. /* Test low level only available in TCP mode */
  165. /* Catch the reply and send reply byte a byte */
  166. uint8_t req[] = "\x00\x1C\x00\x00\x00\x05\xFF\x03\x02\x00\x00";
  167. int req_length = 11;
  168. int w_s = modbus_get_socket(ctx);
  169. if (w_s == -1) {
  170. fprintf(stderr, "Unable to get a valid socket in special test\n");
  171. continue;
  172. }
  173. /* Copy TID */
  174. req[1] = query[1];
  175. for (i = 0; i < req_length; i++) {
  176. printf("(%.2X)", req[i]);
  177. usleep(5000);
  178. rc = send(w_s, (const char *) (req + i), 1, MSG_NOSIGNAL);
  179. if (rc == -1) {
  180. break;
  181. }
  182. }
  183. continue;
  184. }
  185. }
  186. rc = modbus_reply(ctx, query, rc, mb_mapping);
  187. if (rc == -1) {
  188. break;
  189. }
  190. }
  191. printf("Quit the loop: %s\n", modbus_strerror(errno));
  192. if (use_backend == TCP) {
  193. if (s != -1) {
  194. close(s);
  195. }
  196. }
  197. modbus_mapping_free(mb_mapping);
  198. free(query);
  199. /* For RTU */
  200. modbus_close(ctx);
  201. modbus_free(ctx);
  202. return 0;
  203. }