837 lines
26 KiB
C
837 lines
26 KiB
C
/*
|
||
* zl_protocol.c
|
||
*
|
||
* Created on: 2024年6月20日
|
||
* Author: psx
|
||
*/
|
||
|
||
#include "sl_protocol.h"
|
||
#include <string.h>
|
||
#include "inflash.h"
|
||
#include "pdebug.h"
|
||
#include "mppt_control.h"
|
||
#include <stdlib.h>
|
||
#include "task.h"
|
||
#include "tim.h"
|
||
|
||
|
||
/* 静态函数申明 */
|
||
static void SL_MsgProcFunc_Read_Register(device_handle device, void *pMsg, uint32_t MsgLen);
|
||
static void SL_MsgProcFunc_Write_Register(device_handle device, void *pMsg, uint32_t MsgLen);
|
||
static void SL_MsgProcFunc_Broadcast_Scan(device_handle device, void *pMsg, uint32_t MsgLen);
|
||
static void SL_MsgProcFunc_Registration_request(device_handle device, void *pMsg, uint32_t MsgLen);
|
||
static void SL_MsgProcFunc_Update_Profile(device_handle device, void *pMsg, uint32_t MsgLen);
|
||
static void SL_MsgProcFunc_Remote_Upgrade(device_handle device, void *pMsg, uint32_t MsgLen);
|
||
|
||
static uint16_t SL_ReadRegisterRegistrationStatus(void *pMsg);
|
||
static uint16_t SL_ReadRegisteraddress(void *pMsg);
|
||
static uint16_t SL_ReadRegisterAccessNodeType(void *pMsg);
|
||
static uint16_t SL_ReadRegisterCommunicationMethods(void *pMsg);
|
||
static uint16_t SL_ReadRegisterBatteryVoltage(void *pMsg);
|
||
static uint16_t SL_ReadRegisterBatterytemperature(void *pMsg);
|
||
static uint16_t SL_ReadRegisterRemainingBatteryBower(void *pMsg);
|
||
static uint16_t SL_ReadRegisterSolarOpenCircuitVoltage(void *pMsg);
|
||
//static uint16_t SL_ReadRegisterSolarOpenCircuitVoltage2(void *pMsg);
|
||
|
||
//static uint16_t SL_WriteRegisterRegistrationStatus(void *pMsg);
|
||
//static uint16_t SL_WriteRegisteraddress(void *pMsg);
|
||
//static uint16_t SL_WriteRegisterAccessNodeType(void *pMsg);
|
||
//static uint16_t SL_WriteRegisterCommunicationMethods(void *pMsg);
|
||
//static uint16_t SL_WriteRegisterBatteryVoltage(void *pMsg);
|
||
//static uint16_t SL_WriteRegisterBatterytemperature(void *pMsg);
|
||
//static uint16_t SL_WriteRegisterRemainingBatteryBower(void *pMsg);
|
||
//static uint16_t SL_WriteRegisterSolarOpenCircuitVoltage1(void *pMsg);
|
||
//static uint16_t SL_WriteRegisterSolarOpenCircuitVoltage2(void *pMsg);
|
||
|
||
/* 读取串口数据时用该数组解析 */
|
||
static uint8_t rs485_buff[50]={0x00};
|
||
|
||
/* 一次最多读写的寄存器个数,由设备决定 */
|
||
#define Register_Number_Max 5
|
||
/* 寄存器个数,由设备决定 */
|
||
#define MPPT_Register_Number 5
|
||
/* 读写的寄存器的最大起始位置,由设备决定 */
|
||
#define Register_Start_Address_Max 0x0200
|
||
|
||
/* 用于解析串口包时的长度 */
|
||
#define analyzeStartFlag 2 //长度为2时解析起始标志
|
||
#define analyzeAddress 9 //长度为9时解析地址
|
||
#define analyzeFunctionCode 10 //长度为10时解析功能码
|
||
#define analyzeWritelen 14 //功能码为写入寄存器且buffer长度为14时,可以解析出写入寄存器包的长度
|
||
|
||
|
||
/* 功能码处理表 */
|
||
SL_FuncionMsgProcTable g_MsgTbl[] =
|
||
{
|
||
{SL_Function_Code_Read_Register, SL_MsgProcFunc_Read_Register},
|
||
{SL_Function_Code_Write_Register, SL_MsgProcFunc_Write_Register},
|
||
{SL_Function_Code_Broadcast_Scan, SL_MsgProcFunc_Broadcast_Scan},
|
||
{SL_Function_Code_Registration_request, SL_MsgProcFunc_Registration_request},
|
||
{SL_Function_Code_Update_Profile, SL_MsgProcFunc_Update_Profile},
|
||
{SL_Function_Code_Remote_Upgrade, SL_MsgProcFunc_Remote_Upgrade},
|
||
};
|
||
|
||
|
||
/* 寄存器处理表 */
|
||
SL_RegProcTable g_RegTblR[] =
|
||
{
|
||
{SL_Register_Registration_Status, SL_ReadRegisterRegistrationStatus},
|
||
{SL_Register_address, SL_ReadRegisteraddress},
|
||
{SL_Register_Access_Node_Type, SL_ReadRegisterAccessNodeType},
|
||
{SL_Register_Communication_Methods, SL_ReadRegisterCommunicationMethods},
|
||
{SL_Register_Battery_Voltage, SL_ReadRegisterBatteryVoltage},
|
||
{SL_Register_Battery_temperature, SL_ReadRegisterBatterytemperature},
|
||
{SL_Register_Remaining_Battery_Bower, SL_ReadRegisterRemainingBatteryBower},
|
||
{SL_Register_Solar_Open_Circuit_Voltage1, SL_ReadRegisterSolarOpenCircuitVoltage},
|
||
// {SL_Register_Solar_Open_Circuit_Voltage2, SL_ReadRegisterSolarOpenCircuitVoltage2},
|
||
};
|
||
|
||
/* 寄存器处理表 */
|
||
SL_RegProcTable g_RegTblW[] =
|
||
{
|
||
// {SL_Register_Registration_Status, SL_WriteRegisterRegistrationStatus},
|
||
// {SL_Register_address, SL_WriteRegisteraddress},
|
||
// {SL_Register_Access_Node_Type, SL_WriteRegisterAccessNodeType},
|
||
// {SL_Register_Communication_Methods, SL_WriteRegisterCommunicationMethods},
|
||
// {SL_Register_Battery_Voltage, SL_WriteRegisterBatteryVoltage},
|
||
// {SL_Register_Battery_temperature, SL_WriteRegisterBatterytemperature},
|
||
// {SL_Register_Remaining_Battery_Bower, SL_WriteRegisterRemainingBatteryBower},
|
||
// {SL_Register_Solar_Open_Circuit_Voltage1, SL_WriteRegisterSolarOpenCircuitVoltage1},
|
||
// {SL_Register_Solar_Open_Circuit_Voltage2, SL_WriteRegisterSolarOpenCircuitVoltage2},
|
||
};
|
||
|
||
|
||
|
||
|
||
|
||
/**
|
||
* @brief 校验算法
|
||
* @param
|
||
* @retval
|
||
*/
|
||
uint16_t CheckFunc(uint8_t *arr_buff, uint8_t len)
|
||
{
|
||
uint16_t crc = 0xFFFF;
|
||
uint16_t 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 随机延时函数,延时区间100-2500ms,分辨率10ms
|
||
* @param
|
||
* @retval 延时时间
|
||
*/
|
||
int randomDelay()
|
||
{
|
||
int minSeconds = 10;
|
||
int maxSeconds = 250;
|
||
// srand(SystemCoreClock);//time(NULL)替换为对应操作系统时钟
|
||
srand(TIM_GetCounter(TIM4));
|
||
int delaySeconds = minSeconds + rand() % (maxSeconds - minSeconds + 1);
|
||
return delaySeconds * 10;
|
||
}
|
||
|
||
/**
|
||
* @brief 检测485总线是否繁忙
|
||
* @param
|
||
* @retval 1 繁忙
|
||
* 0 空闲
|
||
*/
|
||
uint8_t Check_485_bus_busy(device_handle device)
|
||
{
|
||
if (device == g_bat485_uart3_handle) {
|
||
USART_ITConfig(USART3, USART_IT_RXNE, ENABLE);
|
||
} else {
|
||
USART_ITConfig(USART4, USART_IT_RXNE, ENABLE);
|
||
}
|
||
|
||
uint16_t num_ago = ring_queue_length(device);
|
||
Delay_Ms(2);
|
||
uint16_t num_now = ring_queue_length(device);
|
||
|
||
if (device == g_bat485_uart3_handle) {
|
||
USART_ITConfig(USART3, USART_IT_RXNE, DISABLE);
|
||
} else {
|
||
USART_ITConfig(USART4, USART_IT_RXNE, DISABLE);
|
||
}
|
||
|
||
if (num_now == num_ago) {
|
||
return 0;
|
||
}
|
||
return 1;
|
||
}
|
||
|
||
/**
|
||
* @brief 读取
|
||
* @param
|
||
* @retval
|
||
*/
|
||
void SL_MsgProcFunc_Read_Register(device_handle device, void *pMsg, uint32_t MsgLen)
|
||
{
|
||
SL_Mppt_Rorecv_pack *rpack = (SL_Mppt_Rorecv_pack *)pMsg;
|
||
|
||
uint16_t Register_Number_16 = chang_8_to_16(rpack->read_Register_Number_L,rpack->read_Register_Number_H);
|
||
if (Register_Number_16 > Register_Number_Max) {
|
||
log_error(" Register_Number error:%x \r\n", Register_Number_16);
|
||
return;
|
||
}
|
||
|
||
uint16_t Start_Address_16 = chang_8_to_16(rpack->read_Register_Start_Address_L,rpack->read_Register_Start_Address_H);
|
||
if (Start_Address_16 > Register_Start_Address_Max) {
|
||
log_error(" Register_Start_Address error : %x \r\n", Start_Address_16);
|
||
return;
|
||
}
|
||
|
||
/* 读取寄存器数据 */
|
||
// uint8_t reply_Data_Content[2 * 5] = {0};
|
||
uint16_t reply_Data_Content[Register_Number_Max] = {0};
|
||
for ( uint16_t pos = 0; pos < Register_Number_16; pos++) {
|
||
for (uint16_t var = 0; var < sizeof(g_RegTblR) / sizeof(SL_RegProcTable); var++) {
|
||
if (g_RegTblR[var].regId == (Start_Address_16 + pos)) {
|
||
// *(uint16_t *)&reply_Data_Content[pos * 2] = g_RegTblR[var].pRegProc(NULL);
|
||
reply_Data_Content[pos] = g_RegTblR[var].pRegProc(NULL);
|
||
}
|
||
}
|
||
}
|
||
|
||
/* 打包 */
|
||
memset(rs485_buff, 0, sizeof(rs485_buff));
|
||
uint8_t *replay_pack = rs485_buff;
|
||
// strlcpy(replay_pack, g_slConfigInfo.start_Flag, 2);
|
||
*(replay_pack) = g_slConfigInfo.start_Flag[0];
|
||
*(replay_pack + 1) = g_slConfigInfo.start_Flag[1];
|
||
|
||
replay_pack += 2;
|
||
// strlcpy(replay_pack, g_slConfigInfo.address, 7);
|
||
*(replay_pack) = g_slConfigInfo.address[0];
|
||
*(replay_pack + 1) = g_slConfigInfo.address[1];
|
||
*(replay_pack + 2) = g_slConfigInfo.address[2];
|
||
*(replay_pack + 3) = g_slConfigInfo.address[3];
|
||
*(replay_pack + 4) = g_slConfigInfo.address[4];
|
||
*(replay_pack + 5) = g_slConfigInfo.address[5];
|
||
*(replay_pack + 6) = g_slConfigInfo.address[6];
|
||
|
||
|
||
replay_pack += 7;
|
||
*replay_pack = SL_Function_Code_Read_Register;
|
||
|
||
replay_pack += 1;
|
||
// *(uint16_t *)&replay_pack = rpack->read_Register_Number;
|
||
*replay_pack = (Register_Number_16 >> 8);
|
||
*(replay_pack + 1) = (Register_Number_16);
|
||
|
||
replay_pack += 2;
|
||
for (uint8_t var = 0; var < Register_Number_16 * 2; var++) {
|
||
if (0 == (var & 0x01)) {
|
||
*(replay_pack + var) = (reply_Data_Content[var / 2] >> 8);
|
||
} else {
|
||
*(replay_pack + var) = (reply_Data_Content[var / 2]);
|
||
}
|
||
}
|
||
|
||
replay_pack += Register_Number_16 * 2;
|
||
uint16_t crc_temp = CheckFunc(rs485_buff, (2 + 7 + 1 + 2 + Register_Number_16 * 2));
|
||
// log_info("CheckFunc crc_temp: %x \r\n", crc_temp);
|
||
*replay_pack = (uint8_t)(crc_temp >> 8);
|
||
replay_pack += 1;
|
||
*replay_pack = (uint8_t)crc_temp;
|
||
|
||
replay_pack += 1;
|
||
*replay_pack = g_slConfigInfo.end_Flag;
|
||
|
||
while (1) {
|
||
Delay_Ms(randomDelay());
|
||
if (!Check_485_bus_busy(device)) {
|
||
// log_info("pack : %s", (uint8_t *)&replay_pack);
|
||
|
||
// uart_dev_write(device, (uint8_t *)&replay_pack, 16 + Register_Number_16 * 2 + 1);
|
||
uart_dev_write(device, rs485_buff, 16 + Register_Number_16 * 2);
|
||
|
||
if (device == g_bat485_uart3_handle) {
|
||
USART_ITConfig(USART3, USART_IT_RXNE, ENABLE);
|
||
} else {
|
||
USART_ITConfig(USART4, USART_IT_RXNE, ENABLE);
|
||
}
|
||
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
void SL_MsgProcFunc_Write_Register(device_handle device, void *pMsg, uint32_t MsgLen)
|
||
{
|
||
SL_Mppt_Worecv_pack *wpack = (SL_Mppt_Worecv_pack *)pMsg;
|
||
uint8_t *buff = (uint8_t *)pMsg;
|
||
|
||
uint16_t Register_Number = \
|
||
chang_8_to_16(wpack->write_Register_Number_L, wpack->write_Register_Number_H);
|
||
if (Register_Number > Register_Number_Max) {
|
||
log_error(" Register_Number error:%x \r\n", Register_Number);
|
||
return;
|
||
}
|
||
|
||
uint16_t Register_Start_Address = \
|
||
chang_8_to_16(wpack->write_Register_Start_Address_L, wpack->write_Register_Start_Address_H);
|
||
if (Register_Start_Address > Register_Start_Address_Max) {
|
||
log_error(" Register_Start_Address error : %x \r\n", Register_Start_Address);
|
||
return;
|
||
}
|
||
|
||
printf("in write register \n");
|
||
|
||
uint16_t content[Register_Number_Max] = {0};
|
||
|
||
for (uint16_t var = 0; var < Register_Number; var++) {
|
||
content[var] = buff[14 + 2 * var] << 8 | buff[14 + 2 * var + 1];
|
||
}
|
||
|
||
for ( uint16_t pos = 0; pos < Register_Number; pos++) {
|
||
for (uint16_t i = 0; i < sizeof(g_RegTblW) / sizeof(SL_RegProcTable); i++) {
|
||
if (g_RegTblW[i].regId == (Register_Start_Address + pos)) {
|
||
g_RegTblW[i].pRegProc(&content[pos]);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
void SL_MsgProcFunc_Broadcast_Scan(device_handle device, void *pMsg, uint32_t MsgLen)
|
||
{
|
||
g_recvBroadcastDevice = device;
|
||
g_recvBroadcastRegisterNumber = MPPT_Register_Number;
|
||
|
||
RegistrationRequestFlag = 1;
|
||
|
||
/* 任务创立后,立即执行一次 */
|
||
TimeSliceOffset_Register(&m_recvbroadcast, Task_recvbroadcast \
|
||
, recvbroadcast_reloadVal, recvbroadcast_offset);
|
||
m_recvbroadcast.runFlag = 1;
|
||
}
|
||
|
||
void SL_MsgProcFunc_Registration_request(device_handle device, void *pMsg, uint32_t MsgLen)
|
||
{
|
||
log_info("Registration success \r\n");
|
||
recvbroadcast_flag = 1;
|
||
RegistrationRequestFlag = 0;
|
||
TimeSliceOffset_Unregister(&m_recvbroadcast);
|
||
SL_Mppt_RegistrationReply_pack *rpack = (SL_Mppt_RegistrationReply_pack *)pMsg;
|
||
g_Mppt_Para.Registration_Status = chang_8_to_16(rpack->registration_Status_L, rpack->registration_Status_H);
|
||
|
||
/* 20s内不再接收广播帧 */
|
||
run_Broadcast = 0;
|
||
TIM2->CNT = 0;
|
||
TIM_Cmd(TIM2, ENABLE); //TIM2使能
|
||
}
|
||
|
||
|
||
void SL_MsgProcFunc_Update_Profile(device_handle device, void *pMsg, uint32_t MsgLen)
|
||
{
|
||
SL_Mppt_SOther_pack SUpdateProfile_pack = {0};
|
||
|
||
SUpdateProfile_pack.start_Flag[0] = g_slConfigInfo.start_Flag[0];
|
||
SUpdateProfile_pack.start_Flag[1] = g_slConfigInfo.start_Flag[1];
|
||
|
||
SUpdateProfile_pack.address[0] = g_slConfigInfo.address[0];
|
||
SUpdateProfile_pack.address[1] = g_slConfigInfo.address[1];
|
||
SUpdateProfile_pack.address[2] = g_slConfigInfo.address[2];
|
||
SUpdateProfile_pack.address[3] = g_slConfigInfo.address[3];
|
||
SUpdateProfile_pack.address[4] = g_slConfigInfo.address[4];
|
||
SUpdateProfile_pack.address[5] = g_slConfigInfo.address[5];
|
||
SUpdateProfile_pack.address[6] = g_slConfigInfo.address[6];
|
||
|
||
SUpdateProfile_pack.function_Code = SL_Function_Code_Update_Profile;
|
||
|
||
SUpdateProfile_pack.state = 0x01;
|
||
|
||
uint16_t crc = CheckFunc((uint8_t *)&SUpdateProfile_pack, SL_MPPT_SOTHER_PACK_SIZE - 3);
|
||
SUpdateProfile_pack.check_Bit_H = crc >> 8;
|
||
SUpdateProfile_pack.check_Bit_L = crc;
|
||
|
||
SUpdateProfile_pack.end_Flag = g_slConfigInfo.end_Flag;
|
||
|
||
while (1) {
|
||
Delay_Ms(randomDelay());
|
||
if (!Check_485_bus_busy(device)) {
|
||
uart_dev_write(device, (uint8_t *)&SUpdateProfile_pack, SL_MPPT_SOTHER_PACK_SIZE + 1);
|
||
|
||
if (device == g_bat485_uart3_handle) {
|
||
USART_ITConfig(USART3, USART_IT_RXNE, ENABLE);
|
||
} else {
|
||
USART_ITConfig(USART4, USART_IT_RXNE, ENABLE);
|
||
}
|
||
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
|
||
void SL_MsgProcFunc_Remote_Upgrade(device_handle device, void *pMsg, uint32_t MsgLen)
|
||
{
|
||
SL_Mppt_SOther_pack SUpdateProfile_pack = {0};
|
||
|
||
SUpdateProfile_pack.start_Flag[0] = g_slConfigInfo.start_Flag[0];
|
||
SUpdateProfile_pack.start_Flag[1] = g_slConfigInfo.start_Flag[1];
|
||
|
||
SUpdateProfile_pack.address[0] = g_slConfigInfo.address[0];
|
||
SUpdateProfile_pack.address[1] = g_slConfigInfo.address[1];
|
||
SUpdateProfile_pack.address[2] = g_slConfigInfo.address[2];
|
||
SUpdateProfile_pack.address[3] = g_slConfigInfo.address[3];
|
||
SUpdateProfile_pack.address[4] = g_slConfigInfo.address[4];
|
||
SUpdateProfile_pack.address[5] = g_slConfigInfo.address[5];
|
||
SUpdateProfile_pack.address[6] = g_slConfigInfo.address[6];
|
||
|
||
SUpdateProfile_pack.function_Code = SL_Function_Code_Remote_Upgrade;
|
||
|
||
SUpdateProfile_pack.state = 0x01;
|
||
|
||
uint16_t crc = CheckFunc((uint8_t *)&SUpdateProfile_pack, SL_MPPT_SOTHER_PACK_SIZE - 3);
|
||
SUpdateProfile_pack.check_Bit_H = crc >> 8;
|
||
SUpdateProfile_pack.check_Bit_L = crc;
|
||
|
||
SUpdateProfile_pack.end_Flag = g_slConfigInfo.end_Flag;
|
||
|
||
while (1) {
|
||
Delay_Ms(randomDelay());
|
||
if (!Check_485_bus_busy(device)) {
|
||
uart_dev_write(device, (uint8_t *)&SUpdateProfile_pack, SL_MPPT_SOTHER_PACK_SIZE + 1);
|
||
|
||
if (device == g_bat485_uart3_handle) {
|
||
USART_ITConfig(USART3, USART_IT_RXNE, ENABLE);
|
||
} else {
|
||
USART_ITConfig(USART4, USART_IT_RXNE, ENABLE);
|
||
}
|
||
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
/**
|
||
* @brief 读取注册状态寄存器
|
||
* @param
|
||
* @retval
|
||
*/
|
||
uint16_t SL_ReadRegisterRegistrationStatus(void *pMsg)
|
||
{
|
||
log_info(" SL_ReadRegisterRegistrationStatus ");
|
||
uint16_t value = g_Mppt_Para.Registration_Status;
|
||
return value;
|
||
}
|
||
|
||
/**
|
||
* @brief 读取地址寄存器
|
||
* @param
|
||
* @retval
|
||
*/
|
||
uint16_t SL_ReadRegisteraddress(void *pMsg)
|
||
{
|
||
log_info(" SL_ReadRegisteraddress ");
|
||
|
||
return 0;
|
||
}
|
||
|
||
/**
|
||
* @brief 读取接入节点类型寄存器
|
||
* @param
|
||
* @retval
|
||
*/
|
||
uint16_t SL_ReadRegisterAccessNodeType(void *pMsg)
|
||
{
|
||
log_info(" SL_ReadRegisterAccessNodeType ");
|
||
uint16_t value = g_Mppt_Para.Access_Node_Type;
|
||
return value;
|
||
}
|
||
|
||
/**
|
||
* @brief 读取通信方式寄存器
|
||
* @param
|
||
* @retval
|
||
*/
|
||
uint16_t SL_ReadRegisterCommunicationMethods(void *pMsg)
|
||
{
|
||
log_info(" SL_ReadRegisterCommunicationMethods ");
|
||
uint16_t value = g_Mppt_Para.Communication_Methods;
|
||
return value;
|
||
}
|
||
|
||
/**
|
||
* @brief 读取电池电压寄存器
|
||
* @param
|
||
* @retval
|
||
*/
|
||
uint16_t SL_ReadRegisterBatteryVoltage(void *pMsg)
|
||
{
|
||
log_info(" SL_ReadRegisterBatteryVoltage ");
|
||
uint16_t value = (uint16_t)g_Mppt_Para.Battery_Voltage * 10;
|
||
return value;
|
||
}
|
||
|
||
/**
|
||
* @brief 读取电池温度寄存器
|
||
* @param
|
||
* @retval
|
||
*/
|
||
uint16_t SL_ReadRegisterBatterytemperature(void *pMsg)
|
||
{
|
||
log_info(" SL_ReadRegisterBatterytemperature ");
|
||
uint16_t value = (uint16_t)g_Mppt_Para.Battery_temperature * 10;
|
||
return value;
|
||
}
|
||
|
||
/**
|
||
* @brief 读取电池剩余电量寄存器
|
||
* @param
|
||
* @retval
|
||
*/
|
||
uint16_t SL_ReadRegisterRemainingBatteryBower(void *pMsg)
|
||
{
|
||
log_info(" SL_ReadRegisterRemainingBatteryBower ");
|
||
uint16_t value = (uint16_t)g_Mppt_Para.Remaining_Battery_Bower * 10;
|
||
return value;
|
||
}
|
||
|
||
/**
|
||
* @brief 读取太阳能开路电压1寄存器
|
||
* @param
|
||
* @retval
|
||
*/
|
||
uint16_t SL_ReadRegisterSolarOpenCircuitVoltage(void *pMsg)
|
||
{
|
||
log_info(" SL_ReadRegisterSolarOpenCircuitVoltage1 ");
|
||
uint16_t value = (uint16_t)g_Mppt_Para.Solar_Open_Circuit_Voltage * 10;
|
||
return value;
|
||
}
|
||
//
|
||
///**
|
||
// * @brief 读取太阳能开路电压2寄存器
|
||
// * @param
|
||
// * @retval
|
||
// */
|
||
//uint16_t SL_ReadRegisterSolarOpenCircuitVoltage2(void *pMsg)
|
||
//{
|
||
// log_info(" SL_ReadRegisterSolarOpenCircuitVoltage2 ");
|
||
// uint16_t value = (uint16_t)g_Mppt_Para.Solar_Open_Circuit_Voltage2 * 10;
|
||
// return value;
|
||
//}
|
||
|
||
///**
|
||
// * @brief 写入注册状态寄存器
|
||
// * @param
|
||
// * @retval
|
||
// */
|
||
//uint16_t SL_WriteRegisterRegistrationStatus(void *pMsg)
|
||
//{
|
||
// log_info(" WriteRegisterRegistrationStatus %x", *(uint16_t *)pMsg);
|
||
//
|
||
// return 0;
|
||
//}
|
||
//
|
||
///**
|
||
// * @brief 写入地址寄存器
|
||
// * @param
|
||
// * @retval
|
||
// */
|
||
//uint16_t SL_WriteRegisteraddress(void *pMsg)
|
||
//{
|
||
// log_info(" WriteRegisteraddress %x", *(uint16_t *)pMsg);
|
||
//
|
||
// return 0;
|
||
//}
|
||
//
|
||
///**
|
||
// * @brief 写入接入节点类型寄存器
|
||
// * @param
|
||
// * @retval
|
||
// */
|
||
//uint16_t SL_WriteRegisterAccessNodeType(void *pMsg)
|
||
//{
|
||
// log_info(" WriteRegisterAccessNodeType %x", *(uint16_t *)pMsg);
|
||
//
|
||
// return 0;
|
||
//}
|
||
//
|
||
///**
|
||
// * @brief 写入通信方式寄存器
|
||
// * @param
|
||
// * @retval
|
||
// */
|
||
//uint16_t SL_WriteRegisterCommunicationMethods(void *pMsg)
|
||
//{
|
||
// log_info(" WriteRegisterCommunicationMethods %x", *(uint16_t *)pMsg);
|
||
//
|
||
// return 0;
|
||
//}
|
||
|
||
/**
|
||
* @brief 匹配起始标志"SL"
|
||
* @param start_buff 起始标志
|
||
* @retval 1 匹配成功
|
||
* 0 匹配失败
|
||
*/
|
||
static int Match_Startflag(uint8_t start_buff[2])
|
||
{
|
||
// if (!strcmp(start_buff, g_slConfigInfo.start_Flag)) {
|
||
// log_info("Match_Startflag fail \r\n");
|
||
// return 1;
|
||
// }
|
||
if ((start_buff[0] == g_slConfigInfo.start_Flag[0]) && \
|
||
(start_buff[1] == g_slConfigInfo.start_Flag[1])) {
|
||
log_info("Match_Startflag success \r\n");
|
||
return 1;
|
||
}
|
||
return 0;
|
||
}
|
||
|
||
/**
|
||
* @brief 匹配设备地址
|
||
* @param address 地址
|
||
* @retval 1 匹配成功
|
||
* 0 匹配失败
|
||
*/
|
||
static int Match_address(u_int8_t address[7])
|
||
{
|
||
// if (!strcmp(address, g_slConfigInfo.address)) {
|
||
// log_info("Match_address fail \r\n");
|
||
// return 1;
|
||
// }
|
||
if ((address[0] == g_slConfigInfo.address[0]) && \
|
||
(address[1] == g_slConfigInfo.address[1]) && \
|
||
(address[2] == g_slConfigInfo.address[2]) && \
|
||
(address[3] == g_slConfigInfo.address[3]) && \
|
||
(address[4] == g_slConfigInfo.address[4]) && \
|
||
(address[5] == g_slConfigInfo.address[5]) && \
|
||
(address[6] == g_slConfigInfo.address[6])) {
|
||
log_info("Match_address success \r\n");
|
||
return 1;
|
||
}
|
||
return 0;
|
||
}
|
||
|
||
/**
|
||
* @brief 匹配广播地址
|
||
* @param address 地址
|
||
* @retval 1 匹配成功
|
||
* 0 匹配失败
|
||
*/
|
||
static int Match_Broadcastaddress(u_int8_t address[7])
|
||
{
|
||
if (address[0] == 0xFF && \
|
||
address[1] == 0xFF && \
|
||
address[2] == 0xFF && \
|
||
address[3] == 0xFF && \
|
||
address[4] == 0xFF && \
|
||
address[5] == 0xFF && \
|
||
address[6] == 0xFF) {
|
||
log_info("Match_Broadcastaddress success\r\n");
|
||
return 1;
|
||
}
|
||
return 0;
|
||
}
|
||
|
||
/**
|
||
* @brief 读取串口数据
|
||
* @param uart_handle 串口句柄
|
||
* @param buff 缓冲区
|
||
* @param buff_size 缓冲区长度
|
||
* @retval
|
||
*/
|
||
static int uart_read_climate_pack(device_handle uart_handle,uint8_t *buff, uint32_t buff_size)
|
||
{
|
||
uint32_t offset = 0;
|
||
uint32_t len = 0;
|
||
uint8_t flag_run = 0;
|
||
char c = 0;
|
||
|
||
SL_Mppt_Recv_pack *pack = (SL_Mppt_Recv_pack *)buff;
|
||
|
||
buff_size--; //预留一个'\0'位置
|
||
for (; offset < buff_size;){
|
||
if (ring_queue_length(uart_handle) == 0) {
|
||
break;
|
||
}
|
||
|
||
c = uart_dev_in_char(uart_handle);
|
||
|
||
buff[offset++] = c;
|
||
|
||
/* 匹配起始标志位 */
|
||
if (offset == analyzeStartFlag || (flag_run > 0)) {
|
||
if (!Match_Startflag(pack->start_Flag)) {
|
||
memcpy(buff, buff+1, offset-1);
|
||
offset--;
|
||
continue;
|
||
}
|
||
}
|
||
|
||
/* 匹配地址 */
|
||
if (offset == analyzeAddress || (flag_run > 1)) {
|
||
if (!((((g_Mppt_Para.Registration_Status == 2) || RegistrationRequestFlag) && Match_address(pack->address))
|
||
|| (run_Broadcast && Match_Broadcastaddress(pack->address)))) {
|
||
if (flag_run < 1) {
|
||
flag_run = 1;
|
||
}
|
||
memcpy(buff, buff+1, offset-1);
|
||
offset--;
|
||
continue;
|
||
}
|
||
}
|
||
|
||
/* 匹配功能码 */
|
||
if (offset == analyzeFunctionCode || (flag_run > 2)) {
|
||
/* 未注册时,不处理读写和其他帧 */
|
||
if (g_Mppt_Para.Registration_Status == 2) {
|
||
/* 读寄存器数据 */
|
||
if (pack->function_Code == SL_Function_Code_Read_Register) {
|
||
log_info("Read_Register\r\n");
|
||
len = SL_MPPT_RORECV_PACK_SIZE;
|
||
}
|
||
|
||
/* 写寄存器数据 */
|
||
else if (pack->function_Code == SL_Function_Code_Write_Register) {
|
||
log_info("Write_Register\r\n");
|
||
}
|
||
|
||
/* 其他帧格式 */
|
||
else if (pack->function_Code == SL_Function_Code_Update_Profile
|
||
|| pack->function_Code == SL_Function_Code_Remote_Upgrade) {
|
||
len = SL_MPPT_ROTHER_PACK_SIZE;
|
||
log_info("Other frames\r\n");
|
||
}
|
||
|
||
else if (run_Broadcast) {
|
||
/* 广播扫描 */
|
||
if (pack->function_Code == SL_Function_Code_Broadcast_Scan) {
|
||
log_info("Broadcast_Scan\r\n");
|
||
len = SL_MPPT_SCAN_BROADCAST_PACK_SIZE;
|
||
}
|
||
|
||
/* 注册请求 */
|
||
else if (pack->function_Code == SL_Function_Code_Registration_request) {
|
||
log_info("Registration_request\r\n");
|
||
len = SL_MPPT_REGISTRATIONREPLY_PACK_SIZE;
|
||
}
|
||
|
||
else {
|
||
if (flag_run < 2) {
|
||
flag_run = 2;
|
||
}
|
||
log_info("funcode error %x\r\n", pack->function_Code);
|
||
memcpy(buff, buff+1, offset-1);
|
||
offset--;
|
||
continue;
|
||
}
|
||
}
|
||
|
||
else {
|
||
if (flag_run < 2) {
|
||
flag_run = 2;
|
||
}
|
||
log_info("funcode error %x\r\n", pack->function_Code);
|
||
memcpy(buff, buff+1, offset-1);
|
||
offset--;
|
||
continue;
|
||
}
|
||
}
|
||
|
||
/* 广播扫描 */
|
||
else if (pack->function_Code == SL_Function_Code_Broadcast_Scan) {
|
||
log_info("Broadcast_Scan\r\n");
|
||
len = SL_MPPT_SCAN_BROADCAST_PACK_SIZE;
|
||
}
|
||
|
||
/* 注册请求 */
|
||
else if (pack->function_Code == SL_Function_Code_Registration_request) {
|
||
log_info("Registration_request\r\n");
|
||
len = SL_MPPT_REGISTRATIONREPLY_PACK_SIZE;
|
||
}
|
||
|
||
else {
|
||
if (flag_run < 2) {
|
||
flag_run = 2;
|
||
}
|
||
log_info("funcode error %x\r\n", pack->function_Code);
|
||
memcpy(buff, buff+1, offset-1);
|
||
offset--;
|
||
continue;
|
||
}
|
||
|
||
}
|
||
|
||
if ((pack->function_Code == SL_Function_Code_Write_Register) && (offset >= 14)) {
|
||
SL_Mppt_Worecv_pack *wpack = (SL_Mppt_Worecv_pack *)buff;
|
||
uint8_t Register_Number = (wpack->write_Register_Number_H << 8) | wpack->write_Register_Number_L;
|
||
len = Register_Number * 2 + SL_MPPT_WORECV_PACK_SIZE - 4;
|
||
continue;
|
||
}
|
||
|
||
if (offset == len) {
|
||
uint16_t crc_16 = chang_8_to_16(buff[offset - 2], buff[offset - 3]);
|
||
|
||
if ((CheckFunc(buff, offset - 3) != crc_16) || (buff[offset - 1] != 0x16)) {
|
||
if (flag_run < 3) {
|
||
flag_run = 3;
|
||
}
|
||
memcpy(buff, buff+1, offset-1);
|
||
offset--;
|
||
|
||
} else {
|
||
return offset;
|
||
}
|
||
}
|
||
}
|
||
return 0;
|
||
}
|
||
|
||
/**
|
||
* @brief 处理一条消息
|
||
* @param
|
||
* @retval
|
||
*/
|
||
void FRT_MsgHandler(device_handle device, uint8_t *pMsg, uint32_t MsgLen)
|
||
{
|
||
SL_Mppt_Recv_pack *pack = (SL_Mppt_Recv_pack *)pMsg;
|
||
|
||
for (u_int16_t i = 0; i < sizeof(g_MsgTbl) / sizeof(SL_FuncionMsgProcTable); i++){
|
||
if (pack->function_Code == g_MsgTbl[i].msgId){
|
||
g_MsgTbl[i].pMsgProc(device, pMsg, MsgLen);
|
||
}
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @brief 读取并解析串口数据
|
||
* @param
|
||
* @retval
|
||
*/
|
||
void read_and_process_uart_data(device_handle device)
|
||
{
|
||
// printf("ring_queue_length = %d \n", ring_queue_length(device));
|
||
// if (ring_queue_length(device) > 10) {uart_dev_char_present(device_handle device)
|
||
if (uart_dev_char_present(device)) {
|
||
Delay_Ms(20);
|
||
// printf("ring_queue_length = %d \n", ring_queue_length(device));
|
||
memset(rs485_buff, 0, sizeof(rs485_buff));
|
||
int ret = uart_read_climate_pack(device, rs485_buff, sizeof(rs485_buff));
|
||
if(ret > 0){
|
||
FRT_MsgHandler(device, rs485_buff, ret);
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
|
||
|