rtmp-windows.c 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351
  1. #ifdef _WIN32
  2. #include "rtmp-stream.h"
  3. #include <winsock2.h>
  4. static void fatal_sock_shutdown(struct rtmp_stream *stream)
  5. {
  6. closesocket(stream->rtmp.m_sb.sb_socket);
  7. stream->rtmp.m_sb.sb_socket = -1;
  8. stream->write_buf_len = 0;
  9. os_event_signal(stream->buffer_space_available_event);
  10. }
  11. static bool socket_event(struct rtmp_stream *stream, bool *can_write,
  12. uint64_t last_send_time)
  13. {
  14. WSANETWORKEVENTS net_events;
  15. bool success;
  16. success = !WSAEnumNetworkEvents(stream->rtmp.m_sb.sb_socket, NULL,
  17. &net_events);
  18. if (!success) {
  19. blog(LOG_ERROR, "socket_thread_windows: Aborting due to "
  20. "WSAEnumNetworkEvents failure, %d",
  21. WSAGetLastError());
  22. fatal_sock_shutdown(stream);
  23. return false;
  24. }
  25. if (net_events.lNetworkEvents & FD_WRITE)
  26. *can_write = true;
  27. if (net_events.lNetworkEvents & FD_CLOSE) {
  28. if (last_send_time) {
  29. uint32_t diff =
  30. (os_gettime_ns() / 1000000) - last_send_time;
  31. blog(LOG_ERROR, "socket_thread_windows: Received "
  32. "FD_CLOSE, %u ms since last send "
  33. "(buffer: %d / %d)",
  34. diff,
  35. stream->write_buf_len,
  36. stream->write_buf_size);
  37. }
  38. if (os_event_try(stream->stop_event) != EAGAIN)
  39. blog(LOG_ERROR, "socket_thread_windows: Aborting due "
  40. "to FD_CLOSE during shutdown, "
  41. "%d bytes lost, error %d",
  42. stream->write_buf_len,
  43. net_events.iErrorCode[FD_CLOSE_BIT]);
  44. else
  45. blog(LOG_ERROR, "socket_thread_windows: Aborting due "
  46. "to FD_CLOSE, error %d",
  47. net_events.iErrorCode[FD_CLOSE_BIT]);
  48. fatal_sock_shutdown(stream);
  49. return false;
  50. }
  51. if (net_events.lNetworkEvents & FD_READ) {
  52. char discard[16384];
  53. int err_code;
  54. bool fatal = false;
  55. for (;;) {
  56. int ret = recv(stream->rtmp.m_sb.sb_socket,
  57. discard, sizeof(discard), 0);
  58. if (ret == -1) {
  59. err_code = WSAGetLastError();
  60. if (err_code == WSAEWOULDBLOCK)
  61. break;
  62. fatal = true;
  63. } else if (ret == 0) {
  64. err_code = 0;
  65. fatal = true;
  66. }
  67. if (fatal) {
  68. blog(LOG_ERROR, "socket_thread_windows: "
  69. "Socket error, recv() returned "
  70. "%d, GetLastError() %d",
  71. ret, err_code);
  72. fatal_sock_shutdown(stream);
  73. return false;
  74. }
  75. }
  76. }
  77. return true;
  78. }
  79. static void ideal_send_backlog_event(struct rtmp_stream *stream,
  80. bool *can_write)
  81. {
  82. ULONG ideal_send_backlog;
  83. int ret;
  84. ret = idealsendbacklogquery(
  85. stream->rtmp.m_sb.sb_socket,
  86. &ideal_send_backlog);
  87. if (ret == 0) {
  88. int cur_tcp_bufsize;
  89. int size = sizeof(cur_tcp_bufsize);
  90. ret = getsockopt(stream->rtmp.m_sb.sb_socket,
  91. SOL_SOCKET,
  92. SO_SNDBUF,
  93. (char *)&cur_tcp_bufsize,
  94. &size);
  95. if (ret == 0) {
  96. if (cur_tcp_bufsize < (int)ideal_send_backlog) {
  97. int bufsize = (int)ideal_send_backlog;
  98. setsockopt(stream->rtmp.m_sb.sb_socket,
  99. SOL_SOCKET,
  100. SO_SNDBUF,
  101. (const char *)&bufsize,
  102. sizeof(bufsize));
  103. blog(LOG_INFO, "socket_thread_windows: "
  104. "Increasing send buffer to "
  105. "ISB %d (buffer: %d / %d)",
  106. ideal_send_backlog,
  107. stream->write_buf_len,
  108. stream->write_buf_size);
  109. }
  110. } else {
  111. blog(LOG_ERROR, "socket_thread_windows: Got "
  112. "send_backlog_event but "
  113. "getsockopt() returned %d",
  114. WSAGetLastError());
  115. }
  116. } else {
  117. blog(LOG_ERROR, "socket_thread_windows: Got "
  118. "send_backlog_event but WSAIoctl() "
  119. "returned %d",
  120. WSAGetLastError());
  121. }
  122. }
  123. enum data_ret {
  124. RET_BREAK,
  125. RET_FATAL,
  126. RET_CONTINUE
  127. };
  128. static enum data_ret write_data(struct rtmp_stream *stream, bool *can_write,
  129. uint64_t *last_send_time, size_t latency_packet_size,
  130. int delay_time)
  131. {
  132. bool exit_loop = false;
  133. pthread_mutex_lock(&stream->write_buf_mutex);
  134. if (!stream->write_buf_len) {
  135. /* this is now an expected occasional condition due to use of
  136. * auto-reset events, we could end up emptying the buffer as
  137. * it's filled in a previous loop cycle, especially if using
  138. * low latency mode. */
  139. pthread_mutex_unlock(&stream->write_buf_mutex);
  140. /* blog(LOG_DEBUG, "socket_thread_windows: Trying to send, "
  141. "but no data available"); */
  142. return RET_BREAK;
  143. }
  144. int ret;
  145. if (stream->low_latency_mode) {
  146. size_t send_len =
  147. min(latency_packet_size, stream->write_buf_len);
  148. ret = send(stream->rtmp.m_sb.sb_socket,
  149. (const char *)stream->write_buf,
  150. (int)send_len, 0);
  151. } else {
  152. ret = send(stream->rtmp.m_sb.sb_socket,
  153. (const char *)stream->write_buf,
  154. (int)stream->write_buf_len, 0);
  155. }
  156. if (ret > 0) {
  157. if (stream->write_buf_len - ret)
  158. memmove(stream->write_buf,
  159. stream->write_buf + ret,
  160. stream->write_buf_len - ret);
  161. stream->write_buf_len -= ret;
  162. *last_send_time = os_gettime_ns() / 1000000;
  163. os_event_signal(stream->buffer_space_available_event);
  164. } else {
  165. int err_code;
  166. bool fatal_err = false;
  167. if (ret == -1) {
  168. err_code = WSAGetLastError();
  169. if (err_code == WSAEWOULDBLOCK) {
  170. *can_write = false;
  171. pthread_mutex_unlock(&stream->write_buf_mutex);
  172. return RET_BREAK;
  173. }
  174. fatal_err = true;
  175. } else if (ret == 0) {
  176. err_code = 0;
  177. fatal_err = true;
  178. }
  179. if (fatal_err) {
  180. /* connection closed, or connection was aborted /
  181. * socket closed / etc, that's a fatal error. */
  182. blog(LOG_ERROR, "socket_thread_windows: "
  183. "Socket error, send() returned %d, "
  184. "GetLastError() %d",
  185. ret, err_code);
  186. pthread_mutex_unlock(&stream->write_buf_mutex);
  187. fatal_sock_shutdown(stream);
  188. return RET_FATAL;
  189. }
  190. }
  191. /* finish writing for now */
  192. if (stream->write_buf_len <= 1000)
  193. exit_loop = true;
  194. pthread_mutex_unlock(&stream->write_buf_mutex);
  195. if (delay_time)
  196. os_sleep_ms(delay_time);
  197. return exit_loop ? RET_BREAK : RET_CONTINUE;
  198. }
  199. #define LATENCY_FACTOR 20
  200. static inline void socket_thread_windows_internal(struct rtmp_stream *stream)
  201. {
  202. bool can_write = false;
  203. int delay_time;
  204. size_t latency_packet_size;
  205. uint64_t last_send_time = 0;
  206. HANDLE send_backlog_event;
  207. OVERLAPPED send_backlog_overlapped;
  208. SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_ABOVE_NORMAL);
  209. WSAEventSelect(stream->rtmp.m_sb.sb_socket,
  210. stream->socket_available_event,
  211. FD_READ|FD_WRITE|FD_CLOSE);
  212. send_backlog_event = CreateEvent(NULL, true, false, NULL);
  213. if (stream->low_latency_mode) {
  214. delay_time = 1000 / LATENCY_FACTOR;
  215. latency_packet_size = stream->write_buf_size / (LATENCY_FACTOR - 2);
  216. } else {
  217. latency_packet_size = stream->write_buf_size;
  218. delay_time = 0;
  219. }
  220. if (!stream->disable_send_window_optimization) {
  221. memset(&send_backlog_overlapped, 0,
  222. sizeof(send_backlog_overlapped));
  223. send_backlog_overlapped.hEvent = send_backlog_event;
  224. idealsendbacklognotify(stream->rtmp.m_sb.sb_socket,
  225. &send_backlog_overlapped, NULL);
  226. } else {
  227. blog(LOG_INFO, "socket_thread_windows: Send window "
  228. "optimization disabled by user.");
  229. }
  230. HANDLE objs[3];
  231. objs[0] = stream->socket_available_event;
  232. objs[1] = stream->buffer_has_data_event;
  233. objs[2] = send_backlog_event;
  234. for (;;) {
  235. if (os_event_try(stream->send_thread_signaled_exit) != EAGAIN) {
  236. pthread_mutex_lock(&stream->write_buf_mutex);
  237. if (stream->write_buf_len == 0) {
  238. //blog(LOG_DEBUG, "Exiting on empty buffer");
  239. pthread_mutex_unlock(&stream->write_buf_mutex);
  240. os_event_reset(stream->send_thread_signaled_exit);
  241. break;
  242. }
  243. pthread_mutex_unlock(&stream->write_buf_mutex);
  244. }
  245. int status = WaitForMultipleObjects(3, objs, false, INFINITE);
  246. if (status == WAIT_ABANDONED || status == WAIT_FAILED) {
  247. blog(LOG_ERROR, "socket_thread_windows: Aborting due "
  248. "to WaitForMultipleObjects failure");
  249. fatal_sock_shutdown(stream);
  250. return;
  251. }
  252. if (status == WAIT_OBJECT_0) {
  253. /* Socket event */
  254. if (!socket_event(stream, &can_write, last_send_time))
  255. return;
  256. } else if (status == WAIT_OBJECT_0 + 2) {
  257. /* Ideal send backlog event */
  258. ideal_send_backlog_event(stream, &can_write);
  259. ResetEvent(send_backlog_event);
  260. idealsendbacklognotify(stream->rtmp.m_sb.sb_socket,
  261. &send_backlog_overlapped, NULL);
  262. continue;
  263. }
  264. if (can_write) {
  265. for (;;) {
  266. enum data_ret ret = write_data(
  267. stream,
  268. &can_write,
  269. &last_send_time,
  270. latency_packet_size,
  271. delay_time);
  272. switch (ret) {
  273. case RET_BREAK:
  274. goto exit_write_loop;
  275. case RET_FATAL:
  276. return;
  277. case RET_CONTINUE:;
  278. }
  279. }
  280. }
  281. exit_write_loop:;
  282. }
  283. if (stream->rtmp.m_sb.sb_socket != INVALID_SOCKET)
  284. WSAEventSelect(stream->rtmp.m_sb.sb_socket,
  285. stream->socket_available_event, 0);
  286. blog(LOG_INFO, "socket_thread_windows: Normal exit");
  287. }
  288. void *socket_thread_windows(void *data)
  289. {
  290. struct rtmp_stream *stream = data;
  291. socket_thread_windows_internal(stream);
  292. return NULL;
  293. }
  294. #endif