195 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			195 lines
		
	
	
		
			6.9 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. | ||
|  | 
 | ||
|  | ****************************************************************************/ | ||
|  | 
 | ||
|  | /* os shim includes */ | ||
|  | #include "os_types_api.h"
 | ||
|  | #include "os_mem_api.h"
 | ||
|  | 
 | ||
|  | /* common includes */ | ||
|  | #include "iot_utils_api.h"
 | ||
|  | #include "iot_errno_api.h"
 | ||
|  | #include "iot_crc_api.h"
 | ||
|  | #include "iot_rtc_api.h"
 | ||
|  | 
 | ||
|  | /* protocol includes */ | ||
|  | #include "proto_mwc.h"
 | ||
|  | 
 | ||
|  | /* sequence number for dl raw read low meter commands sent out */ | ||
|  | static uint8_t psdu_sn = 0; | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * @brief:  get sn for dl raw read low power meter. | ||
|  |  * @return: psdu dl sn, it ranges from 1 to 255. | ||
|  |  */ | ||
|  | static uint8_t proto_mwc_get_psdu_dl_sn() | ||
|  | { | ||
|  |     psdu_sn++; | ||
|  |     if (psdu_sn == 0) { | ||
|  |         psdu_sn = 1; | ||
|  |     } | ||
|  |     return psdu_sn; | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * @brief:  convert hplc net snid to mwc proto channel index. | ||
|  |  * @param   plc_nid: hplc net snid. | ||
|  |  * @param   return:  channel index. | ||
|  |  */ | ||
|  | static uint8_t proto_mwc_convert_channel_id(uint32_t plc_nid) | ||
|  | { | ||
|  |     // nid is the result of plc_nid mod 32, it ranges from 1 to 32.
 | ||
|  |     uint8_t nid = (plc_nid & 0x1F) + 1; | ||
|  | 
 | ||
|  |     return nid; | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * @brief:  get real time. | ||
|  |  * @param   real_time: year, BIN format. | ||
|  |  */ | ||
|  | static void proto_mwc_get_real_time(proto_mwc_real_time_t *real_time) | ||
|  | { | ||
|  |     uint8_t       real_week; | ||
|  |     iot_time_tm_t tm; | ||
|  |     iot_rtc_get(&tm); | ||
|  | 
 | ||
|  |     real_week = iot_calc_week(tm.tm_year, tm.tm_mon, tm.tm_mday); | ||
|  |     real_time->sec   = iot_byte_to_bcd(tm.tm_sec); | ||
|  |     real_time->min   = iot_byte_to_bcd(tm.tm_min); | ||
|  |     real_time->hour  = iot_byte_to_bcd(tm.tm_hour); | ||
|  |     real_time->week  = iot_byte_to_bcd(real_week); | ||
|  |     real_time->day   = iot_byte_to_bcd(tm.tm_mday); | ||
|  |     real_time->month = iot_byte_to_bcd(tm.tm_mon); | ||
|  |     real_time->year  = iot_byte_to_bcd(tm.tm_year % 100); | ||
|  | } | ||
|  | 
 | ||
|  | uint32_t proto_mwc_handle_raw_mr_lp_meter(uint8_t *mr_data, uint8_t *data, | ||
|  |     uint16_t data_len, uint32_t plc_nid, uint8_t *lp_meter_mac, | ||
|  |     uint8_t *dau_mac) | ||
|  | { | ||
|  |     uint16_t  len; | ||
|  |     uint16_t crc_itu = 0; | ||
|  |     proto_mwc_phy_phr_t       *phr = (proto_mwc_phy_phr_t *)mr_data; | ||
|  |     proto_mwc_psdu_header_t   *psdu_head = NULL; | ||
|  |     proto_mwc_psdu_clct_raw_t *psdu_data = NULL; | ||
|  |     proto_mwc_phy_phr_tail_t  *phr_tail = NULL; | ||
|  |     len = PROTO_MWC_RAW_MR_FRAME_LEN + data_len - | ||
|  |         sizeof(phr->data_len) - sizeof(crc_itu); | ||
|  |     if (len > 0xFF) { | ||
|  |         return ERR_FAIL; | ||
|  |     } | ||
|  |     phr->data_len = (uint8_t)len; | ||
|  |     phr->channel_idx = proto_mwc_convert_channel_id(plc_nid) * 2; | ||
|  |     phr->proto_ver = PROTO_MWC_PHR_PROTO_VER; | ||
|  |     phr->phr_cs = (phr->data_len ^ phr->channel_idx) ^ phr->proto_ver; | ||
|  | 
 | ||
|  |     psdu_head = (proto_mwc_psdu_header_t *)phr->data; | ||
|  |     psdu_head->control = PROTO_MWC_PSDU_CTRL_FIELD; | ||
|  |     psdu_head->sn = proto_mwc_get_psdu_dl_sn(); | ||
|  |     psdu_head->panid = PROTO_MWC_PSDU_PANID_DEFAULT; | ||
|  |     os_mem_cpy(psdu_head->dest_addr, lp_meter_mac, PROTO_MWC_PSDU_ADDR_LEN); | ||
|  |     os_mem_set(psdu_head->source_addr, 0, PROTO_MWC_PSDU_ADDR_LEN); | ||
|  |     os_mem_cpy(psdu_head->source_addr, dau_mac, PROTO_MWC_ADDR_LEN); | ||
|  |     psdu_head->payload.cmd_type = PROTO_MWC_PSDU_CMD_CLCT_RAW; | ||
|  | 
 | ||
|  |     psdu_data = (proto_mwc_psdu_clct_raw_t *)psdu_head->payload.data; | ||
|  |     psdu_data->baud_rate = 0; | ||
|  |     psdu_data->timeout = 0; | ||
|  |     psdu_data->data_len = (uint8_t)data_len; | ||
|  |     os_mem_cpy(psdu_data->data, data, data_len); | ||
|  | 
 | ||
|  |     phr_tail = (proto_mwc_phy_phr_tail_t *)(psdu_data->data + data_len); | ||
|  |     crc_itu = iot_getcrc16(mr_data, phr->data_len + sizeof(phr->data_len), | ||
|  |         IOT_CRC16_TYPE_X25); | ||
|  |     phr_tail->crc = crc_itu; | ||
|  |     return ERR_OK; | ||
|  | } | ||
|  | 
 | ||
|  | void proto_mwc_handle_cfg_lp_meter(uint8_t *cfg_data, uint32_t plc_nid, | ||
|  |     uint16_t slot_id, uint16_t panid, uint8_t *lp_meter_mac, uint8_t *dau_mac) | ||
|  | { | ||
|  |     uint8_t  len = PROTO_MWC_CFG_LP_METER_LEN; | ||
|  |     uint16_t crc_itu = 0; | ||
|  |     proto_mwc_phy_phr_t           *phr = (proto_mwc_phy_phr_t *)cfg_data; | ||
|  |     proto_mwc_psdu_header_t       *psdu_head = NULL; | ||
|  |     proto_mwc_psdu_cfg_lp_meter_t *psdu_data = NULL; | ||
|  |     proto_mwc_phy_phr_tail_t      *phr_tail = NULL; | ||
|  | 
 | ||
|  |     phr->data_len = len - sizeof(phr->data_len) - sizeof(crc_itu); | ||
|  |     phr->channel_idx = PROTO_MWC_PSDU_CHANNEL_DEFAULT; | ||
|  |     phr->proto_ver = PROTO_MWC_PHR_PROTO_VER; | ||
|  |     phr->phr_cs = (phr->data_len ^ phr->channel_idx) ^ phr->proto_ver; | ||
|  | 
 | ||
|  |     psdu_head = (proto_mwc_psdu_header_t *)phr->data; | ||
|  |     psdu_head->control = PROTO_MWC_PSDU_CTRL_FIELD; | ||
|  |     psdu_head->sn = proto_mwc_get_psdu_dl_sn(); | ||
|  |     psdu_head->panid = PROTO_MWC_PSDU_PANID_DEFAULT; | ||
|  |     os_mem_cpy(psdu_head->dest_addr, lp_meter_mac, PROTO_MWC_PSDU_ADDR_LEN); | ||
|  |     os_mem_set(psdu_head->source_addr, 0, PROTO_MWC_PSDU_ADDR_LEN); | ||
|  |     os_mem_cpy(psdu_head->source_addr, dau_mac, PROTO_MWC_ADDR_LEN); | ||
|  |     psdu_head->payload.cmd_type = PROTO_MWC_PSDU_CMD_CONFIG_LP_METER; | ||
|  | 
 | ||
|  |     psdu_data = (proto_mwc_psdu_cfg_lp_meter_t *)psdu_head->payload.data; | ||
|  |     psdu_data->channel_group = proto_mwc_convert_channel_id(plc_nid); | ||
|  |     os_mem_cpy(psdu_data->dau_mac, dau_mac, PROTO_MWC_ADDR_LEN); | ||
|  |     psdu_data->panid = panid; | ||
|  |     psdu_data->slot_time = slot_id; | ||
|  |     proto_mwc_get_real_time(&psdu_data->real_time); | ||
|  |     psdu_data->work_mode = PROTO_MWC_PSDU_WORK_MODE_VALID; | ||
|  |     psdu_data->work_time = PROTO_MWC_PSDU_WORK_TIME_DEFAULT; | ||
|  | 
 | ||
|  |     phr_tail = (proto_mwc_phy_phr_tail_t *)(psdu_data + 1); | ||
|  |     crc_itu = iot_getcrc16(cfg_data, len - sizeof(crc_itu), IOT_CRC16_TYPE_X25); | ||
|  |     phr_tail->crc = crc_itu; | ||
|  | } | ||
|  | 
 | ||
|  | proto_mwc_psdu_data_t *proto_mwc_get_psdu_data(uint8_t *cfg_ack_data, | ||
|  |     uint32_t len) | ||
|  | { | ||
|  |     proto_mwc_psdu_data_t *psdu_data; | ||
|  |     proto_mwc_psdu_lp_meter_ack_t *mr_ack; | ||
|  |     proto_mwc_psdu_cfg_ack_t *cfg_ack; | ||
|  |     if (len < sizeof(proto_mwc_phy_phr_t) + sizeof(proto_mwc_psdu_header_t)) { | ||
|  |         return NULL; | ||
|  |     } | ||
|  |     len = len - sizeof(proto_mwc_phy_phr_t) - sizeof(proto_mwc_psdu_header_t); | ||
|  | 
 | ||
|  |     psdu_data = (proto_mwc_psdu_data_t *)(cfg_ack_data + | ||
|  |         sizeof(proto_mwc_phy_phr_t) + sizeof(proto_mwc_psdu_header_t) - | ||
|  |         sizeof(proto_mwc_psdu_data_t)); | ||
|  | 
 | ||
|  |     switch (psdu_data->cmd_type) { | ||
|  |     case PROTO_MWC_PSDU_CMD_LP_METER_ACK: | ||
|  |     { | ||
|  |         mr_ack = (proto_mwc_psdu_lp_meter_ack_t *)psdu_data->data; | ||
|  |         if (len < sizeof(*mr_ack) + mr_ack->data_len) { | ||
|  |             return NULL; | ||
|  |         } | ||
|  |         break; | ||
|  |     } | ||
|  |     case PROTO_MWC_PSDU_CMD_CONFIG_ACK: | ||
|  |     { | ||
|  |         cfg_ack = (proto_mwc_psdu_cfg_ack_t *)psdu_data->data; | ||
|  |         if (len < sizeof(*cfg_ack)) { | ||
|  |             return NULL; | ||
|  |         } | ||
|  |         break; | ||
|  |     } | ||
|  |     default: | ||
|  |         return NULL; | ||
|  |     } | ||
|  | 
 | ||
|  |     return psdu_data; | ||
|  | } |