inwudriver-weibo/app/mrs/cco/mrs_cco_xr.c

167 lines
5.9 KiB
C
Raw Normal View History

/*
* Copyright (c) CompanyNameMagicTag 2019-2020. All rights reserved.
* Description: MRS cco mr - AFN=13H-F1.
*/
#include "mrs_cco_xr.h"
#include "mrs_common_plc.h"
#include "mrs_cco_mr_list.h"
#include "mrs_common_tools.h"
#if defined(PRODUCT_CFG_PRODUCT_TYPE_CCO)
typedef td_void (*mrs_xr_fill_uart_frame)(td_pbyte data, td_u16 data_len, mr_task_node *node, td_pbyte buf,
td_u16 buf_len);
/* reply 376.2 transparent read meter frame */
td_void mrs_xr_fill_frame_13_f1(td_pbyte data, td_u16 data_len, mr_task_node *node, td_pbyte buf, td_u16 buf_len);
td_void mrs_xr_fill_frame_13_f255(td_pbyte data, td_u16 data_len, mr_task_node *node, td_pbyte buf,
td_u16 buf_len);
td_void mrs_xr_fill_frame_02_f1(td_pbyte data, td_u16 data_len, mr_task_node *node, td_pbyte buf, td_u16 buf_len);
td_void mrs_xr_fill_frame_02_f255(td_pbyte data, td_u16 data_len, mr_task_node *node, td_pbyte buf,
td_u16 buf_len);
#define MRS_AFN_COMM_DELAY (1)
td_u32 mrs_cco_create_xr_plc_frame(mr_task_node *node, td_pbyte buffer, td_u16 length)
{
mrs_plc_frame_data *new_frame = TD_NULL;
mrs_data_transmit_xr *xr = TD_NULL;
td_u16 payload_len = sizeof(mrs_plc_frame_data) + sizeof(mrs_data_transmit_xr) + length;
td_pbyte payload = (td_pbyte)mrs_malloc(payload_len);
if (payload == TD_NULL) {
return EXT_ERR_MALLOC_FAILUE;
}
(td_void) memset_s(payload, payload_len, 0, payload_len);
new_frame = (mrs_plc_frame_data *)payload;
mrs_convert_meter_to_mac(node->dst_addr, sizeof(node->dst_addr), new_frame->addr, sizeof(new_frame->addr));
new_frame->id = PLC_CMD_ID_XR;
new_frame->payload_len = sizeof(mrs_data_transmit_xr) + length;
new_frame->payload = payload + sizeof(mrs_plc_frame_data);
xr = (mrs_data_transmit_xr *)new_frame->payload;
xr->stru_ver = MRS_PLC_PROTO_VERSION;
xr->stru_size = sizeof(mrs_data_transmit_xr);
xr->protocol = node->protocol;
xr->data_len = length;
xr->seq = node->plc_seq;
xr->timeout = MRS_CFG_METER_TIMEOUT;
xr->dir = MRS_PLC_DN_FLG;
if (xr->data_len > 0) {
if (memcpy_s(xr->data, xr->data_len, buffer, length) != EOK) {
mrs_free(payload);
return EXT_ERR_MEMCPY_FAIL;
}
}
node->plc_frame = new_frame;
return EXT_ERR_SUCCESS;
}
td_u32 mrs_cco_create_xr_1376_2_frame(td_pbyte *out_buffer, td_u16 *out_len,
mr_task_node *node, td_pbyte buffer, td_u16 length)
{
mrs_proto_1376_2_encode encode;
td_u32 ret;
td_u16 data_length = length;
td_u8 head_length[] = { 4, 5, 2, 3 };
td_u8 afn_tbl[] = { 0x13, 0x13, 0x02, 0x02 };
td_u8 tbl[] = { 0x01, 0xff, 0x01, 0xff };
mrs_xr_fill_uart_frame fill_func[] = {
mrs_xr_fill_frame_13_f1, mrs_xr_fill_frame_13_f255,
mrs_xr_fill_frame_02_f1, mrs_xr_fill_frame_02_f255
};
if (node->option > MRS_XR_02H_F255) {
return EXT_ERR_INVALID_PARAMETER;
}
data_length += head_length[node->option];
(td_void) memset_s(&encode, sizeof(encode), 0, sizeof(encode));
encode.length = data_length;
encode.data = mrs_malloc(encode.length);
if (encode.data == TD_NULL) {
return EXT_ERR_MALLOC_FAILUE;
}
fill_func[node->option](encode.data, encode.length, node, buffer, length);
encode.afn = afn_tbl[node->option];
encode.fn = tbl[node->option];
encode.seq = node->seq;
encode.module_flag = 1;
(td_void) memcpy_s(encode.src_addr, sizeof(encode.src_addr), node->dst_addr, sizeof(node->dst_addr));
(td_void) memcpy_s(encode.dst_addr, sizeof(encode.dst_addr), node->src_addr, sizeof(node->src_addr));
ret = mrs_proto_1376_2_create_frame(&encode, out_buffer, out_len);
mrs_free(encode.data);
return ret;
}
td_void mrs_xr_fill_frame_13_f1(td_pbyte data, td_u16 data_len, mr_task_node *node, td_pbyte buf, td_u16 buf_len)
{
td_u16 offset = 0;
td_u8 len = (td_u8)buf_len;
if (data_len < len + 4) { /* 4: time 2B, protocol 1B, length 1B */
return;
}
data[offset++] = MRS_AFN_COMM_DELAY;
data[offset++] = 0;
data[offset++] = node->protocol;
data[offset++] = len;
if (memcpy_s(data + offset, data_len - offset, buf, len) != EOK) {}
}
td_void mrs_xr_fill_frame_13_f255(td_pbyte data, td_u16 data_len, mr_task_node *node, td_pbyte buf, td_u16 buf_len)
{
td_u16 offset = 0;
td_u16 len = buf_len;
if ((data_len < len + 5) || (len > 512)) { /* 5: time 2B, protocol 1B, length 1B, and data length less than 512 */
return;
}
data[offset++] = MRS_AFN_COMM_DELAY;
data[offset++] = 0;
data[offset++] = node->protocol;
data[offset++] = (td_u8)(len & 0xff);
data[offset++] = (td_u8)((len >> 8) & 0xff); /* high 8bits */
if (memcpy_s(data + offset, data_len - offset, buf, len) != EOK) {}
}
td_void mrs_xr_fill_frame_02_f1(td_pbyte data, td_u16 data_len, mr_task_node *node, td_pbyte buf, td_u16 buf_len)
{
td_u16 offset = 0;
td_u8 len = (td_u8)buf_len;
if (data_len < len + 2) { /* 2: protocol 1B, length 1B */
return;
}
data[offset++] = node->protocol;
data[offset++] = len;
if (memcpy_s(data + offset, data_len - offset, buf, len) != EOK) {}
}
td_void mrs_xr_fill_frame_02_f255(td_pbyte data, td_u16 data_len, mr_task_node *node, td_pbyte buf, td_u16 buf_len)
{
td_u16 offset = 0;
td_u16 len = buf_len;
if ((data_len < len + 3) || (len > 512)) { /* 3: protocol 1B, length 2B ,length less than 512 */
return;
}
data[offset++] = node->protocol;
data[offset++] = (td_u8)(len & 0xff);
data[offset++] = (td_u8)((len >> 8) & 0xff); /* high 8bits */
if (memcpy_s(data + offset, data_len - offset, buf, len) != EOK) {}
}
#endif /* defined(PRODUCT_CFG_PRODUCT_TYPE_CCO) */