modbus_master.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  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_FFX_ADDR = 1,
  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. #define STR(fieldname) ((const char *)( fieldname ))
  26. const mb_parameter_descriptor_t device_parameters[] = {
  27. // { CID, Param Name, Units, Modbus Slave Addr, Modbus Reg Type, Reg Start, Reg Size, Instance Offset, Data Type, Data Size, Parameter Options, Access Mode}
  28. {CID_POWER, STR("power"), STR(""), MB_DEVICE_FFX_ADDR, MB_PARAM_HOLDING,
  29. POWER_REG_ADDRESS, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  30. {CID_MODE, STR("mode"), STR(""), MB_DEVICE_FFX_ADDR, MB_PARAM_HOLDING,
  31. WORK_MOD_REG_ADDRESS, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  32. {CID_FAN_MODE, STR("fan_mode"), STR(""), MB_DEVICE_FFX_ADDR, MB_PARAM_HOLDING,
  33. FAN_MODE_REG_ADDRESS, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  34. {CID_FAN_SPEED, STR("fan_speed"), STR(""), MB_DEVICE_FFX_ADDR, MB_PARAM_HOLDING,
  35. FAN_LEVEL_REG_ADDRESS, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  36. {CID_FAN_1_DEGREE, STR("fan_1_degree"), STR(""), MB_DEVICE_FFX_ADDR, MB_PARAM_HOLDING,
  37. FAN_VALVE_1_REG_ADDRESS, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  38. {CID_FAN_2_DEGREE, STR("fan_2_degree"), STR(""), MB_DEVICE_FFX_ADDR, MB_PARAM_HOLDING,
  39. FAN_VALVE_2_REG_ADDRESS, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  40. {CID_FAN_3_DEGREE, STR("fan_3_degree"), STR(""), MB_DEVICE_FFX_ADDR, MB_PARAM_HOLDING,
  41. FAN_VALVE_3_REG_ADDRESS, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  42. {CID_FAN_4_DEGREE, STR("fan_4_degree"), STR(""), MB_DEVICE_FFX_ADDR, MB_PARAM_HOLDING,
  43. FAN_VALVE_4_REG_ADDRESS, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  44. {CID_FAN_5_DEGREE, STR("fan_5_degree"), STR(""), MB_DEVICE_FFX_ADDR, MB_PARAM_HOLDING,
  45. FAN_VALVE_5_REG_ADDRESS, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  46. {CID_SET_TEMP, STR("set_temp"), STR(""), MB_DEVICE_FFX_ADDR, MB_PARAM_HOLDING,
  47. SET_TEMP_REG_ADDRESS, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  48. {CID_VOLTAGE_1, STR("voltage_1"), STR(""), MB_DEVICE_FFX_ADDR, MB_PARAM_HOLDING,
  49. VOLTAGE_1_REG_ADDRESS, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  50. {CID_VOLTAGE_2, STR("voltage_2"), STR(""), MB_DEVICE_FFX_ADDR, MB_PARAM_HOLDING,
  51. VOLTAGE_2_REG_ADDRESS, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  52. {CID_VOLTAGE_3, STR("voltage_3"), STR(""), MB_DEVICE_FFX_ADDR, MB_PARAM_HOLDING,
  53. VOLTAGE_3_REG_ADDRESS, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  54. {CID_VOLTAGE_4, STR("voltage_4"), STR(""), MB_DEVICE_FFX_ADDR, MB_PARAM_HOLDING,
  55. VOLTAGE_4_REG_ADDRESS, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  56. {CID_VOLTAGE_5, STR("voltage_5"), STR(""), MB_DEVICE_FFX_ADDR, MB_PARAM_HOLDING,
  57. VOLTAGE_5_REG_ADDRESS, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  58. {CID_ENV_TEMP, STR("env_temp"), STR(""), MB_DEVICE_FFX_ADDR, MB_PARAM_HOLDING,
  59. ENV_TEMP_REG_ADDRESS, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  60. {CID_ENV_HUMIDITY, STR("env_humidity"), STR(""), MB_DEVICE_FFX_ADDR, MB_PARAM_HOLDING,
  61. ENV_HUMIDITY_REG_ADDRESS, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  62. {CID_PM25, STR("pm25"), STR(""), MB_DEVICE_FFX_ADDR, MB_PARAM_HOLDING,
  63. PM25_NUMBER_REG_ADDRESS, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ},
  64. {CID_CO2, STR("co2"), STR(""), MB_DEVICE_FFX_ADDR, MB_PARAM_HOLDING,
  65. CO2_NUMBER_REG_ADDRESS, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ},
  66. {CID_INNER_ADDR, STR("addr"), STR(""), MB_DEVICE_FFX_ADDR, MB_PARAM_HOLDING,
  67. GMV_INTERNAL_NUMBER_REG_ADDRESS, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
  68. };
  69. // Calculate number of parameters in the table
  70. const uint16_t num_device_parameters = (sizeof(device_parameters) / sizeof(device_parameters[0]));
  71. IRAM_ATTR static esp_err_t master_init() {
  72. // Initialize Modbus controller
  73. mb_communication_info_t comm = {
  74. .ser_opts.port = MB_PORT_NUM,
  75. .ser_opts.mode = MB_RTU,
  76. .ser_opts.baudrate = MB_DEV_SPEED,
  77. .ser_opts.parity = MB_PARITY_NONE,
  78. .ser_opts.response_tout_ms = 1000,
  79. .ser_opts.data_bits = UART_DATA_8_BITS,
  80. .ser_opts.stop_bits = UART_STOP_BITS_1
  81. };
  82. ESP_LOGI(TAG, "Modbus controller initialized...");
  83. esp_err_t err = mbc_master_create_serial(&comm, &master_handle);
  84. MB_RETURN_ON_FALSE((master_handle != NULL), ESP_ERR_INVALID_STATE, TAG,
  85. "mb controller initialization fail.");
  86. MB_RETURN_ON_FALSE((err == ESP_OK), ESP_ERR_INVALID_STATE, TAG,
  87. "mb controller initialization fail, returns(0x%x).", (int) err);
  88. // Set UART pin numbers
  89. err = uart_set_pin(MB_PORT_NUM, MB_GPIO_TX, MB_GPIO_RX,
  90. UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE);
  91. MB_RETURN_ON_FALSE((err == ESP_OK), ESP_ERR_INVALID_STATE, TAG,
  92. "mb serial set pin failure, uart_set_pin() returned (0x%x).", (int) err);
  93. err = mbc_master_start(master_handle);
  94. MB_RETURN_ON_FALSE((err == ESP_OK), ESP_ERR_INVALID_STATE, TAG,
  95. "mb controller start fail, returned (0x%x).", (int) err);
  96. // Set driver mode to Half Duplex
  97. err = uart_set_mode(MB_PORT_NUM, UART_MODE_UART);
  98. MB_RETURN_ON_FALSE((err == ESP_OK), ESP_ERR_INVALID_STATE, TAG,
  99. "mb serial set mode failure, uart_set_mode() returned (0x%x).", (int) err);
  100. vTaskDelay(5);
  101. err = mbc_master_set_descriptor(master_handle, &device_parameters[0], num_device_parameters);
  102. MB_RETURN_ON_FALSE((err == ESP_OK), ESP_ERR_INVALID_STATE, TAG,
  103. "mb controller set descriptor fail, returns(0x%x).", (int) err);
  104. ESP_LOGI(TAG, "Modbus master stack initialized...");
  105. return err;
  106. }
  107. //static void *master_get_param_data(const mb_parameter_descriptor_t *param_descriptor)
  108. //{
  109. // assert(param_descriptor != NULL);
  110. // void *instance_ptr = NULL;
  111. // if (param_descriptor->param_offset != 0) {
  112. // switch(param_descriptor->mb_param_type)
  113. // {
  114. // case MB_PARAM_HOLDING:
  115. // instance_ptr = ((void *)&holding_reg_params + param_descriptor->param_offset - 1);
  116. // break;
  117. // default:
  118. // instance_ptr = NULL;
  119. // break;
  120. // }
  121. // } else {
  122. // ESP_LOGE(TAG, "Wrong parameter offset for CID #%u", (unsigned)param_descriptor->cid);
  123. // assert(instance_ptr != NULL);
  124. // }
  125. // return instance_ptr;
  126. //}
  127. void modbus_master_task(void *arg) {
  128. vTaskDelay(5000 / portTICK_PERIOD_MS);
  129. ESP_LOGI(TAG, "Modbus master task started...");
  130. esp_err_t err = ESP_OK;
  131. uint8_t temp_data[2] = {0}; // temporary buffer to hold maximum CID size
  132. uint8_t type = 0;
  133. const mb_parameter_descriptor_t *param_descriptor = NULL;
  134. for (;;) {
  135. mb_param_request_t req = {
  136. .slave_addr = 0x05,
  137. .reg_size = 10,
  138. .reg_start = 0x23,
  139. .command = 0x10
  140. };
  141. uint16_t data[20] = {0, 1 , 2, 3, 4, 5, 6, 7, 8, 9};
  142. send_request(&req, &data[0]);
  143. err = mbc_master_get_cid_info(master_handle, 55, &param_descriptor);
  144. for (uint16_t cid = 0; (err != ESP_ERR_NOT_FOUND) && cid < 1; cid++) {
  145. err = mbc_master_get_cid_info(master_handle, cid, &param_descriptor);
  146. if ((err != ESP_ERR_NOT_FOUND) && (param_descriptor != NULL)) {
  147. err = mbc_master_get_parameter(master_handle, param_descriptor->cid, (uint8_t *) temp_data, &type);
  148. if (err == ESP_OK) {
  149. ESP_LOGI(TAG, "Characteristic #%d %s (%s) value = %d read successful.",
  150. param_descriptor->cid,
  151. (char *) param_descriptor->param_key,
  152. (char *) param_descriptor->param_units,
  153. *(uint16_t *) temp_data);
  154. } else {
  155. ESP_LOGE(TAG, "Characteristic #%d (%s) read fail, err = 0x%x (%s).",
  156. param_descriptor->cid,
  157. (char *) param_descriptor->param_key,
  158. (int) err,
  159. (char *) esp_err_to_name(err));
  160. }
  161. }
  162. vTaskDelay(1000 / portTICK_PERIOD_MS);
  163. }
  164. vTaskDelay(5000 / portTICK_PERIOD_MS);
  165. }
  166. }
  167. void modbus_master_init() {
  168. ESP_LOGI(TAG, "Modbus master init...");
  169. ESP_ERROR_CHECK(master_init());
  170. vTaskDelay(10);
  171. xTaskCreatePinnedToCore(modbus_master_task, "modbus_master_task", 4 * 1024, NULL, 5, NULL, 1);
  172. }
  173. esp_err_t mm_set_param(uint16_t cid, uint8_t *value) {
  174. uint8_t type = PARAM_TYPE_U16;
  175. return mbc_master_set_parameter(master_handle, cid, value, &type);
  176. }
  177. esp_err_t mm_get_param(uint16_t cid, uint8_t *value) {
  178. uint8_t type = PARAM_TYPE_U16;
  179. return mbc_master_get_parameter(master_handle, cid, value, &type);
  180. }
  181. esp_err_t send_request(mb_param_request_t *request, void * data_prt) {
  182. return mbc_master_send_request(master_handle, request, data_prt);
  183. }
  184. void modbus_master_destroy() {
  185. mbc_master_delete(master_handle);
  186. }