| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207 |
- /**
- * @Author: 李建
- * @Date: 2025/4/27 15:35
- * Description: 实现485主站
- * Copyright: Copyright (©) 2025 永续绿建. All rights reserved.
- */
- #include <esp_err.h>
- #include "modbus_master.h"
- #include "main.h"
- #include "mbcontroller.h"
- #define OPTS(min_val, max_val, step_val) { .opt1 = min_val, .opt2 = max_val, .opt3 = step_val }
- static const char *TAG = "MODBUS_MASTER";
- enum {
- //
- MB_DEVICE_JS_ADDR = 7,
- MB_DEVICE_CS_ADDR = 20,
- };
- static void *master_handle = NULL;
- // The number of parameters that intended to be used in the particular control process
- #define MASTER_MAX_CIDS num_device_parameters
- // Number of reading of parameters from slave
- #define MASTER_MAX_RETRY (10)
- // Timeout to update cid over Modbus
- #define UPDATE_CIDS_TIMEOUT_MS (500)
- #define UPDATE_CIDS_TIMEOUT_TICS (UPDATE_CIDS_TIMEOUT_MS / portTICK_PERIOD_MS)
- // Timeout between polls
- #define POLL_TIMEOUT_MS (1)
- #define POLL_TIMEOUT_TICS (POLL_TIMEOUT_MS / portTICK_PERIOD_MS)
- // The macro to get offset for parameter in the appropriate structure
- #define HOLD_OFFSET(field) ((uint16_t)(offsetof(holding_reg_params_t, field) + 1))
- #define INPUT_OFFSET(field) ((uint16_t)(offsetof(input_reg_params_t, field) + 1))
- // Discrete offset macro
- #define STR(fieldname) ((const char *)( fieldname ))
- mb_parameter_descriptor_t device_parameters[] = {
- // { CID, Param Name, Units, Modbus Slave Addr, Modbus Reg Type, Reg Start, Reg Size, Instance Offset, Data Type, Data Size, Parameter Options, Access Mode}
- {CID_CS_POWER, STR("cs_power"), STR(""), MB_DEVICE_CS_ADDR, MB_PARAM_HOLDING,
- CS_POWER_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- {CID_JS_POWER, STR("js_power"), STR(""), MB_DEVICE_JS_ADDR, MB_PARAM_HOLDING,
- JS_POWER_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- {CID_CS_MODE, STR("cs_mode"), STR(""), MB_DEVICE_CS_ADDR, MB_PARAM_HOLDING,
- CS_MODE_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- {CID_JS_MODE, STR("js_mode"), STR(""), MB_DEVICE_JS_ADDR, MB_PARAM_HOLDING,
- JS_MODE_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- {CID_CS_FAN_SPEED, STR("cs_fan_speed"), STR(""), MB_DEVICE_CS_ADDR, MB_PARAM_HOLDING,
- FAN_CS_SPEED_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- {CID_JS_FAN_SPEED, STR("cs_fan_speed"), STR(""), MB_DEVICE_JS_ADDR, MB_PARAM_HOLDING,
- FAN_JS_SPEED_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- {CID_JS_MIN_HUMIDITY, STR("js_min_humidity"), STR(""), MB_DEVICE_JS_ADDR, MB_PARAM_HOLDING,
- JS_MIN_HUMIDITY_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- {CID_JS_MAX_HUMIDITY, STR("js_max_humidity"), STR(""), MB_DEVICE_JS_ADDR, MB_PARAM_HOLDING,
- JS_MAX_HUMIDITY_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- {CID_CS_SET_HUMIDITY, STR("cs_set_humidity"), STR(""), MB_DEVICE_CS_ADDR, MB_PARAM_HOLDING,
- CS_SET_HUMIDITY_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- {CID_CS_ERROR_CODE, STR("cs_error_code"), STR(""), MB_DEVICE_CS_ADDR, MB_PARAM_HOLDING,
- CS_ERROR_CODE_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- {CID_JS_ERROR_CODE, STR("js_error_code"), STR(""), MB_DEVICE_JS_ADDR, MB_PARAM_HOLDING,
- JS_ERROR_CODE_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- {CID_CS_INEFFICIENT_FILTER_SIGN, STR("inefficient_filter_sign"), STR(""), MB_DEVICE_CS_ADDR, MB_PARAM_HOLDING,
- CS_INEFFICIENT_FILTER_SIGN_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- {CID_CS_HEPA_FILTER_SIGN, STR("hepa_filer_sigh"), STR(""), MB_DEVICE_CS_ADDR, MB_PARAM_HOLDING,
- CS_HEPA_FILTER_SIGN_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- {CID_JS_EXTERNAL_HUMIDITY, STR("external_humidity"), STR(""), MB_DEVICE_JS_ADDR, MB_PARAM_HOLDING,
- JS_EXTERNAL_HUMIDITY_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- {CID_JS_MOULD_PROOF_MODE_TIME, STR("proof_mode_time"), STR(""), MB_DEVICE_JS_ADDR, MB_PARAM_HOLDING,
- JS_MOULD_PROOF_MODE_TIME_ADDRESS_REGISTER, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- {CID_CS_INEFFICIENT_FILTER_SET_TIME, STR("inefficient_fileter_set"), STR(""), MB_DEVICE_CS_ADDR, MB_PARAM_HOLDING,
- CS_INEFFICIENT_FILTER_SET_TIME, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- {CID_CS_HEPA_FILTER_SET_TIME, STR("hepa_fileter_set"), STR(""), MB_DEVICE_CS_ADDR, MB_PARAM_HOLDING,
- CS_HEPA_FILTER_SET_TIME, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- {CID_CS_INEFFICIENT_FILTER_USED_TIME, STR("inefficient_fileter_used"), STR(""), MB_DEVICE_CS_ADDR, MB_PARAM_HOLDING,
- CS_INEFFICIENT_FILTER_USED_TIME, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- {CID_CS_HEPA_FILTER_USED_TIME, STR("hepa_fileter_used"), STR(""), MB_DEVICE_CS_ADDR, MB_PARAM_HOLDING,
- CS_HEPA_FILTER_USED_TIME, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- {CID_CS_INEFFICIENT_FILTER_RESET, STR("inefficient_fileter_reset"), STR(""), MB_DEVICE_CS_ADDR, MB_PARAM_HOLDING,
- CS_INEFFICIENT_FILTER_RESET, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- {CID_CS_HEPA_FILTER_RESET, STR("hepa_fileter_reset"), STR(""), MB_DEVICE_CS_ADDR, MB_PARAM_HOLDING,
- CS_HEPA_FILTER_RESET, 1, 0, PARAM_TYPE_U16, 2, OPTS(0, 0, 0), PAR_PERMS_READ_WRITE},
- };
- // Calculate number of parameters in the table
- const uint16_t num_device_parameters = (sizeof(device_parameters) / sizeof(device_parameters[0]));
- IRAM_ATTR static esp_err_t master_init() {
- // Initialize Modbus controller
- mb_communication_info_t comm = {
- .ser_opts.port = MB_PORT_NUM,
- .ser_opts.mode = MB_RTU,
- .ser_opts.baudrate = MB_DEV_SPEED,
- .ser_opts.parity = MB_PARITY_NONE,
- .ser_opts.response_tout_ms = 400,
- .ser_opts.data_bits = UART_DATA_8_BITS,
- .ser_opts.stop_bits = UART_STOP_BITS_1
- };
- ESP_LOGI(TAG, "Modbus controller initialized...");
- esp_err_t err = mbc_master_create_serial(&comm, &master_handle);
- MB_RETURN_ON_FALSE((master_handle != NULL), ESP_ERR_INVALID_STATE, TAG,
- "mb controller initialization fail.");
- MB_RETURN_ON_FALSE((err == ESP_OK), ESP_ERR_INVALID_STATE, TAG,
- "mb controller initialization fail, returns(0x%x).", (int) err);
- // Set UART pin numbers
- err = uart_set_pin(MB_PORT_NUM, MB_GPIO_TX, MB_GPIO_RX,
- UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE);
- MB_RETURN_ON_FALSE((err == ESP_OK), ESP_ERR_INVALID_STATE, TAG,
- "mb serial set pin failure, uart_set_pin() returned (0x%x).", (int) err);
- err = mbc_master_start(master_handle);
- MB_RETURN_ON_FALSE((err == ESP_OK), ESP_ERR_INVALID_STATE, TAG,
- "mb controller start fail, returned (0x%x).", (int) err);
- // Set driver mode to Half Duplex
- err = uart_set_mode(MB_PORT_NUM, UART_MODE_UART);
- MB_RETURN_ON_FALSE((err == ESP_OK), ESP_ERR_INVALID_STATE, TAG,
- "mb serial set mode failure, uart_set_mode() returned (0x%x).", (int) err);
- vTaskDelay(5);
- err = mbc_master_set_descriptor(master_handle, &device_parameters[0], num_device_parameters);
- MB_RETURN_ON_FALSE((err == ESP_OK), ESP_ERR_INVALID_STATE, TAG,
- "mb controller set descriptor fail, returns(0x%x).", (int) err);
- ESP_LOGI(TAG, "Modbus master stack initialized...");
- return err;
- }
- //static void *master_get_param_data(const mb_parameter_descriptor_t *param_descriptor)
- //{
- // assert(param_descriptor != NULL);
- // void *instance_ptr = NULL;
- // if (param_descriptor->param_offset != 0) {
- // switch(param_descriptor->mb_param_type)
- // {
- // case MB_PARAM_HOLDING:
- // instance_ptr = ((void *)&holding_reg_params + param_descriptor->param_offset - 1);
- // break;
- // default:
- // instance_ptr = NULL;
- // break;
- // }
- // } else {
- // ESP_LOGE(TAG, "Wrong parameter offset for CID #%u", (unsigned)param_descriptor->cid);
- // assert(instance_ptr != NULL);
- // }
- // return instance_ptr;
- //}
- void modbus_master_task(void *arg) {
- vTaskDelay(5000 / portTICK_PERIOD_MS);
- ESP_LOGI(TAG, "Modbus master task started...");
- esp_err_t err = ESP_OK;
- uint8_t temp_data[2] = {0}; // temporary buffer to hold maximum CID size
- uint8_t type = 0;
- const mb_parameter_descriptor_t *param_descriptor = NULL;
- for (;;) {
- for (uint16_t cid = 0; (err != ESP_ERR_NOT_FOUND) && cid < 1; cid++) {
- err = mbc_master_get_cid_info(master_handle, cid, ¶m_descriptor);
- if ((err != ESP_ERR_NOT_FOUND) && (param_descriptor != NULL)) {
- err = mbc_master_get_parameter(master_handle, param_descriptor->cid, (uint8_t *) temp_data, &type);
- if (err == ESP_OK) {
- ESP_LOGI(TAG, "Characteristic #%d %s (%s) value = %d read successful.",
- param_descriptor->cid,
- (char *) param_descriptor->param_key,
- (char *) param_descriptor->param_units,
- *(uint16_t *) temp_data);
- } else {
- ESP_LOGE(TAG, "Characteristic #%d (%s) read fail, err = 0x%x (%s).",
- param_descriptor->cid,
- (char *) param_descriptor->param_key,
- (int) err,
- (char *) esp_err_to_name(err));
- }
- }
- vTaskDelay(1000 / portTICK_PERIOD_MS);
- }
- vTaskDelay(5000 / portTICK_PERIOD_MS);
- }
- }
- void modbus_master_init() {
- ESP_LOGI(TAG, "Modbus master init...1");
- ESP_ERROR_CHECK(master_init());
- vTaskDelay(10);
- //xTaskCreatePinnedToCore(modbus_master_task, "modbus_master_task", 4 * 1024, NULL, 5, NULL, 1);
- }
- esp_err_t mm_set_param(uint16_t cid, uint8_t *value) {
- uint8_t type = PARAM_TYPE_U16;
- return mbc_master_set_parameter(master_handle, cid, value, &type);
- }
- esp_err_t mm_get_param(uint16_t cid, uint8_t *value) {
- uint8_t type = PARAM_TYPE_U16;
- return mbc_master_get_parameter(master_handle, cid, value, &type);
- }
- void modbus_master_destroy() {
- mbc_master_delete(master_handle);
- }
|