2024-07-05 03:52:43 +00:00
|
|
|
|
#include "frt_protocol.h"
|
|
|
|
|
#include "cmsis_os.h"
|
|
|
|
|
#include "assertions.h"
|
|
|
|
|
#include "inflash.h"
|
|
|
|
|
#include "anemometer_dev.h"
|
|
|
|
|
#include "sht30.h"
|
2024-07-17 02:58:39 +00:00
|
|
|
|
#include "hp203b.h"
|
2024-07-05 03:52:43 +00:00
|
|
|
|
|
|
|
|
|
extern u_int8_t rs485_out_buff[50];
|
2024-07-09 08:08:38 +00:00
|
|
|
|
extern uart_device_info uart_devices[];
|
2024-07-05 03:52:43 +00:00
|
|
|
|
|
|
|
|
|
/* 静态函数申明 */
|
|
|
|
|
static void send_uart_pack(device_handle device,FRT_MsgFunctionCode_e cmd_type, const void *data, u_int16_t len);
|
|
|
|
|
static void FRT_MsgProc_ReadRegister(device_handle device, void *pMsg);
|
|
|
|
|
static void FRT_MsgProc_WriteRegister(device_handle device, void *pMsg);
|
|
|
|
|
static u_int16_t FRT_ReadReg(unsigned char regId);
|
|
|
|
|
static u_int16_t FRT_ReadRegMinWindDiretion(void *pMsg);
|
|
|
|
|
static u_int16_t FRT_ReadRegAverageWindDirection(void *pMsg);
|
|
|
|
|
static u_int16_t FRT_ReadRegMaxWindDirection(void *pMsg);
|
|
|
|
|
static u_int16_t FRT_ReadRegMinWindSpeed(void *pMsg);
|
|
|
|
|
static u_int16_t FRT_ReadRegAverageWindSpeed(void *pMsg);
|
|
|
|
|
static u_int16_t FRT_ReadRegMaxWindSpeed(void *pMsg);
|
|
|
|
|
static u_int16_t FRT_ReadRegTemperature(void *pMsg);
|
|
|
|
|
static u_int16_t FRT_ReadRegHumidity(void *pMsg);
|
|
|
|
|
static u_int16_t FRT_ReadRegPressure(void *pMsg);
|
|
|
|
|
static u_int16_t FRT_ReadRegRain(void *pMsg);
|
|
|
|
|
static u_int16_t FRT_ReadRegPrecipitationIntensity(void *pMsg);
|
|
|
|
|
static u_int16_t FRT_ReadRegDeviceAddr(void *pMsg);
|
2024-07-08 02:09:40 +00:00
|
|
|
|
static u_int16_t FRT_ReadRegCommuBaudRate(void *pMsg);
|
|
|
|
|
static u_int16_t FRT_ReadRegSpeedAverageTime(void *pMsg);
|
|
|
|
|
static u_int16_t FRT_ReadRegTempHumUpdateTime(void *pMsg);
|
|
|
|
|
static u_int16_t FRT_ReadRegRainPowerCtl(void *pMsg);
|
2024-07-08 02:55:37 +00:00
|
|
|
|
static u_int16_t FRT_WriteRegDeviceAddr(void *pMsg);
|
|
|
|
|
static u_int16_t FRT_WriteRegCommuBaudRate(void *pMsg);
|
|
|
|
|
static u_int16_t FRT_WriteRegSpeedAverageTime(void *pMsg);
|
|
|
|
|
static u_int16_t FRT_WriteRegTempHumUpdateTime(void *pMsg);
|
|
|
|
|
static u_int16_t FRT_WriteRegSoftReset(void *pMsg);
|
|
|
|
|
static u_int16_t FRT_WriteRegResetDefault(void *pMsg);
|
2024-07-05 03:52:43 +00:00
|
|
|
|
static void pdebug_mcs_info();
|
|
|
|
|
|
|
|
|
|
/* 功能码处理表 */
|
|
|
|
|
FRT_FuncionMsgProcTable_s g_MsgTbl[] =
|
|
|
|
|
{
|
|
|
|
|
{ FRT_FUNCTION_CODE_READ_REGISTER, FRT_MsgProc_ReadRegister },
|
|
|
|
|
{ FRT_FUNCTION_CODE_WRITE_REGISTER, FRT_MsgProc_WriteRegister },
|
|
|
|
|
};
|
|
|
|
|
|
2024-07-08 02:09:40 +00:00
|
|
|
|
/* 读寄存器处理表 */
|
2024-07-05 03:52:43 +00:00
|
|
|
|
FRT_RegProcTable_s g_RegTbl[] =
|
|
|
|
|
{
|
2024-07-08 02:55:37 +00:00
|
|
|
|
{ FRT_REGISTER_MIN_WIND_DIRECTION, FRT_ReadRegMinWindDiretion }, /* 最小风向 */
|
|
|
|
|
{ FRT_REGISTER_AVERAGE_WIND_DIRECTION, FRT_ReadRegAverageWindDirection }, /* 平均风向 */
|
|
|
|
|
{ FRT_REGISTER_MAX_WIND_DIRECTION, FRT_ReadRegMaxWindDirection }, /* 最大风向 */
|
|
|
|
|
{ FRT_REGISTER_MIN_WIND_SPEED, FRT_ReadRegMinWindSpeed }, /* 最小风速 */
|
|
|
|
|
{ FRT_REGISTER_AVERAGE_WIND_SPEED, FRT_ReadRegAverageWindSpeed }, /* 平均风速 */
|
|
|
|
|
{ FRT_REGISTER_MAX_WIND_SPEED, FRT_ReadRegMaxWindSpeed }, /* 最大风速 */
|
|
|
|
|
{ FRT_REGISTER_TEMPERATURE, FRT_ReadRegTemperature }, /* 大气温度 */
|
|
|
|
|
{ FRT_REGISTER_HUMIDITY, FRT_ReadRegHumidity }, /* 大气湿度 */
|
|
|
|
|
{ FRT_REGISTER_PRESSURE, FRT_ReadRegPressure }, /* 大气压 */
|
|
|
|
|
{ FRT_REGISTER_RAIN, FRT_ReadRegRain }, /* 雨量 */
|
|
|
|
|
{ FRT_REGISTER_PRECIPITATION_INTENSITY, FRT_ReadRegPrecipitationIntensity }, /* 总辐射 */
|
|
|
|
|
{ FRT_REGISTER_DEVICE_ADDR, FRT_ReadRegDeviceAddr }, /* 设备地址 */
|
|
|
|
|
{ FRT_REGISTER_COMMU_BAUDRATE, FRT_ReadRegCommuBaudRate }, /* 波特率 */
|
|
|
|
|
{ FRT_REGISTER_SPEED_AVERAGE_TIME, FRT_ReadRegSpeedAverageTime }, /* 风速平均时间 */
|
|
|
|
|
{ FRT_REGISTER_TEMPHUM_UPDATE_TIME, FRT_ReadRegTempHumUpdateTime }, /* 温湿度更新时间 */
|
|
|
|
|
{ FRT_REGISTER_RAIN_POWER_CONTROL, FRT_ReadRegRainPowerCtl }, /* 雨量电源时间 */
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/* 写寄存器处理表 */
|
|
|
|
|
FRT_RegProcTable_s g_Write_RegTbl[] =
|
|
|
|
|
{
|
|
|
|
|
{ FRT_REGISTER_DEVICE_ADDR, FRT_WriteRegDeviceAddr }, /* 写设备地址 */
|
|
|
|
|
{ FRT_REGISTER_COMMU_BAUDRATE, FRT_WriteRegCommuBaudRate }, /* 写波特率 */
|
|
|
|
|
{ FRT_REGISTER_SPEED_AVERAGE_TIME, FRT_WriteRegSpeedAverageTime }, /* 写风速平均时间 */
|
|
|
|
|
{ FRT_REGISTER_TEMPHUM_UPDATE_TIME, FRT_WriteRegTempHumUpdateTime }, /* 写温湿度更新时间 */
|
|
|
|
|
{ FRT_REGISTER_SOFT_REST, FRT_WriteRegSoftReset }, /* 软件复位 */
|
|
|
|
|
{ FRT_REGISTER_REST_DEFAULT_SETTING, FRT_WriteRegResetDefault }, /* 恢复出厂设置 */
|
2024-07-05 03:52:43 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief modbus crc16算法
|
|
|
|
|
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
unsigned short CRC16(unsigned char *arr_buff, unsigned char len)
|
|
|
|
|
{
|
|
|
|
|
unsigned short crc=0xFFFF;
|
|
|
|
|
unsigned char i, j;
|
|
|
|
|
for ( j=0; j<len; j++){
|
|
|
|
|
crc=crc ^*arr_buff++;
|
|
|
|
|
for ( i=0; i<8; i++){
|
|
|
|
|
if( ( crc&0x0001) >0){
|
|
|
|
|
crc=crc>>1;
|
|
|
|
|
crc=crc^ 0xa001;
|
|
|
|
|
}else{
|
|
|
|
|
crc=crc>>1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return crc;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 交换高低字节
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_swap_endian_16(u_int16_t value)
|
|
|
|
|
{
|
|
|
|
|
return ((value << 8) | (value >> 8));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 读最小风向寄存器值
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_ReadRegMinWindDiretion(void *pMsg)
|
|
|
|
|
{
|
|
|
|
|
u_int16_t value = (u_int16_t)(g_stMcs_Para.min_wind_direction *10);
|
|
|
|
|
return FRT_swap_endian_16(value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 读平均风向寄存器值
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_ReadRegAverageWindDirection(void *pMsg)
|
|
|
|
|
{
|
|
|
|
|
u_int16_t value = (u_int16_t)(g_stMcs_Para.average_wind_direction *10);
|
|
|
|
|
return FRT_swap_endian_16(value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 读最大风向寄存器值
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_ReadRegMaxWindDirection(void *pMsg)
|
|
|
|
|
{
|
|
|
|
|
u_int16_t value=(u_int16_t)(g_stMcs_Para.max_wind_direction *10);
|
|
|
|
|
return FRT_swap_endian_16(value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 读最小风速寄存器值
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_ReadRegMinWindSpeed(void *pMsg)
|
|
|
|
|
{
|
|
|
|
|
u_int16_t value=(u_int16_t)(g_stMcs_Para.min_wind_speed *10);
|
|
|
|
|
return FRT_swap_endian_16(value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 读平均风速寄存器值
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_ReadRegAverageWindSpeed(void *pMsg)
|
|
|
|
|
{
|
|
|
|
|
u_int16_t value=(u_int16_t)(g_stMcs_Para.average_wind_speed *10);
|
|
|
|
|
return FRT_swap_endian_16(value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 读最大风速寄存器值
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_ReadRegMaxWindSpeed(void *pMsg)
|
|
|
|
|
{
|
|
|
|
|
u_int16_t value=(u_int16_t)(g_stMcs_Para.max_wind_speed *10);
|
|
|
|
|
return FRT_swap_endian_16(value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 读温度寄存器值
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_ReadRegTemperature(void *pMsg)
|
|
|
|
|
{
|
|
|
|
|
u_int16_t value= (u_int16_t)(g_stMcs_Para.temperature*10);
|
|
|
|
|
return FRT_swap_endian_16(value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 读湿度寄存器值
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_ReadRegHumidity(void *pMsg)
|
|
|
|
|
{
|
|
|
|
|
u_int16_t value=(u_int16_t)(g_stMcs_Para.humidity*10);
|
|
|
|
|
return FRT_swap_endian_16(value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 读气压寄存器值
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_ReadRegPressure(void *pMsg)
|
|
|
|
|
{
|
2024-07-08 02:09:40 +00:00
|
|
|
|
u_int16_t value=(u_int16_t)(g_stMcs_Para.pressure*10);
|
2024-07-05 03:52:43 +00:00
|
|
|
|
return FRT_swap_endian_16(value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 读雨量寄存器值
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_ReadRegRain(void *pMsg)
|
|
|
|
|
{
|
|
|
|
|
u_int16_t value=9;
|
|
|
|
|
return FRT_swap_endian_16(value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 读光辐射强度寄存器值
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_ReadRegPrecipitationIntensity(void *pMsg)
|
|
|
|
|
{
|
|
|
|
|
u_int16_t value=10;
|
|
|
|
|
return FRT_swap_endian_16(value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 读设备地址寄存器值
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_ReadRegDeviceAddr(void *pMsg)
|
2024-07-08 02:09:40 +00:00
|
|
|
|
{
|
2024-07-09 08:08:38 +00:00
|
|
|
|
read_config_info();
|
2024-07-05 03:52:43 +00:00
|
|
|
|
u_int16_t value=g_stConfigInfo.addr;
|
|
|
|
|
return FRT_swap_endian_16(value);
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-08 02:09:40 +00:00
|
|
|
|
/**
|
|
|
|
|
* @brief 读波特率寄存器值
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_ReadRegCommuBaudRate(void *pMsg)
|
|
|
|
|
{
|
2024-07-09 08:08:38 +00:00
|
|
|
|
read_config_info();
|
2024-07-08 02:09:40 +00:00
|
|
|
|
u_int16_t value=g_stConfigInfo.uart_baud;
|
2024-07-09 02:48:29 +00:00
|
|
|
|
// term_printf("%x", value);
|
2024-07-08 02:09:40 +00:00
|
|
|
|
return FRT_swap_endian_16(value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 读风速平均时间值
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_ReadRegSpeedAverageTime(void *pMsg)
|
|
|
|
|
{
|
2024-07-09 08:08:38 +00:00
|
|
|
|
read_config_info();
|
2024-07-08 02:09:40 +00:00
|
|
|
|
u_int16_t value=g_stConfigInfo.speed_average_time;
|
|
|
|
|
return FRT_swap_endian_16(value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 读温湿度更新时间值
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_ReadRegTempHumUpdateTime(void *pMsg)
|
|
|
|
|
{
|
2024-07-09 08:08:38 +00:00
|
|
|
|
read_config_info();
|
2024-07-08 02:09:40 +00:00
|
|
|
|
u_int16_t value=g_stConfigInfo.temp_hum_update_time;
|
|
|
|
|
return FRT_swap_endian_16(value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 读雨量电源控制值
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_ReadRegRainPowerCtl(void *pMsg)
|
|
|
|
|
{
|
2024-07-08 02:55:37 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 写设备地址
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_WriteRegDeviceAddr(void *pMsg)
|
|
|
|
|
{
|
2024-07-08 08:09:54 +00:00
|
|
|
|
uint16_t *pMsgAddr = (uint16_t *)pMsg;
|
|
|
|
|
uint16_t data = *pMsgAddr;
|
2024-07-09 02:48:29 +00:00
|
|
|
|
g_stConfigInfo.addr = data;
|
2024-07-09 08:08:38 +00:00
|
|
|
|
save_config_info(g_stConfigInfo);
|
2024-07-08 08:09:54 +00:00
|
|
|
|
|
2024-07-08 02:55:37 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 写波特率
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_WriteRegCommuBaudRate(void *pMsg)
|
|
|
|
|
{
|
2024-07-08 08:09:54 +00:00
|
|
|
|
uint16_t *pMsgAddr = (uint16_t *)pMsg;
|
|
|
|
|
uint16_t data = *pMsgAddr;
|
2024-07-09 02:48:29 +00:00
|
|
|
|
g_stConfigInfo.uart_baud = data;
|
2024-07-09 08:08:38 +00:00
|
|
|
|
save_config_info(g_stConfigInfo);
|
|
|
|
|
|
|
|
|
|
MX_USART3_UART_Init(g_stConfigInfo.uart_baud);
|
|
|
|
|
MX_USART1_UART_Init(g_stConfigInfo.uart_baud);
|
2024-07-08 08:09:54 +00:00
|
|
|
|
|
2024-07-08 02:55:37 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 写风速平均时间
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_WriteRegSpeedAverageTime(void *pMsg)
|
|
|
|
|
{
|
2024-07-08 08:09:54 +00:00
|
|
|
|
uint16_t *pMsgAddr = (uint16_t *)pMsg;
|
|
|
|
|
uint16_t data = *pMsgAddr;
|
|
|
|
|
|
|
|
|
|
g_stConfigInfo.speed_average_time = data;
|
2024-07-09 08:08:38 +00:00
|
|
|
|
save_config_info(g_stConfigInfo);
|
2024-07-08 08:09:54 +00:00
|
|
|
|
|
2024-07-08 02:55:37 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 写温湿度更新时间
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_WriteRegTempHumUpdateTime(void *pMsg)
|
|
|
|
|
{
|
2024-07-08 08:09:54 +00:00
|
|
|
|
uint16_t *pMsgAddr = (uint16_t *)pMsg;
|
|
|
|
|
uint16_t data = *pMsgAddr;
|
|
|
|
|
|
|
|
|
|
g_stConfigInfo.temp_hum_update_time = data;
|
2024-07-09 08:08:38 +00:00
|
|
|
|
save_config_info(g_stConfigInfo);
|
2024-07-08 08:09:54 +00:00
|
|
|
|
|
2024-07-08 02:55:37 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 软件复位
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_WriteRegSoftReset(void *pMsg)
|
|
|
|
|
{
|
2024-07-09 02:48:29 +00:00
|
|
|
|
|
|
|
|
|
uint16_t *pMsgAddr = (uint16_t *)pMsg;
|
|
|
|
|
uint16_t data = *pMsgAddr;
|
|
|
|
|
if(data == 0x0001)
|
|
|
|
|
{
|
|
|
|
|
__iar_builtin_set_FAULTMASK(1);
|
|
|
|
|
NVIC_SystemReset();
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-08 02:55:37 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2024-07-09 02:48:29 +00:00
|
|
|
|
* @brief 恢复出厂设置
|
2024-07-08 02:55:37 +00:00
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int16_t FRT_WriteRegResetDefault(void *pMsg)
|
|
|
|
|
{
|
2024-07-08 08:09:54 +00:00
|
|
|
|
uint16_t *pMsgAddr = (uint16_t *)pMsg;
|
|
|
|
|
uint16_t data = *pMsgAddr;
|
2024-07-09 02:48:29 +00:00
|
|
|
|
|
|
|
|
|
if(data == 0x0001)
|
|
|
|
|
{
|
2024-07-09 08:08:38 +00:00
|
|
|
|
delete_config_info();
|
|
|
|
|
|
2024-07-09 02:48:29 +00:00
|
|
|
|
__iar_builtin_set_FAULTMASK(1);
|
|
|
|
|
NVIC_SystemReset();
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-08 02:09:40 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-05 03:52:43 +00:00
|
|
|
|
/**
|
|
|
|
|
* @brief 封装协议发送
|
|
|
|
|
|
|
|
|
|
* @param
|
|
|
|
|
* @retval 数据正确返回TRUE,数据异常返回FALSE
|
|
|
|
|
*/
|
|
|
|
|
static void send_uart_pack(device_handle device,FRT_MsgFunctionCode_e cmd_type, const void *data, u_int16_t len)
|
|
|
|
|
{
|
|
|
|
|
memset(rs485_out_buff,0,sizeof(rs485_out_buff));
|
|
|
|
|
frt_climate_pack_resp *pack = (frt_climate_pack_resp*)rs485_out_buff;
|
|
|
|
|
|
|
|
|
|
pack->addr = g_stConfigInfo.addr;
|
|
|
|
|
pack->func = FRT_FUNCTION_CODE_READ_REGISTER;
|
|
|
|
|
pack->data_len = len;
|
|
|
|
|
memcpy(pack->data, data, len);
|
|
|
|
|
*(u_int16_t*)&pack->data[len] = CRC16((u_int8_t *)&pack->addr,pack->data_len+3);
|
|
|
|
|
|
|
|
|
|
uart_dev_write(device,(u_int8_t*)pack,pack->data_len+5);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2024-07-08 02:55:37 +00:00
|
|
|
|
* @brief 遍历读取寄存器
|
2024-07-05 03:52:43 +00:00
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
u_int16_t FRT_ReadReg(unsigned char regId)
|
|
|
|
|
{
|
|
|
|
|
for(u_int16_t i = 0; i < sizeof(g_RegTbl) / sizeof(FRT_RegProcTable_s); i++){
|
|
|
|
|
//term_printf("regId:%d, g_RegTbl.regId :%d\r\n",regId,g_RegTbl[i].regId);
|
|
|
|
|
if (regId == g_RegTbl[i].regId){
|
|
|
|
|
return g_RegTbl[i].pRegProc(NULL);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 处理读取寄存器数据
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
void FRT_MsgProc_ReadRegister(device_handle device, void *pMsg)
|
|
|
|
|
{
|
2024-07-06 07:26:21 +00:00
|
|
|
|
// static u_int8_t reg_value_buff[2*100]={0x00};
|
|
|
|
|
//
|
|
|
|
|
// memset(reg_value_buff,0,sizeof(reg_value_buff));
|
|
|
|
|
// frt_climate_pack *pack = (frt_climate_pack*)pMsg;
|
|
|
|
|
//
|
|
|
|
|
// u_int16_t start_reg_addr = (pack->addr_begin_high_byte<<8)| pack->addr_begin_low_byte;
|
|
|
|
|
// u_int16_t reg_num= (pack->regnum_begin_high_byte<<8)| pack->regnum_begin_low_byte;
|
|
|
|
|
//
|
|
|
|
|
// AssertError(start_reg_addr < 100,return, "读取寄存器起始地址错误%d",start_reg_addr);
|
|
|
|
|
// AssertError(reg_num < 100,return, "读取寄存器数量错误%d",reg_num);
|
|
|
|
|
//
|
|
|
|
|
// /* 采集sht30数据 */
|
|
|
|
|
// AssertError(get_temp_humi_data(&g_stMcs_Para.temperature, &g_stMcs_Para.humidity),g_stMcs_Para.temperature=0;g_stMcs_Para.humidity=0,"采集sht30温湿度数据失败" );
|
|
|
|
|
//
|
|
|
|
|
//// /* 调试信息 */
|
|
|
|
|
//// pdebug_mcs_info();
|
|
|
|
|
//
|
|
|
|
|
// for(u_int16_t pos=0; pos <reg_num; pos++){
|
|
|
|
|
// *(u_int16_t*)®_value_buff[pos*2] = FRT_ReadReg((start_reg_addr+pos));
|
|
|
|
|
// }
|
|
|
|
|
//
|
|
|
|
|
// if(device == g_term_uart_handle)
|
|
|
|
|
// {
|
|
|
|
|
// pdebug_mcs_info();
|
|
|
|
|
// }
|
|
|
|
|
// if(device == g_rs485_uart_handle)
|
|
|
|
|
// {
|
|
|
|
|
// send_uart_pack(device, FRT_FUNCTION_CODE_READ_REGISTER, (u_int8_t*)®_value_buff, reg_num*2);
|
|
|
|
|
// pdebug_mcs_info();
|
|
|
|
|
// }
|
2024-07-05 03:52:43 +00:00
|
|
|
|
|
2024-07-06 07:26:21 +00:00
|
|
|
|
uint8_t *data = (uint8_t *)pMsg;
|
|
|
|
|
u_int16_t start_reg_addr = (data[2] << 8)| data[3];
|
|
|
|
|
u_int16_t reg_num= (data[4] << 8)| data[5];
|
2024-07-08 02:09:40 +00:00
|
|
|
|
if (start_reg_addr < 0x00 || (start_reg_addr > 0x08 && start_reg_addr < 0x14) || start_reg_addr > 0x17)
|
2024-07-06 07:26:21 +00:00
|
|
|
|
{
|
2024-07-08 02:09:40 +00:00
|
|
|
|
term_printf("start_reg_addr error:%d", start_reg_addr);
|
2024-07-06 07:26:21 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2024-07-08 02:09:40 +00:00
|
|
|
|
if (reg_num < 0x01 || (((reg_num + start_reg_addr - 1) > 0x08) && ((reg_num + start_reg_addr - 1) < 0x14)) || ((reg_num + start_reg_addr -1) > 0x17))
|
2024-07-06 07:26:21 +00:00
|
|
|
|
{
|
2024-07-08 02:09:40 +00:00
|
|
|
|
term_printf("reg_num error:%d", reg_num);
|
2024-07-06 07:26:21 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2024-07-05 03:52:43 +00:00
|
|
|
|
|
2024-07-17 02:58:39 +00:00
|
|
|
|
//采集气压数据
|
|
|
|
|
uint8_t tttt[] = {0xff, 0xff};
|
|
|
|
|
HAL_I2C_Master_Transmit(&hi2c3, 0xEE, tttt, 1, 0xFF);
|
2024-07-05 03:52:43 +00:00
|
|
|
|
/* 采集sht30数据 */
|
2024-07-12 09:30:01 +00:00
|
|
|
|
// AssertError(get_temp_humi_data(&g_stMcs_Para.temperature, &g_stMcs_Para.humidity),g_stMcs_Para.temperature=0;g_stMcs_Para.humidity=0,"采集sht30温湿度数据失败" );
|
2024-07-22 02:24:58 +00:00
|
|
|
|
//采集HP203B数据(大气压)
|
2024-07-18 10:15:50 +00:00
|
|
|
|
Hp203bReadPressure();
|
2024-07-06 07:26:21 +00:00
|
|
|
|
|
2024-07-05 03:52:43 +00:00
|
|
|
|
// /* 调试信息 */
|
|
|
|
|
// pdebug_mcs_info();
|
|
|
|
|
|
2024-07-06 07:26:21 +00:00
|
|
|
|
static u_int8_t reg_value_buff[2*100]={0x00};
|
|
|
|
|
|
|
|
|
|
memset(reg_value_buff,0,sizeof(reg_value_buff));
|
2024-07-05 03:52:43 +00:00
|
|
|
|
for(u_int16_t pos=0; pos <reg_num; pos++){
|
|
|
|
|
*(u_int16_t*)®_value_buff[pos*2] = FRT_ReadReg((start_reg_addr+pos));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(device == g_term_uart_handle)
|
|
|
|
|
{
|
2024-07-08 02:55:37 +00:00
|
|
|
|
send_uart_pack(device, FRT_FUNCTION_CODE_READ_REGISTER, (u_int8_t*)®_value_buff, reg_num*2);
|
|
|
|
|
// pdebug_mcs_info();
|
2024-07-05 03:52:43 +00:00
|
|
|
|
}
|
|
|
|
|
if(device == g_rs485_uart_handle)
|
|
|
|
|
{
|
|
|
|
|
send_uart_pack(device, FRT_FUNCTION_CODE_READ_REGISTER, (u_int8_t*)®_value_buff, reg_num*2);
|
2024-07-09 02:48:29 +00:00
|
|
|
|
// pdebug_mcs_info();
|
2024-07-05 03:52:43 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 处理写入寄存器数据
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
void FRT_MsgProc_WriteRegister(device_handle device, void *pMsg)
|
|
|
|
|
{
|
2024-07-06 08:20:58 +00:00
|
|
|
|
uint8_t *data = (uint8_t *)pMsg;
|
|
|
|
|
u_int16_t start_reg_addr = (data[2] << 8)| data[3];
|
|
|
|
|
u_int16_t reg_num= (data[4] << 8)| data[5];
|
2024-07-08 08:09:54 +00:00
|
|
|
|
u_int16_t byte_num= data[6];
|
2024-07-09 08:08:38 +00:00
|
|
|
|
unsigned short return_crc_value;
|
2024-07-08 08:09:54 +00:00
|
|
|
|
|
|
|
|
|
// 校验
|
2024-07-06 08:20:58 +00:00
|
|
|
|
if (start_reg_addr < 0x14 || start_reg_addr > 0x1C )
|
|
|
|
|
{
|
2024-07-08 08:09:54 +00:00
|
|
|
|
term_printf("start_reg_addr error:%d", start_reg_addr);
|
2024-07-06 08:20:58 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (reg_num < 0x01 || ((reg_num + start_reg_addr - 1) < 0x14) || ((reg_num + start_reg_addr -1) > 0x1C))
|
|
|
|
|
{
|
2024-07-08 08:09:54 +00:00
|
|
|
|
term_printf("reg_num error:%d", reg_num);
|
2024-07-06 08:20:58 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2024-07-08 02:55:37 +00:00
|
|
|
|
|
2024-07-08 08:09:54 +00:00
|
|
|
|
// 取出数据
|
|
|
|
|
uint16_t content[50] = {0};
|
|
|
|
|
for (uint16_t var = 0; var < reg_num; var++)
|
2024-07-06 08:20:58 +00:00
|
|
|
|
{
|
2024-07-08 08:09:54 +00:00
|
|
|
|
content[var] = data[7 + 2 * var] << 8 | data[7 + 2 * var + 1];
|
2024-07-06 08:20:58 +00:00
|
|
|
|
}
|
2024-07-08 08:09:54 +00:00
|
|
|
|
//// 调试
|
|
|
|
|
// for (int i = 0; i < 20; i++)
|
|
|
|
|
// {
|
|
|
|
|
// term_printf("%x", content[i]);
|
|
|
|
|
// }
|
|
|
|
|
|
2024-07-09 02:48:29 +00:00
|
|
|
|
// 发回数据
|
2024-07-09 08:08:38 +00:00
|
|
|
|
uint8_t Trans_data[6];
|
2024-07-09 02:48:29 +00:00
|
|
|
|
Trans_data[0] = g_stConfigInfo.addr;
|
|
|
|
|
Trans_data[1] = data[1];
|
|
|
|
|
Trans_data[2] = start_reg_addr;
|
|
|
|
|
Trans_data[3] = reg_num;
|
2024-07-09 08:08:38 +00:00
|
|
|
|
|
|
|
|
|
return_crc_value = CRC16(Trans_data, 4);
|
|
|
|
|
Trans_data[4] = return_crc_value;
|
|
|
|
|
Trans_data[5] = return_crc_value >> 8;
|
|
|
|
|
uart_dev_write(device, Trans_data, 6);
|
2024-07-09 02:48:29 +00:00
|
|
|
|
|
2024-07-08 08:09:54 +00:00
|
|
|
|
for(u_int16_t pos=0; pos <reg_num; pos++)
|
2024-07-06 08:20:58 +00:00
|
|
|
|
{
|
2024-07-08 08:09:54 +00:00
|
|
|
|
for (uint16_t i = 0; i < sizeof(g_Write_RegTbl) / sizeof(FRT_RegProcTable_s); i++)
|
|
|
|
|
{
|
|
|
|
|
if (g_Write_RegTbl[i].regId == (start_reg_addr + pos))
|
|
|
|
|
{
|
|
|
|
|
g_Write_RegTbl[i].pRegProc(&content[pos]);
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-07-05 03:52:43 +00:00
|
|
|
|
}
|
2024-07-08 08:09:54 +00:00
|
|
|
|
|
|
|
|
|
// if(device == g_term_uart_handle)
|
|
|
|
|
// {
|
|
|
|
|
// pdebug_mcs_info();
|
|
|
|
|
// }
|
|
|
|
|
// if(device == g_rs485_uart_handle)
|
|
|
|
|
// {
|
|
|
|
|
// pdebug_mcs_info();
|
|
|
|
|
// }
|
2024-07-05 03:52:43 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 读取串口数据
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static int uart_read_frt_climate_pack(device_handle uart_handle,u_int8_t *buff, u_int32_t buff_size)
|
|
|
|
|
{
|
|
|
|
|
u_int32_t offset = 0;
|
|
|
|
|
char c = 0;
|
2024-07-06 05:48:52 +00:00
|
|
|
|
// frt_climate_pack *pack = (frt_climate_pack *)buff;
|
2024-07-05 03:52:43 +00:00
|
|
|
|
|
2024-07-06 05:48:52 +00:00
|
|
|
|
unsigned char new_buff[50];
|
2024-07-29 03:33:16 +00:00
|
|
|
|
buff_size--; //预留一个'\0'位置
|
2024-07-06 05:48:52 +00:00
|
|
|
|
|
2024-07-29 03:33:16 +00:00
|
|
|
|
for (int offset = 0; offset < buff_size;)
|
2024-07-06 05:48:52 +00:00
|
|
|
|
{
|
2024-07-29 03:33:16 +00:00
|
|
|
|
// 逐字符读取
|
2024-07-05 03:52:43 +00:00
|
|
|
|
c = uart_dev_in_char(uart_handle);
|
2024-07-29 03:33:16 +00:00
|
|
|
|
buff[offset++] = c;
|
|
|
|
|
|
|
|
|
|
// 判断首字符是否是地址,是地址再开始读取,不是则将索引退一步
|
|
|
|
|
if(offset == sizeof(unsigned char))
|
|
|
|
|
{
|
|
|
|
|
if(buff[0] != g_stConfigInfo.addr)
|
|
|
|
|
{
|
|
|
|
|
memcpy(buff, buff + 1, offset - 1);
|
|
|
|
|
offset--;
|
|
|
|
|
buff_size--;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// 读寄存器
|
|
|
|
|
else if (buff[1] == FRT_FUNCTION_CODE_READ_REGISTER & offset == 8)
|
|
|
|
|
{
|
|
|
|
|
return offset;
|
|
|
|
|
}
|
|
|
|
|
// 写寄存器
|
|
|
|
|
else if (buff[1] == FRT_FUNCTION_CODE_WRITE_REGISTER & offset == 7 + buff[6] * 2 +2)
|
|
|
|
|
{
|
|
|
|
|
return offset;
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-07-05 03:52:43 +00:00
|
|
|
|
|
2024-07-17 02:58:39 +00:00
|
|
|
|
|
|
|
|
|
// for (offset = 0; offset < buff_size;){
|
|
|
|
|
// c = uart_dev_in_char(uart_handle);
|
|
|
|
|
// buff[offset++] = c;
|
|
|
|
|
// if (offset == sizeof(pack->addr)){
|
|
|
|
|
// if (pack->addr != g_stConfigInfo.addr){
|
|
|
|
|
// memcpy(buff, buff+1, offset-1);
|
|
|
|
|
// offset--;
|
|
|
|
|
// buff_size--;
|
|
|
|
|
// }
|
|
|
|
|
// }else if (offset == FRT_CLIMATE_PACK_SIZE(pack)){
|
|
|
|
|
// return offset;
|
|
|
|
|
// }
|
|
|
|
|
// }
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-29 03:33:16 +00:00
|
|
|
|
|
2024-07-05 03:52:43 +00:00
|
|
|
|
/**
|
|
|
|
|
* @brief 处理一条消息
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
void FRT_MsgHandler(device_handle device, u_int8_t *pMsg, u_int32_t MsgLen)
|
|
|
|
|
{
|
2024-07-06 06:17:39 +00:00
|
|
|
|
// frt_climate_pack *pack = (frt_climate_pack*)pMsg;
|
|
|
|
|
//
|
|
|
|
|
// //AssertErrorNoPrint((CRC16(pMsg, MsgLen-2) == FRT_CLIMATE_PACK_CRC16(pHeader)),return);
|
|
|
|
|
//// AssertError((CRC16(pMsg, MsgLen-2) == FRT_CLIMATE_PACK_CRC16(pack)),return,"crc16校验失败");
|
|
|
|
|
//// AssertError((pack->func == FRT_FUNCTION_CODE_READ_REGISTER) ||\
|
|
|
|
|
//// (pack->func == FRT_FUNCTION_CODE_WRITE_REGISTER),return,"crc16校验失败");
|
|
|
|
|
//// 断言有问题,CRC出错卡死
|
|
|
|
|
// if(CRC16(pMsg, MsgLen-2) != FRT_CLIMATE_PACK_CRC16(pack))
|
|
|
|
|
// {
|
|
|
|
|
//// term_printf("CRC不过");
|
|
|
|
|
// return;
|
|
|
|
|
// }
|
|
|
|
|
// if((pack->func != FRT_FUNCTION_CODE_READ_REGISTER) && (pack->func != FRT_FUNCTION_CODE_WRITE_REGISTER))
|
|
|
|
|
// {
|
|
|
|
|
//// term_printf("功能码不过");
|
|
|
|
|
// return;
|
|
|
|
|
// }
|
|
|
|
|
//
|
|
|
|
|
// for (u_int16_t i = 0; i < sizeof(g_MsgTbl) / sizeof(FRT_FuncionMsgProcTable_s); i++){
|
|
|
|
|
// if (pack->func == g_MsgTbl[i].msgId){
|
|
|
|
|
// g_MsgTbl[i].pMsgProc(device, pMsg);
|
|
|
|
|
// }
|
|
|
|
|
// }
|
|
|
|
|
|
2024-07-05 03:52:43 +00:00
|
|
|
|
|
2024-07-06 06:17:39 +00:00
|
|
|
|
if(CRC16(pMsg, MsgLen-2) != FRT_CLIMATE_BUFF_CRC16(pMsg))
|
2024-07-05 03:52:43 +00:00
|
|
|
|
{
|
|
|
|
|
// term_printf("CRC不过");
|
|
|
|
|
return;
|
|
|
|
|
}
|
2024-07-06 06:17:39 +00:00
|
|
|
|
if((pMsg[1] != FRT_FUNCTION_CODE_READ_REGISTER) && (pMsg[1] != FRT_FUNCTION_CODE_WRITE_REGISTER))
|
2024-07-05 03:52:43 +00:00
|
|
|
|
{
|
|
|
|
|
// term_printf("功能码不过");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
for (u_int16_t i = 0; i < sizeof(g_MsgTbl) / sizeof(FRT_FuncionMsgProcTable_s); i++){
|
2024-07-06 06:17:39 +00:00
|
|
|
|
if (pMsg[1] == g_MsgTbl[i].msgId){
|
2024-07-05 03:52:43 +00:00
|
|
|
|
g_MsgTbl[i].pMsgProc(device, pMsg);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 读取并解析串口数据
|
|
|
|
|
* @param
|
|
|
|
|
* @retval
|
|
|
|
|
*/
|
|
|
|
|
static u_int8_t rs485_buff[50]={0x00};
|
|
|
|
|
void read_and_process_uart_data(device_handle device)
|
|
|
|
|
{
|
|
|
|
|
if(uart_dev_char_present(device)){
|
|
|
|
|
osDelay(20);
|
2024-07-17 02:58:39 +00:00
|
|
|
|
memset(rs485_buff,0,sizeof(rs485_buff));
|
2024-07-05 03:52:43 +00:00
|
|
|
|
int ret = uart_read_frt_climate_pack(device, rs485_buff, sizeof(rs485_buff));
|
2024-07-29 03:33:16 +00:00
|
|
|
|
if(ret > 0){
|
|
|
|
|
FRT_MsgHandler(device, rs485_buff, ret);
|
|
|
|
|
}
|
2024-07-05 03:52:43 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 打印微气象数据 */
|
|
|
|
|
static void pdebug_mcs_info()
|
|
|
|
|
{
|
|
|
|
|
term_printf("min_wind_direction: %.2f\r\n", g_stMcs_Para.min_wind_direction);
|
|
|
|
|
term_printf("average_wind_direction: %.2f\r\n", g_stMcs_Para.average_wind_direction);
|
|
|
|
|
term_printf("max_wind_direction: %.2f\r\n", g_stMcs_Para.max_wind_direction);
|
|
|
|
|
term_printf("min_wind_speed: %.2f\r\n", g_stMcs_Para.min_wind_speed);
|
|
|
|
|
term_printf("average_wind_speed: %.2f\r\n", g_stMcs_Para.average_wind_speed);
|
|
|
|
|
term_printf("max_wind_speed: %.2f\r\n", g_stMcs_Para.max_wind_speed);
|
|
|
|
|
term_printf("temperature: %.2f\r\n", g_stMcs_Para.temperature);
|
|
|
|
|
term_printf("humidity: %.2f\r\n", g_stMcs_Para.humidity);
|
|
|
|
|
term_printf("pressure: %.2f\r\n", g_stMcs_Para.pressure);
|
|
|
|
|
term_printf("rain: %.2f\r\n", g_stMcs_Para.precipitation);
|
|
|
|
|
// term_printf("precipitation_intensity: %.2f\r\n", g_stMcs_Para.precipitation_intensity);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// 30 03 00 00 00 0B 00 2C
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|