2001-07-22 16:18:56 +02:00
/*
* UDP prototype streaming system
2009-01-19 16:46:40 +01:00
* Copyright ( c ) 2000 , 2001 , 2002 Fabrice Bellard
2001-07-22 16:18:56 +02:00
*
2006-10-07 17:30:46 +02:00
* This file is part of FFmpeg .
*
* FFmpeg is free software ; you can redistribute it and / or
2002-05-26 00:34:32 +02:00
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation ; either
2006-10-07 17:30:46 +02:00
* version 2.1 of the License , or ( at your option ) any later version .
2001-07-22 16:18:56 +02:00
*
2006-10-07 17:30:46 +02:00
* FFmpeg is distributed in the hope that it will be useful ,
2001-07-22 16:18:56 +02:00
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
2002-05-26 00:34:32 +02:00
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
* Lesser General Public License for more details .
2001-07-22 16:18:56 +02:00
*
2002-05-26 00:34:32 +02:00
* You should have received a copy of the GNU Lesser General Public
2006-10-07 17:30:46 +02:00
* License along with FFmpeg ; if not , write to the Free Software
2006-01-12 23:43:26 +01:00
* Foundation , Inc . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 USA
2001-07-22 16:18:56 +02:00
*/
2008-04-16 18:52:35 +02:00
/**
2010-04-20 16:45:34 +02:00
* @ file
2008-04-16 18:52:35 +02:00
* UDP protocol
*/
2008-08-25 08:20:00 +02:00
# define _BSD_SOURCE /* Needed for using struct ip_mreq with recent glibc */
2011-05-11 17:52:51 +02:00
2001-08-13 23:37:10 +02:00
# include "avformat.h"
2011-03-08 10:35:52 +01:00
# include "avio_internal.h"
2011-02-16 09:52:38 +01:00
# include "libavutil/parseutils.h"
2011-05-13 17:16:15 +02:00
# include "libavutil/fifo.h"
2011-08-25 21:33:54 +02:00
# include "libavutil/intreadwrite.h"
2011-11-09 10:45:01 +01:00
# include "libavutil/avstring.h"
2012-09-13 13:10:26 +02:00
# include "libavutil/opt.h"
# include "libavutil/log.h"
2013-03-12 15:25:59 +01:00
# include "libavutil/time.h"
2010-03-15 00:59:48 +01:00
# include "internal.h"
2007-02-04 18:05:44 +01:00
# include "network.h"
2007-11-21 17:33:06 +01:00
# include "os_support.h"
2011-04-04 20:40:38 +02:00
# include "url.h"
2011-06-04 16:55:22 +02:00
2012-05-08 19:36:06 +02:00
# if HAVE_PTHREAD_CANCEL
2011-05-13 16:28:42 +02:00
# include <pthread.h>
2011-06-04 16:55:22 +02:00
# endif
2012-07-20 14:58:11 +02:00
# ifndef HAVE_PTHREAD_CANCEL
# define HAVE_PTHREAD_CANCEL 0
# endif
2005-08-12 16:01:03 +02:00
# ifndef IPV6_ADD_MEMBERSHIP
# define IPV6_ADD_MEMBERSHIP IPV6_JOIN_GROUP
# define IPV6_DROP_MEMBERSHIP IPV6_LEAVE_GROUP
# endif
2011-08-25 21:33:54 +02:00
# define UDP_TX_BUF_SIZE 32768
# define UDP_MAX_PKT_SIZE 65536
2001-07-22 16:18:56 +02:00
typedef struct {
2012-09-13 13:10:26 +02:00
const AVClass * class ;
2002-07-24 19:45:41 +02:00
int udp_fd ;
int ttl ;
2008-11-12 22:40:53 +01:00
int buffer_size ;
2002-07-24 19:45:41 +02:00
int is_multicast ;
int local_port ;
2006-10-28 19:16:18 +02:00
int reuse_socket ;
2012-03-12 16:09:53 +01:00
int overrun_nonfatal ;
2004-11-09 01:27:16 +01:00
struct sockaddr_storage dest_addr ;
2008-05-11 12:53:21 +02:00
int dest_addr_len ;
2010-10-08 10:42:30 +02:00
int is_connected ;
2011-05-13 16:28:42 +02:00
/* Circular Buffer variables for use in UDP receive code */
int circular_buffer_size ;
2011-05-13 17:16:15 +02:00
AVFifoBuffer * fifo ;
2011-05-13 16:28:42 +02:00
int circular_buffer_error ;
2012-05-08 19:36:06 +02:00
# if HAVE_PTHREAD_CANCEL
2011-05-13 16:28:42 +02:00
pthread_t circular_buffer_thread ;
2011-12-23 01:17:18 +01:00
pthread_mutex_t mutex ;
pthread_cond_t cond ;
2012-01-14 17:14:18 +01:00
int thread_started ;
2011-06-04 16:55:22 +02:00
# endif
2011-08-25 21:33:54 +02:00
uint8_t tmp [ UDP_MAX_PKT_SIZE + 4 ] ;
int remaining_in_dg ;
2012-09-13 13:10:26 +02:00
char * local_addr ;
int packet_size ;
int timeout ;
2001-07-22 16:18:56 +02:00
} UDPContext ;
2012-09-13 13:10:26 +02:00
# define OFFSET(x) offsetof(UDPContext, x)
# define D AV_OPT_FLAG_DECODING_PARAM
# define E AV_OPT_FLAG_ENCODING_PARAM
static const AVOption options [ ] = {
{ " buffer_size " , " Socket buffer size in bytes " , OFFSET ( buffer_size ) , AV_OPT_TYPE_INT , { . i64 = 0 } , 0 , INT_MAX , D | E } ,
{ " localport " , " Set local port to bind to " , OFFSET ( local_port ) , AV_OPT_TYPE_INT , { . i64 = 0 } , 0 , INT_MAX , D | E } ,
{ " localaddr " , " Choose local IP address " , OFFSET ( local_addr ) , AV_OPT_TYPE_STRING , { . str = " " } , 0 , 0 , D | E } ,
{ " pkt_size " , " Set size of UDP packets " , OFFSET ( packet_size ) , AV_OPT_TYPE_INT , { . i64 = 1472 } , 0 , INT_MAX , D | E } ,
{ " reuse " , " Explicitly allow or disallow reusing UDP sockets " , OFFSET ( reuse_socket ) , AV_OPT_TYPE_INT , { . i64 = 0 } , 0 , 1 , D | E } ,
{ " ttl " , " Set the time to live value (for multicast only) " , OFFSET ( ttl ) , AV_OPT_TYPE_INT , { . i64 = 16 } , 0 , INT_MAX , E } ,
{ " connect " , " Should connect() be called on socket " , OFFSET ( is_connected ) , AV_OPT_TYPE_INT , { . i64 = 0 } , 0 , 1 , D | E } ,
/* TODO 'sources', 'block' option */
{ " fifo_size " , " Set the UDP receiving circular buffer size, expressed as a number of packets with size of 188 bytes " , OFFSET ( circular_buffer_size ) , AV_OPT_TYPE_INT , { . i64 = 7 * 4096 } , 0 , INT_MAX , D } ,
{ " overrun_nonfatal " , " Survive in case of UDP receiving circular buffer overrun " , OFFSET ( overrun_nonfatal ) , AV_OPT_TYPE_INT , { . i64 = 0 } , 0 , 1 , D } ,
{ " timeout " , " In read mode: if no data arrived in more than this time interval, raise error " , OFFSET ( timeout ) , AV_OPT_TYPE_INT , { . i64 = 0 } , 0 , INT_MAX , D } ,
{ NULL }
} ;
static const AVClass udp_context_class = {
. class_name = " udp " ,
. item_name = av_default_item_name ,
. option = options ,
. version = LIBAVUTIL_VERSION_INT ,
} ;
2012-06-19 14:37:06 +02:00
static void log_net_error ( void * ctx , int level , const char * prefix )
{
char errbuf [ 100 ] ;
av_strerror ( ff_neterrno ( ) , errbuf , sizeof ( errbuf ) ) ;
av_log ( ctx , level , " %s: %s \n " , prefix , errbuf ) ;
}
2010-01-22 17:12:55 +01:00
static int udp_set_multicast_ttl ( int sockfd , int mcastTTL ,
struct sockaddr * addr )
{
2007-11-14 08:42:46 +01:00
# ifdef IP_MULTICAST_TTL
2004-11-09 01:27:16 +01:00
if ( addr - > sa_family = = AF_INET ) {
if ( setsockopt ( sockfd , IPPROTO_IP , IP_MULTICAST_TTL , & mcastTTL , sizeof ( mcastTTL ) ) < 0 ) {
2012-06-19 14:37:06 +02:00
log_net_error ( NULL , AV_LOG_ERROR , " setsockopt(IP_MULTICAST_TTL) " ) ;
2004-11-09 01:27:16 +01:00
return - 1 ;
}
}
2007-11-14 08:42:46 +01:00
# endif
2010-01-21 16:42:05 +01:00
# if defined(IPPROTO_IPV6) && defined(IPV6_MULTICAST_HOPS)
2004-11-09 01:27:16 +01:00
if ( addr - > sa_family = = AF_INET6 ) {
if ( setsockopt ( sockfd , IPPROTO_IPV6 , IPV6_MULTICAST_HOPS , & mcastTTL , sizeof ( mcastTTL ) ) < 0 ) {
2012-06-19 14:37:06 +02:00
log_net_error ( NULL , AV_LOG_ERROR , " setsockopt(IPV6_MULTICAST_HOPS) " ) ;
2004-11-09 01:27:16 +01:00
return - 1 ;
}
}
2007-11-14 08:42:46 +01:00
# endif
2004-11-09 01:27:16 +01:00
return 0 ;
}
2012-03-12 06:48:32 +01:00
static int udp_join_multicast_group ( int sockfd , struct sockaddr * addr )
2010-01-22 17:12:55 +01:00
{
2007-11-14 08:42:46 +01:00
# ifdef IP_ADD_MEMBERSHIP
2004-11-09 01:27:16 +01:00
if ( addr - > sa_family = = AF_INET ) {
2007-11-14 08:42:46 +01:00
struct ip_mreq mreq ;
2004-11-09 01:27:16 +01:00
mreq . imr_multiaddr . s_addr = ( ( struct sockaddr_in * ) addr ) - > sin_addr . s_addr ;
mreq . imr_interface . s_addr = INADDR_ANY ;
if ( setsockopt ( sockfd , IPPROTO_IP , IP_ADD_MEMBERSHIP , ( const void * ) & mreq , sizeof ( mreq ) ) < 0 ) {
2012-06-19 14:37:06 +02:00
log_net_error ( NULL , AV_LOG_ERROR , " setsockopt(IP_ADD_MEMBERSHIP) " ) ;
2004-11-09 01:27:16 +01:00
return - 1 ;
}
}
2007-11-14 08:42:46 +01:00
# endif
2010-09-20 08:38:40 +02:00
# if HAVE_STRUCT_IPV6_MREQ && defined(IPPROTO_IPV6)
2004-11-09 01:27:16 +01:00
if ( addr - > sa_family = = AF_INET6 ) {
2007-11-14 08:42:46 +01:00
struct ipv6_mreq mreq6 ;
2004-11-09 01:27:16 +01:00
memcpy ( & mreq6 . ipv6mr_multiaddr , & ( ( ( struct sockaddr_in6 * ) addr ) - > sin6_addr ) , sizeof ( struct in6_addr ) ) ;
mreq6 . ipv6mr_interface = 0 ;
if ( setsockopt ( sockfd , IPPROTO_IPV6 , IPV6_ADD_MEMBERSHIP , & mreq6 , sizeof ( mreq6 ) ) < 0 ) {
2012-06-19 14:37:06 +02:00
log_net_error ( NULL , AV_LOG_ERROR , " setsockopt(IPV6_ADD_MEMBERSHIP) " ) ;
2004-11-09 01:27:16 +01:00
return - 1 ;
}
}
2007-11-14 08:42:46 +01:00
# endif
2004-11-09 01:27:16 +01:00
return 0 ;
}
2010-01-22 17:12:55 +01:00
static int udp_leave_multicast_group ( int sockfd , struct sockaddr * addr )
{
2007-11-14 08:42:46 +01:00
# ifdef IP_DROP_MEMBERSHIP
2004-11-09 01:27:16 +01:00
if ( addr - > sa_family = = AF_INET ) {
2007-11-14 08:42:46 +01:00
struct ip_mreq mreq ;
2004-11-09 01:27:16 +01:00
mreq . imr_multiaddr . s_addr = ( ( struct sockaddr_in * ) addr ) - > sin_addr . s_addr ;
mreq . imr_interface . s_addr = INADDR_ANY ;
if ( setsockopt ( sockfd , IPPROTO_IP , IP_DROP_MEMBERSHIP , ( const void * ) & mreq , sizeof ( mreq ) ) < 0 ) {
2012-06-19 14:37:06 +02:00
log_net_error ( NULL , AV_LOG_ERROR , " setsockopt(IP_DROP_MEMBERSHIP) " ) ;
2004-11-09 01:27:16 +01:00
return - 1 ;
}
}
2007-11-14 08:42:46 +01:00
# endif
2010-09-20 08:38:40 +02:00
# if HAVE_STRUCT_IPV6_MREQ && defined(IPPROTO_IPV6)
2004-11-09 01:27:16 +01:00
if ( addr - > sa_family = = AF_INET6 ) {
2007-11-14 08:42:46 +01:00
struct ipv6_mreq mreq6 ;
2004-11-09 01:27:16 +01:00
memcpy ( & mreq6 . ipv6mr_multiaddr , & ( ( ( struct sockaddr_in6 * ) addr ) - > sin6_addr ) , sizeof ( struct in6_addr ) ) ;
mreq6 . ipv6mr_interface = 0 ;
if ( setsockopt ( sockfd , IPPROTO_IPV6 , IPV6_DROP_MEMBERSHIP , & mreq6 , sizeof ( mreq6 ) ) < 0 ) {
2012-06-19 14:37:06 +02:00
log_net_error ( NULL , AV_LOG_ERROR , " setsockopt(IPV6_DROP_MEMBERSHIP) " ) ;
2004-11-09 01:27:16 +01:00
return - 1 ;
}
}
2007-11-14 08:42:46 +01:00
# endif
2004-11-09 01:27:16 +01:00
return 0 ;
}
2010-01-22 17:12:55 +01:00
static struct addrinfo * udp_resolve_host ( const char * hostname , int port ,
int type , int family , int flags )
{
2011-06-06 14:13:02 +02:00
struct addrinfo hints = { 0 } , * res = 0 ;
2004-11-09 01:27:16 +01:00
int error ;
char sport [ 16 ] ;
2007-10-31 08:02:30 +01:00
const char * node = 0 , * service = " 0 " ;
2004-11-09 01:27:16 +01:00
if ( port > 0 ) {
2005-01-12 01:59:42 +01:00
snprintf ( sport , sizeof ( sport ) , " %d " , port ) ;
2004-11-09 01:27:16 +01:00
service = sport ;
}
if ( ( hostname ) & & ( hostname [ 0 ] ! = ' \0 ' ) & & ( hostname [ 0 ] ! = ' ? ' ) ) {
node = hostname ;
}
2007-10-31 08:08:12 +01:00
hints . ai_socktype = type ;
hints . ai_family = family ;
hints . ai_flags = flags ;
if ( ( error = getaddrinfo ( node , service , & hints , & res ) ) ) {
2010-02-28 23:33:22 +01:00
res = NULL ;
2010-01-22 17:10:08 +01:00
av_log ( NULL , AV_LOG_ERROR , " udp_resolve_host: %s \n " , gai_strerror ( error ) ) ;
2007-10-31 08:08:12 +01:00
}
2004-11-09 01:27:16 +01:00
return res ;
}
2012-06-21 13:19:56 +02:00
static int udp_set_multicast_sources ( int sockfd , struct sockaddr * addr ,
int addr_len , char * * sources ,
int nb_sources , int include )
{
# if HAVE_STRUCT_GROUP_SOURCE_REQ && defined(MCAST_BLOCK_SOURCE) && !defined(_WIN32)
/* These ones are available in the microsoft SDK, but don't seem to work
* as on linux , so just prefer the v4 - only approach there for now . */
int i ;
for ( i = 0 ; i < nb_sources ; i + + ) {
struct group_source_req mreqs ;
int level = addr - > sa_family = = AF_INET ? IPPROTO_IP : IPPROTO_IPV6 ;
struct addrinfo * sourceaddr = udp_resolve_host ( sources [ i ] , 0 ,
SOCK_DGRAM , AF_UNSPEC ,
AI_NUMERICHOST ) ;
if ( ! sourceaddr )
return AVERROR ( ENOENT ) ;
mreqs . gsr_interface = 0 ;
memcpy ( & mreqs . gsr_group , addr , addr_len ) ;
memcpy ( & mreqs . gsr_source , sourceaddr - > ai_addr , sourceaddr - > ai_addrlen ) ;
freeaddrinfo ( sourceaddr ) ;
if ( setsockopt ( sockfd , level ,
include ? MCAST_JOIN_SOURCE_GROUP : MCAST_BLOCK_SOURCE ,
( const void * ) & mreqs , sizeof ( mreqs ) ) < 0 ) {
if ( include )
log_net_error ( NULL , AV_LOG_ERROR , " setsockopt(MCAST_JOIN_SOURCE_GROUP) " ) ;
else
log_net_error ( NULL , AV_LOG_ERROR , " setsockopt(MCAST_BLOCK_SOURCE) " ) ;
return ff_neterrno ( ) ;
}
}
# elif HAVE_STRUCT_IP_MREQ_SOURCE && defined(IP_BLOCK_SOURCE)
int i ;
if ( addr - > sa_family ! = AF_INET ) {
av_log ( NULL , AV_LOG_ERROR ,
" Setting multicast sources only supported for IPv4 \n " ) ;
return AVERROR ( EINVAL ) ;
}
for ( i = 0 ; i < nb_sources ; i + + ) {
struct ip_mreq_source mreqs ;
struct addrinfo * sourceaddr = udp_resolve_host ( sources [ i ] , 0 ,
SOCK_DGRAM , AF_UNSPEC ,
AI_NUMERICHOST ) ;
if ( ! sourceaddr )
return AVERROR ( ENOENT ) ;
if ( sourceaddr - > ai_addr - > sa_family ! = AF_INET ) {
freeaddrinfo ( sourceaddr ) ;
av_log ( NULL , AV_LOG_ERROR , " %s is of incorrect protocol family \n " ,
sources [ i ] ) ;
return AVERROR ( EINVAL ) ;
}
mreqs . imr_multiaddr . s_addr = ( ( struct sockaddr_in * ) addr ) - > sin_addr . s_addr ;
mreqs . imr_interface . s_addr = INADDR_ANY ;
mreqs . imr_sourceaddr . s_addr = ( ( struct sockaddr_in * ) sourceaddr - > ai_addr ) - > sin_addr . s_addr ;
freeaddrinfo ( sourceaddr ) ;
if ( setsockopt ( sockfd , IPPROTO_IP ,
include ? IP_ADD_SOURCE_MEMBERSHIP : IP_BLOCK_SOURCE ,
( const void * ) & mreqs , sizeof ( mreqs ) ) < 0 ) {
if ( include )
log_net_error ( NULL , AV_LOG_ERROR , " setsockopt(IP_ADD_SOURCE_MEMBERSHIP) " ) ;
else
log_net_error ( NULL , AV_LOG_ERROR , " setsockopt(IP_BLOCK_SOURCE) " ) ;
return ff_neterrno ( ) ;
}
}
# else
return AVERROR ( ENOSYS ) ;
# endif
return 0 ;
}
2010-01-22 17:12:55 +01:00
static int udp_set_url ( struct sockaddr_storage * addr ,
const char * hostname , int port )
{
2004-11-09 01:27:16 +01:00
struct addrinfo * res0 ;
2007-11-14 09:16:51 +01:00
int addr_len ;
2010-01-22 17:10:08 +01:00
res0 = udp_resolve_host ( hostname , port , SOCK_DGRAM , AF_UNSPEC , 0 ) ;
2007-07-19 17:23:32 +02:00
if ( res0 = = 0 ) return AVERROR ( EIO ) ;
2007-11-14 09:16:51 +01:00
memcpy ( addr , res0 - > ai_addr , res0 - > ai_addrlen ) ;
addr_len = res0 - > ai_addrlen ;
2004-11-09 01:27:16 +01:00
freeaddrinfo ( res0 ) ;
2007-11-14 09:16:51 +01:00
return addr_len ;
2004-11-09 01:27:16 +01:00
}
2011-11-09 10:45:01 +01:00
static int udp_socket_create ( UDPContext * s , struct sockaddr_storage * addr ,
2012-10-26 20:46:37 +02:00
socklen_t * addr_len , const char * localaddr )
2007-11-15 15:26:52 +01:00
{
2004-11-09 01:27:16 +01:00
int udp_fd = - 1 ;
2005-07-21 23:10:23 +02:00
struct addrinfo * res0 = NULL , * res = NULL ;
2007-10-31 08:27:38 +01:00
int family = AF_UNSPEC ;
2005-12-17 19:14:38 +01:00
2007-10-31 08:27:38 +01:00
if ( ( ( struct sockaddr * ) & s - > dest_addr ) - > sa_family )
family = ( ( struct sockaddr * ) & s - > dest_addr ) - > sa_family ;
2011-11-09 10:45:01 +01:00
res0 = udp_resolve_host ( localaddr [ 0 ] ? localaddr : NULL , s - > local_port ,
SOCK_DGRAM , family , AI_PASSIVE ) ;
2007-10-31 08:08:12 +01:00
if ( res0 = = 0 )
goto fail ;
for ( res = res0 ; res ; res = res - > ai_next ) {
udp_fd = socket ( res - > ai_family , SOCK_DGRAM , 0 ) ;
2012-06-19 16:59:57 +02:00
if ( udp_fd ! = - 1 ) break ;
2012-06-19 14:37:06 +02:00
log_net_error ( NULL , AV_LOG_ERROR , " socket " ) ;
2007-10-31 08:08:12 +01:00
}
2005-07-21 23:10:23 +02:00
if ( udp_fd < 0 )
2004-11-09 01:27:16 +01:00
goto fail ;
2005-12-17 19:14:38 +01:00
2007-11-15 15:26:52 +01:00
memcpy ( addr , res - > ai_addr , res - > ai_addrlen ) ;
* addr_len = res - > ai_addrlen ;
2004-11-09 01:27:16 +01:00
2007-11-15 15:26:52 +01:00
freeaddrinfo ( res0 ) ;
2005-12-17 19:14:38 +01:00
2004-11-09 01:27:16 +01:00
return udp_fd ;
2005-12-17 19:14:38 +01:00
2004-11-09 01:27:16 +01:00
fail :
if ( udp_fd > = 0 )
closesocket ( udp_fd ) ;
2005-02-24 20:08:50 +01:00
if ( res0 )
freeaddrinfo ( res0 ) ;
2004-11-09 01:27:16 +01:00
return - 1 ;
}
2007-11-15 15:26:52 +01:00
static int udp_port ( struct sockaddr_storage * addr , int addr_len )
{
2008-02-11 04:30:42 +01:00
char sbuf [ sizeof ( int ) * 3 + 1 ] ;
2012-06-19 14:28:48 +02:00
int error ;
2007-11-15 15:26:52 +01:00
2012-06-19 14:28:48 +02:00
if ( ( error = getnameinfo ( ( struct sockaddr * ) addr , addr_len , NULL , 0 , sbuf , sizeof ( sbuf ) , NI_NUMERICSERV ) ) ! = 0 ) {
av_log ( NULL , AV_LOG_ERROR , " getnameinfo: %s \n " , gai_strerror ( error ) ) ;
2007-11-15 15:26:52 +01:00
return - 1 ;
}
return strtol ( sbuf , NULL , 10 ) ;
}
2004-11-09 01:27:16 +01:00
2002-07-24 19:45:41 +02:00
/**
* If no filename is given to av_open_input_file because you want to
* get the local port first , then you must call this function to set
* the remote server address .
*
* url syntax : udp : //host:port[?option=val...]
2008-04-15 13:23:07 +02:00
* option : ' ttl = n ' : set the ttl value ( for multicast only )
2002-07-24 19:45:41 +02:00
* ' localport = n ' : set the local port
2003-02-09 19:06:23 +01:00
* ' pkt_size = n ' : set max packet size
2006-10-28 19:16:18 +02:00
* ' reuse = 1 ' : enable reusing the socket
2012-03-12 16:09:53 +01:00
* ' overrun_nonfatal = 1 ' : survive in case of circular buffer overrun
2002-07-24 19:45:41 +02:00
*
2010-07-02 12:49:29 +02:00
* @ param h media file context
2002-07-24 19:45:41 +02:00
* @ param uri of the remote server
* @ return zero if no error .
*/
2011-03-08 10:35:52 +01:00
int ff_udp_set_remote_url ( URLContext * h , const char * uri )
2002-07-24 19:45:41 +02:00
{
UDPContext * s = h - > priv_data ;
2011-01-06 16:16:50 +01:00
char hostname [ 256 ] , buf [ 10 ] ;
2002-07-24 19:45:41 +02:00
int port ;
2011-01-06 16:16:50 +01:00
const char * p ;
2005-12-17 19:14:38 +01:00
2010-06-27 16:16:46 +02:00
av_url_split ( NULL , 0 , NULL , 0 , hostname , sizeof ( hostname ) , & port , NULL , 0 , uri ) ;
2001-07-22 16:18:56 +02:00
2002-07-24 19:45:41 +02:00
/* set the destination address */
2007-11-14 09:16:51 +01:00
s - > dest_addr_len = udp_set_url ( & s - > dest_addr , hostname , port ) ;
if ( s - > dest_addr_len < 0 ) {
2007-07-19 17:23:32 +02:00
return AVERROR ( EIO ) ;
2007-11-14 09:16:51 +01:00
}
2010-10-07 09:58:56 +02:00
s - > is_multicast = ff_is_multicast_address ( ( struct sockaddr * ) & s - > dest_addr ) ;
2011-01-06 16:16:50 +01:00
p = strchr ( uri , ' ? ' ) ;
if ( p ) {
2011-02-16 09:52:38 +01:00
if ( av_find_info_tag ( buf , sizeof ( buf ) , " connect " , p ) ) {
2011-01-06 16:16:50 +01:00
int was_connected = s - > is_connected ;
s - > is_connected = strtol ( buf , NULL , 10 ) ;
if ( s - > is_connected & & ! was_connected ) {
if ( connect ( s - > udp_fd , ( struct sockaddr * ) & s - > dest_addr ,
s - > dest_addr_len ) ) {
s - > is_connected = 0 ;
2012-06-19 14:37:06 +02:00
log_net_error ( h , AV_LOG_ERROR , " connect " ) ;
2011-01-06 16:16:50 +01:00
return AVERROR ( EIO ) ;
}
}
}
}
2007-11-14 09:16:51 +01:00
2002-07-24 19:45:41 +02:00
return 0 ;
}
/**
2010-06-14 11:09:59 +02:00
* Return the local port used by the UDP connection
2010-07-02 12:49:29 +02:00
* @ param h media file context
2002-07-24 19:45:41 +02:00
* @ return the local port number
*/
2011-03-08 10:35:52 +01:00
int ff_udp_get_local_port ( URLContext * h )
2002-07-24 19:45:41 +02:00
{
UDPContext * s = h - > priv_data ;
return s - > local_port ;
}
/**
* Return the udp file handle for select ( ) usage to wait for several RTP
* streams at the same time .
* @ param h media file context
*/
2011-02-03 12:39:11 +01:00
static int udp_get_file_handle ( URLContext * h )
2002-07-24 19:45:41 +02:00
{
UDPContext * s = h - > priv_data ;
return s - > udp_fd ;
}
2012-05-08 19:36:06 +02:00
# if HAVE_PTHREAD_CANCEL
2011-05-13 16:28:42 +02:00
static void * circular_buffer_task ( void * _URLContext )
{
URLContext * h = _URLContext ;
UDPContext * s = h - > priv_data ;
2012-03-15 13:03:38 +01:00
int old_cancelstate ;
2011-05-13 16:28:42 +02:00
2012-03-15 13:03:38 +01:00
pthread_setcancelstate ( PTHREAD_CANCEL_DISABLE , & old_cancelstate ) ;
2012-03-15 13:35:27 +01:00
pthread_mutex_lock ( & s - > mutex ) ;
2012-11-05 21:21:04 +01:00
if ( ff_socket_nonblock ( s - > udp_fd , 0 ) < 0 ) {
av_log ( h , AV_LOG_ERROR , " Failed to set blocking mode " ) ;
s - > circular_buffer_error = AVERROR ( EIO ) ;
goto end ;
}
2012-03-15 13:03:38 +01:00
while ( 1 ) {
2011-05-13 16:28:42 +02:00
int len ;
2012-03-15 13:35:27 +01:00
pthread_mutex_unlock ( & s - > mutex ) ;
2012-03-15 13:03:38 +01:00
/* Blocking operations are always cancellation points;
see " General Information " / " Thread Cancelation Overview "
in Single Unix . */
pthread_setcancelstate ( PTHREAD_CANCEL_ENABLE , & old_cancelstate ) ;
2011-08-25 21:33:54 +02:00
len = recv ( s - > udp_fd , s - > tmp + 4 , sizeof ( s - > tmp ) - 4 , 0 ) ;
2012-03-15 13:03:38 +01:00
pthread_setcancelstate ( PTHREAD_CANCEL_DISABLE , & old_cancelstate ) ;
2012-03-15 13:35:27 +01:00
pthread_mutex_lock ( & s - > mutex ) ;
2011-05-13 16:28:42 +02:00
if ( len < 0 ) {
if ( ff_neterrno ( ) ! = AVERROR ( EAGAIN ) & & ff_neterrno ( ) ! = AVERROR ( EINTR ) ) {
2012-03-15 13:42:34 +01:00
s - > circular_buffer_error = ff_neterrno ( ) ;
2011-12-23 01:17:18 +01:00
goto end ;
2011-05-13 16:28:42 +02:00
}
2011-08-25 21:43:30 +02:00
continue ;
2011-05-13 16:28:42 +02:00
}
2011-08-25 21:33:54 +02:00
AV_WL32 ( s - > tmp , len ) ;
2012-03-15 13:11:06 +01:00
if ( av_fifo_space ( s - > fifo ) < len + 4 ) {
2012-03-12 16:09:53 +01:00
/* No Space left */
if ( s - > overrun_nonfatal ) {
av_log ( h , AV_LOG_WARNING , " Circular buffer overrun. "
" Surviving due to overrun_nonfatal option \n " ) ;
continue ;
} else {
av_log ( h , AV_LOG_ERROR , " Circular buffer overrun. "
" To avoid, increase fifo_size URL option. "
" To survive in such case, use overrun_nonfatal option \n " ) ;
s - > circular_buffer_error = AVERROR ( EIO ) ;
goto end ;
}
}
2011-08-25 21:33:54 +02:00
av_fifo_generic_write ( s - > fifo , s - > tmp , len + 4 , NULL ) ;
2011-12-23 01:17:18 +01:00
pthread_cond_signal ( & s - > cond ) ;
2011-05-13 16:28:42 +02:00
}
2011-12-23 01:17:18 +01:00
end :
pthread_cond_signal ( & s - > cond ) ;
pthread_mutex_unlock ( & s - > mutex ) ;
2011-05-13 16:28:42 +02:00
return NULL ;
}
2011-12-23 01:17:18 +01:00
# endif
2011-05-13 16:28:42 +02:00
2002-07-24 19:45:41 +02:00
/* put it in UDP context */
2001-07-22 16:18:56 +02:00
/* return non zero if error */
static int udp_open ( URLContext * h , const char * uri , int flags )
{
2011-11-09 10:45:01 +01:00
char hostname [ 1024 ] , localaddr [ 1024 ] = " " ;
2008-12-07 18:29:09 +01:00
int port , udp_fd = - 1 , tmp , bind_ret = - 1 ;
2011-12-01 10:44:21 +01:00
UDPContext * s = h - > priv_data ;
2005-02-24 20:08:50 +01:00
int is_output ;
2002-07-24 19:45:41 +02:00
const char * p ;
char buf [ 256 ] ;
2007-11-15 15:26:52 +01:00
struct sockaddr_storage my_addr ;
2012-10-26 20:46:37 +02:00
socklen_t len ;
2011-02-07 22:14:43 +01:00
int reuse_specified = 0 ;
2012-06-21 13:19:56 +02:00
int i , include = 0 , num_sources = 0 ;
char * sources [ 32 ] ;
2001-07-22 16:18:56 +02:00
h - > is_streamed = 1 ;
2011-05-09 15:56:56 +02:00
is_output = ! ( flags & AVIO_FLAG_READ ) ;
2012-09-13 13:10:26 +02:00
if ( ! s - > buffer_size ) /* if not set explicitly */
s - > buffer_size = is_output ? UDP_TX_BUF_SIZE : UDP_MAX_PKT_SIZE ;
2011-05-13 16:28:42 +02:00
2002-07-24 19:45:41 +02:00
p = strchr ( uri , ' ? ' ) ;
if ( p ) {
2011-02-16 09:52:38 +01:00
if ( av_find_info_tag ( buf , sizeof ( buf ) , " reuse " , p ) ) {
2011-06-17 09:08:23 +02:00
char * endptr = NULL ;
2011-02-07 22:14:43 +01:00
s - > reuse_socket = strtol ( buf , & endptr , 10 ) ;
/* assume if no digits were found it is a request to enable it */
if ( buf = = endptr )
s - > reuse_socket = 1 ;
reuse_specified = 1 ;
}
2012-03-12 16:09:53 +01:00
if ( av_find_info_tag ( buf , sizeof ( buf ) , " overrun_nonfatal " , p ) ) {
char * endptr = NULL ;
s - > overrun_nonfatal = strtol ( buf , & endptr , 10 ) ;
/* assume if no digits were found it is a request to enable it */
if ( buf = = endptr )
s - > overrun_nonfatal = 1 ;
2012-07-18 16:33:24 +02:00
if ( ! HAVE_PTHREAD_CANCEL )
av_log ( h , AV_LOG_WARNING ,
" 'overrun_nonfatal' option was set but it is not supported "
" on this build (pthread support is required) \n " ) ;
2012-03-12 16:09:53 +01:00
}
2011-02-16 09:52:38 +01:00
if ( av_find_info_tag ( buf , sizeof ( buf ) , " ttl " , p ) ) {
2002-07-24 19:45:41 +02:00
s - > ttl = strtol ( buf , NULL , 10 ) ;
}
2011-02-16 09:52:38 +01:00
if ( av_find_info_tag ( buf , sizeof ( buf ) , " localport " , p ) ) {
2002-07-24 19:45:41 +02:00
s - > local_port = strtol ( buf , NULL , 10 ) ;
}
2011-02-16 09:52:38 +01:00
if ( av_find_info_tag ( buf , sizeof ( buf ) , " pkt_size " , p ) ) {
2012-09-13 13:10:26 +02:00
s - > packet_size = strtol ( buf , NULL , 10 ) ;
2003-02-09 19:06:23 +01:00
}
2011-02-16 09:52:38 +01:00
if ( av_find_info_tag ( buf , sizeof ( buf ) , " buffer_size " , p ) ) {
2008-11-12 22:40:53 +01:00
s - > buffer_size = strtol ( buf , NULL , 10 ) ;
}
2011-02-16 09:52:38 +01:00
if ( av_find_info_tag ( buf , sizeof ( buf ) , " connect " , p ) ) {
2010-10-08 10:42:30 +02:00
s - > is_connected = strtol ( buf , NULL , 10 ) ;
}
2011-07-25 15:50:13 +02:00
if ( av_find_info_tag ( buf , sizeof ( buf ) , " fifo_size " , p ) ) {
2012-09-13 13:10:26 +02:00
s - > circular_buffer_size = strtol ( buf , NULL , 10 ) ;
2012-07-18 16:33:24 +02:00
if ( ! HAVE_PTHREAD_CANCEL )
av_log ( h , AV_LOG_WARNING ,
" 'circular_buffer_size' option was set but it is not supported "
" on this build (pthread support is required) \n " ) ;
2011-05-13 16:28:42 +02:00
}
2011-11-09 10:45:01 +01:00
if ( av_find_info_tag ( buf , sizeof ( buf ) , " localaddr " , p ) ) {
av_strlcpy ( localaddr , buf , sizeof ( localaddr ) ) ;
}
2012-06-21 13:19:56 +02:00
if ( av_find_info_tag ( buf , sizeof ( buf ) , " sources " , p ) )
include = 1 ;
if ( include | | av_find_info_tag ( buf , sizeof ( buf ) , " block " , p ) ) {
char * source_start ;
source_start = buf ;
while ( 1 ) {
char * next = strchr ( source_start , ' , ' ) ;
if ( next )
* next = ' \0 ' ;
sources [ num_sources ] = av_strdup ( source_start ) ;
if ( ! sources [ num_sources ] )
goto fail ;
source_start = next + 1 ;
num_sources + + ;
if ( num_sources > = FF_ARRAY_ELEMS ( sources ) | | ! next )
break ;
}
}
2012-08-27 15:31:09 +02:00
if ( ! is_output & & av_find_info_tag ( buf , sizeof ( buf ) , " timeout " , p ) )
2012-09-13 13:10:26 +02:00
s - > timeout = strtol ( buf , NULL , 10 ) ;
2001-07-22 16:18:56 +02:00
}
2012-09-13 13:10:26 +02:00
/* handling needed to support options picking from both AVOption and URL */
s - > circular_buffer_size * = 188 ;
2013-06-28 23:58:13 +02:00
if ( flags & AVIO_FLAG_WRITE ) {
h - > max_packet_size = s - > packet_size ;
} else {
h - > max_packet_size = UDP_MAX_PKT_SIZE ;
}
2012-09-13 13:10:26 +02:00
h - > rw_timeout = s - > timeout ;
2002-07-24 19:45:41 +02:00
/* fill the dest addr */
2010-06-27 16:16:46 +02:00
av_url_split ( NULL , 0 , NULL , 0 , hostname , sizeof ( hostname ) , & port , NULL , 0 , uri ) ;
2005-12-17 19:14:38 +01:00
2010-06-27 16:16:46 +02:00
/* XXX: fix av_url_split */
2002-07-24 19:45:41 +02:00
if ( hostname [ 0 ] = = ' \0 ' | | hostname [ 0 ] = = ' ? ' ) {
/* only accepts null hostname if input */
2011-05-09 15:56:56 +02:00
if ( ! ( flags & AVIO_FLAG_READ ) )
2002-07-24 19:45:41 +02:00
goto fail ;
} else {
2011-03-08 10:35:52 +01:00
if ( ff_udp_set_remote_url ( h , uri ) < 0 )
2010-07-28 18:27:16 +02:00
goto fail ;
2002-07-24 19:45:41 +02:00
}
2001-07-22 16:18:56 +02:00
2011-06-17 07:58:01 +02:00
if ( ( s - > is_multicast | | ! s - > local_port ) & & ( h - > flags & AVIO_FLAG_READ ) )
2007-11-15 14:34:56 +01:00
s - > local_port = port ;
2012-09-13 13:10:26 +02:00
udp_fd = udp_socket_create ( s , & my_addr , & len , localaddr [ 0 ] ? localaddr : s - > local_addr ) ;
2002-07-24 19:45:41 +02:00
if ( udp_fd < 0 )
goto fail ;
2011-02-07 22:14:43 +01:00
/* Follow the requested reuse option, unless it's multicast in which
2011-10-05 14:12:42 +02:00
* case enable reuse unless explicitly disabled .
2011-02-07 22:14:43 +01:00
*/
if ( s - > reuse_socket | | ( s - > is_multicast & & ! reuse_specified ) ) {
s - > reuse_socket = 1 ;
2006-10-28 19:16:18 +02:00
if ( setsockopt ( udp_fd , SOL_SOCKET , SO_REUSEADDR , & ( s - > reuse_socket ) , sizeof ( s - > reuse_socket ) ) ! = 0 )
goto fail ;
2011-02-07 22:14:43 +01:00
}
2006-10-28 19:16:18 +02:00
2012-03-08 14:15:11 +01:00
/* If multicast, try binding the multicast address first, to avoid
* receiving UDP packets from other sources aimed at the same UDP
2012-03-08 14:17:15 +01:00
* port . This fails on windows . This makes sending to the same address
* using sendto ( ) fail , so only do it if we ' re opened in read - only mode . */
if ( s - > is_multicast & & ! ( h - > flags & AVIO_FLAG_WRITE ) ) {
2008-12-07 18:29:09 +01:00
bind_ret = bind ( udp_fd , ( struct sockaddr * ) & s - > dest_addr , len ) ;
}
/* bind to the local address if not multicast or if the multicast
* bind failed */
2012-03-07 23:38:02 +01:00
/* the bind is needed to give a port to the socket now */
2012-03-07 20:40:29 +01:00
if ( bind_ret < 0 & & bind ( udp_fd , ( struct sockaddr * ) & my_addr , len ) < 0 ) {
2012-06-19 14:37:06 +02:00
log_net_error ( h , AV_LOG_ERROR , " bind failed " ) ;
2001-07-22 16:18:56 +02:00
goto fail ;
2012-03-07 20:40:29 +01:00
}
2001-07-22 16:18:56 +02:00
2007-11-15 14:41:51 +01:00
len = sizeof ( my_addr ) ;
getsockname ( udp_fd , ( struct sockaddr * ) & my_addr , & len ) ;
2007-11-15 15:26:52 +01:00
s - > local_port = udp_port ( & my_addr , len ) ;
2004-11-09 01:27:16 +01:00
if ( s - > is_multicast ) {
2012-03-08 14:31:45 +01:00
if ( h - > flags & AVIO_FLAG_WRITE ) {
2007-11-14 08:42:46 +01:00
/* output */
2007-11-14 08:43:51 +01:00
if ( udp_set_multicast_ttl ( udp_fd , s - > ttl , ( struct sockaddr * ) & s - > dest_addr ) < 0 )
2004-11-09 01:27:16 +01:00
goto fail ;
2012-03-08 14:31:45 +01:00
}
if ( h - > flags & AVIO_FLAG_READ ) {
2007-11-14 08:42:46 +01:00
/* input */
2012-06-21 13:19:56 +02:00
if ( num_sources = = 0 | | ! include ) {
if ( udp_join_multicast_group ( udp_fd , ( struct sockaddr * ) & s - > dest_addr ) < 0 )
goto fail ;
if ( num_sources ) {
if ( udp_set_multicast_sources ( udp_fd , ( struct sockaddr * ) & s - > dest_addr , s - > dest_addr_len , sources , num_sources , 0 ) < 0 )
goto fail ;
}
} else if ( include & & num_sources ) {
if ( udp_set_multicast_sources ( udp_fd , ( struct sockaddr * ) & s - > dest_addr , s - > dest_addr_len , sources , num_sources , 1 ) < 0 )
goto fail ;
} else {
av_log ( NULL , AV_LOG_ERROR , " invalid udp settings: inclusive multicast but no sources given \n " ) ;
2004-11-09 01:27:16 +01:00
goto fail ;
2012-06-21 13:19:56 +02:00
}
2004-11-09 01:27:16 +01:00
}
}
2001-07-22 16:18:56 +02:00
2002-07-24 19:45:41 +02:00
if ( is_output ) {
/* limit the tx buf size to limit latency */
2008-11-12 22:40:53 +01:00
tmp = s - > buffer_size ;
2002-07-24 19:45:41 +02:00
if ( setsockopt ( udp_fd , SOL_SOCKET , SO_SNDBUF , & tmp , sizeof ( tmp ) ) < 0 ) {
2012-06-19 14:37:06 +02:00
log_net_error ( h , AV_LOG_ERROR , " setsockopt(SO_SNDBUF) " ) ;
2002-07-24 19:45:41 +02:00
goto fail ;
}
2007-08-17 00:12:03 +02:00
} else {
/* set udp recv buffer size to the largest possible udp packet size to
* avoid losing data on OSes that set this too low by default . */
2008-11-12 22:40:53 +01:00
tmp = s - > buffer_size ;
if ( setsockopt ( udp_fd , SOL_SOCKET , SO_RCVBUF , & tmp , sizeof ( tmp ) ) < 0 ) {
2012-06-19 14:37:06 +02:00
log_net_error ( h , AV_LOG_WARNING , " setsockopt(SO_RECVBUF) " ) ;
2008-11-12 22:40:53 +01:00
}
2008-11-17 08:50:25 +01:00
/* make the socket non-blocking */
ff_socket_nonblock ( udp_fd , 1 ) ;
2002-07-24 19:45:41 +02:00
}
2010-10-08 10:42:30 +02:00
if ( s - > is_connected ) {
if ( connect ( udp_fd , ( struct sockaddr * ) & s - > dest_addr , s - > dest_addr_len ) ) {
2012-06-19 14:37:06 +02:00
log_net_error ( h , AV_LOG_ERROR , " connect " ) ;
2010-10-08 10:42:30 +02:00
goto fail ;
}
}
2002-07-24 19:45:41 +02:00
2012-06-21 13:19:56 +02:00
for ( i = 0 ; i < num_sources ; i + + )
2012-06-28 01:09:43 +02:00
av_freep ( & sources [ i ] ) ;
2012-06-21 13:19:56 +02:00
2002-07-24 19:45:41 +02:00
s - > udp_fd = udp_fd ;
2011-05-13 16:28:42 +02:00
2012-05-08 19:36:06 +02:00
# if HAVE_PTHREAD_CANCEL
2011-05-13 16:28:42 +02:00
if ( ! is_output & & s - > circular_buffer_size ) {
2012-01-14 17:14:18 +01:00
int ret ;
2011-05-13 16:28:42 +02:00
/* start the task going */
2011-05-13 17:16:15 +02:00
s - > fifo = av_fifo_alloc ( s - > circular_buffer_size ) ;
2012-01-14 17:14:18 +01:00
ret = pthread_mutex_init ( & s - > mutex , NULL ) ;
if ( ret ! = 0 ) {
av_log ( h , AV_LOG_ERROR , " pthread_mutex_init failed : %s \n " , strerror ( ret ) ) ;
2011-05-13 16:28:42 +02:00
goto fail ;
}
2012-01-14 17:14:18 +01:00
ret = pthread_cond_init ( & s - > cond , NULL ) ;
if ( ret ! = 0 ) {
av_log ( h , AV_LOG_ERROR , " pthread_cond_init failed : %s \n " , strerror ( ret ) ) ;
goto cond_fail ;
}
ret = pthread_create ( & s - > circular_buffer_thread , NULL , circular_buffer_task , h ) ;
if ( ret ! = 0 ) {
av_log ( h , AV_LOG_ERROR , " pthread_create failed : %s \n " , strerror ( ret ) ) ;
goto thread_fail ;
}
s - > thread_started = 1 ;
2011-05-13 16:28:42 +02:00
}
2011-06-04 16:55:22 +02:00
# endif
2011-05-13 16:28:42 +02:00
2001-07-22 16:18:56 +02:00
return 0 ;
2012-05-08 19:36:06 +02:00
# if HAVE_PTHREAD_CANCEL
2012-01-14 17:14:18 +01:00
thread_fail :
pthread_cond_destroy ( & s - > cond ) ;
cond_fail :
pthread_mutex_destroy ( & s - > mutex ) ;
# endif
2001-07-22 16:18:56 +02:00
fail :
2002-07-24 19:45:41 +02:00
if ( udp_fd > = 0 )
2002-11-02 11:35:07 +01:00
closesocket ( udp_fd ) ;
2011-06-03 16:08:16 +02:00
av_fifo_free ( s - > fifo ) ;
2012-06-21 13:19:56 +02:00
for ( i = 0 ; i < num_sources ; i + + )
2012-06-28 01:09:43 +02:00
av_freep ( & sources [ i ] ) ;
2007-07-19 17:23:32 +02:00
return AVERROR ( EIO ) ;
2001-07-22 16:18:56 +02:00
}
2003-02-11 17:35:48 +01:00
static int udp_read ( URLContext * h , uint8_t * buf , int size )
2001-07-22 16:18:56 +02:00
{
UDPContext * s = h - > priv_data ;
2008-11-14 20:33:22 +01:00
int ret ;
2012-03-15 12:19:37 +01:00
int avail , nonblock = h - > flags & AVIO_FLAG_NONBLOCK ;
2011-05-13 16:28:42 +02:00
2012-05-08 19:36:06 +02:00
# if HAVE_PTHREAD_CANCEL
2011-05-14 13:34:08 +02:00
if ( s - > fifo ) {
2011-12-23 01:17:18 +01:00
pthread_mutex_lock ( & s - > mutex ) ;
2011-05-13 16:28:42 +02:00
do {
2011-05-13 17:16:15 +02:00
avail = av_fifo_size ( s - > fifo ) ;
2011-05-13 16:28:42 +02:00
if ( avail ) { // >=size) {
2011-08-25 21:33:54 +02:00
uint8_t tmp [ 4 ] ;
av_fifo_generic_read ( s - > fifo , tmp , 4 , NULL ) ;
avail = AV_RL32 ( tmp ) ;
if ( avail > size ) {
av_log ( h , AV_LOG_WARNING , " Part of datagram lost due to insufficient buffer size \n " ) ;
avail = size ;
}
2011-05-13 16:28:42 +02:00
2011-08-25 21:33:54 +02:00
av_fifo_generic_read ( s - > fifo , buf , avail , NULL ) ;
2011-11-09 03:51:08 +01:00
av_fifo_drain ( s - > fifo , AV_RL32 ( tmp ) - avail ) ;
2012-03-15 13:35:27 +01:00
pthread_mutex_unlock ( & s - > mutex ) ;
2011-08-25 21:33:54 +02:00
return avail ;
2011-12-23 01:41:15 +01:00
} else if ( s - > circular_buffer_error ) {
2012-03-15 13:35:27 +01:00
int err = s - > circular_buffer_error ;
2011-12-23 01:17:18 +01:00
pthread_mutex_unlock ( & s - > mutex ) ;
2012-03-15 13:35:27 +01:00
return err ;
2012-03-15 12:19:37 +01:00
} else if ( nonblock ) {
2011-12-23 01:51:47 +01:00
pthread_mutex_unlock ( & s - > mutex ) ;
return AVERROR ( EAGAIN ) ;
2011-05-13 16:28:42 +02:00
}
else {
2012-03-15 12:19:37 +01:00
/* FIXME: using the monotonic clock would be better,
but it does not exist on all supported platforms . */
int64_t t = av_gettime ( ) + 100000 ;
struct timespec tv = { . tv_sec = t / 1000000 ,
. tv_nsec = ( t % 1000000 ) * 1000 } ;
2012-10-20 23:31:32 +02:00
if ( pthread_cond_timedwait ( & s - > cond , & s - > mutex , & tv ) < 0 ) {
pthread_mutex_unlock ( & s - > mutex ) ;
2012-03-15 12:19:37 +01:00
return AVERROR ( errno = = ETIMEDOUT ? EAGAIN : errno ) ;
2012-10-20 23:31:32 +02:00
}
2012-03-15 12:19:37 +01:00
nonblock = 1 ;
2011-05-13 16:28:42 +02:00
}
} while ( 1 ) ;
}
2011-12-23 01:17:18 +01:00
# endif
2002-07-24 19:45:41 +02:00
2011-04-04 20:11:19 +02:00
if ( ! ( h - > flags & AVIO_FLAG_NONBLOCK ) ) {
2011-04-04 18:17:12 +02:00
ret = ff_network_wait_fd ( s - > udp_fd , 0 ) ;
if ( ret < 0 )
return ret ;
2002-07-24 19:45:41 +02:00
}
2011-04-04 18:17:12 +02:00
ret = recv ( s - > udp_fd , buf , size , 0 ) ;
2011-05-13 16:28:42 +02:00
2011-04-04 18:17:12 +02:00
return ret < 0 ? ff_neterrno ( ) : ret ;
2001-07-22 16:18:56 +02:00
}
2010-06-01 09:46:23 +02:00
static int udp_write ( URLContext * h , const uint8_t * buf , int size )
2001-07-22 16:18:56 +02:00
{
UDPContext * s = h - > priv_data ;
2002-07-24 19:45:41 +02:00
int ret ;
2011-04-04 20:11:19 +02:00
if ( ! ( h - > flags & AVIO_FLAG_NONBLOCK ) ) {
2011-04-04 18:17:12 +02:00
ret = ff_network_wait_fd ( s - > udp_fd , 1 ) ;
if ( ret < 0 )
return ret ;
2001-07-22 16:18:56 +02:00
}
2011-04-04 18:17:12 +02:00
if ( ! s - > is_connected ) {
ret = sendto ( s - > udp_fd , buf , size , 0 ,
( struct sockaddr * ) & s - > dest_addr ,
s - > dest_addr_len ) ;
} else
ret = send ( s - > udp_fd , buf , size , 0 ) ;
return ret < 0 ? ff_neterrno ( ) : ret ;
2002-07-24 19:45:41 +02:00
}
static int udp_close ( URLContext * h )
{
UDPContext * s = h - > priv_data ;
2012-01-14 17:14:18 +01:00
int ret ;
2002-07-24 19:45:41 +02:00
2011-05-09 15:56:56 +02:00
if ( s - > is_multicast & & ( h - > flags & AVIO_FLAG_READ ) )
2007-11-14 08:43:51 +01:00
udp_leave_multicast_group ( s - > udp_fd , ( struct sockaddr * ) & s - > dest_addr ) ;
2002-11-02 11:35:07 +01:00
closesocket ( s - > udp_fd ) ;
2012-05-08 19:36:06 +02:00
# if HAVE_PTHREAD_CANCEL
2012-01-14 17:14:18 +01:00
if ( s - > thread_started ) {
2012-03-15 13:03:38 +01:00
pthread_cancel ( s - > circular_buffer_thread ) ;
2012-01-14 17:14:18 +01:00
ret = pthread_join ( s - > circular_buffer_thread , NULL ) ;
if ( ret ! = 0 )
av_log ( h , AV_LOG_ERROR , " pthread_join(): %s \n " , strerror ( ret ) ) ;
2012-08-06 09:56:50 +02:00
pthread_mutex_destroy ( & s - > mutex ) ;
pthread_cond_destroy ( & s - > cond ) ;
2012-01-14 17:14:18 +01:00
}
2011-12-23 01:17:18 +01:00
# endif
2012-06-27 15:09:40 +02:00
av_fifo_free ( s - > fifo ) ;
2002-07-24 19:45:41 +02:00
return 0 ;
2001-07-22 16:18:56 +02:00
}
2011-01-25 23:03:28 +01:00
URLProtocol ff_udp_protocol = {
2011-04-08 07:41:47 +02:00
. name = " udp " ,
. url_open = udp_open ,
. url_read = udp_read ,
. url_write = udp_write ,
. url_close = udp_close ,
2009-03-03 18:04:51 +01:00
. url_get_file_handle = udp_get_file_handle ,
2011-12-01 10:44:21 +01:00
. priv_data_size = sizeof ( UDPContext ) ,
2012-09-13 13:10:26 +02:00
. priv_data_class = & udp_context_class ,
2011-12-30 10:38:05 +01:00
. flags = URL_PROTOCOL_FLAG_NETWORK ,
2001-07-22 16:18:56 +02:00
} ;