tcp_server.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434
  1. #include "lwip/opt.h"
  2. #include "lwip/sys.h"
  3. #include "tcp_server.h"
  4. #include <string.h>
  5. #include <stdio.h>
  6. #include "gd32f30x.h"
  7. #include "main.h"
  8. #include "lwip/tcp.h"
  9. #include "lwip/memp.h"
  10. #include "lwip/api.h"
  11. #include "sockets.h"
  12. #include "log.h"
  13. #include "netconf.h"
  14. #include "gateway_message.h"
  15. #include "gd32_flash.h"
  16. #include "stdlib.h"
  17. #include "protocol.h"
  18. #include "delay.h"
  19. #include "sys_mqtt.h"
  20. #include "updata.h"
  21. uint8_t closeFlag = 0; // DHCP关闭标志位
  22. uint8_t tcp_config = 0; // 存储config标志位
  23. uint8_t ProtocolsModeFlag = 1;// 开启内置协议模式
  24. uint8_t TransparentModeFlag = 0; // 表示用串口透传 将数据原封不动的发给客户端
  25. ip_config load_ip_config = {0};
  26. ip_config *get_ip_config()
  27. {
  28. return &load_ip_config;
  29. }
  30. // 解析设备当前的硬件信息(结构体内的数据)
  31. void get_device_params(char* device_params)
  32. {
  33. GATEWAY_PARAMS *get;
  34. get= get_gateway_config_params();
  35. DEVICE_PARAMS *current_device=get->device_params;
  36. sprintf(device_params, "{\"read_config\":\"success\",\"baudrate\":%d,\"checkBit\":%d,\"commandTopic\":\"%s\",\"dataBit\":%d,\
  37. \"deviceId\":\"%s\",\"host\":\"%s\",\"inboundTime\":%d,\"messageTopic\":\"%s\",\"port\":%d,\"stopBit\":%d,\"deviceList\":[",get->baudrate,
  38. get->checkBit,get->commandTopic,get->dataBits,get->deviceId,get->host,get->inboundTime,get->messageTopic,get->port,get->stopBit);
  39. while(current_device != NULL)
  40. {
  41. sprintf(device_params + strlen(device_params),"{\"protocol\":%d,\"bigLittleFormat\":%d,\"deviceId\":\"%s\",\"sensorData\":[",
  42. current_device->protocol,current_device->MDBbigLittleFormat,current_device->deviceID);
  43. GATEWAY_READ_DLT645_COMMAND *read_dlt645_command = current_device->params->gateway_read_dlt645_command;
  44. GATEWAY_READ_MODBUS_COMMAND *read_modbus_command = current_device->params->gateway_read_modbus_command;
  45. GATEWAY_WRITE_MODBUS_COMMAND *write_modbus_command = current_device->params->gateway_write_modbus_command;
  46. // dlt645 read
  47. while(read_dlt645_command != NULL)
  48. {
  49. sprintf(device_params + strlen(device_params),"{\"identifier645\":%d,\"identifier\":\"%s\",\"deviceID645\":\"%s\"},",
  50. read_dlt645_command->Identification,read_dlt645_command->keyword,read_dlt645_command->deviceID645);
  51. read_dlt645_command = read_dlt645_command->nextParams;
  52. }
  53. // modbus read
  54. while(read_modbus_command != NULL)
  55. {
  56. sprintf(device_params + strlen(device_params),"{\"rFunctionCode\":%d,\"registerAddress\":%d,\"slaveAddress\":%d,\"registerByteNum\":%d,\"identifier\":\"%s\",\"precise\":%d},",
  57. read_modbus_command->functionCode, read_modbus_command->registerAddress,read_modbus_command->slaveAddress,
  58. read_modbus_command->registerByteNum,read_modbus_command->keyword,read_modbus_command->decimalPoint);
  59. read_modbus_command = read_modbus_command->nextParams;
  60. }
  61. // modbus write
  62. sprintf(device_params + strlen(device_params)-1,"], \"commandData\":[");//sensorData:[
  63. while(write_modbus_command != NULL)
  64. {
  65. sprintf(device_params + strlen(device_params),"{\"registerAddress\":%d,\"slaveAddress\":%d,\"wFunctionCode\":%d,\"registerByteNum\":%d},",
  66. write_modbus_command->registerAddress,write_modbus_command->slaveAddress,write_modbus_command->functionCode,write_modbus_command->registerByteNum);
  67. write_modbus_command = write_modbus_command->nextParams;
  68. }
  69. sprintf(device_params + strlen(device_params)-1,"]},");// commandData:[
  70. current_device = current_device->nextDevice;
  71. }
  72. sprintf(device_params + strlen(device_params) -1 ,"]}");
  73. }
  74. int siee;
  75. // 储存上位机发送的config数据,并返回上位机操作结果
  76. void save_config(int client_socket,char* buf)
  77. {
  78. GATEWAY_PARAMS *get;
  79. get= get_gateway_config_params();
  80. char* saveData = pvPortMalloc(7* 1024);//(RECV_BUF_SIZE);// 存储config数据 最大20K,暂定3K
  81. if(saveData == NULL)
  82. {
  83. char* retMsg = pvPortMalloc(32);
  84. memset(retMsg, 0, strlen(retMsg));
  85. retMsg = "{\"write_config\":\"error_full\"}";
  86. send(client_socket, retMsg, strlen(retMsg), 0);
  87. vPortFree(retMsg);
  88. tcp_config = 0;
  89. LogPrint(LOG_ERROR,__FILE__,__FUNCTION__,__LINE__,"recv buf malloc fail");
  90. }
  91. else
  92. {
  93. memset(saveData,0,strlen(saveData));
  94. sprintf(saveData,"%s",buf);
  95. // 获取上位机发送的数据
  96. do{
  97. memset(buf, 0, strlen(buf));
  98. recv(client_socket,buf,1460,0);
  99. memcpy(saveData + strlen(saveData), buf, strlen(buf));
  100. }while(buf[1] != 0 && buf[1459] != 0);
  101. // 插入tcp_config标志位,后续不通过http获取默认配置数据
  102. sprintf(saveData + strlen(saveData),"tcp_config");
  103. portENTER_CRITICAL();
  104. save_config_params(saveData);// 储存到flash 不用判断失败
  105. // vPortFree(get->device_params->params->gateway_read_modbus_command);
  106. // vPortFree(get->device_params->params->gateway_write_modbus_command);
  107. // vPortFree(get->device_params->params->gateway_read_dlt645_command);
  108. // vPortFree(get->device_params->params);
  109. // vPortFree(get->device_params);
  110. addGatewayParams(saveData);
  111. vPortFree(saveData);
  112. portEXIT_CRITICAL();
  113. char* retMsg = pvPortMalloc(32);
  114. memset(retMsg, 0, strlen(retMsg));
  115. retMsg = "{\"write_config\":\"success\"}";
  116. send(client_socket, retMsg, strlen(retMsg), 0);
  117. vPortFree(retMsg);
  118. tcp_config = 1;
  119. }
  120. }
  121. // 储存上位机发送的config_add数据,并返回上位机操作结果
  122. void add_config(int client_socket, char* dataBuf)
  123. {
  124. GATEWAY_PARAMS *get;
  125. get= get_gateway_config_params();
  126. DEVICE_PARAMS *device=get->device_params;
  127. char* retMsg = pvPortMalloc(32);
  128. memset(retMsg, 0, strlen(retMsg));
  129. while(device != NULL)// 一直轮询到当前为NULL
  130. {
  131. device = device->nextDevice;
  132. }
  133. addDevice(dataBuf);
  134. // 再检查更新后的deviceId是否为NULL
  135. if(device == NULL)// error
  136. {
  137. retMsg = "{\"write_config\":\"error\"}";
  138. send(client_socket, retMsg, strlen(retMsg), 0);
  139. }
  140. else// success
  141. {
  142. retMsg = "{\"write_config\":\"success\"}";
  143. send(client_socket, retMsg, strlen(retMsg), 0);
  144. }
  145. vPortFree(retMsg);
  146. }
  147. // 设备嗅探
  148. void find_device(int client_socket)
  149. {
  150. char deviceId[50];// 发送设备名
  151. GATEWAY_PARAMS *get;
  152. get= get_gateway_config_params();
  153. if(get->device_params == NULL)
  154. {
  155. sprintf(deviceId, "{\"find_device\":\"%s\"}", gatewayId);
  156. send(client_socket, deviceId, strlen(deviceId), 0);
  157. }
  158. else
  159. {
  160. sprintf(deviceId, "{\"find_device\":\"%s\"}", get->deviceId);
  161. send(client_socket, deviceId, strlen(deviceId), 0);
  162. }
  163. memset(deviceId, 0, 50);
  164. }
  165. int sie;
  166. // 发送设备当前的config数据
  167. void send_config(int client_socket)
  168. {
  169. GATEWAY_PARAMS *get;
  170. get= get_gateway_config_params();
  171. char* device_params = pvPortMalloc(8 * 1024);// 目前最大分配8K
  172. memset(device_params,0,8 * 1024);
  173. if(get->device_params == NULL)
  174. {
  175. sprintf(device_params, "{\"read_config\":\"error\"}");
  176. send(client_socket, device_params, strlen(device_params), 0);
  177. }
  178. else
  179. {
  180. get_device_params(device_params);
  181. for(uint8_t i = 0; i < 8; i++)
  182. {
  183. send(client_socket, device_params + (i*1024),1024, 0);
  184. }
  185. }
  186. vPortFree(device_params);
  187. }
  188. // 解析上位机发送的ip_config数据
  189. void analysis_ip_config(int client_socket, int sockfd,struct sockaddr_in tcpServerSock, char* buf)
  190. {
  191. // 打开DHCP
  192. if (strstr(buf, "\"dhcpMode\":\"open\"") != NULL)
  193. {
  194. DHCP_open();
  195. char* retMsg = pvPortMalloc(32);
  196. retMsg = "{\"ip_config\":\"success\"}";
  197. send(client_socket,retMsg,strlen(retMsg),0);
  198. vPortFree(retMsg);
  199. }
  200. // 关闭DHCP
  201. else if (strstr(buf, "\"dhcpMode\":\"close\"") != NULL)
  202. {
  203. uint8_t prot[6];
  204. // 设置静态IP地址,并关闭DHCP
  205. set_ipaddr(buf);
  206. // 解析buf内的数据,保存到load_ip_config结构体
  207. parseStringField(buf, "\"ipv4\":\"", (char *)&load_ip_config.host);// eg:192.168.1.100
  208. parseStringField(buf, "\"subnetMask\":\"", (char *)&load_ip_config.subnetMask);// eg:255.255.255.0
  209. parseStringField(buf, "\"defaultGateway\":\"", (char *)&load_ip_config.defaultGateway);// eg:192.168.1.1
  210. parseStringField(buf, "\"udpLogPort\":\"", (char *)&prot);
  211. load_ip_config.udpLogPort = atoi((char *)&prot);
  212. // 关闭服务端和客户端
  213. lwip_close(sockfd);
  214. lwip_close(client_socket);
  215. // DHCP关闭标志位置 1
  216. closeFlag = 1;
  217. }
  218. }
  219. // 切换工作模式
  220. void work_mode(char* buf)
  221. {
  222. /* 用标志位开关data_task任务中,发送数据的方式 */
  223. // 内置协议模式
  224. if(strstr(buf,"protocolsMode") != NULL)
  225. {
  226. ProtocolsModeFlag = 1;// 开启内置协议模式
  227. TransparentModeFlag = 0; // 关闭透明传输模式
  228. LogPrint(LOG_INFO,__FILE__,__FUNCTION__,__LINE__,"ProtocolsMode");
  229. }
  230. // 透明传输模式
  231. if(strstr(buf,"TransparentMode") != NULL)
  232. {
  233. ProtocolsModeFlag = 0;// 关闭内置协议模式
  234. TransparentModeFlag = 1; // 开启透明传输模式
  235. LogPrint(LOG_INFO,__FILE__,__FUNCTION__,__LINE__,"TransparentModeFlag");
  236. }
  237. }
  238. #define RECV_BUF_SIZE 3 * 1024
  239. void tcp_server_task(void *pvParameters)
  240. {
  241. int ret,sockfd;
  242. int recv_size;
  243. struct sockaddr_in tcpServerSock;
  244. struct sockaddr_in client_sock;
  245. // 命令集
  246. char* recv_cmd[] = {"\"cmd\":\"write_config\"","\"cmd\":\"write_config_add\"",
  247. "\"cmd\":\"read_config\"","\"cmd\":\"find_device\"",
  248. "\"cmd\":\"ip_config\"","\"cmd\":\"toggle_work_mode\"",
  249. "\"cmd\":\"software_update\"","\"cmd\":\"reboot\""};// cmd:error_cmd 仅为了判断错误的命令
  250. while(dhcp_done!=1)
  251. {
  252. vTaskDelay(100);
  253. }
  254. tcpServerSock.sin_family = AF_INET;
  255. inet_aton("192.168.0.100",&(tcpServerSock.sin_addr));
  256. // tcpServerSock.sin_addr.s_addr = htonl(IPADDR_ANY);
  257. tcpServerSock.sin_port = htons(8080);
  258. tcp_server_begin:
  259. sockfd = socket(AF_INET, SOCK_STREAM, 0);
  260. if (sockfd < 0)
  261. {
  262. goto tcp_server_begin;
  263. }
  264. ret = bind(sockfd, (struct sockaddr *)&tcpServerSock, sizeof(tcpServerSock));
  265. if (ret < 0)
  266. {
  267. lwip_close(sockfd);
  268. sockfd = -1;
  269. goto tcp_server_begin;
  270. }
  271. ret = listen(sockfd, 10);
  272. if (ret < 0)
  273. {
  274. lwip_close(sockfd);
  275. sockfd = -1;
  276. goto tcp_server_begin;
  277. }
  278. socklen_t len = sizeof(client_sock);
  279. int client_socket = accept(sockfd, (struct sockaddr*)&client_sock,&len);
  280. if (client_socket<0)
  281. {
  282. printf("error");
  283. }
  284. LogPrint(LOG_INFO,__FILE__,__FUNCTION__,__LINE__,"PC connect success");
  285. while (1)
  286. {
  287. vTaskDelay(100);
  288. char* dataBuf = pvPortMalloc(RECV_BUF_SIZE);// 存储上位机数据
  289. if(dataBuf == NULL)
  290. LogPrint(LOG_ERROR,__FILE__,__FUNCTION__,__LINE__,"recv buf malloc fail");
  291. memset(dataBuf,0,strlen(dataBuf));
  292. // 获取上位机发送的数据
  293. recv_size=recv(client_socket,dataBuf,RECV_BUF_SIZE,0);
  294. // 接收到消息
  295. // 解析上位机发送的CMD命令
  296. if(recv_size>0)
  297. {
  298. int j = 0;
  299. for(int i = 0; i < sizeof(recv_cmd)/sizeof(recv_cmd[0]); i++)
  300. {
  301. if(strstr(dataBuf,recv_cmd[i]) != NULL)
  302. {
  303. i = sizeof(recv_cmd)/sizeof(recv_cmd[0]);
  304. }
  305. j++;
  306. }
  307. if (ProtocolsModeFlag) {
  308. switch (j) {
  309. case WRITE_CONFIG:
  310. save_config(client_socket, dataBuf);
  311. vPortFree(dataBuf);
  312. LogPrint(LOG_INFO, __FILE__, __FUNCTION__, __LINE__, "write config");
  313. __set_PRIMASK(1);
  314. NVIC_SystemReset();
  315. break;
  316. case WRITE_CONFIG_ADD:
  317. add_config(client_socket, dataBuf);
  318. vPortFree(dataBuf);
  319. LogPrint(LOG_INFO, __FILE__, __FUNCTION__, __LINE__, "write config add");
  320. break;
  321. case READ_CONFIG:
  322. send_config(client_socket);
  323. vPortFree(dataBuf);
  324. LogPrint(LOG_INFO, __FILE__, __FUNCTION__, __LINE__, "read config");
  325. break;
  326. case FIND_DEVICE:
  327. find_device(client_socket);
  328. vPortFree(dataBuf);
  329. LogPrint(LOG_INFO, __FILE__, __FUNCTION__, __LINE__, "find device");
  330. break;
  331. case IP_CONFIG:
  332. LogPrint(LOG_INFO, __FILE__, __FUNCTION__, __LINE__, "ipconfig");
  333. analysis_ip_config(client_socket, sockfd, tcpServerSock, dataBuf);
  334. if (closeFlag == 1) {
  335. // 更新新的地址
  336. inet_aton((char*)&load_ip_config.host, &(tcpServerSock.sin_addr));
  337. closeFlag = 0;
  338. vPortFree(dataBuf);
  339. goto tcp_server_begin;
  340. }
  341. vPortFree(dataBuf);
  342. break;
  343. case TOGGLE_MODE:
  344. work_mode(dataBuf);
  345. vPortFree(dataBuf);
  346. LogPrint(LOG_INFO, __FILE__, __FUNCTION__, __LINE__, "toggle work mode");
  347. break;
  348. case UPDATE:
  349. LogPrint(LOG_INFO, __FILE__, __FUNCTION__, __LINE__, "update");
  350. vPortFree(dataBuf);
  351. updata_task_creat(client_socket);
  352. break;
  353. case REBOOT:
  354. send(client_socket,"{\"reboot\":\"success\"}", 20, 0);
  355. LogPrint(LOG_INFO, __FILE__, __FUNCTION__, __LINE__, "reboot");
  356. __set_PRIMASK(1);
  357. NVIC_SystemReset();
  358. break;
  359. }
  360. } else {
  361. switch (j) {
  362. case FIND_DEVICE:
  363. find_device(client_socket);
  364. vPortFree(dataBuf);
  365. LogPrint(LOG_INFO, __FILE__, __FUNCTION__, __LINE__, "find device");
  366. break;
  367. case IP_CONFIG:
  368. LogPrint(LOG_INFO, __FILE__, __FUNCTION__, __LINE__, "ip config");
  369. analysis_ip_config(client_socket, sockfd, tcpServerSock, dataBuf);
  370. if (closeFlag == 1) {
  371. // 更新新的地址
  372. inet_aton((char*)&load_ip_config.host, &(tcpServerSock.sin_addr));
  373. closeFlag = 0;
  374. vPortFree(dataBuf);
  375. goto tcp_server_begin;
  376. }
  377. vPortFree(dataBuf);
  378. break;
  379. case TOGGLE_MODE:
  380. work_mode(dataBuf);
  381. vPortFree(dataBuf);
  382. LogPrint(LOG_INFO, __FILE__, __FUNCTION__, __LINE__, "toggle work mode");
  383. break;
  384. case UPDATE:
  385. LogPrint(LOG_INFO, __FILE__, __FUNCTION__, __LINE__, "update");
  386. vPortFree(dataBuf);
  387. updata_task_creat(client_socket);
  388. break;
  389. case REBOOT:
  390. send(client_socket,"{\"reboot\":\"success\"}", 20, 0);
  391. LogPrint(LOG_INFO, __FILE__, __FUNCTION__, __LINE__, "reboot");
  392. __set_PRIMASK(1);
  393. NVIC_SystemReset();
  394. break;
  395. }
  396. }
  397. }
  398. // 没接收到消息
  399. else if(recv_size==0)
  400. {
  401. lwip_close(client_socket);
  402. LOG_PRINT(LOG_ERROR,"PC disconnect,wait reconnect");
  403. vPortFree(dataBuf);
  404. client_socket= accept(sockfd, (struct sockaddr*)&client_sock,&len);
  405. }
  406. vPortFree(dataBuf);
  407. }
  408. }
  409. /*!
  410. \brief initialize the tcp_client application
  411. \param[in] none
  412. \param[out] none
  413. \retval none
  414. */
  415. void tcp_server_init(void)
  416. {
  417. xTaskCreate(tcp_server_task, "TCP_SERVER", DEFAULT_THREAD_STACKSIZE, NULL, 1, NULL);
  418. }