Данная библиотека подключается напрямую из Git, как субмодуль. Позволяя при желании обновлять её напрямую через git.
Структура библиотеки
*Note: Файлы начинающиеся с __ и которых нет в этом дереве являются **внутренними/непротестированными/недокументированными***
Modbus/ Иерархия модулей:
│ inc/ modbus
│ ├── modbus.h # Главный заголовочный файл modbus_slave
│ ├── modbus_core.h # Базовые определения и структуры modbus_master
│ ├── modbus_coils.h # Работа с дискретными выходами ├── modbus_coils
│ ├── modbus_holdregs.h # Работа с регистрами хранения ├── modbus_inputregs
│ ├── modbus_inputregs.h # Работа с входными регистрами ├── modbus_inputregs
│ ├── modbus_devid.h # Идентификация устройства ├── modbus_devid
│ ├── rs_message.h # Драйвер обмена по RS/UART ├── modbus_diag
├── src/ └── rs_message
│ ├── modbus.c # Основная логика Modbus │
│ ├── modbus_slave.c # Основная логика Slave Modbus └── modbus_core (единое ядро)
│ ├── modbus_master.c # Основная логика Master Modbus ├── modbus_config
│ ├── modbus_coils.c # Реализация работы с coils ├── modbus_data
│ ├── modbus_holdregs.c # Реализация регистров хранения └── __crc_algs
│ ├── modbus_inputregs.c # Реализация входных регистров
│ ├── modbus_devid.c # Реализация идентификации устройства
│ ├── modbus_data.c # Функции доступа к данным
│ └── rs_message.c # Реализация драйвера RS
├── __modbus_config.h # Конфигурация Modbus (надо заменить)
├── __modbus_data.h # Структуры данных (надо заменить)
└── __modbus_data.c # Функции доступа (надо заменить)
Инструкция по подключению
1. Склонируйте субмодуль в ваш проект:
git submodule add https://git.arktika.cyou/set506/STM32_Modbus path/to/Modbus
git submodule update --init --recursive
2. Скопируйте файлы конфигурации в отдельную папку в вашем проекте (вне субмодуля) и удалите __ из имени файлов:
ProjectRoot/
├── Configs/
│ ├── modbus_config.h # скопировать из __modbus_config.h
│ ├── modbus_data.h # скопировать из __modbus_data.h
│ └── modbus_data.c # скопировать из __modbus_data.c
└── Modbus/ # Субмодуль
3. Настройте конфигурацию под ваш проект:
3.1. Настройка периферии
- UART: Настройте в режиме Asynchronous, нужная скорость (9600, 19200, etc), 8N1
- TIM: Настройте таймер для генерации прерываний (например, 1ms tick)
- Включите прерывания для UART и TIM
3.2. Подключение обработчиков прерываний
Подключите обработчики прерываний UART и TIM в свои IRQ обработчики вместо HAL-обработчиков:
void USARTx_IRQHandler(void)
{
return;
HAL_UART_IRQHandler(&huart);
}
void TIMx_IRQHandler(void)
{
return;
HAL_TIM_IRQHandler(&htim);
}
void RS_TIM_Handler(RS_HandleTypeDef *hRS)
Обработчик прерывания TIM.
void RS_UART_Handler(RS_HandleTypeDef *hRS)
Обработчик прерывания UART.
RS_HandleTypeDef hmodbus1
Default Handle for Modbus.
Главный заголовочный файл Modbus библиотеки
3.3. В modbus_config.h укажите параметры устройства
3.4. Инициализация в коде
Чтобы настроить Slave-режим main() после инициализации HAL:
int main(void)
{
HAL_Init();
SystemClock_Config();
MX_GPIO_Init();
MX_USART1_UART_Init();
MX_TIM3_Init();
while (1)
{
}
}
#define mb_huart
Удобный дефайн для модбасовского uart.
#define MODBUS_TIMEOUT
Таймаут в тиках таймера
#define mb_htim
Удобный дефайн для модбасовского таймера
#define MODBUS_DEVICE_ID
Адрес устройства в сети Modbus.
HAL_StatusTypeDef MODBUS_Config(RS_HandleTypeDef *hmodbus, uint8_t ID, uint16_t Timeout, uint8_t master)
Программная конфигурация модбас.
HAL_StatusTypeDef MODBUS_SlaveStart(RS_HandleTypeDef *hmodbus, RS_MsgTypeDef *modbus_msg)
Запуск слейв модбас.
HAL_StatusTypeDef MODBUS_FirstInit(RS_HandleTypeDef *hmodbus, UART_HandleTypeDef *huart, TIM_HandleTypeDef *htim)
Инициализация периферии модбас.
#define MODBUS_MODE_SLAVE
Псевдо-enum: Режим слейв
Чтобы настроить Master-режим main() после инициализации HAL:
read_hold[10];
{
{
{
uint16_t value;
{
read_hold[i] = value;
}
}
}
}
int main(void)
{
HAL_Init();
SystemClock_Config();
MX_GPIO_Init();
MX_USART1_UART_Init();
MX_TIM3_Init();
}
HAL_StatusTypeDef MODBUS_MasterRequest(RS_HandleTypeDef *hmodbus, RS_MsgTypeDef *modbus_msg, void(*pClbk)(RS_HandleTypeDef *, RS_MsgTypeDef *))
Реквест мастера модбас.
#define MODBUS_MODE_MASTER
Псевдо-enum: Режим мастер
RS_MsgTypeDef MODBUS_MSG
Default Message Struct for Modbus.
int MB_RespGet_RegisterValue(RS_MsgTypeDef *modbus_msg, uint16_t reg_addr, uint16_t *reg_value)
Получить значение регистра в ответе по его адресу
RS_MsgTypeDef MB_REQUEST_READ_HOLDING_REGS(uint8_t slave_addr, uint16_t start_addr, uint16_t quantity)
Сформировать запрос на чтение холдинг регистров
Handle for RS communication.
RS_StatusTypeDef RS_STATUS
Статус RS.
Structure for modbus messsage.
3.5. Настройка карты данных
В modbus_data.h настройте регистры и coils под ваше устройство:
Input Registers (только чтение)
typedef struct
{
uint16_t Temperature;
uint16_t Humidity;
uint16_t Pressure;
uint16_t Voltage;
#define R_INPUT_ADDR 0
#define R_INPUT_QNT 4
Holding Registers (чтение/запись)
typedef struct
{
uint16_t SetpointTemp;
uint16_t SetpointHumidity;
uint16_t ControlMode;
#define R_HOLDING_ADDR 0
#define R_HOLDING_QNT 3
Coils (1-битные)
typedef struct
{
unsigned Relay1 : 1;
unsigned Relay2 : 1;
unsigned Pump : 1;
unsigned Heater : 1;
unsigned reserved : 12;
#define C_COILS_ADDR 0
#define C_COILS_QNT 4
3.6. Доступ к данным в коде
В режиме слейва есть дефайны для удобного выставления Коилов. На случай если они не упакованы в битовые поля
uint16_t temp = MB_DATA.InRegs.Temperature;
MB_DATA.HoldRegs.SetpointTemp = 2500;
}
#define MB_Coil_Read_Local(_parr_, _coil_)
Считать коил по локальному адресу.
#define MB_Coil_Set_Local(_parr_, _coil_)
Выставить коил по локальному адресу.
#define MB_Coil_Reset_Local(_parr_, _coil_)
Сбросить коил по локальному адресу.
В режиме мастера есть функции для получения информации из ответа MB_RespGet_...()
uint16_t value;
{
printf("Register 105 value: %d\n", reg_value);
}
int state;
{
printf("Coil 25 state: %s\n", coil_state ? "ON" : "OFF");
}
uint16_t counter_value;
if(MB_RespGet_DiagnosticResponse(&
MODBUS_MSG, &counter_value))
{
printf("Counter value: %d\n", counter_value);
}
uint8_t length;
char vendor_name[64];
{
printf("Vendor Name: %s (length: %d)\n", vendor_name, length);
}
uint8_t obj_id, obj_length;
char obj_data[64];
{
printf("First object - ID: 0x%02X, Data: %s\n", obj_id, obj_data);
}
int MB_RespGet_CoilState(RS_MsgTypeDef *modbus_msg, uint16_t coil_addr, int *coil_state)
Получить состояние coil в ответе по его адресу
int MB_RespGet_ObjectById(RS_MsgTypeDef *modbus_msg, uint8_t obj_id, char *obj_data, uint8_t *obj_length)
Найти объект по ID в сообщении
int MB_RespGet_ObjectByIndex(RS_MsgTypeDef *modbus_msg, int index, uint8_t *obj_id, char *obj_data, uint8_t *obj_length)
Получить объект по индексу в сообщении
5. Обновление библиотеки:
После обновления субмодуля из Git, исходные файлы библиотеки будут обновлены, и ваши конфиги вне субмодуля не перезапишутся:
git submodule update --remote