/* * Copyright (c) 2006-2021, RT-Thread Development Team * * SPDX-License-Identifier: Apache-2.0 * * Change Logs: * Date Author Notes * 2023-07-06 murmur the first version */ #include "func.h" #define LOG_TAG "func" #define LOG_LVL LOG_LVL_DBG #include extern SYS_CFG scfg; extern int cryptSingleMsg(uint8_t *din, size_t len, uint8_t *dout); extern int pointInPolygon(int polyCorners,float polyX[], float polyY[],float x,float y); /** * 将收发数据以ASCII字符形式存入log * @param din 待存储数据 * @param len 数据长度 * @return 1-正常,0-异常 */ int trDataTolog(uint8_t *din, size_t len, uint8_t isTx) { // char rootDir[22] = ROOT_PATH_LOG; // mkdir(rootDir, 0); // // char tstmp[30] = ""; // char fn[50] = ""; // strcat(fn,rootDir); // strcat(fn,"20"); // strcat(fn,date2Str(tstmp));//以小时为独立文件 // fn[strlen(fn)-3]='\0'; // strcat(fn,"/"); // mkdir(fn, 0); // // strcat(fn,tstmp); // strcat(fn,".log"); char fn[60] = "\n"; initDataLogPath(fn); // LOG_D("fn=%s",fn); // LOG_HEX("d",16,din,len); int fd = open(fn, O_WRONLY | O_CREAT | O_APPEND); if (fd < 0) { LOG_E("open file %s failed!", fn); return -RT_ERROR; } else { char log[30]="\n"; getTimestmp(log+1); strcat(log,isTx?" [T]: ":" [R]: "); write(fd, log, strlen(log)); int rst = write(fd, din, len); if (rst != len) { LOG_E("write to file %s failed!", fn); close(fd); return -RT_ERROR; } close(fd); return RT_EOK; } } //3.2.1双模通信功能 //1为TT通信,0为BD短报文通信 void setCommMode(int isTT) { LOG_I("FUNC = setCommMode"); if (isTT) { //change mode } else { } //write to cfg file set_cfg("commMode", isTT); LOG_D("set commMode to %s",isTT?"TT":"BD"); } int getCommMode() { //load from cfg file int flag = get_cfg("commMode"); if (flag < 0) { LOG_W("get mode fault."); } return flag; } //3.2.2状态自检 /** * 获取磁盘剩余空间大小,单位MB * @param path指定磁盘挂载的路径,为空时表示挂载在根目录的磁盘 * @return 返回结果,单位MB */ static uint16_t getFreeSpace(const char *path) { long long cap; struct statfs buffer; int result = dfs_statfs(path ? path : "/", &buffer); if (result != 0) { LOG_E("dfs_statfs failed."); return 0; } cap = (uint16_t)((long long)buffer.f_bsize) * ((long long)buffer.f_bfree) / 1024LL / 1024LL ;//转换为MB return cap; } void d_getFreeSpace() { LOG_D("free space of flash is %d MB.",getFreeSpace(NULL)); uint16_t rst = getFreeSpace("/sd"); LOG_D("free space of sd is %d MB.%02X,%02X",rst,rst>>8,rst&0xff); } /** * 獲取電池電量。 * @return 返回電量百分比 */ RT_WEAK int getPowerLevel(void) { return 0; } typedef struct { rt_uint8_t len; rt_uint8_t data[]; } SMSG;//single messgae void upSend_thread_entry(void* parameter) { SMSG* msg = RT_NULL; msg = (SMSG*) parameter; LOG_D("1--%p",msg); // LOG_HEX("--",16,msg->data,msg->len); //check status uint8_t sta = 1; while (1 && !isTTjh()) //判断TT状态 { static uint8_t trycnt = 0; rt_thread_mdelay(4000); trycnt += 1; if (trycnt > 3) { sta = 0; //try 3 time break; } } if (!sta) { //cache to file LOG_W("TT is not ready, try to cache %d bytes data to file.",msg->len); cacheDataToFile(msg->data, msg->len); return ; } //打包数据 uint8_t dout[200]; static MSG cfg; rt_memset(&cfg, 0, sizeof(MSG)); // 分配空间 char fin[30]; time2Str(fin); strcat(fin,".bin"); packInit(&cfg, fin, 0); //写入配置 cfg.fcurpiece[0] = 1; cfg.fallpiece[0] = 1; #ifdef CRYPT_BEFRE_PACK //crypt before pack uint8_t tmp[200]; uint8_t len = cryptSingleMsg(msg->data, msg->len, tmp); size_t rst = packMsg(&cfg, tmp, len, dout);//packMsgs #else size_t rst = packMsg(&cfg, tmpmsg->data, msg->len, dout);//packMsgs #endif LOG_HEX("upSend", 27, dout, rst); if (rst) { if (sendMsg(dout, rst) == RT_EOK) { LOG_I("send to TT Done."); } else//发送失败,实例不存在此种情况 { LOG_E("send error, try to cache %d bytes data to file.",msg->len); cacheDataToFile(msg->data, msg->len); } } list_thread(); return ; } /** * 上傳天通數據 * @param din 待發送數據 * @param len 待發送數據的長度 * @return */ RT_WEAK int upSend(uint8_t *din, size_t len) { LOG_D("try to upsend to TT."); static SMSG msg; memset(&msg, 0, sizeof(SMSG)); memcpy(msg.data,din,len); msg.len=len; LOG_D("0--%p",&msg); /* 创建 serial 线程 */ rt_thread_t thread = rt_thread_create("upSend", upSend_thread_entry, (void *) &msg, 1024 * 5, 27, 10); /* 创建成功则启动线程 */ if (thread != RT_NULL) { rt_thread_startup(thread); return RT_EOK; } else { LOG_E("thread 'upSend' create failure."); return -RT_ERROR; } } /** * 系统自检,自动发送自检结果 */ void selfTest() { LOG_I("FUNC = selftest"); rt_uint8_t rst[100]={0x5A, 0xA5, ADDR_ANJI, ADDR_TT, _CMD_SELF_TEST>>8, _CMD_SELF_TEST & 0xff}; int p = 6; rt_uint8_t sysSta=1,commSpeed=0; //长度 rst[p++]=0x00; rst[p++]=0x09; rst[p++] = sysSta; rst[p++] = getXh();//xh,jh rst[p++] = isTTjh(); rst[p++] = commSpeed; rst[p++] = getPowerLevel(); //flash 剩餘空間 uint16_t cap = getFreeSpace("/"); rst[p++] = (uint8_t)(cap >> 8); rst[p++] = (uint8_t)(cap & 0xff); //SD卡剩餘空間 cap = getFreeSpace("/sd"); rst[p++] = (uint8_t)(cap >> 8); rst[p++] = (uint8_t)(cap & 0xff); //添加配置文件信息 uint8_t size = sizeof(scfg); memcpy(rst + p,&scfg, size); p+=size; rst[7] = p-8;//更新数据长度 rst[p++] = bccCRC(rst+2, p-1); ////校验位为1个字节,采用异或运算,从指令的第3个字节开始,到奇偶校验位的前一个字节结束 rst[p++] = 0xED; //结束位 LOG_HEX("selfTestRes",16,rst,p); //发送结果 upSend(rst,p); } //3.2.3日志记录 //日志功能由各函数通过LOG_D()实现 //3.2.4自毁功能 /** * 设置自毁功能开关 * @param setON 1-自毁功能开启,0-关闭 */ void setSelfDestructSWT(int setON) { LOG_I("FUNC = set selfdestruct"); //write to cfg file set_cfg("selfDesSW", setON); LOG_D("set SelfDesSW to %s",setON?"ON":"OFF"); } /** * 获取自毁开关状态 * @return 1-自毁功能开启,0-关闭 */ int getSelfDestructSWT() { LOG_I("FUNC = get selfdestruct"); //load from cfg file int flag = get_cfg("selfDesSW"); if (flag < 0) { LOG_W("get mode fault."); } return flag; } /** * 启动自毁 */ void selfDestruct() { if (getSelfDestructSWT()) { //硬件自毁 LOG_W("SELF DESTRUCT START."); } } //3.2.5开、关窗功能 extern void updateAlarm(uint8_t *t, size_t len); /** * 更新开窗时间,目前支持两组开窗时段。更新会清除之前的开窗设置 */ void setCommWindow(uint8_t *t, size_t len) { LOG_I("FUNC = setCommWindow"); LOG_HEX("new rtc",16,t,len); //not working here // char tmpstr[20]; // bytes2str(t,len, 10, ",", tmpstr); // set_cfgs("openWindowTime",tmpstr); updateAlarm(t,len); // LOG_D("updated new CommWindow zone"); } void d_sw(void) { uint8_t cfg[]={0x03, 0x1F, 0x03, 0x20, 0x07, 0x1F, 0x09, 0x1E}; setCommWindow(cfg, 8); } /** * 手动控制开窗 * @param t 开窗时长,单位分钟,时间到则自动关窗。t=0时需要手动关窗。 */ void openWindow(int t) { //开启TT // pwTT_thread_entry("1");//开机 initTT(); setWindowMode(); if (!t) { LOG_D("手动开窗完成,需手动关窗。"); return; } //设置定时器,定时器到则关窗 /* 创建定时器,单次定时器 */ rt_timer_t timer1; timer1 = rt_timer_create("window", deInitTT(), RT_NULL, rt_tick_from_millisecond(t*60*1000), RT_TIMER_FLAG_ONE_SHOT); /* 启动定时器 */ if (timer1 != RT_NULL) { rt_timer_start(timer1); LOG_D("手动开窗完成,%d分钟后自动关窗。",t); } } /** * 手动关窗 */ void closeWindow() { // pwTT_thread_entry("0");//关 机 deInitTT(); LOG_D("手动关窗完成。"); } //3.2.6工作参数配置、状态查询 //包含浮体自身、标体透传、两者结合 //这里主要实现浮体自身 //浮体自身的参数配置各功能函数有实现,此处需规定参数下发协议和解码实现 //获取当前位置 /** * 获取当前经纬度信息 * @param dout 存储结果的数组 * @param cnt 获取几组位置数据 * @return 数组长度。0表示位置数据未准备好。 */ RT_WEAK int getLoc(uint8_t * dout, size_t cnt) { LOG_D("待实现获取位置函数,此处以0xCD代替"); uint8_t tmp[200]; memset(tmp,0xCD,200); memcpy(dout,tmp,cnt*10); return cnt*10;//4+4+2 } /** * 确保获取正常的位置数据 * @param dout 位置数据结果 * @param pairCnt 获取几组位置数据 * @return 数组长度 */ static int getAndCheckLoc(uint8_t *dout, size_t pairCnt) { uint8_t loc[200]; size_t cnt = getLoc(loc,pairCnt); if (!cnt) { return 0; // LOG_W("位置信息还未准备好。"); } // else { // memcpy(dout,loc,cnt); // } // return cnt; // while (!cnt) // { // return 0; // static uint8_t i=0; // LOG_W("位置信息还未准备好。"); // rt_thread_mdelay(4000);//状态数据默认3s更新一次 // cnt = getLoc(loc,pairCnt); // if (i++ > 20) { // LOG_E("位置信息获取异常"); //// break; // memset(dout,0x37,pairCnt*10); // return pairCnt*10; // } // } memcpy(dout,loc,cnt); return cnt; } /** * c回应深度数据,含位置信息 * @param din * @param len * @return */ int reportDepth(uint8_t *din,size_t len) { // 000000000000000000005AA53E320608000A170407101B33FFFFFFFF24ED LOG_I("FUNC = reportDepth"); //获取并更新位置信息 uint8_t dout[200]; memset(dout,0,200); size_t nlen = getAndCheckLoc(dout, 1); rt_memcpy(dout+10, din, len);//5aa5从第11字节开始,共len个字节 LOG_HEX("depth",16,dout,len+10); //加密。因加密后数据长度会变化,故不能只加密位置数据。 nlen = cryptSingleMsg(din, len+10, dout); LOG_HEX("crypt",16,dout,nlen); // LOG_D("位置数据加密完成"); upSend(dout, nlen); } //天通指令类型:0x70 0x01 /** * 从数组更新系统各项工作参数 * @param cfg 参数数组 * @param len 数组长度 */ void updateAllSysCfg(uint8_t *cfg, size_t len) { LOG_I("FUNC = update SYSCFG"); SYS_CFG ts; memcpy(&scfg,cfg,len); //详见SYS_CFG的定义 /** * uint8_t sendInterval; uint8_t maxTTWaitTime; uint8_t maxTTRetryCnt; uint8_t minTTPeriCnt; uint8_t minTTsinal; uint8_t timeout; uint8_t openWindowTime[8]; uint16_t maxSizePerFile; uint8_t commMode; uint8_t selfDesSW; uint8_t locAlert; uint8_t locRepInterval; */ set_cfg("sendInterval",scfg.sendInterval); set_cfg("maxTTWaitTime",scfg.maxTTWaitTime); set_cfg("maxTTRetryCnt",scfg.maxTTRetryCnt); set_cfg("minTTPeriCnt",scfg.minTTPeriCnt); set_cfg("minTTsinal",scfg.minTTsinal); set_cfg("timeout",scfg.timeout); char tmpstr[20]; bytes2str(scfg.openWindowTime, 8, 10, ",", tmpstr); set_cfgs("openWindowTime",tmpstr); set_cfg("maxSizePerFile", scfg.maxSizePerFile); set_cfg("commMode",scfg.commMode); set_cfg("selfDesSW",scfg.selfDesSW); set_cfg("locAlert",scfg.locAlert); set_cfg("locRepInterval",scfg.locRepInterval); } /** * 上报系统各项工作参数 */ void reportSysCfg() { LOG_I("FUNC = report SYSCFG"); // scfg.commMode=1; // scfg.timeout=5; // scfg.maxSizePerFile=1024; // uint8_t w[]={0x01,2,3,4,5,6,7,8}; // memcpy(scfg.openWindowTime,w,8); uint8_t size = sizeof(scfg); uint8_t rst[200] = { 0x5A, 0xA5, ADDR_ANJI, ADDR_TT, _CFG_UPDATE_CFG >> 8, _CFG_UPDATE_CFG & 0xff }; int p = 6; rst[p++] = 0; rst[p++] = size; memcpy(rst + p,&scfg, size); p+=size; rst[p] = bccCRC(rst + 2, p-2); rst[++p] = 0xED; size = p+1; LOG_HEX("cfg",16,rst,size); upSend(rst, size); // LOG_HEX("scfg",16,&scfg,size); // LOG_HEX("cfg",16,rst,p+1); // updateAllSysCfg(rst+8, rst[7]); } //3.2.8定时自报位置信息 //每小时传数据时同步传位置信息 //与告警信息共用编码方式 //首字节为是否越界,默认为00,此处为预留。 //单次最多可上报15条位置信息 /** * 按15条经纬度打包位置信息。由[是否越界]+[时戳]+[15组数据]组成。 * 是否越界为预留,不在此判断。 * 15组数据采集顺序为MOB(most oldest bit),越新的数据离时戳越近 * 由于加密后数据最大会增加16字节,后位置数据条数改为10 * @param dout 存储位置信息的数组 * @return 数组大小 */ #define _LOC_CNT 10 static int packLocMsg(uint8_t *dout) { uint8_t alertMsg[200] = { 0x5A, 0xA5, ADDR_ANJI, ADDR_TT, _CFG_LOCATION_ALERT >> 8, _CFG_LOCATION_ALERT & 0xFF, 0, 0 }; alertMsg[8] = 0; //首字节00为定时发送,未检测围栏;其它为在围栏外 int len = time2Byte(alertMsg+9);//添加时间戳 len += getAndCheckLoc(alertMsg + 9 + len, _LOC_CNT) + 1; //p指向第2个数据 if (len == 0x08) {//无有效位置数据 return 0; } alertMsg[7] = len; //update len of raw data alertMsg[8 + len] = bccCRC(alertMsg + 2, 8 + len - 2); //update CRC alertMsg[8 + len + 1] = 0xED; len = 8 + len + 2; LOG_HEX("packLocMsg", 16, alertMsg, len); //bccCRC+Tail memcpy(dout, alertMsg, len); return len; } static void d_packLocMsg(void) { uint8_t tmp[200]; size_t len = packLocMsg(tmp); // len = cryptSingleMsg(tmp, len, tmp); // LOG_HEX("c",16,tmp,len); upSend(tmp, len); } ///** // * 加密位置信息。对位置信息的更改需在加密前操作 // * @param din 存储待加密位置信息的数组 // * @param len 带加密长度 // * @param dout 存储加密结果的数组 // * @return 加密后的长度 // */ //int cryptSingleMsg(uint8_t *din, size_t len, uint8_t *dout) //{ //加密。因加密后数据长度会变化,故不能只加密位置数据。 // uint8_t cd[200]={0xAB,0xBA}; // size_t nlen = 0; // ////#define FULL_DATA_CRYPT //#ifdef FULL_DATA_CRYPT // nlen = cryp_data(din, len, cd); //#else // //单独加密时在加密后数据头部添加0xABBA便于识别、解析 // nlen = cryp_data(din, len, cd + 2) + 2; //#endif // memcpy(dout, cd, nlen); // return nlen; //} //static uint8_t locMsg[200]; //static rt_timer_t repLoc; //static rt_sem_t isReadyToSendLoc; int isInFence(uint8_t *loc); /** * 打包并检测位置数据 默认10s更新一次 * @return */ static void packAndSendLoc_thread_entry(void *parameter) { while (isTCPok()) { static size_t i = 0; int isReadyToSendLoc=0; uint8_t rst[200]; int len = packLocMsg(rst); if (!len) {//无有效位置数据 return; continue; } // LOG_D("len=%d", len); // len = cryptSingleMsg(rst, len, rst); // LOG_D("len=%d", len); // LOG_HEX("crypt", 16, rst, len); //检测是否在围栏内 if (!isInFence(rst+len-11))//不在围栏内 { rst[8]=1; isReadyToSendLoc = 1; } rt_thread_mdelay(10000); //默认10s刷新一次 if (i++ > scfg.locRepInterval * 60)//定时发送,默认5分钟 { i=0; isReadyToSendLoc = 1; } if (isReadyToSendLoc) { upSend(rst, len); } } } //创建定时任务,默认时间5分钟,采集发送 /** * 上报时间到或位置异常时上报 */ //static void upSendLoc_thread_entry(void *parameter) //{ // while(1) // { // if (rt_sem_take(isReadyToSendLoc, RT_WAITING_FOREVER)) { // uint8_t rst[200]; // size_t len = packAndChkLoc(rst); // upSend(rst, len); // } // } //} //cb_upSendLoc() //{ // if (isTCPok()) { //// packAndSendLoc(); // rt_sem_release(isReadyToSendLoc); // } // else { // rt_timer_delete(repLoc); // } //} void upSendLoc() { // isReadyToSendLoc = rt_sem_create("SendLoc", 0, RT_IPC_FLAG_PRIO); // repLoc = rt_timer_create("repLoc", cb_upSendLoc, // RT_NULL, 10, // RT_TIMER_FLAG_PERIODIC); /* 创建 serial 线程 */ // rt_thread_t thread = rt_thread_create("sendLoc", upSendLoc_thread_entry, RT_NULL, 1024 * 1, 27, 10); // /* 创建成功则启动线程 */ // if (thread != RT_NULL) // { // rt_thread_startup(thread); // } // else // { // LOG_E("thread 'sendLoc' create failure."); // return; // } rt_thread_t thread = rt_thread_create("PCLoc", packAndSendLoc_thread_entry, RT_NULL, 1024 * 2, 27, 10); /* 创建成功则启动线程 */ if (thread != RT_NULL) { rt_thread_startup(thread); } else { LOG_E("thread 'PCLoc' create failure."); return; } } //3.2.9深度异常告警 /** * 将位置信息合入深度信息 * @param din * @param len * @param dout * @return */ int depthAlert(uint8_t *din, int len) { //获取并更新位置信息 // uint8_t loc[10]; // uint8_t dout[200]; // size_t nlen = getAndCheckLoc(dout, 1); // // rt_memcpy(dout+10, din, len);//5aa5从第11字节开始,共len个字节 // LOG_HEX("depth",16,dout,len+10); // //加密。因加密后数据长度会变化,故不能只加密位置数据。 // nlen = cryptSingleMsg(din, len+10, dout); // LOG_HEX("crypt",16,dout,nlen); // LOG_D("位置数据加密完成"); // upSend(dout, nlen); } //3.2.10位置异常告警 //map.c中实现 /** * 判断是否在电子围栏内部 * @param x 当前位置经度 * @param y 当前位置纬度 * @return 在内部则返回true,反之false */ static int isInPolgon(float x, float y) { float polyX[10]={},polyY[10]={}; int polyCorners = mapParse("/map.geojson",polyX,polyY); return pointInPolygon(polyCorners,polyX,polyY,x,y); } /** * 设置位置告警功能开关 * @param setON 1-告警功能开启,0-关闭 */ void setLocationAlertSWT(int setON) { } /** * 判断给定位置是否在电子围栏内部 * @param loc 经纬度数组,8个字节 * @return 1- */ int isInFence(uint8_t *loc) { // LOG_HEX("x=",16,loc,8); float x,y =0; x=*((float *)loc); y=*((float *)(loc+4)); // LOG_D("%lf-%lf", x, y); int isIN = isInPolgon(x, y); if (isIN) { LOG_I("设备在围栏内,位置正常。"); return 1; } else { LOG_W("警告!设备不在围栏内!"); return 0; } } void d_isInFence(void) { uint8_t rst[200]; int len = packLocMsg(rst); if (!len) { //无有效位置数据 LOG_W("无有效位置数据"); return; } //检测是否在围栏内 isInFence(rst + len - 11); //不在围栏内,1+4+4+1+bcc+0x4D } //创建定时任务,默认时间10秒钟,异常则发送 //3.2.12数据存储区清空 // /** * 采用格式化命令对存储区进行清空,谨慎使用! */ void clearAllData() { mkfs("elm","sd0");//format SD mkfs("elm", "W25Q128");//format flash } /** * @brief TT根据下发的指令执行对应的功能 * * @param din 待执行的指令数据 * @param len 数据长度,单位字节 */ void ttRunCMD(uint8_t *din, size_t len) { /** +--------+--------+------------+------------+---------+--------+------+------+-------+-----+-------+---------+------+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | ... | 8+N | 9+N | 10+N | +--------+--------+------------+------------+---------+--------+------+------+-------+-----+-------+---------+------+ | Header | Header | targetAddr | sourceAddr | mainCMD | subCMD | lenH | lenL | data1 | ... | dataN | BbccCRC | tail | +--------+--------+------------+------------+---------+--------+------+------+-------+-----+-------+---------+------+ */ int cmd = (din[4] << 8) + din[5]; // int para[] switch (cmd) { case _CFG_COMM_MODE: setCommMode(din[8]); break; case _CMD_SELF_TEST: selfTest(); break; case _CFG_SELF_DESTRUCT: setSelfDestructSWT(din[8]); break; case _CMD_SELF_DESTRUCT: selfDestruct(); break; case _CFG_COMM_WINDOW: setCommWindow(din+8,din[7]);//只支持两组开窗,开窗时间为小时、分钟,UTC+0 char tmpstr[20]; bytes2str(din+8,din[7], 10, ",", tmpstr); set_cfgs("openWindowTime",tmpstr); break; case _CMD_OPEN_WINDOW: openWindow(din[7]?((din[8] << 8) + din[9]):0);//两字节开窗时间 break; case _CMD_CLOSE_WINDOW: closeWindow(); break; case _CFG_LOCATION_ALERT: setLocationAlertSWT(din[8]); break; case _CMD_CLEAR_DATA: clearAllData(); break; case _CFG_UPDATE_CFG: if (din[7]) {//数据长度不为0则为覆写参数 updateAllSysCfg(din+8, din[7]); } else {//数据长度为0则为请求参数 reportSysCfg(); } break; default: LOG_W("0x%04X=未支持的指令。",cmd); break; } } RT_WEAK int formatAndSendTo3S(uint8_t * din, size_t len) { // LOG_D("直接调用小彭的函数进行处理。"); LOG_I("FUNC = forward to 3S"); char str[200] = "RCV:"; int cmd = (din[4] << 8) + din[5]; switch (cmd) { case _CMD_RTC_REQUEST: //不加前后缀的指令,仅有少数 bytes2str(din, len, 16, "", str); break; default: //默认加前后缀 bytes2str(din, len, 16, "", str + 4); strcat(str, "\r\n"); break; } LOG_D("send '%s' to 3S.", str); sendTo3S(str, strlen(str));//作为字符串发送 return 0; } int chk3SDataValid(uint8_t *din, size_t count) { uint8_t head[]={0x5A,0xA5}; //header[4] addr[2]  func[2]  len[2]  data[N]  fcrc[1] tail[1] if (memcmp(din,head,sizeof(head))) { LOG_W("0x5AA5[√] != 0x%02X%02X[×],帧头不匹配",din[0],din[1]); return -RT_ERROR; } if (din[count-2] != bccCRC(din+2,count-2-2))//校验位为1个字节,采用异或运算,从指令的第3个字节开始,到奇偶校验位的前一个字节结束 { LOG_W("0x%02X[√] != 0x%02X[×],校验值不匹配", bccCRC(din+2,count-2-2),din[count-2] ); return -RT_ERROR; } if (din[count-1] != 0xED) { LOG_W("0xED[√] != 0x%02X[×],帧尾不匹配",din[count-1]); return -RT_ERROR; } // LOG_D("valid data."); return RT_EOK; } //原计划将指令粗解析放在上位机,考虑到上位机到位时间晚,现放到MCU端 /** * @brief 校验、解析3S数据,以0x5AA5开头0xED结尾。 * * @param din 待解析数据 * @param count 待解析数据长度 */ void parse3SData(uint8_t *din, size_t count) { /** +--------+--------+------------+------------+---------+--------+------+------+-------+-----+-------+---------+------+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | ... | 8+N | 9+N | 10+N | +--------+--------+------------+------------+---------+--------+------+------+-------+-----+-------+---------+------+ | Header | Header | targetAddr | sourceAddr | mainCMD | subCMD | lenH | lenL | data1 | ... | dataN | BbccCRC | tail | +--------+--------+------------+------------+---------+--------+------+------+-------+-----+-------+---------+------+ */ if (chk3SDataValid(din, count) != RT_EOK) { return; } // uint8_t dout[200]; // 未采用switch case if (din[2] == ADDR_TT)//仅给TT的消息 { ttRunCMD(din,count); } else if (din[2] == ADDR_3S)//给3S的指令,需要再加工,返回数据可能也需要再加工 { // formatAndSendTo3S(din,count); } else if (din[2] == ADDR_ANJI) { //可能需要对回传信息再加工,如查询深度需要加入位置坐标 //或是缓存任务数据 //故需要对数据进行简单判断 int cmd = (din[4] << 8) + din[5]; switch (cmd) { case _CMD_DEPTH_REQUEST: reportDepth(din, count); break; default: LOG_I("pass-through raw data."); if (isTTon()) {//如果TT为开机状态(具备通信状态?待商榷)则不缓存直接发 upSend(din, count); } else { cacheDataToFile(din, count); } } // upSend(din, count); } else { LOG_W("无法解析的目标地址"); } } void chkACK(uint8_t *msg, size_t size) { rt_uint8_t ackgood[] = { 0x88, 0xAA, 0xBB, 0x88, 0x41, 0x43, 0x4B }; //前四字节=帧头、后三字节=ACK if (rt_memcmp(msg, ackgood, 4) == 0 && rt_memcmp(msg + size - 3, ackgood + 4, 3) == 0) { LOG_I("data is ACK."); } else { LOG_W("INVALID DATA."); } } /** * @brief 解析TT数据,TT收到的指令必是单指令,解析容易。 * * @param din * @param len */ void parseTTData(uint8_t *din, size_t len) { /** * +---------------------+-----------+-----------+-----------+-----------+---------+---------------------+------------+ * | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | * +---------------------+-----------+-----------+-----------+-----------+---------+---------------------+------------+ * | fstart[4] | fnum[2] | bak[2] | ftype[2] | fdlen[2] | fcrc[1] | ftccid[4] | rawData[N] | * | 0x88,0xAA,0xBB,0x88 | 0x00,0x01 | 0x00,0x00 | 0x70,0x21 | 0x00,0xAA | 0x00 | 0x27,0x22,0x22,0x22 | | * +---------------------+-----------+-----------+-----------+-----------+---------+---------------------+------------+ */ uint8_t head[]={0x88,0xAA,0xBB,0x88, 0xFF,0xFF, 0x00,0x00, 0x70,0x21, 0x00,0xaa, 0x00, 0x27,0x22,0x22,0x22 }; //fstart[4] fnum[2]  bak[2]  ftype[2]  fdlen[2]  fcrc[1] ftccid[4] uint8_t index[10]; size_t n=isInByte(din, len, head, 4, index);//仅判断帧头,因ACK不一样 uint8_t ndin[200]; if (!n) { LOG_W("TTData中无匹配帧"); return; } for (size_t i = 0; i < n; i++) { //按帧头分割 int cnt=(i+11) { LOG_HEX("frame",16,ndin,cnt); } //判断是否为ACK if ((ndin[10]<<8) | ndin[11] == 0x03) {//数据长度只有3 chkACK(ndin, cnt); } //数据 else { resetTM();//非ACK信号才能触发延时逻辑 uint8_t rst = memcmp(ndin,head,10);//只比较到ftype if (rst) { LOG_W("帧头不匹配"); return; } uint8_t id[30]=""; LOG_I("data info: id=\"%s\", cur/all=[%d/%d]",bytes2str(ndin+17,7,10,"_",id),ndin[25],ndin[26]); if (ndin[24] >> 7) // fcfg=数据类型。解析TT收到的数据时仅需解析“命令”,“数据”传输是单向的。 { LOG_W("浮标端仅接受指令,暂不支持数据。"); return; } uint8_t rawData[200]; uint8_t rawDataLen=cnt-27; memcpy(rawData, ndin + 27, rawDataLen); parse3SData(rawData,rawDataLen); } } } /** * get file size in byte * @param file file name whith fullpath */ unsigned long getFileSize(char *file) { struct stat stat; char *fullpath, *path; unsigned long rst=0; #ifdef DFS_USING_WORKDIR /* open current working directory */ path = rt_strdup("/"); #endif /* build full path for each file */ fullpath = dfs_normalize_path(path, file); rt_memset(&stat, 0, sizeof(struct stat)); if (dfs_file_stat(fullpath, &stat) == 0) { rst = stat.st_size; } rt_free(fullpath); rt_free(path); return rst; } void d_getFileSize(int argc, char ** argv) { // char *f; uint16_t size = (uint16_t)getFileSize(argv[1]); LOG_D("size = %d Byte.",size); } extern SYS_CFG scfg; static uint8_t iscdlock=0; /** * 缓存任务数据 * @param din 单次收到的任务数据 * @param len 任务数据长度 * @return 0--正常,-1--异常 */ int cacheDataToFile(uint8_t *din, size_t len) { LOG_I("FUNC = cache to upsend"); // static uint16_t cnt = 0; // LOG_D("already cached %d bytes.", cnt); // char rootDir[22] = "/sd/rxdata/"; // mkdir(rootDir, 0); // // strcat(rootDir, "2023_07_19/"); // // //name of cachefile // static char f[60] = ""; // // if (cnt == 0 || cnt > 1024) // { // if (cnt) { // //加入待发列表 // postFileInfo(f,0); // } // //广播待发送文件 // cnt = 0; // //更新时戳 // char ts[30] = ""; // time2Str(ts); // //更新文件夹 // strncpy(rootDir + strlen(rootDir) - 9, ts, 8); // mkdir(rootDir, 0); // // //更新文件名 // f[0] = '\0'; // strcat(f, rootDir); //// strcpy(f,rootDir); // strcat(f, "23_07_19_16_38_36_36.bin"); //// LOG_D("need to creat new file"); // strncpy(f + strlen(rootDir), ts, strlen(ts)); // } // LOG_D("f=%s", f); char f[60]=""; getLstCacheFileName(f); //lock file if (iscdlock) { rt_thread_mdelay(1000); } iscdlock=1; int fd = open(f, O_WRONLY | O_CREAT | O_APPEND); if (fd < 0) { LOG_E("open file %s failed!", f); return -RT_ERROR; } else { int rst = write(fd, din, len); if (rst != len) { LOG_E("write to file %s failed!", f); close(fd); iscdlock=0; return -RT_ERROR; } // cnt += len; close(fd); unsigned long size = getFileSize(f); LOG_I("cached %d bytes data to '%s', new size is %ld bytes.",len,f,size); if (size > scfg.maxSizePerFile) { postFileInfo(f,0);//加入待发列表 updateCacheFileName();//更新文件名 } } iscdlock=0; return RT_EOK; } void d_cacheData() { uint8_t demo[200]; size_t len = sizeof(demo); memset(demo,0xAB,len); cacheDataToFile(demo, len); } void parseRS232(uint8_t *din, size_t len) { //有HEX有ASCII,统一按HEX解析 //部分数据以10字节(20字符)的0数据开始,如深度查询,心跳包等 //处理思路是先不管前导0,对应指令手动加前导数据 // uint8_t asciiHead[]={0x41, 0x54, 0x2B, 0x53, 0x4E, 0x44,};//"AT+SND" uint8_t hexHead[]={0x5a, 0xa5};//"5AA5" //由于帧头有多种,且ascii和hex混发,无法处理粘包 // LOG_D("data is %s",isDataASCII(din, len)?"ASCII":"BIN"); //check // if (memcmp(din,asciiHead,sizeof(asciiHead)) == 0) { if (isDataASCII(din, len)) { //ascii LOG_I("type = ASCII"); trDataTolog(din, len, 0); uint8_t index[10]; uint8_t tmpHead[]={0x35, 0x41, 0x41, 0x35}; size_t n=isInByte(din, len, tmpHead, sizeof(tmpHead), index); // LOG_D("n=%d",n); if (!n) { LOG_W("RS232中无匹配帧"); return; } for (size_t i = 0; i < n; i++) { //按帧头分割 uint8_t ndin[400]; int cnt=(i+1 //#include #include #include /** * check eth * @return 1-UP,0-DOWN */ int isEthUP() { //netdev.c extern struct netdev *netdev_default; int rst = (netdev_default->flags & 0x01U) ? 1:0; //0x01 = NETDEV_FLAG_UP // LOG_D("eth is %s.",rst?"up":"down"); // rst &= (netdev_default->flags & 0x04U) ? 1:0;//0x04 NETDEV_FLAG_LINK_UP return rst; } #define FUNC_DEMO #ifdef FUNC_DEMO //测试时导出命令到控制台 MSH_CMD_EXPORT(d_getFreeSpace,getFreeSpace); MSH_CMD_EXPORT(selfTest,sysSelfTest); MSH_CMD_EXPORT(d_packLocMsg,dpackLocMsg); MSH_CMD_EXPORT(upSendLoc,采集并发送位置信息); MSH_CMD_EXPORT(d_cacheData,d_cacheData); MSH_CMD_EXPORT(d_getFileSize,d_getFileSize); MSH_CMD_EXPORT(isEthUP,isEthUP); MSH_CMD_EXPORT(reportSysCfg,reportSysCfg); MSH_CMD_EXPORT(d_isInFence,d_isInFence); MSH_CMD_EXPORT(d_sw,msw); #endif