917 lines
27 KiB
C
917 lines
27 KiB
C
|
/****************************************************************************
|
||
|
|
||
|
Copyright(c) 2019 by Aerospace C.Power (Chongqing) Microelectronics. ALL RIGHTS RESERVED.
|
||
|
|
||
|
This Information is proprietary to Aerospace C.Power (Chongqing) Microelectronics and MAY NOT
|
||
|
be copied by any method or incorporated into another program without
|
||
|
the express written consent of Aerospace C.Power. This Information or any portion
|
||
|
thereof remains the property of Aerospace C.Power. The Information contained herein
|
||
|
is believed to be accurate and Aerospace C.Power assumes no responsibility or
|
||
|
liability for its use in any way and conveys no license or title under
|
||
|
any patent or copyright and makes no representation or warranty that this
|
||
|
Information is free from patent or copyright infringement.
|
||
|
|
||
|
****************************************************************************/
|
||
|
|
||
|
#include "iot_errno_api.h"
|
||
|
#include "os_utils_api.h"
|
||
|
#include "iot_uart_api.h"
|
||
|
#include "iot_version_api.h"
|
||
|
#include "iot_system_api.h"
|
||
|
#include "proto_spg_cco.h"
|
||
|
#include "iot_sg_ctrl_api.h"
|
||
|
#include "proto_645.h"
|
||
|
|
||
|
#if (IOT_SG_CONTROLLER_ENABLE)
|
||
|
|
||
|
/* report max scan node count */
|
||
|
#define IOT_CTRL_SCAN_NODE_MAX_CNT (200)
|
||
|
|
||
|
/* command from contrl application */
|
||
|
typedef struct _ctrl_spg_frame_pkt_t {
|
||
|
/* main application function code */
|
||
|
uint8_t afn;
|
||
|
/* seccondary application function code */
|
||
|
uint8_t fn;
|
||
|
/* sequence number of ctrl command */
|
||
|
uint8_t app_sn;
|
||
|
/* length of the applied data (unit: byte)*/
|
||
|
uint16_t app_data_len;
|
||
|
/* applied data */
|
||
|
uint8_t *app_data;
|
||
|
}ctrl_spg_frame_pkt_t;
|
||
|
|
||
|
typedef struct _spg_cctt_drv_data {
|
||
|
/* seq number */
|
||
|
uint8_t sn;
|
||
|
/* requested meter mac */
|
||
|
uint8_t req_mac[IOT_MAC_ADDR_LEN];
|
||
|
/* destination mac in address field */
|
||
|
uint8_t dest_mac[IOT_MAC_ADDR_LEN];
|
||
|
} spg_cctt_drv_data;
|
||
|
|
||
|
static spg_cctt_drv_data *drv_data = NULL;
|
||
|
|
||
|
static uint32_t iot_spg_ctrl_send_uart_sn_ext(uint8_t role, uint8_t afn,
|
||
|
uint8_t fn, iot_pkt_t *pkt, uint8_t sn, uint8_t mac_used, uint8_t *req_mac,
|
||
|
uint8_t *dec_mac, uint8_t di2)
|
||
|
{
|
||
|
uint8_t *data = NULL;
|
||
|
proto_spg_hdr_t *pkt_hdr = NULL;
|
||
|
proto_spg_addr_field_t *spg_addr_ptr = NULL;
|
||
|
proto_spg_app_data_t *spg_app_ptr = NULL;
|
||
|
uint8_t *cs_ptr = NULL; //pointer to checksum
|
||
|
uint16_t index = 0;
|
||
|
uint16_t app_data_len = (uint16_t)iot_pkt_data_len(pkt);
|
||
|
|
||
|
data = iot_pkt_push(pkt, proto_spg_get_header_len(mac_used));
|
||
|
IOT_ASSERT(data);
|
||
|
|
||
|
if ((iot_pkt_block_len(pkt, IOT_PKT_BLOCK_DATA) + PROTO_SPG_CS_BCODE_LEN) !=
|
||
|
proto_spg_get_proto_pkt_len(mac_used, app_data_len)) {
|
||
|
iot_pkt_free(pkt);
|
||
|
return ERR_NOMEM;
|
||
|
}
|
||
|
pkt_hdr = (proto_spg_hdr_t*)data;
|
||
|
pkt_hdr->sof_byte = PROTO_SPG_SOF_BYTE;
|
||
|
|
||
|
pkt_hdr->ctrl_byte.dir = PROTO_SPG_DIR_UP_LINK;
|
||
|
if (role == PROTO_SPG_PRM_SLAVE) {
|
||
|
/* response Message */
|
||
|
pkt_hdr->ctrl_byte.prm = PROTO_SPG_PRM_SLAVE;
|
||
|
} else {
|
||
|
/* report message */
|
||
|
pkt_hdr->ctrl_byte.prm = PROTO_SPG_PRM_MASTER;
|
||
|
}
|
||
|
pkt_hdr->ctrl_byte.addr = PROTO_SPG_NO_ADDRESS;
|
||
|
pkt_hdr->ctrl_byte.ver = PROTO_SPG_DEFAULT_VERSION;
|
||
|
pkt_hdr->ctrl_byte.rsvd = 0;
|
||
|
|
||
|
index += sizeof(proto_spg_hdr_t);
|
||
|
|
||
|
if (mac_used && req_mac && dec_mac) {
|
||
|
spg_addr_ptr = (proto_spg_addr_field_t *)(data+index);
|
||
|
pkt_hdr->ctrl_byte.addr = PROTO_SPG_WITH_ADDRESS;
|
||
|
|
||
|
/* only src mac and dest mac for up link */
|
||
|
os_mem_cpy(spg_addr_ptr->src_mac, req_mac, PROTO_SPG_MAC_ADDR_LEN);
|
||
|
index += PROTO_SPG_MAC_ADDR_LEN;
|
||
|
os_mem_cpy(spg_addr_ptr->dest_mac, dec_mac, PROTO_SPG_MAC_ADDR_LEN);
|
||
|
index += PROTO_SPG_MAC_ADDR_LEN;
|
||
|
}
|
||
|
|
||
|
spg_app_ptr = (proto_spg_app_data_t *)(data+index);
|
||
|
spg_app_ptr->afn = afn;
|
||
|
spg_app_ptr->seq = sn;
|
||
|
spg_app_ptr->di[0] = fn;
|
||
|
spg_app_ptr->di[1] = afn;
|
||
|
spg_app_ptr->di[2] = di2;
|
||
|
spg_app_ptr->di[3] = (uint8_t)PROTO_SPG_APP_CTRL_DI3;
|
||
|
index += sizeof(proto_spg_app_data_t);
|
||
|
|
||
|
/* skip app data as app data is already in iot_pkt */
|
||
|
index += app_data_len;
|
||
|
|
||
|
/* check sum field */
|
||
|
cs_ptr = (data +index);
|
||
|
index++;
|
||
|
|
||
|
/* tail of the frame */
|
||
|
data[index] = PROTO_SPG_EOF_BYTE;
|
||
|
index++;
|
||
|
pkt_hdr->len = index;
|
||
|
|
||
|
*cs_ptr = proto_spg_get_checksum(data + PROTO_SPG_CTRL_OFFSET,
|
||
|
index - PROTO_SPG_CTRL_OFFSET - PROTO_SPG_CHECKSUM_LEN -
|
||
|
PROTO_SPG_BACKCODE_LEN);
|
||
|
|
||
|
// put checksum and back code. 2 byte.
|
||
|
iot_pkt_put(pkt, PROTO_SPG_CS_BCODE_LEN);
|
||
|
iot_ctrl_uart_send(pkt);
|
||
|
return ERR_OK;
|
||
|
}
|
||
|
|
||
|
static uint32_t iot_spg_ctrl_send_uart_sn(uint8_t role, uint8_t afn,
|
||
|
uint8_t fn, iot_pkt_t *pkt, uint8_t di2, uint8_t mac_used, uint8_t sn)
|
||
|
{
|
||
|
if (mac_used) {
|
||
|
return iot_spg_ctrl_send_uart_sn_ext(role, afn, fn, pkt, sn,
|
||
|
SPG_USED_ADDR, drv_data->dest_mac, drv_data->req_mac, di2);
|
||
|
} else {
|
||
|
return iot_spg_ctrl_send_uart_sn_ext(role, afn, fn, pkt, sn,
|
||
|
SPG_NO_USED_ADDR, NULL, NULL, di2);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static uint32_t iot_spg_ctrl_send_uart(uint8_t role, uint8_t afn, uint8_t fn,
|
||
|
const uint8_t *app_data, uint16_t len, uint8_t di2, uint8_t mac_used,
|
||
|
uint8_t sn)
|
||
|
{
|
||
|
iot_pkt_t *pkt = NULL;
|
||
|
uint8_t *pkt_data = NULL;
|
||
|
|
||
|
pkt = iot_pkt_alloc(proto_spg_get_pkt_len(mac_used, len),
|
||
|
IOT_SMART_GRID_MID);
|
||
|
IOT_ASSERT(pkt);
|
||
|
|
||
|
pkt_data = iot_pkt_reserve(pkt, proto_spg_get_rsvd_len(mac_used));
|
||
|
|
||
|
if (app_data && len) {
|
||
|
os_mem_cpy(pkt_data, app_data, len);
|
||
|
iot_pkt_put(pkt, len);
|
||
|
}
|
||
|
|
||
|
return iot_spg_ctrl_send_uart_sn(role, afn, fn, pkt, di2, mac_used, sn);
|
||
|
}
|
||
|
|
||
|
|
||
|
static uint32_t iot_spg_ctrl_drv_init()
|
||
|
{
|
||
|
uint32_t ret = 0;
|
||
|
uint32_t result = ERR_OK;
|
||
|
iot_frame_fmt fmt = { 0 };
|
||
|
|
||
|
if (!drv_data) {
|
||
|
drv_data = os_mem_malloc(IOT_SMART_GRID_MID,
|
||
|
sizeof(*drv_data));
|
||
|
if (!drv_data) {
|
||
|
return ERR_NOMEM;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* config the UART port settings */
|
||
|
fmt.preamble_code[0] = PROTO_SPG_SOF_BYTE;
|
||
|
fmt.preamble_codelen = PROTO_SPG_PRECODE_LEN;
|
||
|
fmt.datalen_offset = 0;
|
||
|
fmt.datalen_size = PROTO_SPG_LENGTH_SIZE;
|
||
|
fmt.backcode_offset = 0;
|
||
|
fmt.backcode_len = PROTO_SPG_BACKCODE_LEN;
|
||
|
/* in uart data parsing, datalen means raw data to be send,
|
||
|
* excluding preamble/datalen/backcode. For datalen field in
|
||
|
* spg datagram means the length of the whole datagram, we shall
|
||
|
* set the datalen_fix as below.
|
||
|
*/
|
||
|
fmt.datalen_fix = fmt.preamble_codelen +
|
||
|
fmt.datalen_size + fmt.backcode_len;
|
||
|
fmt.backcode[0] = PROTO_SPG_EOF_BYTE;
|
||
|
fmt.frame_timeout = CTRL_FRAME_PARSE_TIMEOUT;
|
||
|
fmt.timeout_mode = TIMEOUT_PERDATA;
|
||
|
fmt.check_frame_func = proto_spg_check_frame_handler;
|
||
|
ret = iot_sg_ctrl_config_uart(PROTO_SPG_DEFAULT_BAUD,
|
||
|
(uint8_t)IOT_UART_PARITY_EVEN, (uint8_t)IOT_UART_DLEN_8_BITS,
|
||
|
(uint8_t)IOT_UART_STOP_1_BITS, &fmt);
|
||
|
if (!ret) {
|
||
|
result = ERR_FAIL;
|
||
|
}
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
static void iot_spg_ctrl_drv_deinit()
|
||
|
{
|
||
|
if (drv_data) {
|
||
|
os_mem_free(drv_data);
|
||
|
drv_data = NULL;
|
||
|
}
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
static void iot_spg_ctrl_freq_band_handler(
|
||
|
iot_plc_freq_band_info_query_rpt_t *rpt)
|
||
|
{
|
||
|
spg_query_freq_band_t rsp = { 0 };
|
||
|
rsp.freq_band = rpt->freq_band;
|
||
|
/* report this response to UART/APP */
|
||
|
iot_spg_ctrl_send_uart(PROTO_SPG_PRM_SLAVE, PROTO_SPG_AFN_03,
|
||
|
PROTO_SPG_F90, (uint8_t *)&rsp, (uint16_t)sizeof(rsp),
|
||
|
PROTO_SPG_APP_DI2_00, SPG_NO_USED_ADDR, drv_data->sn);
|
||
|
}
|
||
|
|
||
|
static void iot_spg_ctrl_send_00h02_nack(uint8_t reason, uint8_t sn)
|
||
|
{
|
||
|
spg_afn00_fn02_reject_t reject = { 0 };
|
||
|
reject.err_no = reason;
|
||
|
iot_spg_ctrl_send_uart(PROTO_SPG_PRM_SLAVE, PROTO_SPG_AFN_00, PROTO_SPG_F2,
|
||
|
(uint8_t*)&reject, sizeof(reject), PROTO_SPG_APP_DI2_01,
|
||
|
SPG_NO_USED_ADDR, sn);
|
||
|
}
|
||
|
|
||
|
static void iot_spg_ctrl_send_00h01_ack(uint8_t sn)
|
||
|
{
|
||
|
spg_afn00_fn01_conf_t conf = { 0 };
|
||
|
|
||
|
conf.wait_duration = PROTO_SPG_DEFAULT_WAIT_TIME;
|
||
|
iot_spg_ctrl_send_uart(PROTO_SPG_PRM_SLAVE, PROTO_SPG_AFN_00, PROTO_SPG_F1,
|
||
|
(uint8_t*)&conf, sizeof(conf), PROTO_SPG_APP_DI2_01,
|
||
|
SPG_NO_USED_ADDR, sn);
|
||
|
}
|
||
|
|
||
|
static void iot_spg_ctrl_set_freq_band_result(uint8_t result)
|
||
|
{
|
||
|
if (!result) {
|
||
|
iot_spg_ctrl_send_00h01_ack(drv_data->sn);
|
||
|
} else {
|
||
|
iot_spg_ctrl_send_00h02_nack(PROTO_SPG_ERR_COMM_TIMEOUT, drv_data->sn);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void iot_spg_ctrl_01h_f01(ctrl_spg_frame_pkt_t *frm)
|
||
|
{
|
||
|
iot_spg_ctrl_send_00h01_ack(frm->app_sn);
|
||
|
os_delay(1000);
|
||
|
iot_system_restart(IOT_SYS_RST_REASON_CCTT_REQ);
|
||
|
}
|
||
|
|
||
|
static void iot_spg_ctrl_rpt_data_transfer_handle(uint8_t sn,
|
||
|
uint8_t *data, uint16_t len, uint8_t data_type)
|
||
|
{
|
||
|
spg_afn22_fn01_data_send_ul_t *rsp;
|
||
|
iot_pkt_t *rpt_pkt;
|
||
|
uint16_t rpt_pkt_len;
|
||
|
|
||
|
rpt_pkt_len = len + sizeof(*rsp);
|
||
|
rpt_pkt = iot_pkt_alloc(proto_spg_get_pkt_len(SPG_NO_USED_ADDR,
|
||
|
rpt_pkt_len), IOT_SMART_GRID_MID);
|
||
|
IOT_ASSERT(rpt_pkt);
|
||
|
|
||
|
rsp = (spg_afn22_fn01_data_send_ul_t *)iot_pkt_reserve(rpt_pkt,
|
||
|
proto_spg_get_rsvd_len(SPG_NO_USED_ADDR));
|
||
|
|
||
|
rsp->data_type = data_type;
|
||
|
rsp->data_len = len;
|
||
|
os_mem_cpy(rsp->data, data, len);
|
||
|
proto_645_add33_handle(rsp->data, len);
|
||
|
iot_pkt_put(rpt_pkt, rpt_pkt_len);
|
||
|
|
||
|
iot_spg_ctrl_send_uart_sn(PROTO_SPG_PRM_SLAVE, PROTO_SPG_AFN_22,
|
||
|
PROTO_SPG_F1, rpt_pkt, PROTO_SPG_APP_DI2_05, SPG_NO_USED_ADDR, sn);
|
||
|
}
|
||
|
|
||
|
static uint32_t iot_spg_ctrl_event_report_handle(uint8_t sn,
|
||
|
uint8_t *addr, uint8_t dev_type, uint32_t proto_type, uint8_t evt_type,
|
||
|
uint8_t *data, uint16_t len)
|
||
|
{
|
||
|
(void)addr;
|
||
|
(void)dev_type;
|
||
|
(void)proto_type;
|
||
|
|
||
|
switch (evt_type) {
|
||
|
case IOT_CTRL_EVENT_DEVICE_CONNECT_CONF:
|
||
|
iot_sg_ctrl_printf("%s connect conf\n", __FUNCTION__);
|
||
|
break;
|
||
|
case IOT_CTRL_EVENT_DEVICE_CONNECT_FAIL:
|
||
|
iot_spg_ctrl_send_00h02_nack(PROTO_SPG_ERR_COMM_TIMEOUT, sn);
|
||
|
break;
|
||
|
case IOT_CTRL_EVENT_DEVICE_CONNECTED:
|
||
|
iot_spg_ctrl_send_00h01_ack(sn);
|
||
|
break;
|
||
|
case IOT_CTRL_EVENT_DEVICE_DISCONNECTED:
|
||
|
iot_sg_ctrl_printf("%s disconnected\n", __FUNCTION__);
|
||
|
break;
|
||
|
case IOT_CTRL_EVENT_CCO_DATA_RPT:
|
||
|
iot_spg_ctrl_rpt_data_transfer_handle(sn, data, len,
|
||
|
PROTO_SPG_CTRL_DATA_TYPE_CCO_CTRL);
|
||
|
break;
|
||
|
case IOT_CTRL_EVENT_PASSTHROUGH_RPT:
|
||
|
iot_spg_ctrl_rpt_data_transfer_handle(sn, data, len,
|
||
|
PROTO_SPG_CTRL_DATA_TYPE_PASSTHROUGH);
|
||
|
break;
|
||
|
default:
|
||
|
iot_sg_ctrl_printf("%s event:%d invalid\n", __FUNCTION__, evt_type);
|
||
|
break;
|
||
|
}
|
||
|
return ERR_OK;
|
||
|
}
|
||
|
|
||
|
static void iot_spg_ctrl_01h_init(ctrl_spg_frame_pkt_t *frm,
|
||
|
iot_pkt_t *pkt)
|
||
|
{
|
||
|
uint32_t reason= PROTO_SPG_ERR_MASTER_UNSUPPORTED_CMD;
|
||
|
|
||
|
if (PROTO_SPG_F1 == frm->fn) {
|
||
|
iot_spg_ctrl_01h_f01(frm);
|
||
|
} else {
|
||
|
iot_sg_ctrl_printf("%s:data_handle error,reason is %lu,afn : %x,"
|
||
|
"fn : %x\n", __FUNCTION__, reason, frm->afn, frm->fn);
|
||
|
iot_spg_ctrl_send_00h02_nack((uint8_t)reason, frm->app_sn);
|
||
|
}
|
||
|
iot_pkt_free(pkt);
|
||
|
}
|
||
|
|
||
|
static void iot_spg_ctrl_03h_f01()
|
||
|
{
|
||
|
spg_afn03_fn01_hw_info_ul_t rsp_data;
|
||
|
iot_build_info_t ts_info;
|
||
|
|
||
|
/* hw info */
|
||
|
uint16_t sw_version = (iot_version_packed() & 0xFFFF);
|
||
|
|
||
|
rsp_data.info.vendor_id = iot_sg_ctrl_get_vendor_id();
|
||
|
rsp_data.info.chip_code = iot_sg_ctrl_get_chip_code();
|
||
|
iot_version_get_user_build_info(&ts_info);
|
||
|
rsp_data.info.day = iot_byte_to_bcd(ts_info.day);
|
||
|
rsp_data.info.month = iot_byte_to_bcd(ts_info.month);
|
||
|
rsp_data.info.year = iot_byte_to_bcd(ts_info.year);
|
||
|
rsp_data.info.version[0] = (sw_version & 0xFF);
|
||
|
rsp_data.info.version[1] = (sw_version >> 8);
|
||
|
|
||
|
iot_spg_ctrl_send_uart(PROTO_SPG_PRM_SLAVE, PROTO_SPG_AFN_03, PROTO_SPG_F1,
|
||
|
(const uint8_t *)&rsp_data, sizeof(rsp_data), PROTO_SPG_APP_DI2_00,
|
||
|
SPG_NO_USED_ADDR, drv_data->sn);
|
||
|
}
|
||
|
|
||
|
static void iot_spg_ctrl_03h_read_param(ctrl_spg_frame_pkt_t *frm,
|
||
|
iot_pkt_t *pkt)
|
||
|
{
|
||
|
uint32_t reason;
|
||
|
|
||
|
switch (frm->fn) {
|
||
|
case PROTO_SPG_F1:
|
||
|
iot_spg_ctrl_03h_f01();
|
||
|
break;
|
||
|
case PROTO_SPG_F90:
|
||
|
iot_sg_ctrl_get_freq_band_msg();
|
||
|
break;
|
||
|
default:
|
||
|
reason = PROTO_SPG_ERR_MASTER_UNSUPPORTED_CMD;
|
||
|
iot_sg_ctrl_printf("%s:data_handle error,reason is %lu,afn : %x,"
|
||
|
"fn : %x\n", __FUNCTION__, reason, frm->afn, frm->fn);
|
||
|
iot_spg_ctrl_send_00h02_nack((uint8_t)reason, frm->app_sn);
|
||
|
break;
|
||
|
}
|
||
|
iot_pkt_free(pkt);
|
||
|
}
|
||
|
|
||
|
static void iot_spg_ctrl_04h_f90(ctrl_spg_frame_pkt_t *frm)
|
||
|
{
|
||
|
uint8_t reason = PROTO_SPG_ERR_OTHER;
|
||
|
uint32_t result = ERR_FAIL;
|
||
|
spg_set_freq_band_t *app_data =
|
||
|
(spg_set_freq_band_t *)frm->app_data;
|
||
|
uint16_t len = frm->app_data_len;
|
||
|
|
||
|
if (!app_data) {
|
||
|
reason = PROTO_SPG_ERR_INVALID_DATA;
|
||
|
goto out;
|
||
|
}
|
||
|
|
||
|
if (len < sizeof(*app_data)) {
|
||
|
reason = PROTO_SPG_ERR_INVALID_DATA_LEN;
|
||
|
goto out;
|
||
|
}
|
||
|
|
||
|
iot_sg_ctrl_set_freq_band_msg(app_data->freq_band);
|
||
|
result = ERR_OK;
|
||
|
out:
|
||
|
if (ERR_FAIL == result) {
|
||
|
iot_spg_ctrl_send_00h02_nack(reason, frm->app_sn);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void iot_spg_ctrl_04h_set_param(ctrl_spg_frame_pkt_t *frm,
|
||
|
iot_pkt_t *pkt)
|
||
|
{
|
||
|
uint32_t reason;
|
||
|
|
||
|
switch (frm->fn) {
|
||
|
case PROTO_SPG_F90:
|
||
|
iot_spg_ctrl_04h_f90(frm);
|
||
|
break;
|
||
|
default:
|
||
|
reason = PROTO_SPG_ERR_MASTER_UNSUPPORTED_CMD;
|
||
|
iot_sg_ctrl_printf("%s:data_handle error,reason is %lu,afn : %x,"
|
||
|
"fn : %x\n", __FUNCTION__, reason, frm->afn, frm->fn);
|
||
|
iot_spg_ctrl_send_00h02_nack((uint8_t)reason, frm->app_sn);
|
||
|
break;
|
||
|
}
|
||
|
iot_pkt_free(pkt);
|
||
|
}
|
||
|
|
||
|
uint8_t iot_spg_ctrl_get_proto_data_sn(uint8_t *data, uint16_t data_len,
|
||
|
uint8_t *app_sn)
|
||
|
{
|
||
|
uint8_t result = ERR_FAIL, *spg_hdr;
|
||
|
proto_spg_hdr_t *headr_data = NULL;
|
||
|
proto_spg_addr_field_t *addr_data = NULL;
|
||
|
proto_spg_app_data_t *app_data = NULL;
|
||
|
|
||
|
spg_hdr = proto_spg_sanity_check(data, data_len, &result,
|
||
|
PROTO_SPG_APP_CTRL_DI3);
|
||
|
if (spg_hdr) {
|
||
|
headr_data = (proto_spg_hdr_t*)spg_hdr;
|
||
|
if (headr_data->ctrl_byte.addr) {
|
||
|
addr_data = (proto_spg_addr_field_t*)(spg_hdr +
|
||
|
PROTO_SPG_DATA_OFFSET);
|
||
|
app_data = (proto_spg_app_data_t*)(spg_hdr + PROTO_SPG_DATA_OFFSET +
|
||
|
sizeof(*addr_data));
|
||
|
} else {
|
||
|
app_data = (proto_spg_app_data_t*)(spg_hdr + PROTO_SPG_DATA_OFFSET);
|
||
|
}
|
||
|
*app_sn = app_data->seq;
|
||
|
return ERR_OK;
|
||
|
}
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
static void iot_sg_ctrl_cco_ctrl_data_send(uint8_t *data, uint16_t data_len,
|
||
|
uint8_t app_sn, iot_pkt_t *pkt)
|
||
|
{
|
||
|
uint8_t reason = PROTO_SPG_ERR_INVALID_DATA;
|
||
|
uint32_t result = ERR_OK;
|
||
|
uint8_t sn;
|
||
|
|
||
|
proto_645_sub33_handle(data, data_len);
|
||
|
if (iot_spg_ctrl_get_proto_data_sn(data, data_len, &sn) != ERR_OK) {
|
||
|
iot_sg_ctrl_data_print("error data", data, data_len);
|
||
|
reason = PROTO_SPG_ERR_INVALID_DATA;
|
||
|
result = ERR_FAIL;
|
||
|
goto out;
|
||
|
}
|
||
|
iot_pkt_set_data(pkt, data);
|
||
|
iot_pkt_set_tail(pkt, data + data_len);
|
||
|
if (iot_sg_ctrl_send_data(PROTO_CTRL_CMD_TYPE_MR_CCO, sn, pkt)) {
|
||
|
reason = PROTO_SPG_ERR_OTHER;
|
||
|
result = ERR_FAIL;
|
||
|
}
|
||
|
|
||
|
out:
|
||
|
if (ERR_OK != result) {
|
||
|
iot_spg_ctrl_send_00h02_nack((uint8_t)reason, app_sn);
|
||
|
iot_pkt_free(pkt);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void iot_sg_ctrl_passthrough_data_send(uint8_t *data, uint16_t data_len,
|
||
|
uint8_t app_sn, iot_pkt_t *pkt)
|
||
|
{
|
||
|
uint8_t reason = PROTO_SPG_ERR_INVALID_DATA;
|
||
|
uint8_t sn = 0;
|
||
|
uint32_t result = ERR_OK;
|
||
|
|
||
|
proto_645_sub33_handle(data, data_len);
|
||
|
switch (iot_sg_ctrl_get_connect_role()) {
|
||
|
case IOT_PLC_DEV_ROLE_CCO:
|
||
|
{
|
||
|
if (iot_spg_ctrl_get_proto_data_sn(data, data_len, &sn) != ERR_OK) {
|
||
|
reason = PROTO_SPG_ERR_INVALID_DATA;
|
||
|
result = ERR_FAIL;
|
||
|
goto out;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
case IOT_PLC_DEV_ROLE_STA:
|
||
|
case IOT_PLC_DEV_ROLE_PCO:
|
||
|
{
|
||
|
if (iot_sg_ctrl_get_sta_passthrough_state()) {
|
||
|
reason = PROTO_SPG_ERR_MASTER_IS_BUSY;
|
||
|
result = ERR_FAIL;
|
||
|
goto out;
|
||
|
}
|
||
|
|
||
|
if (proto_645_format_check(data, data_len, PROTO_645_DIR_MASTER)
|
||
|
== NULL) {
|
||
|
reason = PROTO_SPG_ERR_INVALID_DATA;
|
||
|
result = ERR_FAIL;
|
||
|
goto out;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
default:
|
||
|
reason = PROTO_SPG_ERR_MASTER_UNSUPPORTED_CMD;
|
||
|
result = ERR_FAIL;
|
||
|
goto out;
|
||
|
}
|
||
|
|
||
|
iot_pkt_set_data(pkt, data);
|
||
|
iot_pkt_set_tail(pkt, data + data_len);
|
||
|
if (sn) {
|
||
|
result = iot_sg_ctrl_send_data(PROTO_CTRL_CMD_TYPE_PASSTHROUGH, sn,
|
||
|
pkt);
|
||
|
} else {
|
||
|
result = iot_sg_ctrl_send_data(PROTO_CTRL_CMD_TYPE_PASSTHROUGH,
|
||
|
IOT_CTRL_PASSTHROUGH_SN, pkt);
|
||
|
}
|
||
|
if (result != ERR_OK) {
|
||
|
reason = PROTO_SPG_ERR_INVALID_DATA;
|
||
|
}
|
||
|
|
||
|
out:
|
||
|
if (ERR_OK != result) {
|
||
|
iot_sg_ctrl_data_print("error data", data, data_len);
|
||
|
iot_spg_ctrl_send_00h02_nack((uint8_t)reason, app_sn);
|
||
|
iot_pkt_free(pkt);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void iot_sg_ctrl_data_send_handle(ctrl_spg_frame_pkt_t *frm,
|
||
|
iot_pkt_t *pkt)
|
||
|
{
|
||
|
uint32_t reason= PROTO_SPG_ERR_MASTER_UNSUPPORTED_CMD;
|
||
|
uint8_t consumed = 0;
|
||
|
spg_afn22_fn01_data_send_dl_t *req;
|
||
|
|
||
|
req = (spg_afn22_fn01_data_send_dl_t *)frm->app_data;
|
||
|
switch (req->data_type) {
|
||
|
case PROTO_SPG_CTRL_DATA_TYPE_CCO_CTRL:
|
||
|
{
|
||
|
iot_sg_ctrl_cco_ctrl_data_send(req->data, req->data_len, frm->app_sn,
|
||
|
pkt);
|
||
|
consumed = 1;
|
||
|
break;
|
||
|
}
|
||
|
case PROTO_SPG_CTRL_DATA_TYPE_PASSTHROUGH:
|
||
|
{
|
||
|
iot_sg_ctrl_passthrough_data_send(req->data, req->data_len, frm->app_sn,
|
||
|
pkt);
|
||
|
consumed = 1;
|
||
|
break;
|
||
|
}
|
||
|
default:
|
||
|
iot_sg_ctrl_printf("%s:data type error,reason is %lu, data type is %lu",
|
||
|
__FUNCTION__, reason, req->data_type);
|
||
|
iot_spg_ctrl_send_00h02_nack((uint8_t)reason, frm->app_sn);
|
||
|
break;
|
||
|
}
|
||
|
if (!consumed) {
|
||
|
iot_pkt_free(pkt);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void iot_spg_ctrl_22h_data_send(ctrl_spg_frame_pkt_t *frm,
|
||
|
iot_pkt_t *pkt)
|
||
|
{
|
||
|
uint32_t reason= PROTO_SPG_ERR_MASTER_UNSUPPORTED_CMD;
|
||
|
uint8_t consumed = 0;
|
||
|
|
||
|
switch (frm->fn) {
|
||
|
case PROTO_SPG_F1:
|
||
|
{
|
||
|
iot_sg_ctrl_data_send_handle(frm, pkt);
|
||
|
consumed = 1;
|
||
|
break;
|
||
|
}
|
||
|
default:
|
||
|
iot_sg_ctrl_printf("%s:data_handle error,reason is %lu,afn : %x,"
|
||
|
"fn : %x\n", __FUNCTION__, reason, frm->afn, frm->fn);
|
||
|
iot_spg_ctrl_send_00h02_nack((uint8_t)reason, frm->app_sn);
|
||
|
break;
|
||
|
}
|
||
|
if (!consumed) {
|
||
|
iot_pkt_free(pkt);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void iot_spg_ctrl_41h_f01(ctrl_spg_frame_pkt_t *frm)
|
||
|
{
|
||
|
uint8_t cnt;
|
||
|
uint8_t i;
|
||
|
uint16_t rsp_len;
|
||
|
spg_afn41_fn01_rpt_node_info_ul *rsp_data = NULL;
|
||
|
ctrl_node_info_t *node_info = NULL;
|
||
|
iot_pkt_t *tmp_pkt;
|
||
|
iot_pkt_t *tmp2_pkt = NULL;
|
||
|
|
||
|
cnt = iot_sg_ctrl_get_scan_node_cnt();
|
||
|
if (cnt > IOT_CTRL_SCAN_NODE_MAX_CNT) {
|
||
|
iot_sg_ctrl_printf("WARNING: %s scan count: %d exceed \n", cnt);
|
||
|
cnt = IOT_CTRL_SCAN_NODE_MAX_CNT;
|
||
|
} else if (!cnt) {
|
||
|
goto out;
|
||
|
}
|
||
|
|
||
|
tmp2_pkt = iot_pkt_alloc(cnt * sizeof(ctrl_node_info_t),
|
||
|
IOT_SMART_GRID_MID);
|
||
|
IOT_ASSERT(tmp2_pkt);
|
||
|
|
||
|
node_info = (ctrl_node_info_t *)iot_pkt_data(tmp2_pkt);
|
||
|
cnt = iot_sg_ctrl_get_scan_node_info(node_info, 1, cnt);
|
||
|
out:
|
||
|
rsp_len = sizeof(*rsp_data) + cnt * sizeof(spg_ctrl_node_info_t);
|
||
|
tmp_pkt = iot_pkt_alloc(proto_spg_get_pkt_len(SPG_NO_USED_ADDR,
|
||
|
rsp_len), IOT_SMART_GRID_MID);
|
||
|
IOT_ASSERT(tmp_pkt);
|
||
|
|
||
|
rsp_data = (spg_afn41_fn01_rpt_node_info_ul*)iot_pkt_reserve(tmp_pkt,
|
||
|
proto_spg_get_rsvd_len(SPG_NO_USED_ADDR));
|
||
|
rsp_data->total_cnt = cnt;
|
||
|
for (i = 0; i < cnt; i++, node_info++) {
|
||
|
rsp_data->node_info[i].nid = node_info->nid;
|
||
|
rsp_data->node_info[i].band_id = node_info->band_id;
|
||
|
iot_mac_addr_cpy(rsp_data->node_info[i].addr, node_info->addr);
|
||
|
}
|
||
|
if (tmp2_pkt) {
|
||
|
iot_pkt_free(tmp2_pkt);
|
||
|
}
|
||
|
iot_pkt_put(tmp_pkt, rsp_len);
|
||
|
iot_spg_ctrl_send_uart_sn(PROTO_SPG_PRM_SLAVE, PROTO_SPG_AFN_41,
|
||
|
PROTO_SPG_F1, tmp_pkt, PROTO_SPG_APP_DI2_00, SPG_NO_USED_ADDR,
|
||
|
frm->app_sn);
|
||
|
}
|
||
|
|
||
|
static void iot_spg_ctrl_41h_f02(ctrl_spg_frame_pkt_t *frm)
|
||
|
{
|
||
|
uint32_t reason = PROTO_SPG_ERR_INVALID_DATA;
|
||
|
uint32_t result = ERR_OK;
|
||
|
spg_afn41_fn02_req_connect_node_dl_t *req =
|
||
|
(spg_afn41_fn02_req_connect_node_dl_t *)frm->app_data;
|
||
|
uint16_t len = frm->app_data_len;
|
||
|
|
||
|
if (len < sizeof(spg_afn41_fn02_req_connect_node_dl_t)) {
|
||
|
/* invalid data length */
|
||
|
reason = PROTO_SPG_ERR_INVALID_DATA_LEN;
|
||
|
result = ERR_FAIL;
|
||
|
goto out;
|
||
|
}
|
||
|
if (ERR_OK != iot_sg_ctrl_proto_connect_from_scan_node(req->addr,
|
||
|
frm->app_sn)) {
|
||
|
reason = PROTO_SPG_ERR_INVALID_DATA;
|
||
|
result = ERR_FAIL;
|
||
|
}
|
||
|
out:
|
||
|
if (ERR_OK != result) {
|
||
|
iot_sg_ctrl_printf("%s:data_handle error,reason is %lu,afn : %x,"
|
||
|
"fn : %x\n", __FUNCTION__, reason, frm->afn, frm->fn);
|
||
|
iot_spg_ctrl_send_00h02_nack((uint8_t)reason, frm->app_sn);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void iot_spg_ctrl_41h_scan_node_mgr(ctrl_spg_frame_pkt_t *frm,
|
||
|
iot_pkt_t *pkt)
|
||
|
{
|
||
|
uint32_t reason= PROTO_SPG_ERR_MASTER_UNSUPPORTED_CMD;
|
||
|
|
||
|
switch (frm->fn) {
|
||
|
case PROTO_SPG_F1:
|
||
|
{
|
||
|
iot_spg_ctrl_41h_f01(frm);
|
||
|
break;
|
||
|
}
|
||
|
case PROTO_SPG_F2:
|
||
|
{
|
||
|
iot_spg_ctrl_41h_f02(frm);
|
||
|
break;
|
||
|
}
|
||
|
default:
|
||
|
iot_sg_ctrl_printf("%s:data_handle error,reason is %lu,afn : %x,"
|
||
|
"fn : %x\n", __FUNCTION__, reason, frm->afn, frm->fn);
|
||
|
iot_spg_ctrl_send_00h02_nack((uint8_t)reason, frm->app_sn);
|
||
|
break;
|
||
|
}
|
||
|
iot_pkt_free(pkt);
|
||
|
}
|
||
|
|
||
|
static void iot_spg_ctrl_f0h_f01(ctrl_spg_frame_pkt_t *frm)
|
||
|
{
|
||
|
spg_afnf0_fn01_req_connect_node_dl_t *cmd = NULL;
|
||
|
|
||
|
uint32_t result = ERR_OK;
|
||
|
uint8_t reason = PROTO_SPG_ERR_INVALID_DATA;
|
||
|
|
||
|
cmd = (spg_afnf0_fn01_req_connect_node_dl_t*)frm->app_data;
|
||
|
uint16_t len = frm->app_data_len;
|
||
|
|
||
|
if (len < sizeof(spg_afnf0_fn01_req_connect_node_dl_t)) {
|
||
|
reason = PROTO_SPG_ERR_INVALID_DATA_LEN;
|
||
|
result = ERR_FAIL;
|
||
|
goto out;
|
||
|
}
|
||
|
if (ERR_OK != iot_sg_ctrl_proto_connect(cmd->addr,
|
||
|
(uint8_t)frm->app_sn)) {
|
||
|
reason = PROTO_SPG_ERR_INVALID_DATA;
|
||
|
result = ERR_FAIL;
|
||
|
}
|
||
|
out:
|
||
|
if (ERR_OK != result) {
|
||
|
iot_spg_ctrl_send_00h02_nack((uint8_t)reason, frm->app_sn);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void iot_spg_ctrl_f0h_f02(ctrl_spg_frame_pkt_t *frm)
|
||
|
{
|
||
|
iot_sg_ctrl_reset_connet_status();
|
||
|
iot_spg_ctrl_send_00h01_ack(frm->app_sn);
|
||
|
}
|
||
|
|
||
|
static void iot_spg_ctrl_f0h_cmd(ctrl_spg_frame_pkt_t *frm, iot_pkt_t *pkt)
|
||
|
{
|
||
|
uint32_t reason= PROTO_SPG_ERR_MASTER_UNSUPPORTED_CMD;
|
||
|
uint8_t consumed = 0;
|
||
|
|
||
|
switch (frm->fn) {
|
||
|
case PROTO_SPG_F1:
|
||
|
{
|
||
|
iot_spg_ctrl_f0h_f01(frm);
|
||
|
break;
|
||
|
}
|
||
|
case PROTO_SPG_F2:
|
||
|
{
|
||
|
iot_spg_ctrl_f0h_f02(frm);
|
||
|
break;
|
||
|
}
|
||
|
default:
|
||
|
iot_sg_ctrl_printf("%s:data_handle error,reason is %lu,afn : 0x%02X,"
|
||
|
"fn : 0x%02X\n", __FUNCTION__, reason, frm->afn, frm->fn);
|
||
|
iot_spg_ctrl_send_00h02_nack((uint8_t)reason, frm->app_sn);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if (!consumed) {
|
||
|
iot_pkt_free(pkt);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void iot_spg_ctrl_send_data_handle(ctrl_spg_frame_pkt_t *frm,
|
||
|
iot_pkt_t *pkt)
|
||
|
{
|
||
|
uint32_t reason= PROTO_SPG_ERR_MASTER_UNSUPPORTED_CMD;
|
||
|
uint32_t result = ERR_OK;
|
||
|
|
||
|
iot_sg_ctrl_printf("%s: afn = 0x%02X, fn = 0n%lu, sn: %d \n", __FUNCTION__,
|
||
|
frm->afn, frm->fn, frm->app_sn);
|
||
|
switch (frm->afn) {
|
||
|
case PROTO_SPG_AFN_01:
|
||
|
{
|
||
|
iot_spg_ctrl_01h_init(frm, pkt);
|
||
|
break;
|
||
|
}
|
||
|
case PROTO_SPG_AFN_03:
|
||
|
{
|
||
|
iot_spg_ctrl_03h_read_param(frm, pkt);
|
||
|
break;
|
||
|
}
|
||
|
case PROTO_SPG_AFN_04:
|
||
|
{
|
||
|
iot_spg_ctrl_04h_set_param(frm, pkt);
|
||
|
break;
|
||
|
}
|
||
|
case PROTO_SPG_AFN_22:
|
||
|
{
|
||
|
iot_spg_ctrl_22h_data_send(frm, pkt);
|
||
|
break;
|
||
|
}
|
||
|
case PROTO_SPG_AFN_41:
|
||
|
{
|
||
|
iot_spg_ctrl_41h_scan_node_mgr(frm, pkt);
|
||
|
break;
|
||
|
}
|
||
|
case PROTO_SPG_AFN_F0:
|
||
|
{
|
||
|
iot_spg_ctrl_f0h_cmd(frm, pkt);
|
||
|
break;
|
||
|
}
|
||
|
default:
|
||
|
result = ERR_FAIL;
|
||
|
break;
|
||
|
}
|
||
|
if (ERR_OK != result) {
|
||
|
iot_sg_ctrl_printf("%s:data_handle error,reason is %lu,afn : %x,"
|
||
|
"fn : %x\n", __FUNCTION__, reason, frm->afn, frm->fn);
|
||
|
iot_spg_ctrl_send_00h02_nack((uint8_t)reason, frm->app_sn);
|
||
|
iot_pkt_free(pkt);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief iot_sg_ctrl_uart_data_recv_handle - receive uart data
|
||
|
* @param pkt: include full 3762 frame, the pkt whill be free
|
||
|
*/
|
||
|
static uint32_t iot_spg_ctrl_uart_data_recv_handle(iot_pkt_t *pkt)
|
||
|
{
|
||
|
uint8_t *spg_hdr;
|
||
|
uint16_t len = (uint16_t)iot_pkt_data_len(pkt);
|
||
|
proto_spg_hdr_t *headr_data = NULL;
|
||
|
proto_spg_addr_field_t *addr_data = NULL;
|
||
|
proto_spg_app_data_t *app_data = NULL;
|
||
|
ctrl_spg_frame_pkt_t spg_cmd;
|
||
|
uint32_t result = ERR_OK;
|
||
|
|
||
|
if(len < sizeof(proto_spg_hdr_t)){
|
||
|
goto err;
|
||
|
}
|
||
|
|
||
|
spg_hdr = proto_spg_sanity_check(iot_pkt_data(pkt), len,
|
||
|
(uint8_t *)&result, PROTO_SPG_APP_CTRL_DI3);
|
||
|
if (!spg_hdr) {
|
||
|
goto err;
|
||
|
}
|
||
|
|
||
|
headr_data = (proto_spg_hdr_t*)spg_hdr;
|
||
|
if (headr_data->ctrl_byte.addr) {
|
||
|
addr_data = (proto_spg_addr_field_t*)(spg_hdr + PROTO_SPG_DATA_OFFSET);
|
||
|
app_data = (proto_spg_app_data_t*)(spg_hdr + PROTO_SPG_DATA_OFFSET +
|
||
|
sizeof(*addr_data));
|
||
|
/* record mac msg */
|
||
|
os_mem_cpy(drv_data->dest_mac, addr_data->dest_mac,
|
||
|
PROTO_SPG_MAC_ADDR_LEN);
|
||
|
os_mem_cpy(drv_data->req_mac, addr_data->src_mac,
|
||
|
PROTO_SPG_MAC_ADDR_LEN);
|
||
|
/* offset address length */
|
||
|
headr_data->len -= PROTO_SPG_MAC_ADDR_LEN;
|
||
|
headr_data->len -= PROTO_SPG_MAC_ADDR_LEN;
|
||
|
} else {
|
||
|
os_mem_set(drv_data->dest_mac, 0, PROTO_SPG_MAC_ADDR_LEN);
|
||
|
os_mem_set(drv_data->req_mac, 0, PROTO_SPG_MAC_ADDR_LEN);
|
||
|
app_data = (proto_spg_app_data_t*)(spg_hdr + PROTO_SPG_DATA_OFFSET);
|
||
|
}
|
||
|
/* check di data */
|
||
|
if ((app_data->di[3] != PROTO_SPG_APP_CTRL_DI3) ||
|
||
|
(app_data->di[1] != app_data->afn)) {
|
||
|
goto err;
|
||
|
}
|
||
|
/* record seq msg */
|
||
|
drv_data->sn = app_data->seq;
|
||
|
spg_cmd.app_sn = app_data->seq;
|
||
|
spg_cmd.afn = app_data->afn;
|
||
|
spg_cmd.fn = app_data->di[0];
|
||
|
spg_cmd.app_data_len = headr_data->len - PROTO_SPG_FIX_FIELD_LEN -
|
||
|
sizeof(proto_spg_app_data_t);
|
||
|
if (!spg_cmd.app_data_len) {
|
||
|
spg_cmd.app_data = NULL;
|
||
|
} else {
|
||
|
spg_cmd.app_data = app_data->data;
|
||
|
}
|
||
|
iot_spg_ctrl_send_data_handle(&spg_cmd, pkt);
|
||
|
goto out;
|
||
|
err:
|
||
|
iot_pkt_free(pkt);
|
||
|
result = ERR_FAIL;
|
||
|
out:
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
static uint32_t iot_spg_ctrl_check_data_handle(iot_pkt_t *pkt)
|
||
|
{
|
||
|
uint8_t *spg_hdr;
|
||
|
uint16_t len = (uint16_t)iot_pkt_data_len(pkt);
|
||
|
uint8_t ctrl = 0;
|
||
|
uint32_t result = ERR_FAIL;
|
||
|
|
||
|
if(len < sizeof(proto_spg_hdr_t)){
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
spg_hdr = proto_spg_sanity_check(iot_pkt_data(pkt), len,
|
||
|
(uint8_t *)&result, PROTO_SPG_APP_CTRL_DI3);
|
||
|
if (!spg_hdr) {
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
ctrl = spg_hdr[PROTO_SPG_CTRL_OFFSET];
|
||
|
if ((ctrl & 0x07) == PROTO_CTRL_COMMUNICAT_RSVD) {
|
||
|
result = ERR_OK;
|
||
|
}
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
iot_ctrl_drv_t g_spg_ctrl_drv = {
|
||
|
.drv_id = IOT_CTRL_DRV_ID_SPG,
|
||
|
.init = iot_spg_ctrl_drv_init,
|
||
|
.deinit = iot_spg_ctrl_drv_deinit,
|
||
|
.event_report_handle = iot_spg_ctrl_event_report_handle,
|
||
|
.ctrl_drv_data_handle = iot_spg_ctrl_uart_data_recv_handle,
|
||
|
.check_data_handle = iot_spg_ctrl_check_data_handle,
|
||
|
.freq_band_report_cb = iot_spg_ctrl_freq_band_handler,
|
||
|
.freq_band_set_result_cb = iot_spg_ctrl_set_freq_band_result,
|
||
|
};
|
||
|
|
||
|
#endif /* IOT_SG_CONTROLLER_ENABLE */
|