Commit 53aa7668 authored by Marton Balint's avatar Marton Balint

avformat/udp: cancel pending IO on win32 manually

recvfrom() is not a cancellation point in pthreads-win32, see
https://sourceware.org/pthreads-win32/manual/pthread_cancel.html

In order to be able to cancel the reader thread on Win32 properly we first
shutdown the socket then call CancelIoEx to abort pending IO. Subsequent
recvfrom() calls will fail with WSAESHUTDOWN causing the thread to exit.

Fixes ticket #5717.
Signed-off-by: 's avatarMarton Balint <cus@passwd.hu>
parent c2b6493b
...@@ -1069,8 +1069,17 @@ static int udp_close(URLContext *h) ...@@ -1069,8 +1069,17 @@ static int udp_close(URLContext *h)
if (s->thread_started) { if (s->thread_started) {
int ret; int ret;
// Cancel only read, as write has been signaled as success to the user // Cancel only read, as write has been signaled as success to the user
if (h->flags & AVIO_FLAG_READ) if (h->flags & AVIO_FLAG_READ) {
#ifdef _WIN32
/* recvfrom() is not a cancellation point for win32, so we shutdown
* the socket and abort pending IO, subsequent recvfrom() calls
* will fail with WSAESHUTDOWN causing the thread to exit. */
shutdown(s->udp_fd, SD_RECEIVE);
CancelIoEx((HANDLE)(SOCKET)s->udp_fd, NULL);
#else
pthread_cancel(s->circular_buffer_thread); pthread_cancel(s->circular_buffer_thread);
#endif
}
ret = pthread_join(s->circular_buffer_thread, NULL); ret = pthread_join(s->circular_buffer_thread, NULL);
if (ret != 0) if (ret != 0)
av_log(h, AV_LOG_ERROR, "pthread_join(): %s\n", strerror(ret)); av_log(h, AV_LOG_ERROR, "pthread_join(): %s\n", strerror(ret));
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment