modbus_master.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. /**
  2. * @Author: 李建
  3. * @Date: 2025/4/27 15:35
  4. * Description: 实现485主站
  5. * Copyright: Copyright (©) 2025 永续绿建. All rights reserved.
  6. */
  7. #include <esp_err.h>
  8. #include "modbus_master.h"
  9. #include "mbcontroller.h"
  10. #define OPTS(min_val, max_val, step_val) { .opt1 = min_val, .opt2 = max_val, .opt3 = step_val }
  11. static const char *TAG = "MODBUS_MASTER";
  12. enum {
  13. // 分风箱主板从站号
  14. MB_DEVICE_XFJS_ADDR = 10,
  15. // 分控一从站号
  16. MB_DEVICE_FK_1_ADDR = 2,
  17. // 分控二从站号
  18. MB_DEVICE_FK_2_ADDR = 3,
  19. // 分控三从站号
  20. MB_DEVICE_FK_3_ADDR = 4,
  21. // 分控四从站号
  22. MB_DEVICE_FK_4_ADDR = 5,
  23. };
  24. static void *master_handle = NULL;
  25. // The number of parameters that intended to be used in the particular control process
  26. #define MASTER_MAX_CIDS num_device_parameters
  27. // Number of reading of parameters from slave
  28. #define MASTER_MAX_RETRY (10)
  29. // Timeout to update cid over Modbus
  30. #define UPDATE_CIDS_TIMEOUT_MS (500)
  31. #define UPDATE_CIDS_TIMEOUT_TICS (UPDATE_CIDS_TIMEOUT_MS / portTICK_PERIOD_MS)
  32. // Timeout between polls
  33. #define POLL_TIMEOUT_MS (1)
  34. #define POLL_TIMEOUT_TICS (POLL_TIMEOUT_MS / portTICK_PERIOD_MS)
  35. // The macro to get offset for parameter in the appropriate structure
  36. #define HOLD_OFFSET(field) ((uint16_t)(offsetof(holding_reg_params_t, field) + 1))
  37. #define INPUT_OFFSET(field) ((uint16_t)(offsetof(input_reg_params_t, field) + 1))
  38. // Discrete offset macro
  39. #define STR(fieldname) ((const char *)( fieldname ))
  40. const mb_parameter_descriptor_t device_parameters[] = {
  41. // { CID, Param Name, Units, Modbus Slave Addr, Modbus Reg Type, Reg Start, Reg Size, Instance Offset, Data Type, Data Size, Parameter Options, Access Mode}
  42. {CID_POWER, STR("power"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  43. POWER_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  44. {CID_MODE, STR("mode"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  45. MODE_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  46. {CID_FAN_SPEED, STR("fan_speed"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  47. FAN_SPEED_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  48. {CID_EXTERNAL_TEMPERATURE, STR("external_temperature"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  49. EXTERNAL_TEMPERATURE_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  50. {CID_EXTERNAL_HUMIDITY, STR("external_humidity"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  51. EXTERNAL_HUMIDITY_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  52. {CID_RETURN_AIR_TEMPERATURE, STR("return_air_temperature"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  53. RETURN_AIR_TEMPERATURE_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  54. {CID_RETURN_AIR_HUMIDITY, STR("return_air_humidity"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  55. RETURN_AIR_HUMIDITY_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  56. {CID_RETURN_AIR_PM25, STR("return_air_pm25"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  57. RETURN_AIR_PM25_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  58. {CID_RETURN_AIR_CO2, STR("return_air_co2"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  59. RETURN_AIR_CO2_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  60. {CID_SELECT_POSITION, STR("select_position"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  61. SELECT_POSITION_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  62. {CID_MIN_HUMIDITY, STR("min_humidity"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  63. MIN_HUMIDITY_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  64. {CID_MAX_HUMIDITY, STR("max_humidity"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  65. MAX_HUMIDITY_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  66. {CID_AIR_EXHAUST_FIRST_VOL, STR("air_exhaust_first_vol"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  67. AIR_EXHAUST_FIRST_VOL_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  68. {CID_AIR_EXHAUST_SECOND_VOL, STR("air_exhaust_second_vol"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  69. AIR_EXHAUST_SECOND_VOL_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  70. {CID_AIR_EXHAUST_THIRD_VOL, STR("air_exhaust_third_vol"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  71. AIR_EXHAUST_THIRD_VOL_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  72. {CID_AIR_EXHAUST_FOURTH_VOL, STR("air_exhaust_fourth_vol"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  73. AIR_EXHAUST_FOURTH_VOL_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  74. {CID_AIR_EXHAUST_FIFTH_VOL, STR("air_exhaust_fifth_vol"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  75. AIR_EXHAUST_FIFTH_VOL_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  76. {CID_AIR_SUPPLY_FIRST_VOL, STR("air_supply_first_vol"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  77. AIR_SUPPLY_FIRST_VOL_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  78. {CID_AIR_SUPPLY_SECOND_VOL, STR("air_supply_second_vol"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  79. AIR_SUPPLY_SECOND_VOL_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  80. {CID_AIR_SUPPLY_THIRD_VOL, STR("air_supply_third_vol"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  81. AIR_SUPPLY_THIRD_VOL_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  82. {CID_AIR_SUPPLY_FOURTH_VOL, STR("air_supply_fourth_vol"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  83. AIR_SUPPLY_FOURTH_VOL_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  84. // {CID_AIR_SUPPLY_FIFTH_VOL, STR("air_supply_fifth_vol"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  85. // AIR_SUPPLY_FIFTH_VOL_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  86. // {CID_ERROR_CODE, STR("air_supply_fifth_vol"), STR(""), MB_DEVICE_XFJS_ADDR, MB_PARAM_HOLDING,
  87. // ERROR_CODE_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  88. };
  89. // Calculate number of parameters in the table
  90. const uint16_t num_device_parameters = (sizeof(device_parameters) / sizeof(device_parameters[0]));
  91. IRAM_ATTR static esp_err_t master_init() {
  92. // Initialize Modbus controller
  93. mb_communication_info_t comm = {
  94. .ser_opts.port = MB_PORT_NUM,
  95. .ser_opts.mode = MB_RTU,
  96. .ser_opts.baudrate = MB_DEV_SPEED,
  97. .ser_opts.parity = MB_PARITY_NONE,
  98. .ser_opts.response_tout_ms = 400,
  99. .ser_opts.data_bits = UART_DATA_8_BITS,
  100. .ser_opts.stop_bits = UART_STOP_BITS_1
  101. };
  102. ESP_LOGI(TAG, "Modbus controller initialized...");
  103. esp_err_t err = mbc_master_create_serial(&comm, &master_handle);
  104. MB_RETURN_ON_FALSE((master_handle != NULL), ESP_ERR_INVALID_STATE, TAG,
  105. "mb controller initialization fail.");
  106. MB_RETURN_ON_FALSE((err == ESP_OK), ESP_ERR_INVALID_STATE, TAG,
  107. "mb controller initialization fail, returns(0x%x).", (int) err);
  108. // Set UART pin numbers
  109. err = uart_set_pin(MB_PORT_NUM, MB_GPIO_TX, MB_GPIO_RX,
  110. UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE);
  111. MB_RETURN_ON_FALSE((err == ESP_OK), ESP_ERR_INVALID_STATE, TAG,
  112. "mb serial set pin failure, uart_set_pin() returned (0x%x).", (int) err);
  113. err = mbc_master_start(master_handle);
  114. MB_RETURN_ON_FALSE((err == ESP_OK), ESP_ERR_INVALID_STATE, TAG,
  115. "mb controller start fail, returned (0x%x).", (int) err);
  116. // Set driver mode to Half Duplex
  117. err = uart_set_mode(MB_PORT_NUM, UART_MODE_UART);
  118. MB_RETURN_ON_FALSE((err == ESP_OK), ESP_ERR_INVALID_STATE, TAG,
  119. "mb serial set mode failure, uart_set_mode() returned (0x%x).", (int) err);
  120. vTaskDelay(5);
  121. err = mbc_master_set_descriptor(master_handle, &device_parameters[0], num_device_parameters);
  122. MB_RETURN_ON_FALSE((err == ESP_OK), ESP_ERR_INVALID_STATE, TAG,
  123. "mb controller set descriptor fail, returns(0x%x).", (int) err);
  124. ESP_LOGI(TAG, "Modbus master stack initialized...");
  125. return err;
  126. }
  127. //static void *master_get_param_data(const mb_parameter_descriptor_t *param_descriptor)
  128. //{
  129. // assert(param_descriptor != NULL);
  130. // void *instance_ptr = NULL;
  131. // if (param_descriptor->param_offset != 0) {
  132. // switch(param_descriptor->mb_param_type)
  133. // {
  134. // case MB_PARAM_HOLDING:
  135. // instance_ptr = ((void *)&holding_reg_params + param_descriptor->param_offset - 1);
  136. // break;
  137. // default:
  138. // instance_ptr = NULL;
  139. // break;
  140. // }
  141. // } else {
  142. // ESP_LOGE(TAG, "Wrong parameter offset for CID #%u", (unsigned)param_descriptor->cid);
  143. // assert(instance_ptr != NULL);
  144. // }
  145. // return instance_ptr;
  146. //}
  147. void modbus_master_task(void *arg) {
  148. vTaskDelay(5000 / portTICK_PERIOD_MS);
  149. ESP_LOGI(TAG, "Modbus master task started...");
  150. esp_err_t err = ESP_OK;
  151. uint8_t temp_data[2] = {0}; // temporary buffer to hold maximum CID size
  152. uint8_t type = 0;
  153. const mb_parameter_descriptor_t *param_descriptor = NULL;
  154. for (;;) {
  155. for (uint16_t cid = 0; (err != ESP_ERR_NOT_FOUND) && cid < 1; cid++) {
  156. err = mbc_master_get_cid_info(master_handle, cid, &param_descriptor);
  157. if ((err != ESP_ERR_NOT_FOUND) && (param_descriptor != NULL)) {
  158. err = mbc_master_get_parameter(master_handle, param_descriptor->cid, (uint8_t *) temp_data, &type);
  159. if (err == ESP_OK) {
  160. ESP_LOGI(TAG, "Characteristic #%d %s (%s) value = %d read successful.",
  161. param_descriptor->cid,
  162. (char *) param_descriptor->param_key,
  163. (char *) param_descriptor->param_units,
  164. *(uint16_t *) temp_data);
  165. } else {
  166. ESP_LOGE(TAG, "Characteristic #%d (%s) read fail, err = 0x%x (%s).",
  167. param_descriptor->cid,
  168. (char *) param_descriptor->param_key,
  169. (int) err,
  170. (char *) esp_err_to_name(err));
  171. }
  172. }
  173. vTaskDelay(1000 / portTICK_PERIOD_MS);
  174. }
  175. vTaskDelay(5000 / portTICK_PERIOD_MS);
  176. }
  177. }
  178. void modbus_master_init() {
  179. ESP_LOGI(TAG, "Modbus master init...");
  180. ESP_ERROR_CHECK(master_init());
  181. vTaskDelay(10);
  182. //xTaskCreatePinnedToCore(modbus_master_task, "modbus_master_task", 4 * 1024, NULL, 5, NULL, 1);
  183. }
  184. esp_err_t mm_set_param(uint16_t cid, uint8_t *value) {
  185. uint8_t type = PARAM_TYPE_U16;
  186. return mbc_master_set_parameter(master_handle, cid, value, &type);
  187. }
  188. esp_err_t mm_get_param(uint16_t cid, uint8_t *value) {
  189. uint8_t type = PARAM_TYPE_U16;
  190. return mbc_master_get_parameter(master_handle, cid, value, &type);
  191. }
  192. void modbus_master_destroy() {
  193. mbc_master_delete(master_handle);
  194. }