at_module.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457
  1. /********************************************************************************
  2. * @file AT_Module.c
  3. * @author 度云未来 DOIOT
  4. * @brief 针对公司模块AT指令的处理库
  5. ******************************************************************************
  6. * @attention
  7. * <h2><center>&copy; Copyright (c) 2019 成都度云未来
  8. * All rights reserved.</center></h2>
  9. * 注意:本程序针对cubemx 生成的hal库程序快速开发
  10. * 环境:keil5.0
  11. * STM32Cube FW_F1 V1.8.0
  12. * 文件版本:1.3 更新时间2020/10/15 降低了文件之间的耦合度
  13. ********************************************************************************/
  14. #include "includes.h"
  15. #include "trace.h"
  16. #include "at_module.h"
  17. #include "stdarg.h"
  18. #include "string.h"
  19. #include "stdio.h"
  20. #include "stdlib.h"
  21. uint8_t AT_CMD_SEND(AT_CMD task);
  22. extern int uart_blocking_read(char *buffer, u8 uartid, u32 timeout);
  23. char at_rxbuff[1024];
  24. int Module_Blocking_Read(char *buffer, u32 timeout)
  25. {
  26. return uart_blocking_read(buffer, UART_ME3616, timeout);
  27. }
  28. uint8_t AT_TASK_START(AT_CMD task[],uint8_t Size)
  29. {
  30. uint32_t i;
  31. for(i=0;i<Size;i++)
  32. {
  33. AT_INFO("Start Task %d",i+1);
  34. if(AT_CMD_SEND(task[i]))
  35. {
  36. /*校验成功,开始延时*/
  37. HAL_Delay(task[i].delay_time);
  38. }
  39. else
  40. {
  41. /*校验失败,开始处理*/
  42. if(task[i].REF_Count!=0)
  43. {
  44. task[i].REF_Count--;
  45. i--;/*下一次迭代会回到这个地方*/
  46. AT_INFO("开始重发");
  47. }
  48. else/*没有重发次数或者次数已尽*/
  49. {
  50. /*装载跳跃位置*/
  51. if(task[i].jump!=0)
  52. {
  53. i=task[i].jump-2;
  54. AT_INFO("开始跳转");
  55. }
  56. }
  57. }
  58. }
  59. return 1;
  60. }
  61. uint8_t AT_CMD_SEND(AT_CMD task)
  62. {
  63. uint8_t i=0;
  64. rxbuff(buff);
  65. /*先执行前调函数*/
  66. if(task.BeforeCpltCallback!=NULL)
  67. {
  68. task.BeforeCpltCallback();
  69. }
  70. module_printf("%s", task.SEND_CMD);
  71. AT_INFO("SEND:%s",task.SEND_CMD);
  72. for(i=0;i<task.crc_size;i++)
  73. {
  74. Module_Blocking_Read(buff,task.timeout);
  75. AT_INFO("ASK:%s",buff);
  76. if(strstr(buff,task.ASK[i]))
  77. {
  78. }else
  79. {
  80. AT_ERROR("%s",buff);
  81. /*校验失败,执行失败回掉*/
  82. if(task.ErrorCpltCallback!=NULL)
  83. {
  84. task.ErrorCpltCallback(buff);
  85. }
  86. return 0;
  87. }
  88. }
  89. /*校验成功,执行完成回调函数*/
  90. if(task.AfterCpltCallback!=NULL)
  91. {
  92. task.AfterCpltCallback(buff);
  93. }
  94. return 1;
  95. }
  96. /**
  97. * @param sned_cmd : 发送的指令
  98. * @param cmd_arry : 要校验的响应数组
  99. * @param ack_str: 响应的校验的字串
  100. * @param overtime : 校验的数量【如果响应组是3,校验是2,则会读取三次,但只校验到第2组】
  101. */
  102. uint8_t Module_SEND_CRC(char* send_cmd,char *ack_str,uint32_t overtime)
  103. {
  104. //char rxbuf[cmd_length];
  105. module_printf("%s", send_cmd);
  106. if(Module_Read_A_CRC(ack_str,overtime)) return 1;//校验成功;
  107. else return 0;/*阻塞的读取一次响应,超时返回失败*/
  108. }
  109. /**
  110. * @param sned_cmd : 发送的指令
  111. * @param cmd_arry : 要校验的响应数组
  112. * @param arry_size: 响应的校验组数量
  113. * @param crc_size : 校验的数量【如果响应组是3,校验是2,则会读取三次,但只校验到第2组】
  114. */
  115. uint8_t Module_SEND_A_CRC(char* send_cmd,char cmd_arry[][cmd_length],uint8_t arry_size,uint8_t crc_size)
  116. {
  117. char rxbuf[cmd_length];
  118. uint8_t i;
  119. module_printf("%s", send_cmd);
  120. for(i=0;i<arry_size;i++) /*读取完指定次数*/
  121. {
  122. if(Module_Blocking_Read(rxbuf,1000)); /*阻塞的读取一次响应,超时返回失败*/
  123. else return 0;
  124. if(i<crc_size) /*校验指定的次数*/
  125. {
  126. if(strstr(rxbuf,cmd_arry[i]))
  127. {
  128. }else goto cmd_fail;
  129. }
  130. }
  131. return 1;//校验成功
  132. cmd_fail:
  133. {
  134. /*自写错误处理*/
  135. }
  136. return 0;
  137. }
  138. /**
  139. * @param 阻塞读取一次数据并校验
  140. * @param cmd_arry : 要校验的响应数组
  141. * @param overtime : 超时时间
  142. */
  143. uint8_t Module_Read_A_CRC(char cmd_arry[],uint32_t overtime)
  144. {
  145. //rxbuff(at_rxbuff);
  146. memset(at_rxbuff, 0, sizeof(at_rxbuff));
  147. if(Module_Blocking_Read(at_rxbuff,overtime)) /*阻塞的读取一次响应,超时返回失败*/
  148. {
  149. // printf("%s\r\n", at_rxbuff);
  150. if(strstr(at_rxbuff,cmd_arry))
  151. return 1;
  152. else
  153. {
  154. return 0;
  155. }
  156. }
  157. else return 0;
  158. }
  159. uint8_t Module_Read_A_CRC_DA(char cmd_arry[],char head_arry[], uint32_t len, uint32_t overtime)
  160. {
  161. //rxbuff(at_rxbuff);
  162. int i = 0,rcv_size = 0;
  163. memset(at_rxbuff, 0, sizeof(at_rxbuff));
  164. rcv_size = Module_Blocking_Read(at_rxbuff,overtime);
  165. if(rcv_size) /*阻塞的读取一次响应,超时返回失败*/
  166. {
  167. // data_dump("net sendx", (uint8_t *)at_rxbuff, len);
  168. // if(strstr(at_rxbuff+len/2,cmd_arry))
  169. // return 1;
  170. // else
  171. // {
  172. // return 0;
  173. // }
  174. for(i=len/2; i<rcv_size; i++){
  175. if(at_rxbuff[i]==0x0D && at_rxbuff[i+1]==0x0A){
  176. if(strstr(at_rxbuff+i,cmd_arry)){
  177. return 1;
  178. }
  179. else continue;
  180. }
  181. }
  182. return 0;
  183. }
  184. else return 0;
  185. }
  186. /**@brief AT_CMD_Polling:按指定的指令轮询直到响应对应的ACK才返回
  187. * @param send_cmd : 需要轮询的指令
  188. * @param Polling_fre : 轮询间隔
  189. * @param overtime : 超时时间
  190. */
  191. uint8_t AT_CMD_Polling(char *send_cmd,char*ack,uint32_t Polling_delay,uint32_t overtime)
  192. {
  193. uint32_t time_count=0;
  194. while(1)
  195. {
  196. if(time_count%Polling_delay==0)
  197. {
  198. module_printf("%s", send_cmd);
  199. if(Module_Read_A_CRC(ack,100)) return 1;
  200. }
  201. time_count++;
  202. HAL_Delay(1);
  203. if(time_count>overtime) return 0;
  204. }
  205. }
  206. /**@brief AT_GET_data:提取字符串中第Number_of_groups组被特定字符包裹的信息,标志必须是成对闭合的
  207. * @param buff : 待提取的字符串,可以是数组或静态字符串
  208. * @param get_data : 提取后装载的数组,必须是数组,不能是静态字符串因为需要写
  209. * @param sign : 包裹符号,是一个字符
  210. * @param Number_of_groups : 提取的是第几组
  211. * 例:字符串 "AT+MIPOPEN=1,"TCP","122.114.122.174",41838,100,0,1,2,11002"
  212. * 参数Number_of_groups=2提取第二组,装载结果get_data: 122.114.122.174
  213. */
  214. uint8_t AT_GET_Data(char * buff,uint8_t get_data[],char*sign,uint8_t Number_of_groups)
  215. {
  216. uint16_t i=0,count=0;
  217. while(buff[i]!='\0') /*获取分隔符整数*/
  218. {
  219. if(buff[i]==sign[0]) count++;
  220. i++;
  221. }
  222. if(count%2==0&&count>=Number_of_groups*2) /*偶数分隔符数才继续进行*/
  223. {
  224. char chBuffer[512] ; /*按需调整*/
  225. char *pchDilem =sign; /*分隔符*/
  226. char *pchStrTmpIn = NULL; /*剩下的缓存*/
  227. char *pchTmp = NULL; /*提取出的字符*/
  228. i=0;
  229. strcpy(chBuffer,buff);
  230. pchTmp = chBuffer;
  231. while(NULL != ( pchTmp = strtok_r( pchTmp, pchDilem, &pchStrTmpIn) ))
  232. {
  233. // uart1_printf("\n pchTmp[%s] \n",pchTmp);
  234. i++;
  235. if(i==Number_of_groups*2) strcpy((char*)get_data,pchTmp);
  236. pchTmp = NULL;
  237. }
  238. return 1;
  239. }
  240. else return 0; /*非偶数分隔符直接结束*/
  241. }
  242. /**@brief AT_GET_Data_2Param :提取被头尾字符串包裹的字串
  243. * @param buff : 待提取的字符串,可以是数组或静态字符串
  244. * @param get_data : 提取后装载的数组,必须是数组,不能是静态字符串因为需要写
  245. * @param head : 头字符串【任意长度符串】
  246. * @param tail : 尾字符串【任意长度字符串】
  247. * 例:字符串 "+QGPSGNMEA: $GPRMC,021229.00,A,3041.560083,N,10405.832869,E,0.0,247.3,040920,1.9,W,A*28"
  248. * 参数head ",A",tail ",N",装载结果get_data: "3041.560083"
  249. */
  250. uint8_t AT_GET_Data_2Param(char * buff,uint8_t get_data[],char*head,char*tail)
  251. {
  252. if(strstr(buff,head)&&strstr(buff,tail))
  253. {
  254. uint16_t i;
  255. uint16_t len1,len2;
  256. char *t = NULL;
  257. char *h=strstr(buff,head);
  258. len1=strlen(h); /*剩余长度*/
  259. len2=strlen(head); /*头子串长度*/
  260. t=strstr(h+len2,tail);
  261. if(len1>len2)
  262. {
  263. for(i=0;h+i+len2!=t;i++)
  264. {
  265. get_data[i]=h[i+len2];
  266. }
  267. get_data[i]='\0';//加上结束符
  268. }else
  269. {
  270. debug_printf("\r\n字串长度错误\r\n");
  271. return 0;
  272. }
  273. }else
  274. {
  275. if(!strstr(buff,head))debug_printf("\r\n未能检索到头字串\r\n");
  276. if(!strstr(buff,tail))debug_printf("\r\n未能检索到尾字串\r\n");
  277. debug_printf("\r\n请检查你的数据原文:\r\n");
  278. debug_printf("%s", buff);
  279. return 0;
  280. }
  281. return 1;
  282. }
  283. /**@brief AT_Data_Replace :提取被头尾字符包裹的字串
  284. * @param buff : 待提取的字符串,必须是实体数组,需要写入
  285. * @param str1 : 被替换的子串,可以是数组和静态字符串
  286. * @param str2 : 替换成的子串,可以是数组和静态字符串
  287. * 例:字符串 AT+QMTCFG="aliauth",0,"a1XVq3z4DsR"
  288. * 参数str1 "aliauth",str2 "recv/mode",替换结果buff: AT+QMTCFG="recv/mode",0,"a1XVq3z4DsR"
  289. */
  290. uint8_t AT_Data_Replace(char buff[],char *str1, char *str2) {
  291. int i,flag = 0;
  292. char *p,*q,*ts;
  293. for(i = 0; buff[i]; ++i)
  294. {
  295. if(buff[i] == str1[0])
  296. {
  297. p = buff + i;
  298. q = str1;
  299. while(*q && (*p++ == *q++));
  300. if(*q == '\0')
  301. {
  302. ts = (char *)malloc(strlen(buff) + 1);
  303. strcpy(ts,p);
  304. buff[i] = '\0';
  305. strcat(buff,str2);
  306. strcat(buff,ts);
  307. free(ts);
  308. flag = 1;
  309. }
  310. }
  311. }
  312. return flag;
  313. }
  314. /**@brief AT_GET_Data_Param_number :指定的标志位开始,提取后面的数据,直到遇到\r,\n,
  315. * @param buff : 待提取的字符串,可以是数组或静态字符串
  316. * @param get_data : 提取后装载的数组,必须是数组,不能是静态字符串因为需要写
  317. * @param head : 头字符串【任意长度符串】
  318. * 例:字符串 " +ESOERR=1,4 "
  319. * 参数head ",",装载结果get_data: "4"
  320. */
  321. uint8_t AT_GET_Data_Param(char * buff,uint8_t get_data[],char*head)
  322. {
  323. char *h=strstr(buff,head);
  324. if(h!=NULL)
  325. {
  326. uint16_t i;
  327. int len=strlen(head);
  328. for(i=0;h[i+len]!='\r'&&h[i+len]!='\n'&&h[i+len]!=' ';i++)
  329. {
  330. get_data[i]=h[i+len];
  331. }
  332. get_data[i]='\0';//加上结束符
  333. }else
  334. {
  335. if(!strstr(buff,head))debug_printf("\r\n未能检索到标志位\r\n");
  336. debug_printf("\r\n请检查你的数据原文:\r\n");
  337. debug_printf("%s", buff);
  338. return 0;
  339. }
  340. return 1;
  341. }
  342. /**@brief AT_Extract_numbers :导入的数组进行提纯,去掉非数字的部分
  343. * @param buff : 待提纯的字符串,必须是实体数组,需要写入
  344. */
  345. uint8_t AT_Extract_numbers(char buff[]) {
  346. uint16_t i=0,n=0;
  347. int len=strlen(buff);
  348. char fifo[32];/*浪费空间,但是效率更高*/
  349. memset(fifo,0, sizeof(fifo));
  350. for(;buff[i]!='\0';)
  351. {
  352. if(buff[i]>='0'&&buff[i]<='9')
  353. {
  354. fifo[n]=buff[i];
  355. n++;
  356. i++;
  357. }
  358. else
  359. {
  360. i++;
  361. }
  362. }
  363. strcpy(buff,fifo);
  364. return 1;
  365. }
  366. /**@brief AT_Extract_numbers :导入的数组进行提纯,去掉非数字和非字母的部分
  367. * @param buff : 待提纯的字符串,必须是实体数组,需要写入
  368. */
  369. uint8_t AT_Extract_numbers_letters(char buff[]) {
  370. uint16_t i=0,n=0;
  371. int len=strlen(buff);
  372. char fifo[256];/*浪费空间,但是效率更高*/
  373. memset(fifo,0,len+1);
  374. for(;buff[i]!='\0';)
  375. {
  376. if((buff[i]>='0'&&buff[i]<='9')||(buff[i]>='A'&&buff[i]<='Z')||(buff[i]>='a'&&buff[i]<='z'))
  377. {
  378. fifo[n]=buff[i];
  379. n++;
  380. i++;
  381. }
  382. else
  383. {
  384. i++;
  385. }
  386. }
  387. //debug_printf("\r\n");
  388. //debug_printf(fifo);
  389. strcpy(buff,fifo);
  390. return 1;
  391. }
  392. /**@brief AT_Remove_spaces :去掉空字符,\r,\n,space
  393. * @param buff : 待提纯的字符串,必须是实体数组,需要写入
  394. */
  395. uint8_t AT_Remove_spaces(char buff[]) {
  396. uint16_t i=0,n=0;
  397. int len=strlen(buff);
  398. char fifo[256];/*浪费空间,但是效率更高*/
  399. memset(fifo,0,len+1);
  400. for(;buff[i]!='\0';)
  401. {
  402. if((buff[i]!='\r')&&(buff[i]!='\n')&&(buff[i]!=' '))
  403. {
  404. fifo[n]=buff[i];
  405. n++;
  406. i++;
  407. }
  408. else
  409. {
  410. i++;
  411. }
  412. }
  413. //debug_printf("\r\n");
  414. //debug_printf(fifo);
  415. strcpy(buff,fifo);
  416. return 1;
  417. }