platformNetwork.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328
  1. // #define rlogEnable // 是否使能日志
  2. #define rlogColorEnable // 是否使能日志颜色
  3. #define rlogLevel (rlogLvlWarning) // 日志打印等级
  4. #define rlogTag "RyanMqttNet" // 日志tag
  5. #include "platformNetwork.h"
  6. #include "RyanMqttLog.h"
  7. /**
  8. * @brief 连接mqtt服务器
  9. *
  10. * @param userData
  11. * @param platformNetwork
  12. * @param host
  13. * @param port
  14. * @return RyanMqttError_e
  15. * 成功返回RyanMqttSuccessError, 失败返回错误信息
  16. */
  17. RyanMqttError_e platformNetworkConnect(void *userData, platformNetwork_t *platformNetwork, const char *host, uint16_t port)
  18. {
  19. RyanMqttError_e result = RyanMqttSuccessError;
  20. // ?线程安全版本,有些设备没有实现,默认不启用。如果涉及多个客户端解析域名请使用线程安全版本
  21. char buf[256];
  22. int ret;
  23. struct hostent hostinfo, *phost;
  24. if (0 != gethostbyname_r(host, &hostinfo, buf, sizeof(buf), &phost, &ret))
  25. {
  26. result = RyanSocketFailedError;
  27. goto exit;
  28. }
  29. platformNetwork->socket = socket(AF_INET, SOCK_STREAM, IPPROTO_IP);
  30. if (platformNetwork->socket < 0)
  31. {
  32. result = RyanSocketFailedError;
  33. goto exit;
  34. }
  35. struct sockaddr_in server_addr;
  36. memset(&server_addr, 0, sizeof(server_addr));
  37. server_addr.sin_family = AF_INET;
  38. server_addr.sin_port = htons(port); // 指定端口号,这里使用HTTP默认端口80
  39. server_addr.sin_addr = *((struct in_addr *)hostinfo.h_addr_list[0]);
  40. // 绑定套接字到主机地址和端口号
  41. if (connect(platformNetwork->socket, (struct sockaddr *)&server_addr, sizeof(server_addr)) != 0)
  42. {
  43. platformNetworkClose(userData, platformNetwork);
  44. result = RyanMqttSocketConnectFailError;
  45. goto exit;
  46. }
  47. // 非线程安全版本,请根据实际情况选择使用
  48. // struct hostent *hostinfo;
  49. // hostinfo = gethostbyname(host);
  50. // if (NULL == hostinfo)
  51. // {
  52. // result = RyanSocketFailedError;
  53. // goto exit;
  54. // }
  55. // platformNetwork->socket = socket(AF_INET, SOCK_STREAM, IPPROTO_IP);
  56. // if (platformNetwork->socket < 0)
  57. // {
  58. // result = RyanSocketFailedError;
  59. // goto exit;
  60. // }
  61. // struct sockaddr_in server_addr;
  62. // memset(&server_addr, 0, sizeof(server_addr));
  63. // server_addr.sin_family = AF_INET;
  64. // server_addr.sin_port = htons(port); // 指定端口号,这里使用HTTP默认端口80
  65. // server_addr.sin_addr = *((struct in_addr *)hostinfo->h_addr_list[0]);
  66. // // 绑定套接字到主机地址和端口号
  67. // if (connect(platformNetwork->socket, (struct sockaddr *)&server_addr, sizeof(server_addr)) != 0)
  68. // {
  69. // platformNetworkClose(userData, platformNetwork);
  70. // result = RyanMqttSocketConnectFailError;
  71. // goto exit;
  72. // }
  73. exit:
  74. return result;
  75. }
  76. /**
  77. * @brief 非阻塞接收数据
  78. *
  79. * @param userData
  80. * @param platformNetwork
  81. * @param recvBuf
  82. * @param recvLen
  83. * @param timeout
  84. * @return RyanMqttError_e
  85. * socket错误返回 RyanSocketFailedError
  86. * 接收超时或者接收数据长度不等于期待数据接受长度 RyanMqttRecvPacketTimeOutError
  87. * 接收成功 RyanMqttSuccessError
  88. */
  89. RyanMqttError_e platformNetworkRecvAsync(void *userData, platformNetwork_t *platformNetwork, char *recvBuf, int recvLen, int timeout)
  90. {
  91. // int32_t recvResult = 0;
  92. // int32_t offset = 0;
  93. // int32_t timeOut2 = timeout;
  94. // struct timeval tv = {0};
  95. // platformTimer_t timer = {0};
  96. // if (-1 == platformNetwork->socket)
  97. // return RyanSocketFailedError;
  98. // platformTimerCutdown(&timer, timeout);
  99. // while ((offset < recvLen) && (0 != timeOut2))
  100. // {
  101. // tv.tv_sec = timeOut2 / 1000;
  102. // tv.tv_usec = timeOut2 % 1000 * 1000;
  103. // if (tv.tv_sec <= 0 && tv.tv_usec <= 100)
  104. // {
  105. // tv.tv_sec = 0;
  106. // tv.tv_usec = 100;
  107. // }
  108. // setsockopt(platformNetwork->socket, SOL_SOCKET, SO_RCVTIMEO, (char *)&tv, sizeof(struct timeval)); // 设置错做模式为非阻塞
  109. // recvResult = recv(platformNetwork->socket, recvBuf + offset, recvLen - offset, 0);
  110. // if (0 == recvResult)
  111. // {
  112. // rlog_d("对端关闭socket连接");
  113. // return RyanSocketFailedError;
  114. // }
  115. // else if (recvResult < 0) // 小于零,表示错误,个别错误不代表socket错误
  116. // {
  117. // int32_t err = errno;
  118. // // 下列3种表示没问题,但需要推出发送
  119. // if (err == EAGAIN || // 套接字已标记为非阻塞,而接收操作被阻塞或者接收超时
  120. // err == EWOULDBLOCK || // 发送时套接字发送缓冲区已满,或接收时套接字接收缓冲区为空
  121. // err == EINTR) // 操作被信号中断
  122. // break;
  123. // return RyanSocketFailedError;
  124. // }
  125. // offset += recvResult;
  126. // timeOut2 = platformTimerRemain(&timer);
  127. // }
  128. // if (offset != recvLen)
  129. // return RyanMqttRecvPacketTimeOutError;
  130. // return RyanMqttSuccessError;
  131. int32_t recvResult = 0;
  132. int32_t offset = 0;
  133. int32_t timeOut2 = timeout;
  134. struct timeval tv = {0};
  135. platformTimer_t timer = {0};
  136. if (-1 == platformNetwork->socket)
  137. return RyanSocketFailedError;
  138. platformTimerCutdown(&timer, timeout);
  139. while ((offset < recvLen) && (0 != timeOut2))
  140. {
  141. tv.tv_sec = timeOut2 / 1000;
  142. tv.tv_usec = timeOut2 % 1000 * 1000;
  143. if (tv.tv_sec <= 0 && tv.tv_usec <= 100)
  144. {
  145. tv.tv_sec = 0;
  146. tv.tv_usec = 100;
  147. }
  148. fd_set readset;
  149. fd_set exceptset;
  150. int i, maxfdp1;
  151. /* 清空可读事件描述符列表 */
  152. FD_ZERO(&readset);
  153. FD_ZERO(&exceptset);
  154. FD_SET(platformNetwork->socket, &readset); // 监听可读事件
  155. FD_SET(platformNetwork->socket, &exceptset); // 监听异常事件
  156. /* 等待设定的网络描述符有事件发生 */
  157. i = select(platformNetwork->socket + 1, &readset, NULL, &exceptset, &tv);
  158. if (i < 0)
  159. {
  160. int32_t err = errno;
  161. // 下列3种表示没问题,但需要退出接收
  162. if (err == EAGAIN || // 套接字已标记为非阻塞,而接收操作被阻塞或者接收超时
  163. err == EWOULDBLOCK || // 发送时套接字发送缓冲区已满,或接收时套接字接收缓冲区为空
  164. err == EINTR) // 操作被信号中断
  165. break;
  166. return RyanSocketFailedError;
  167. }
  168. /* 查看 sock 描述符上有没有发生可读事件 */
  169. else if (i > 0)
  170. {
  171. if (FD_ISSET(platformNetwork->socket, &readset))
  172. {
  173. recvResult = recv(platformNetwork->socket, recvBuf + offset, recvLen - offset, 0);
  174. if (recvResult <= 0) // 小于零,表示错误,个别错误不代表socket错误
  175. {
  176. int32_t err = errno;
  177. // 下列3种表示没问题,但需要退出接收
  178. if (err == EAGAIN || // 套接字已标记为非阻塞,而接收操作被阻塞或者接收超时
  179. err == EWOULDBLOCK || // 发送时套接字发送缓冲区已满,或接收时套接字接收缓冲区为空
  180. err == EINTR) // 操作被信号中断
  181. break;
  182. return RyanSocketFailedError;
  183. }
  184. offset += recvResult;
  185. }
  186. if (FD_ISSET(platformNetwork->socket, &exceptset))
  187. {
  188. return RyanSocketFailedError;
  189. }
  190. }
  191. timeOut2 = platformTimerRemain(&timer);
  192. }
  193. if (offset != recvLen)
  194. return RyanMqttRecvPacketTimeOutError;
  195. return RyanMqttSuccessError;
  196. }
  197. /**
  198. * @brief 非阻塞发送数据
  199. *
  200. * @param userData
  201. * @param platformNetwork
  202. * @param sendBuf
  203. * @param sendLen
  204. * @param timeout
  205. * @return RyanMqttError_e
  206. * socket错误返回 RyanSocketFailedError
  207. * 接收超时或者接收数据长度不等于期待数据接受长度 RyanMqttRecvPacketTimeOutError
  208. * 接收成功 RyanMqttSuccessError
  209. */
  210. RyanMqttError_e platformNetworkSendAsync(void *userData, platformNetwork_t *platformNetwork, char *sendBuf, int sendLen, int timeout)
  211. {
  212. int32_t sendResult = 0;
  213. int32_t offset = 0;
  214. int32_t timeOut2 = timeout;
  215. struct timeval tv = {0};
  216. platformTimer_t timer = {0};
  217. if (-1 == platformNetwork->socket)
  218. return RyanSocketFailedError;
  219. platformTimerCutdown(&timer, timeout);
  220. while ((offset < sendLen) && (0 != timeOut2))
  221. {
  222. tv.tv_sec = timeOut2 / 1000;
  223. tv.tv_usec = timeOut2 % 1000 * 1000;
  224. if (tv.tv_sec <= 0 && tv.tv_usec <= 100)
  225. {
  226. tv.tv_sec = 0;
  227. tv.tv_usec = 100;
  228. }
  229. setsockopt(platformNetwork->socket, SOL_SOCKET, SO_SNDTIMEO, (char *)&tv, sizeof(struct timeval)); // 设置错做模式为非阻塞
  230. sendResult = send(platformNetwork->socket, sendBuf + offset, sendLen - offset, 0);
  231. if (0 == sendResult)
  232. {
  233. rlog_d("对端关闭socket连接");
  234. return RyanSocketFailedError;
  235. }
  236. else if (sendResult < 0) // 小于零,表示错误,个别错误不代表socket错误
  237. {
  238. int32_t err = errno;
  239. // 下列3种表示没问题,但需要退出发送
  240. if (err == EAGAIN || // 套接字已标记为非阻塞,而接收操作被阻塞或者接收超时
  241. err == EWOULDBLOCK || // 发送时套接字发送缓冲区已满,或接收时套接字接收缓冲区为空
  242. err == EINTR) // 操作被信号中断
  243. break;
  244. return RyanSocketFailedError;
  245. }
  246. offset += sendResult;
  247. timeOut2 = platformTimerRemain(&timer);
  248. }
  249. if (offset != sendLen)
  250. return RyanMqttSendPacketTimeOutError;
  251. return RyanMqttSuccessError;
  252. }
  253. /**
  254. * @brief 断开mqtt服务器连接
  255. *
  256. * @param userData
  257. * @param platformNetwork
  258. * @return RyanMqttError_e
  259. */
  260. RyanMqttError_e platformNetworkClose(void *userData, platformNetwork_t *platformNetwork)
  261. {
  262. if (platformNetwork->socket >= 0)
  263. {
  264. close(platformNetwork->socket);
  265. platformNetwork->socket = -1;
  266. }
  267. return RyanMqttSuccessError;
  268. }