BC260Y_UDP_Client5.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582
  1. #include "CONFIG.h"
  2. #if BC260Y
  3. #include "stm32f10x.h"
  4. #include <stdio.h>
  5. #include <stdarg.h>
  6. #include <string.h>
  7. #include <stdlib.h>
  8. #include "UDP_Client.h"
  9. #include "Timer_Module.h"
  10. #include "Log_Module.h"
  11. #include "mbedtls_util.h"
  12. #include "cJSON.h"
  13. #include "bc260y.h"
  14. #include "Pump_Dicts_Util.h"
  15. #include "Regist.h"
  16. #include "PumpBusiness.h"
  17. #include "Initialize.h"
  18. #include "Tuoreniot.h"
  19. // 测试模式
  20. uint8_t UDPCLIENT_TestModeFlag = 0;
  21. // socket ID
  22. static uint8_t connectID = 1;
  23. // coap报文
  24. static uint8_t coap_message[512];
  25. static uint16_t coap_message_length = 0;
  26. // 待发送数据的地址和长度
  27. static uint8_t data[128];
  28. static uint16_t data_length;
  29. // 信号值
  30. static struct Signal
  31. {
  32. int RSRP;
  33. int RSRQ;
  34. int RSSI;
  35. int SINR;
  36. } signal = {
  37. .RSRP = 99,
  38. .RSRQ = 99,
  39. .RSSI = 99,
  40. .SINR = 99,
  41. };
  42. // 查询信号质量
  43. void UDPCLIENT_QuerySignal(int * RSRP, int * RSRQ, int * RSSI, int * SINR)
  44. {
  45. *RSRP = signal.RSRP;
  46. *RSRQ = signal.RSRQ;
  47. *RSSI = signal.RSSI;
  48. *SINR = signal.SINR;
  49. }
  50. // 函数声明
  51. static void UDPCLIENT_Process(void); // 流程
  52. static void UDPCLIENT_SetStatus(enum UDPCLIENT_StatusEnum status); // 设置状态
  53. static void UDPCLIENT_SetResult(enum UDPCLIENT_ResultEnum result); // 设置结果
  54. static void goto_start(void); // 开始
  55. static void goto_success(void); // 成功
  56. static void goto_failure(char * _info); // 失败
  57. static void goto_finish(void); // 结束
  58. // 流程
  59. enum Step{
  60. STEP_START, // 开始
  61. STEP_EXIT_SLEEP, // 退出睡眠
  62. STEP_ENTER_SLEEP, // 进入睡眠
  63. STEP_WAIT, // 等待
  64. STEP_STATE, // 查询状态
  65. STEP_OPEN, // 打开
  66. STEP_OPEN_WAIT_RESULT, // 等待打开结果
  67. STEP_JUDGE_AUTH_OR_DATA, // 判断认证还是发送
  68. STEP_JOIN_AUTH_MESSAGE, // 拼接认证报文
  69. STEP_JOIN_DATA_MESSAGE, // 拼接数据报文
  70. STEP_QUERY_SLEEP, // 查询休眠
  71. STEP_SET_SLEEP, // 开启休眠
  72. STEP_QUERY_QENG_SERVINGCELL, // 查询信号质量
  73. STEP_SET_QISDE_0, // 关闭发送回显
  74. STEP_SET_QISDE_1,
  75. STEP_SEND, // 发送
  76. STEP_RECV, // 等待发送结果
  77. STEP_QUERY_CFUN,
  78. STEP_SET_CFUN_0, // 设置最小功能模式
  79. STEP_WAIT_SET_CFUN_0, // 等待设置最小功能模式
  80. STEP_SET_CFUN_1, // 设置全功能模式
  81. STEP_WAIT_SET_CFUN_1, // 等待设置全功能模式结果
  82. STEP_SET_CGREG_2, // 设置ps域允许上报网络注册和位置信息
  83. STEP_QUERY_CGREG, // 查询网络注册状态
  84. STEP_DATAFORMAT,//配置发送格式
  85. STEP_CLOSE, // 关闭
  86. STEP_SUCCESS, // 成功
  87. STEP_FAILURE, // 失败
  88. STEP_FINISH, // 流程结束
  89. STEP_VERIFY_REGIST,//验证注册
  90. STEP_VERIFY_INITIALIZE, // 验证初始化
  91. STEP_QUERY_SOCKET, // 查询连接
  92. STEP_QUERY_DNS, // 查询DNS
  93. STEP_SET_DNS, // 设置DNS
  94. STEP_QUERY_CFUN_DNS, // 设置完dns后,查询功能模式
  95. STEP_SET_CFUN_0_DNS, // 设置完dns后,设置最小功能模式
  96. };
  97. // 步骤跳转时的监听
  98. static uint8_t STEP_JOIN_AUTH_MESSAGE_times = 0; // 认证的次数
  99. static uint8_t STEP_SET_CFUN_1_times = 0; // 全功能模式的次数
  100. static void initialize()
  101. {
  102. STEP_JOIN_AUTH_MESSAGE_times = 0;
  103. STEP_SET_CFUN_1_times = 0;
  104. }
  105. static uint16_t goto_step_event(uint16_t ns)
  106. {
  107. // 重置bc260y状态
  108. bc260y.reset();
  109. // 流程控制
  110. uint16_t step = ns;
  111. if(ns == STEP_JOIN_AUTH_MESSAGE && STEP_JOIN_AUTH_MESSAGE_times++ >= 1)
  112. {
  113. goto_failure("认证次数过多");
  114. step = STEP_FAILURE;
  115. }
  116. else if(ns == STEP_SET_CFUN_1 && STEP_SET_CFUN_1_times++ >= 2)
  117. {
  118. goto_failure("重新联网次数过多");
  119. step = STEP_FAILURE;
  120. }
  121. return step;
  122. }
  123. // 流程控制
  124. static struct PCTRL_Struct pctrl = {
  125. .current_step = STEP_FINISH,
  126. .next_step = STEP_FINISH,
  127. .step_wait = STEP_WAIT,
  128. .goto_step_listener = goto_step_event,
  129. };
  130. // 备注
  131. static char info[40];
  132. // 开始
  133. static void goto_start(void)
  134. {
  135. initialize();
  136. PCTRL_GotoStep(&pctrl, STEP_START, "开始");
  137. }
  138. // 成功
  139. static void goto_success(void)
  140. {
  141. UDPCLIENT_SetResult(UDPCLIENT_Result_Success);
  142. PCTRL_GotoStep(&pctrl, STEP_SUCCESS, "成功");
  143. }
  144. // 失败
  145. static void goto_failure(char * _info)
  146. {
  147. UDPCLIENT_SetResult(UDPCLIENT_Result_Failure);
  148. memset(info, 0, sizeof(info));
  149. strcpy(info, _info);
  150. PCTRL_GotoStep(&pctrl, STEP_FAILURE, "失败");
  151. Log_Printf_Debug("info:%s!\r\n", info);
  152. }
  153. // 结束
  154. static void goto_finish(void)
  155. {
  156. UDPCLIENT_SetStatus(UDPCLIENT_Status_Done);
  157. PCTRL_GotoStep(&pctrl, STEP_FINISH, "结束");
  158. }
  159. // 状态
  160. static enum UDPCLIENT_StatusEnum udpclient_status = UDPCLIENT_Status_None;
  161. // 结果
  162. static enum UDPCLIENT_ResultEnum udpclient_result = UDPCLIENT_Result_None;
  163. // 设置状态
  164. static void UDPCLIENT_SetStatus(enum UDPCLIENT_StatusEnum status)
  165. {
  166. udpclient_status = status;
  167. }
  168. // 获取状态
  169. enum UDPCLIENT_StatusEnum UDPCLIENT_GetStatus(void)
  170. {
  171. return udpclient_status;
  172. }
  173. // 设置结果
  174. static void UDPCLIENT_SetResult(enum UDPCLIENT_ResultEnum result)
  175. {
  176. udpclient_result = result;
  177. }
  178. // 获取结果
  179. enum UDPCLIENT_ResultEnum UDPCLIENT_GetResult(void)
  180. {
  181. return udpclient_result;
  182. }
  183. // 重置
  184. enum EXECUTE_ResultEnum UDPCLIENT_Reset(void)
  185. {
  186. if(UDPCLIENT_GetStatus() == UDPCLIENT_Status_Being)
  187. {
  188. Log_Printf_Debug("发送流程重置错误!\r\n");
  189. return EXECUTE_Result_Failure;
  190. }
  191. UDPCLIENT_SetStatus(UDPCLIENT_Status_None);
  192. UDPCLIENT_SetResult(UDPCLIENT_Result_None);
  193. return EXECUTE_Result_Success;
  194. }
  195. // 开始
  196. enum EXECUTE_ResultEnum UDPCLIENT_Start(void)
  197. {
  198. UDPCLIENT_Reset();
  199. if(UDPCLIENT_GetStatus() != UDPCLIENT_Status_None)
  200. {
  201. Log_Printf_Debug("发送流程启动错误!\r\n");
  202. return EXECUTE_Result_Failure;
  203. }
  204. goto_start();
  205. UDPCLIENT_SetStatus(UDPCLIENT_Status_Being);
  206. return EXECUTE_Result_Success;
  207. }
  208. // 流程处理,放到大循环里
  209. void UDPCLIENT_ProcessHandle(void)
  210. {
  211. if(UDPCLIENT_GetStatus() == UDPCLIENT_Status_Being)
  212. {
  213. UDPCLIENT_Process();
  214. }
  215. }
  216. //// 发送数据的逻辑
  217. static enum Result result = Result_None;
  218. static uint8_t cgreg_n;
  219. static uint8_t cgreg_stat;
  220. static uint16_t cgreg_lac;
  221. static uint32_t cgreg_ci;
  222. //static uint8_t query_cgreg_times = 0; // 查询网络状态的次数
  223. static uint16_t socket_err = 0;
  224. // dns
  225. static char pridnsaddr[20];
  226. static uint8_t cfun_mode = 0;
  227. static uint8_t connectstate = 0;
  228. // 开启回显的标志
  229. static void UDPCLIENT_Process(void)
  230. {
  231. // 流程
  232. switch(pctrl.current_step)
  233. {
  234. case STEP_START: // 开始
  235. PCTRL_GotoStep(&pctrl, STEP_EXIT_SLEEP, "退出休眠");
  236. break;
  237. case STEP_EXIT_SLEEP: // 退出休眠
  238. result = bc260y.exit_sleep_2();
  239. if(result == Result_Success)
  240. {
  241. PCTRL_GotoStep(&pctrl, STEP_SET_QISDE_1, "开启回显");
  242. }
  243. else if(result == Result_Failed)
  244. {
  245. goto_failure("退出休眠失败");
  246. }
  247. break;
  248. case STEP_SET_QISDE_1: // 开启回显
  249. result = bc260y.set_qisde(1);
  250. if(result == Result_Success)
  251. {
  252. PCTRL_GotoStep(&pctrl, STEP_SET_SLEEP, "设置休眠模式");
  253. }
  254. else if(result == Result_Failed)
  255. {
  256. goto_failure("开启回显失败");
  257. }
  258. break;
  259. case STEP_SET_SLEEP: // 设置休眠模式
  260. result = bc260y.set_sleep(2);
  261. if(result == Result_Success)
  262. {
  263. PCTRL_GotoStep(&pctrl, STEP_DATAFORMAT, "配置发送格式");
  264. }
  265. else if(result == Result_Failed)
  266. {
  267. goto_failure("设置休眠模式失败");
  268. }
  269. break;
  270. case STEP_DATAFORMAT: // 配置发送格式
  271. result = bc260y.dataformat(1);
  272. if(result == Result_Success)
  273. {
  274. PCTRL_GotoStep(&pctrl, STEP_QUERY_CFUN, "查询功能模式");
  275. }
  276. else if(result == Result_Failed)
  277. {
  278. goto_failure("配置发送格式失败");
  279. }
  280. break;
  281. case STEP_QUERY_CFUN: // 查询功能模式
  282. result = bc260y.query_cfun(&cfun_mode);
  283. if(result == Result_Success)
  284. {
  285. if(cfun_mode == 0)
  286. {
  287. PCTRL_GotoStep(&pctrl, STEP_SET_CFUN_1, "设置全功能模式");
  288. }
  289. else
  290. {
  291. PCTRL_GotoStep(&pctrl, STEP_QUERY_DNS, "查询DNS");
  292. memset(pridnsaddr, 0, sizeof(pridnsaddr));
  293. }
  294. }
  295. else if(result == Result_Failed)
  296. {
  297. goto_failure("查询功能模式失败");
  298. }
  299. break;
  300. case STEP_SET_CFUN_0:
  301. result=bc260y.set_cfun(0);//设置最小功能模式
  302. if(result == Result_Success)
  303. {
  304. if(REGIST_GetResult() != REGIST_Result_None)
  305. {
  306. goto_finish();
  307. }
  308. else
  309. {
  310. PCTRL_GotoStep(&pctrl, STEP_SET_CFUN_1, "设置全功能模式");
  311. }
  312. }
  313. else if(result == Result_Failed)
  314. {
  315. goto_failure("设置最小功能模式失败");
  316. goto_finish();
  317. }
  318. break;
  319. case STEP_SET_CFUN_1: // 设置全功能模式
  320. result = bc260y.set_cfun(1);
  321. if(result == Result_Success)
  322. {
  323. PCTRL_GotoStep(&pctrl, STEP_QUERY_DNS, "查询DNS");
  324. memset(pridnsaddr, 0, sizeof(pridnsaddr));
  325. }
  326. else if(result == Result_Failed)
  327. {
  328. goto_failure("设置全功能模式失败");
  329. }
  330. break;
  331. case STEP_QUERY_DNS: // 查询DNS
  332. result = bc260y.query_dns(pridnsaddr);
  333. if(result == Result_Success)
  334. {
  335. if(strlen(pridnsaddr) <= 1)
  336. {
  337. PCTRL_GotoStep(&pctrl, STEP_SET_DNS, "设置DNS");
  338. }
  339. else
  340. {
  341. PCTRL_GotoStep(&pctrl, STEP_SET_CGREG_2, "设置ps域");
  342. }
  343. }
  344. else if(result == Result_Failed)
  345. {
  346. goto_failure("查询DNS失败");
  347. }
  348. break;
  349. case STEP_SET_DNS:
  350. result = bc260y.set_dns("114.114.114.114", "8.8.8.8"); // 设置DNS
  351. if(result == Result_Success)
  352. {
  353. PCTRL_GotoStep(&pctrl, STEP_SET_CFUN_0, "设置完DNS,设置最小功能模式");
  354. }
  355. else if(result == Result_Failed)
  356. {
  357. goto_failure("设置DNS失败");
  358. }
  359. break;
  360. case STEP_SET_CGREG_2: // 设置ps域
  361. result = bc260y.set_cereg(2);
  362. if(result == Result_Success)
  363. {
  364. PCTRL_GotoStep(&pctrl, STEP_QUERY_CGREG, "查询ps域");
  365. }
  366. else if(result == Result_Failed)
  367. {
  368. goto_failure("设置ps域失败");
  369. }
  370. break;
  371. case STEP_QUERY_CGREG: // 查询ps域
  372. result = bc260y.query_cereg(&cgreg_n, &cgreg_stat, &cgreg_lac, &cgreg_ci);
  373. if(result == Result_Success)
  374. {
  375. if(cgreg_stat == 1)
  376. {
  377. // 下一步
  378. if(UDPCLIENT_TestModeFlag == 0)
  379. {
  380. PCTRL_GotoStep(&pctrl, STEP_QUERY_SOCKET, "查询连接");
  381. }
  382. else
  383. {
  384. PCTRL_GotoStep(&pctrl, STEP_QUERY_QENG_SERVINGCELL, "查询信号质量");
  385. }
  386. }
  387. else
  388. {
  389. goto_failure("网络注册失败");
  390. }
  391. }
  392. else if(result == Result_Failed)
  393. {
  394. goto_failure("查询ps域失败");
  395. }
  396. break;
  397. case STEP_QUERY_QENG_SERVINGCELL: // 查询信号质量
  398. result = bc260y.qeng_servingcell(&signal.RSRP, &signal.RSRQ, &signal.RSSI, &signal.SINR);
  399. if(result == Result_Success)
  400. {
  401. PCTRL_GotoStep(&pctrl, STEP_QUERY_SOCKET, "查询连接");
  402. }
  403. else if(result == Result_Failed)
  404. {
  405. goto_failure("查询信号质量失败");
  406. }
  407. break;
  408. case STEP_QUERY_SOCKET: // 查询连接
  409. result =bc260y.query_socket_state(connectID,&connectstate);
  410. if(result == Result_Success)
  411. {
  412. if(connectstate==2)
  413. {
  414. PCTRL_GotoStep(&pctrl, STEP_JUDGE_AUTH_OR_DATA, "判断认证或发送");
  415. }
  416. else
  417. {
  418. PCTRL_GotoStep(&pctrl, STEP_CLOSE, "关闭连接");
  419. }
  420. }
  421. else if(result == Result_Failed)
  422. {
  423. goto_failure("查询UDP连接失败");
  424. }
  425. break;
  426. case STEP_CLOSE: // 关闭连接
  427. result = bc260y.close_socket(connectID);
  428. if(result == Result_Success)
  429. {
  430. PCTRL_GotoStep(&pctrl, STEP_OPEN, "打开客户端");
  431. }
  432. else if(result == Result_Failed)
  433. {
  434. goto_failure("关闭连接失败");
  435. }
  436. break;
  437. case STEP_OPEN: // 打开客户端
  438. result = bc260y.open_socket(connectID, "UDP", regist_response.remoteAddress, regist_response.remotePort, 1, &socket_err);
  439. if(result == Result_Success)
  440. {
  441. if(socket_err == 0)
  442. {
  443. PCTRL_GotoStep(&pctrl, STEP_JUDGE_AUTH_OR_DATA, "判断认证或发送");
  444. }
  445. else
  446. {
  447. PCTRL_GotoStep(&pctrl, STEP_SET_CFUN_0, "打开客户端失败,设置最小功能模式");
  448. }
  449. }
  450. else if(result == Result_Failed)
  451. {
  452. goto_failure("打开客户端失败");
  453. }
  454. break;
  455. case STEP_JUDGE_AUTH_OR_DATA: // 判断认证或发送
  456. if(TUORENIOT_IsAuthentication() == 1) // 已认证
  457. {
  458. PCTRL_GotoStep(&pctrl, STEP_JOIN_DATA_MESSAGE, "拼接数据报文");
  459. }
  460. else
  461. {
  462. PCTRL_GotoStep(&pctrl, STEP_JOIN_AUTH_MESSAGE, "拼接认证报文");
  463. }
  464. break;
  465. case STEP_JOIN_AUTH_MESSAGE: // 拼接认证报文
  466. memset(coap_message, 0, sizeof(coap_message));
  467. TUORENIOT_PackAuthMessage(coap_message, &coap_message_length);
  468. // Log_Printf_Debug("认证报文:%d\r\n", coap_message_length);
  469. // Log_SendHex(coap_message, coap_message_length);
  470. // Log_Printf_Debug("\r\n");
  471. PCTRL_GotoStep(&pctrl, STEP_SEND, "发送认证报文");
  472. break;
  473. case STEP_JOIN_DATA_MESSAGE: // 拼接数据报文
  474. pump_params.lac = cgreg_lac;
  475. pump_params.ci = cgreg_ci;
  476. PUMPBUSINESS_ParamsRefresh();
  477. // 编码
  478. memset(data, 0, sizeof(data));
  479. PUMPDICTS_ProtocolEncode(&pump_params, data, &data_length);
  480. memset(coap_message, 0, sizeof(coap_message));
  481. TUORENIOT_PackDataMessage(data, data_length, coap_message, &coap_message_length);
  482. // Log_Printf_Debug("数据报文:%d\r\n", coap_message_length);
  483. // Log_SendHex(coap_message, coap_message_length);
  484. // Log_Printf_Debug("\r\n");
  485. PCTRL_GotoStep(&pctrl, STEP_SEND, "发送数据报文");
  486. break;
  487. case STEP_SEND: // 发送send
  488. result = bc260y.send_rai(connectID, coap_message, coap_message_length, 2);
  489. if(result == Result_Success)
  490. {
  491. memset(coap_message, 0, sizeof(coap_message));
  492. PCTRL_GotoStep(&pctrl, STEP_RECV, "等待接收数据");
  493. }
  494. else if(result == Result_Failed)
  495. {
  496. goto_failure("发送send失败");
  497. }
  498. break;
  499. case STEP_RECV: // 等待结果
  500. result = bc260y.recv_with_time(connectID, coap_message, &coap_message_length, 10000);
  501. if(result == Result_Success)
  502. {
  503. uint8_t res = TUORENIOT_AnalysisMessage(coap_message, coap_message_length);
  504. if(res == 0) // 发送失败
  505. {
  506. PCTRL_GotoStep(&pctrl, STEP_JUDGE_AUTH_OR_DATA, "重新认证"); // 重新认证
  507. }
  508. else if(res == 1) // 认证成功
  509. {
  510. PCTRL_GotoStep(&pctrl, STEP_JOIN_DATA_MESSAGE, "拼接数据报文");
  511. }
  512. else if(res == 2) // 发送成功
  513. {
  514. goto_success();
  515. }
  516. }
  517. else if(result == Result_Failed)
  518. {
  519. PCTRL_GotoStep(&pctrl, STEP_SET_CFUN_0, "重新发送,设置最小功能模式");
  520. }
  521. break;
  522. case STEP_WAIT: // 等待
  523. PCTRL_Wait(&pctrl);
  524. break;
  525. case STEP_SUCCESS: // 成功
  526. goto_finish();
  527. break;
  528. case STEP_FAILURE: // 失败
  529. PCTRL_GotoStep(&pctrl, STEP_SET_CFUN_0, "失败,设置最小功能模式");
  530. break;
  531. case STEP_FINISH: // 结束流程
  532. break;
  533. default:
  534. goto_failure("步骤不存在");
  535. break;
  536. }
  537. }
  538. #endif