sys_mqtt.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341
  1. #include "string.h"
  2. #include "stm32f2xx.h"
  3. #include "ucos_ii.h"
  4. #include "hd_eth.h"
  5. #include "lwip/arch.h"
  6. #include "sys_mqtt.h"
  7. #include "transport.h"
  8. #include "MQTTFormat.h"
  9. #include "cJSON.h"
  10. #include "led.h"
  11. #include "MQTTClient.h"
  12. #include "malloc.h"
  13. #include "mmodbus.h"
  14. #include "myFile.h"
  15. #include "gateway_message.h"
  16. #include "task.h"
  17. /********************************************************
  18. * @brief 连接到服务器
  19. * @param sock: sock编号
  20. *********************************************************/
  21. int mqtt_userConnect(void)
  22. {
  23. GATEWAY_PARAMS* get;
  24. get = get_gateway_config_params();
  25. char* MQTT_SERVER_ADDR = (char*)get->host;
  26. int MQTT_SERVER_PORT = get->port;
  27. int sock = -1;
  28. //连接到tcp服务器
  29. sock = transport_open(MQTT_SERVER_ADDR, MQTT_SERVER_PORT);
  30. if(sock < 0)
  31. {
  32. MQTT_PRINTF("connect tcp server error \r\n");
  33. return -1;
  34. }
  35. MQTT_PRINTF("connect tcp server success \r\n");
  36. //连接到mqtt服务器
  37. if(mqtt_connectToMqttServer(sock) <= 0)
  38. {
  39. MQTT_PRINTF("connect mqtt server error \r\n");
  40. return -1;
  41. }
  42. MQTT_PRINTF("connect mqtt server success \r\n");
  43. return sock;
  44. }
  45. /************************************************************
  46. * @brief 订阅主题
  47. * @param sock: sock编号
  48. * @retval 1: 订阅成功
  49. *************************************************************/
  50. int mqtt_userSubscribeTopic(int sock)
  51. {
  52. GATEWAY_PARAMS* get;
  53. get = get_gateway_config_params();
  54. char* TOPICPC = (char*)get->commandTopic;
  55. return mqtt_subscribeTopic(sock, TOPICPC, 2);
  56. }
  57. /*
  58. *接收mqtt下发消息并存储进一个数据队列
  59. *name MQTT主题相关信息
  60. */
  61. void *json_message[10];
  62. #define QUEUE_SIZE 10 //队列深度
  63. //消息队列指针
  64. OS_EVENT *JsonQ;
  65. void mqtt_outputMsg(MQTTString *name, uint8_t *msgbuf, int msglen, uint16_t id, int qos)
  66. {
  67. int lenght=msglen;
  68. MQTT_PRINTF("receive a msg: id=%d qos=%d topic=%s msg=%s \r\n", id, qos, name->lenstring.data, msgbuf);
  69. <<<<<<< HEAD
  70. StringInfo message;
  71. message.p=mymalloc(SRAMEX ,msglen);
  72. memcpy(message.p,msgbuf,msglen);
  73. OSMboxPost(mqtt_recvMseeageMbox,(void*)&message);
  74. =======
  75. StringInfo *message=mymalloc(SRAMIN ,sizeof(StringInfo));
  76. message->stringLength=msglen;
  77. message->p=mymalloc(SRAMIN ,msglen+1);
  78. memcpy(message->p,msgbuf,msglen);
  79. uint8_t err;
  80. OSTimeDly(1500);
  81. err=OSQPost(JsonQ,message);
  82. commd = 0;
  83. switch(err)
  84. {
  85. case OS_ERR_NONE:
  86. break;
  87. case OS_ERR_Q_FULL:
  88. MQTT_PRINTF("receive a msg queue is full \r\n");
  89. break;
  90. default:
  91. break;
  92. }
  93. >>>>>>> 841e7987800910a57cf5275c331a8acbfec24198
  94. }
  95. /************************************************************
  96. * @brief 处理来自平台发布的信息
  97. * @param sock: sock编号
  98. pbuf: 接收的数据
  99. buflen: 数据的长度
  100. * @retval 1: 消息正常处理
  101. * -2: 帧数据包错误
  102. *************************************************************/
  103. int mqtt_recvPublishMessage(int sock, uint8_t *pbuf, int buflen)
  104. {
  105. uint8_t dup, retained;
  106. int qos, payloadlen;
  107. uint16_t packetid;
  108. MQTTString topicName;
  109. uint8_t *payload;
  110. int msg;
  111. if(MQTTDeserialize_publish(&dup, &qos, &retained, &packetid, &topicName, &payload, &payloadlen, pbuf, buflen) == 1)
  112. {
  113. mqtt_outputMsg(&topicName, payload, payloadlen, packetid, qos);
  114. }
  115. else return -2;
  116. if(qos == 1)
  117. {
  118. mqtt_recvPublishQos1_packid = packetid;
  119. msg = MBOX_MQTT_QOS1PUBACK;
  120. OSMboxPost(mqtt_sendMseeageMbox, &msg);
  121. }
  122. if(qos == 2)
  123. {
  124. mqtt_recvPublishQos2_packid = packetid;
  125. msg = MBOX_MQTT_QOS2PUBREC;
  126. OSMboxPost(mqtt_sendMseeageMbox, &msg);
  127. }
  128. return 1;
  129. }
  130. /************************************************************
  131. * @brief 解析所有平台下发的数据包
  132. * @param sock: sock编号
  133. type: 数据帧的类型
  134. pbuf: 数据
  135. buflen: 数据的长度
  136. * @retval 1: 消息正常处理
  137. * -2: 帧数据包错误
  138. *************************************************************/
  139. int mqtt_userReceiveMessage(int sock, int type, uint8_t *pbuf, int len)
  140. {
  141. int rc;
  142. switch(type)
  143. {
  144. case PUBLISH: rc = mqtt_recvPublishMessage(sock, pbuf, len); break; //平台向设备发布消息
  145. case PUBACK: rc = mqtt_publishMessage_qos1_PUBACK(pbuf, len); break; //设备向平台发布 QOS1 确认报文
  146. case PUBREC: rc = mqtt_publishMessage_qos2_PUBREC(pbuf, len); break; //设备向平台发布 QOS2 确认报文
  147. case PUBREL: rc = mqtt_recvPublishMessage_qos2_PUBREL(sock, pbuf, len); break; //平台向设备发布 QOS2 消息释放报文
  148. case PUBCOMP: rc = mqtt_publishMessage_qos2_PUBCOMP(pbuf, len); break; //设备向平台发布 QOS2 消息释放报文的确认报文
  149. case SUBACK: rc = mqtt_subscribeTopic_SUBACK(pbuf, len);break; //设备向平台订阅报文时的响应报文
  150. case UNSUBACK: rc = mqtt_unSubscribeTopic_UNSUBACK(pbuf, len); break; //设备向平台发取消订阅报文时的响应报文
  151. case PINGRESP: rc = mqtt_pingResponse(); break; //心跳响应报文
  152. default: rc = 1; break;
  153. }
  154. return rc;
  155. }
  156. /************************************************************
  157. * @brief 处理发送邮箱 mqtt_recvMbox 中的消息,根据消息的类型,发送对应的数据包
  158. * @param sock: sock编号
  159. boxMsg: 消息内容
  160. * @retval 1: 消息正常处理
  161. * -1: 网络错误
  162. * -2: 组建发送数据时发送错误
  163. *************************************************************/
  164. char pubJsonString[jsonMaxSize];
  165. int mqtt_userSendMessage(int sock, int boxMsg)
  166. {
  167. GATEWAY_PARAMS* get;
  168. get = get_gateway_config_params();
  169. char* TOPIC = (char*)get->messageTopic;
  170. int rc = 1;
  171. //用户发送的数据
  172. if((boxMsg & 0xf0000000) == 0x20000000)
  173. {
  174. switch(boxMsg)
  175. {
  176. case MBOX_USER_PUBLISHQOS0: rc = mqtt_publishMessage_qos0(sock, TOPIC, (uint8_t *)pubJsonString, strlen(pubJsonString)); break;
  177. case MBOX_USER_PUBLISHQOS1: rc = mqtt_publishMessage_qos1(sock, TOPIC, (uint8_t *)pubJsonString, strlen(pubJsonString)); break;
  178. case MBOX_USER_PUBLISHQOS2: rc = mqtt_publishMessage_qos2(sock, TOPIC, (uint8_t *)pubJsonString, strlen(pubJsonString)); break;
  179. }
  180. }
  181. //协议站使用的数据
  182. else if((boxMsg & 0xf0000000) == 0x10000000)
  183. {
  184. switch(boxMsg)
  185. {
  186. case MBOX_MQTT_QOS1PUBACK: rc = mqtt_recvPublishMessage_qos1_PUBACK(sock, mqtt_recvPublishQos1_packid); break;
  187. case MBOX_MQTT_QOS2PUBREC: rc = mqtt_recvPublishMessage_qos2_PUBREC(sock, mqtt_recvPublishQos2_packid); break;
  188. case MBOX_MQTT_QOS2PUBCOMP: rc = mqtt_recvPublishMessage_qos2_PUBCOMP(sock, mqtt_recvPublishQos2_packid); break;
  189. }
  190. }
  191. return rc;
  192. }
  193. OS_EVENT *mqtt_sendMseeageMbox; //发送消息邮箱
  194. OS_EVENT *mqtt_recvMseeageMbox; //接收消息邮箱
  195. int mqtt_connectFlag; //成功连接服务器标志
  196. static int mysock; //连接mqtt服务器的sock编号
  197. /************************************************************
  198. * @brief MQTT主线程
  199. * @param arg: 未使用
  200. * @retval
  201. *************************************************************/
  202. void mqtt_userManThread(void *arg)
  203. {
  204. int rc;
  205. uint8_t err;
  206. void *mboxMsg;
  207. MQTT_PRINTF("mqtt mainthread start \r\n");
  208. __MQTT_START:
  209. //1.建立与服务器的连接
  210. mysock = -1;
  211. mqtt_connectFlag = 0;
  212. while(mysock < 0)
  213. {
  214. mysock = mqtt_userConnect();
  215. OSTimeDly(2000);
  216. }
  217. OSMboxAccept(mqtt_sendMseeageMbox); //清空mbox的数据
  218. mqtt_connectFlag = 1;
  219. //2.订阅服务器的主题
  220. rc = mqtt_userSubscribeTopic(mysock);
  221. if(rc <= 0)
  222. {
  223. MQTT_PRINTF("subscribe error \r\n");
  224. if(rc == -1) goto __MQTT_START; //如果网络发生错误,重新建立连接
  225. }
  226. else MQTT_PRINTF("subscribe success \r\n");
  227. //3.循环发送数据
  228. while(1)
  229. {
  230. mboxMsg = OSMboxPend(mqtt_sendMseeageMbox, 60000, &err);
  231. if(OS_ERR_NONE == err)
  232. {
  233. //接收到了 网络异常消息
  234. if((*(unsigned int *)mboxMsg) == MBOX_NETWORK_ERROR) goto __MQTT_START;
  235. else
  236. {
  237. if(mqtt_userSendMessage(mysock, *(unsigned int *)mboxMsg) == -1)
  238. goto __MQTT_START;
  239. memset(pubJsonString,0,strlen(pubJsonString));
  240. }
  241. }
  242. else //超时没有发送数据包,发送心跳包
  243. {
  244. rc = mqtt_pingReq(mysock);
  245. if(rc == -1) goto __MQTT_START;
  246. }
  247. OSTimeDly(100);
  248. }
  249. }
  250. /************************************************************
  251. * @brief MQTT接收线程,处理所有平台下发的数据帧, CONNACK除外
  252. * @param arg: 未使用
  253. * @retval
  254. *************************************************************/
  255. void mqtt_userReceiveThread(void *arg)
  256. {
  257. int len;
  258. int packetType;
  259. int msg;
  260. MQTT_PRINTF("mqtt receivethread start \r\n");
  261. while(1)
  262. {
  263. if(mqtt_connectFlag == 1)
  264. {
  265. len = transport_receive(mysock); //阻塞接收数据
  266. if(len <=0)
  267. {
  268. if(len == EWOULDBLOCK) //接收数据超时,重新接收
  269. {
  270. MQTT_PRINTF("receive data timeout \r\n");
  271. continue;
  272. }
  273. else //接收数据时,网络发生了异常,给主线程发送消息,重新建立连接
  274. {
  275. MQTT_PRINTF("sock close \r\n");
  276. transport_close(mysock);
  277. mqtt_connectFlag = 0;
  278. msg = MBOX_NETWORK_ERROR;
  279. OSMboxPost(mqtt_sendMseeageMbox, &msg);
  280. }
  281. }
  282. memset(mqtt_recvbuffer, 0, MQTT_RECVBUF_LENTH);
  283. packetType = MQTTPacket_read(mqtt_recvbuffer, len, transport_getdata);
  284. mqtt_userReceiveMessage(mysock, packetType, mqtt_recvbuffer, len);
  285. }
  286. OSTimeDly(100);
  287. }
  288. }
  289. #define APP_TASK_MQTTMAIN_PRIO 6
  290. #define APP_TASK_MQTTMAIN_STK_SIZE 1024
  291. static OS_STK mqttmainTaskStack[APP_TASK_MQTTMAIN_STK_SIZE];
  292. #define APP_TASK_MQTTRECEIVE_PRIO 7
  293. #define APP_TASK_MQTTRECEIVE_STK_SIZE 1024
  294. static OS_STK mqttreceiveTaskStack[APP_TASK_MQTTRECEIVE_STK_SIZE];
  295. /************************************************************
  296. * @brief 创建MQTT发送和接收线程,初始化相关数据
  297. * @param 此函数必须在LWIP协议栈初始完成以后再调用
  298. * @retval
  299. *************************************************************/
  300. void mqtt_threadCreate(void)
  301. {
  302. mqtt_connectFlag = 0;
  303. mqtt_sendMseeageMbox = OSMboxCreate(NULL);
  304. mqtt_recvMseeageMbox = OSMboxCreate(NULL);
  305. JsonQ = OSQCreate(json_message[0], QUEUE_SIZE); //创建json队列
  306. OSTaskCreate(mqtt_userManThread, NULL, &mqttmainTaskStack[APP_TASK_MQTTMAIN_STK_SIZE-1], APP_TASK_MQTTMAIN_PRIO);
  307. OSTaskCreate(mqtt_userReceiveThread, NULL, &mqttreceiveTaskStack[APP_TASK_MQTTRECEIVE_STK_SIZE-1], APP_TASK_MQTTRECEIVE_PRIO);
  308. }