modbus.h 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536
  1. /*
  2. * Copyright (c) 2020 PHYTEC Messtechnik GmbH
  3. * Copyright (c) 2021 Nordic Semiconductor ASA
  4. *
  5. * SPDX-License-Identifier: Apache-2.0
  6. */
  7. /*
  8. * Client API in this file is based on mbm_core.c from uC/Modbus Stack.
  9. *
  10. * uC/Modbus
  11. * The Embedded Modbus Stack
  12. *
  13. * Copyright 2003-2020 Silicon Laboratories Inc. www.silabs.com
  14. *
  15. * SPDX-License-Identifier: APACHE-2.0
  16. *
  17. * This software is subject to an open source license and is distributed by
  18. * Silicon Laboratories Inc. pursuant to the terms of the Apache License,
  19. * Version 2.0 available at www.apache.org/licenses/LICENSE-2.0.
  20. */
  21. /**
  22. * @brief MODBUS transport protocol API
  23. * @defgroup modbus MODBUS
  24. * @ingroup io_interfaces
  25. * @{
  26. */
  27. #ifndef ZEPHYR_INCLUDE_MODBUS_H_
  28. #define ZEPHYR_INCLUDE_MODBUS_H_
  29. #include <drivers/uart.h>
  30. #ifdef __cplusplus
  31. extern "C" {
  32. #endif
  33. /** Length of MBAP Header */
  34. #define MODBUS_MBAP_LENGTH 7
  35. /** Length of MBAP Header plus function code */
  36. #define MODBUS_MBAP_AND_FC_LENGTH (MODBUS_MBAP_LENGTH + 1)
  37. /**
  38. * @brief Frame struct used internally and for raw ADU support.
  39. */
  40. struct modbus_adu {
  41. /** Transaction Identifier */
  42. uint16_t trans_id;
  43. /** Protocol Identifier */
  44. uint16_t proto_id;
  45. /** Length of the data only (not the length of unit ID + PDU) */
  46. uint16_t length;
  47. /** Unit Identifier */
  48. uint8_t unit_id;
  49. /** Function Code */
  50. uint8_t fc;
  51. /** Transaction Data */
  52. uint8_t data[CONFIG_MODBUS_BUFFER_SIZE - 4];
  53. /** RTU CRC */
  54. uint16_t crc;
  55. };
  56. /**
  57. * @brief Coil read (FC01)
  58. *
  59. * Sends a Modbus message to read the status of coils from a server.
  60. *
  61. * @param iface Modbus interface index
  62. * @param unit_id Modbus unit ID of the server
  63. * @param start_addr Coil starting address
  64. * @param coil_tbl Pointer to an array of bytes containing the value
  65. * of the coils read.
  66. * The format is:
  67. *
  68. * MSB LSB
  69. * B7 B6 B5 B4 B3 B2 B1 B0
  70. * -------------------------------------
  71. * coil_tbl[0] #8 #7 #1
  72. * coil_tbl[1] #16 #15 #9
  73. * :
  74. * :
  75. *
  76. * Note that the array that will be receiving the coil
  77. * values must be greater than or equal to:
  78. * (num_coils - 1) / 8 + 1
  79. * @param num_coils Quantity of coils to read
  80. *
  81. * @retval 0 If the function was successful
  82. */
  83. int modbus_read_coils(const int iface,
  84. const uint8_t unit_id,
  85. const uint16_t start_addr,
  86. uint8_t *const coil_tbl,
  87. const uint16_t num_coils);
  88. /**
  89. * @brief Read discrete inputs (FC02)
  90. *
  91. * Sends a Modbus message to read the status of discrete inputs from
  92. * a server.
  93. *
  94. * @param iface Modbus interface index
  95. * @param unit_id Modbus unit ID of the server
  96. * @param start_addr Discrete input starting address
  97. * @param di_tbl Pointer to an array that will receive the state
  98. * of the discrete inputs.
  99. * The format of the array is as follows:
  100. *
  101. * MSB LSB
  102. * B7 B6 B5 B4 B3 B2 B1 B0
  103. * -------------------------------------
  104. * di_tbl[0] #8 #7 #1
  105. * di_tbl[1] #16 #15 #9
  106. * :
  107. * :
  108. *
  109. * Note that the array that will be receiving the discrete
  110. * input values must be greater than or equal to:
  111. * (num_di - 1) / 8 + 1
  112. * @param num_di Quantity of discrete inputs to read
  113. *
  114. * @retval 0 If the function was successful
  115. */
  116. int modbus_read_dinputs(const int iface,
  117. const uint8_t unit_id,
  118. const uint16_t start_addr,
  119. uint8_t *const di_tbl,
  120. const uint16_t num_di);
  121. /**
  122. * @brief Read holding registers (FC03)
  123. *
  124. * Sends a Modbus message to read the value of holding registers
  125. * from a server.
  126. *
  127. * @param iface Modbus interface index
  128. * @param unit_id Modbus unit ID of the server
  129. * @param start_addr Register starting address
  130. * @param reg_buf Is a pointer to an array that will receive
  131. * the current values of the holding registers from
  132. * the server. The array pointed to by 'reg_buf' needs
  133. * to be able to hold at least 'num_regs' entries.
  134. * @param num_regs Quantity of registers to read
  135. *
  136. * @retval 0 If the function was successful
  137. */
  138. int modbus_read_holding_regs(const int iface,
  139. const uint8_t unit_id,
  140. const uint16_t start_addr,
  141. uint16_t *const reg_buf,
  142. const uint16_t num_regs);
  143. /**
  144. * @brief Read input registers (FC04)
  145. *
  146. * Sends a Modbus message to read the value of input registers from
  147. * a server.
  148. *
  149. * @param iface Modbus interface index
  150. * @param unit_id Modbus unit ID of the server
  151. * @param start_addr Register starting address
  152. * @param reg_buf Is a pointer to an array that will receive
  153. * the current value of the holding registers
  154. * from the server. The array pointed to by 'reg_buf'
  155. * needs to be able to hold at least 'num_regs' entries.
  156. * @param num_regs Quantity of registers to read
  157. *
  158. * @retval 0 If the function was successful
  159. */
  160. int modbus_read_input_regs(const int iface,
  161. const uint8_t unit_id,
  162. const uint16_t start_addr,
  163. uint16_t *const reg_buf,
  164. const uint16_t num_regs);
  165. /**
  166. * @brief Write single coil (FC05)
  167. *
  168. * Sends a Modbus message to write the value of single coil to a server.
  169. *
  170. * @param iface Modbus interface index
  171. * @param unit_id Modbus unit ID of the server
  172. * @param coil_addr Coils starting address
  173. * @param coil_state Is the desired state of the coil
  174. *
  175. * @retval 0 If the function was successful
  176. */
  177. int modbus_write_coil(const int iface,
  178. const uint8_t unit_id,
  179. const uint16_t coil_addr,
  180. const bool coil_state);
  181. /**
  182. * @brief Write single holding register (FC06)
  183. *
  184. * Sends a Modbus message to write the value of single holding register
  185. * to a server unit.
  186. *
  187. * @param iface Modbus interface index
  188. * @param unit_id Modbus unit ID of the server
  189. * @param start_addr Coils starting address
  190. * @param reg_val Desired value of the holding register
  191. *
  192. * @retval 0 If the function was successful
  193. */
  194. int modbus_write_holding_reg(const int iface,
  195. const uint8_t unit_id,
  196. const uint16_t start_addr,
  197. const uint16_t reg_val);
  198. /**
  199. * @brief Read diagnostic (FC08)
  200. *
  201. * Sends a Modbus message to perform a diagnostic function of a server unit.
  202. *
  203. * @param iface Modbus interface index
  204. * @param unit_id Modbus unit ID of the server
  205. * @param sfunc Diagnostic sub-function code
  206. * @param data Sub-function data
  207. * @param data_out Pointer to the data value
  208. *
  209. * @retval 0 If the function was successful
  210. */
  211. int modbus_request_diagnostic(const int iface,
  212. const uint8_t unit_id,
  213. const uint16_t sfunc,
  214. const uint16_t data,
  215. uint16_t *const data_out);
  216. /**
  217. * @brief Write coils (FC15)
  218. *
  219. * Sends a Modbus message to write to coils on a server unit.
  220. *
  221. * @param iface Modbus interface index
  222. * @param unit_id Modbus unit ID of the server
  223. * @param start_addr Coils starting address
  224. * @param coil_tbl Pointer to an array of bytes containing the value
  225. * of the coils to write.
  226. * The format is:
  227. *
  228. * MSB LSB
  229. * B7 B6 B5 B4 B3 B2 B1 B0
  230. * -------------------------------------
  231. * coil_tbl[0] #8 #7 #1
  232. * coil_tbl[1] #16 #15 #9
  233. * :
  234. * :
  235. *
  236. * Note that the array that will be receiving the coil
  237. * values must be greater than or equal to:
  238. * (num_coils - 1) / 8 + 1
  239. * @param num_coils Quantity of coils to write
  240. *
  241. * @retval 0 If the function was successful
  242. */
  243. int modbus_write_coils(const int iface,
  244. const uint8_t unit_id,
  245. const uint16_t start_addr,
  246. uint8_t *const coil_tbl,
  247. const uint16_t num_coils);
  248. /**
  249. * @brief Write holding registers (FC16)
  250. *
  251. * Sends a Modbus message to write to integer holding registers
  252. * to a server unit.
  253. *
  254. * @param iface Modbus interface index
  255. * @param unit_id Modbus unit ID of the server
  256. * @param start_addr Register starting address
  257. * @param reg_buf Is a pointer to an array containing
  258. * the value of the holding registers to write.
  259. * Note that the array containing the register values must
  260. * be greater than or equal to 'num_regs'
  261. * @param num_regs Quantity of registers to write
  262. *
  263. * @retval 0 If the function was successful
  264. */
  265. int modbus_write_holding_regs(const int iface,
  266. const uint8_t unit_id,
  267. const uint16_t start_addr,
  268. uint16_t *const reg_buf,
  269. const uint16_t num_regs);
  270. /**
  271. * @brief Read floating-point holding registers (FC03)
  272. *
  273. * Sends a Modbus message to read the value of floating-point
  274. * holding registers from a server unit.
  275. *
  276. * @param iface Modbus interface index
  277. * @param unit_id Modbus unit ID of the server
  278. * @param start_addr Register starting address
  279. * @param reg_buf Is a pointer to an array that will receive
  280. * the current values of the holding registers from
  281. * the server. The array pointed to by 'reg_buf' needs
  282. * to be able to hold at least 'num_regs' entries.
  283. * @param num_regs Quantity of registers to read
  284. *
  285. * @retval 0 If the function was successful
  286. */
  287. int modbus_read_holding_regs_fp(const int iface,
  288. const uint8_t unit_id,
  289. const uint16_t start_addr,
  290. float *const reg_buf,
  291. const uint16_t num_regs);
  292. /**
  293. * @brief Write floating-point holding registers (FC16)
  294. *
  295. * Sends a Modbus message to write to floating-point holding registers
  296. * to a server unit.
  297. *
  298. * @param iface Modbus interface index
  299. * @param unit_id Modbus unit ID of the server
  300. * @param start_addr Register starting address
  301. * @param reg_buf Is a pointer to an array containing
  302. * the value of the holding registers to write.
  303. * Note that the array containing the register values must
  304. * be greater than or equal to 'num_regs'
  305. * @param num_regs Quantity of registers to write
  306. *
  307. * @retval 0 If the function was successful
  308. */
  309. int modbus_write_holding_regs_fp(const int iface,
  310. const uint8_t unit_id,
  311. const uint16_t start_addr,
  312. float *const reg_buf,
  313. const uint16_t num_regs);
  314. /** Modbus Server User Callback structure */
  315. struct modbus_user_callbacks {
  316. /** Coil read callback */
  317. int (*coil_rd)(uint16_t addr, bool *state);
  318. /** Coil write callback */
  319. int (*coil_wr)(uint16_t addr, bool state);
  320. /** Discrete Input read callback */
  321. int (*discrete_input_rd)(uint16_t addr, bool *state);
  322. /** Input Register read callback */
  323. int (*input_reg_rd)(uint16_t addr, uint16_t *reg);
  324. /** Floating Point Input Register read callback */
  325. int (*input_reg_rd_fp)(uint16_t addr, float *reg);
  326. /** Holding Register read callback */
  327. int (*holding_reg_rd)(uint16_t addr, uint16_t *reg);
  328. /** Holding Register write callback */
  329. int (*holding_reg_wr)(uint16_t addr, uint16_t reg);
  330. /** Floating Point Holding Register read callback */
  331. int (*holding_reg_rd_fp)(uint16_t addr, float *reg);
  332. /** Floating Point Holding Register write callback */
  333. int (*holding_reg_wr_fp)(uint16_t addr, float reg);
  334. };
  335. /**
  336. * @brief Get Modbus interface index according to interface name
  337. *
  338. * If there is more than one interface, it can be used to clearly
  339. * identify interfaces in the application.
  340. *
  341. * @param iface_name Modbus interface name
  342. *
  343. * @retval Modbus interface index or negative error value.
  344. */
  345. int modbus_iface_get_by_name(const char *iface_name);
  346. /**
  347. * @brief ADU raw callback function signature
  348. *
  349. * @param iface Modbus RTU interface index
  350. * @param adu Pointer to the RAW ADU struct to send
  351. *
  352. * @retval 0 If transfer was successful
  353. */
  354. typedef int (*modbus_raw_cb_t)(const int iface, const struct modbus_adu *adu);
  355. /**
  356. * @brief Modbus interface mode
  357. */
  358. enum modbus_mode {
  359. /** Modbus over serial line RTU mode */
  360. MODBUS_MODE_RTU,
  361. /** Modbus over serial line ASCII mode */
  362. MODBUS_MODE_ASCII,
  363. /** Modbus raw ADU mode */
  364. MODBUS_MODE_RAW,
  365. };
  366. /**
  367. * @brief Modbus serial line parameter
  368. */
  369. struct modbus_serial_param {
  370. /** Baudrate of the serial line */
  371. uint32_t baud;
  372. /** parity UART's parity setting:
  373. * UART_CFG_PARITY_NONE,
  374. * UART_CFG_PARITY_EVEN,
  375. * UART_CFG_PARITY_ODD
  376. */
  377. enum uart_config_parity parity;
  378. };
  379. /**
  380. * @brief Modbus server parameter
  381. */
  382. struct modbus_server_param {
  383. /** Pointer to the User Callback structure */
  384. struct modbus_user_callbacks *user_cb;
  385. /** Modbus unit ID of the server */
  386. uint8_t unit_id;
  387. };
  388. /**
  389. * @brief User parameter structure to configure Modbus interfase
  390. * as client or server.
  391. */
  392. struct modbus_iface_param {
  393. /** Mode of the interface */
  394. enum modbus_mode mode;
  395. union {
  396. struct modbus_server_param server;
  397. /** Amount of time client will wait for
  398. * a response from the server.
  399. */
  400. uint32_t rx_timeout;
  401. };
  402. union {
  403. /** Serial support parameter of the interface */
  404. struct modbus_serial_param serial;
  405. /** Pointer to raw ADU callback function */
  406. modbus_raw_cb_t raw_tx_cb;
  407. };
  408. };
  409. /**
  410. * @brief Configure Modbus Interface as raw ADU server
  411. *
  412. * @param iface Modbus RTU interface index
  413. * @param param Configuration parameter of the server interface
  414. *
  415. * @retval 0 If the function was successful
  416. */
  417. int modbus_init_server(const int iface, struct modbus_iface_param param);
  418. /**
  419. * @brief Configure Modbus Interface as raw ADU client
  420. *
  421. * @param iface Modbus RTU interface index
  422. * @param param Configuration parameter of the client interface
  423. *
  424. * @retval 0 If the function was successful
  425. */
  426. int modbus_init_client(const int iface, struct modbus_iface_param param);
  427. /**
  428. * @brief Disable Modbus Interface
  429. *
  430. * This function is called to disable Modbus interface.
  431. *
  432. * @param iface Modbus interface index
  433. *
  434. * @retval 0 If the function was successful
  435. */
  436. int modbus_disable(const uint8_t iface);
  437. /**
  438. * @brief Submit raw ADU
  439. *
  440. * @param iface Modbus RTU interface index
  441. * @param adu Pointer to the RAW ADU struct that is received
  442. *
  443. * @retval 0 If transfer was successful
  444. */
  445. int modbus_raw_submit_rx(const int iface, const struct modbus_adu *adu);
  446. /**
  447. * @brief Put MBAP header into a buffer
  448. *
  449. * @param adu Pointer to the RAW ADU struct
  450. * @param header Pointer to the buffer in which MBAP header
  451. * will be placed.
  452. *
  453. * @retval 0 If transfer was successful
  454. */
  455. void modbus_raw_put_header(const struct modbus_adu *adu, uint8_t *header);
  456. /**
  457. * @brief Get MBAP header from a buffer
  458. *
  459. * @param adu Pointer to the RAW ADU struct
  460. * @param header Pointer to the buffer containing MBAP header
  461. *
  462. * @retval 0 If transfer was successful
  463. */
  464. void modbus_raw_get_header(struct modbus_adu *adu, const uint8_t *header);
  465. /**
  466. * @brief Set Server Device Failure exception
  467. *
  468. * This function modifies ADU passed by the pointer.
  469. *
  470. * @param adu Pointer to the RAW ADU struct
  471. */
  472. void modbus_raw_set_server_failure(struct modbus_adu *adu);
  473. /**
  474. * @brief Use interface as backend to send and receive ADU
  475. *
  476. * This function overwrites ADU passed by the pointer and generates
  477. * exception responses if backend interface is misconfigured or
  478. * target device is unreachable.
  479. *
  480. * @param iface Modbus client interface index
  481. * @param adu Pointer to the RAW ADU struct
  482. *
  483. * @retval 0 If transfer was successful
  484. */
  485. int modbus_raw_backend_txn(const int iface, struct modbus_adu *adu);
  486. #ifdef __cplusplus
  487. }
  488. #endif
  489. /**
  490. * @}
  491. */
  492. #endif /* ZEPHYR_INCLUDE_MODBUS_H_ */