]> git.sesse.net Git - vlc/commitdiff
* all: added a TCP access input module.
authorLaurent Aimar <fenrir@videolan.org>
Sun, 30 Nov 2003 14:28:07 +0000 (14:28 +0000)
committerLaurent Aimar <fenrir@videolan.org>
Sun, 30 Nov 2003 14:28:07 +0000 (14:28 +0000)
configure.ac
modules/access/Modules.am
modules/access/tcp.c [new file with mode: 0644]

index 605c7e9b0295577a9ecd1c54830fca735207f54d..284bf8b9b3ba5bc470009d766366dcd791d385d4 100644 (file)
@@ -1,5 +1,5 @@
 dnl Autoconf settings for vlc
-dnl $Id: configure.ac,v 1.123 2003/11/27 10:13:11 hartman Exp $
+dnl $Id: configure.ac,v 1.124 2003/11/30 14:28:07 fenrir Exp $
 
 AC_INIT(vlc,0.7.0-test1)
 
@@ -142,7 +142,7 @@ case "${target_os}" in
         # add ws2_32 for closesocket, select, recv
         CPPFLAGS_save="${CPPFLAGS_save} -D_OFF_T_ -D_off_t=long"; CPPFLAGS="${CPPFLAGS_save}"
         AX_ADD_LDFLAGS([vlc],[-lws2_32 -lnetapi32 -lwinmm -mwindows])
-        AX_ADD_LDFLAGS([ipv4 ipv6 access_http access_mms access_udp access_ftp access_output_udp sap slp http httpd stream_out_standard],[-lws2_32])
+        AX_ADD_LDFLAGS([ipv4 ipv6 access_http access_mms access_udp access_tcp access_ftp access_output_udp sap slp http httpd stream_out_standard],[-lws2_32])
     fi
     ;;
   *nto*)
@@ -301,7 +301,7 @@ AC_CHECK_FUNCS(connect,,[
 
 AC_CHECK_FUNCS(send,,[
   AC_CHECK_LIB(socket,send,[
-    AX_ADD_LDFLAGS([access_http access_mms access_udp access_ftp sap httpd access_output_udp stream_out_standard],[-lsocket])
+    AX_ADD_LDFLAGS([access_http access_mms access_udp access_tcp access_ftp sap httpd access_output_udp stream_out_standard],[-lsocket])
   ])
 ])
 
@@ -879,7 +879,7 @@ AX_ADD_PLUGINS([i420_rgb i420_yuy2 i422_yuy2 i420_ymga])
 AX_ADD_PLUGINS([id3 m3u])
 AX_ADD_PLUGINS([rawvideo])
 AX_ADD_PLUGINS([wav araw demuxdump demuxsub adpcm a52sys au])
-AX_ADD_PLUGINS([access_file access_udp access_http ipv4 access_mms])
+AX_ADD_PLUGINS([access_file access_udp access_tcp access_http ipv4 access_mms])
 AX_ADD_PLUGINS([access_ftp access_directory sap httpd http])
 
 dnl
index 32f39d8574273f24ba61afef2f2f680cf026b202..c2a327ac7e5272978534f658923e647c6011ad23 100644 (file)
@@ -1,6 +1,7 @@
 SOURCES_access_file = file.c
 SOURCES_access_directory = directory.c
 SOURCES_access_udp = udp.c
+SOURCES_access_tcp = tcp.c
 SOURCES_access_http = http.c
 SOURCES_access_ftp = ftp.c
 SOURCES_slp = slp.c
diff --git a/modules/access/tcp.c b/modules/access/tcp.c
new file mode 100644 (file)
index 0000000..4658eb0
--- /dev/null
@@ -0,0 +1,268 @@
+/*****************************************************************************
+ * tcp.c: TCP access plug-in
+ *****************************************************************************
+ * Copyright (C) 2003 VideoLAN
+ * $Id: tcp.c,v 1.1 2003/11/30 14:28:07 fenrir Exp $
+ *
+ * Authors: Laurent Aimar <fenrir@via.ecp.fr>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111, USA.
+ *****************************************************************************/
+
+/*****************************************************************************
+ * Preamble
+ *****************************************************************************/
+#include <stdlib.h>
+
+#include <vlc/vlc.h>
+#include <vlc/input.h>
+
+#include <sys/stat.h>
+#include <errno.h>
+#include <fcntl.h>
+
+#ifdef HAVE_SYS_TIME_H
+#    include <sys/time.h>
+#endif
+
+#ifdef HAVE_UNISTD_H
+#   include <unistd.h>
+#endif
+
+#ifdef WIN32
+#   include <winsock2.h>
+#   include <ws2tcpip.h>
+#   ifndef IN_MULTICAST
+#       define IN_MULTICAST(a) IN_CLASSD(a)
+#   endif
+#else
+#   include <sys/socket.h>
+#endif
+
+#include "network.h"
+
+/*****************************************************************************
+ * Module descriptor
+ *****************************************************************************/
+#define CACHING_TEXT N_("caching value in ms")
+#define CACHING_LONGTEXT N_( \
+    "Allows you to modify the default caching value for udp streams. This " \
+    "value should be set in miliseconds units." )
+
+static int  Open ( vlc_object_t * );
+static void Close( vlc_object_t * );
+
+vlc_module_begin();
+    set_description( _("TCP input") );
+    add_category_hint( N_("tcp"), NULL , VLC_TRUE );
+    add_integer( "tcp-caching", DEFAULT_PTS_DELAY / 1000, NULL, CACHING_TEXT, CACHING_LONGTEXT, VLC_TRUE );
+    set_capability( "access", 0 );
+    add_shortcut( "tcp" );
+    add_shortcut( "tcp4" );
+    add_shortcut( "tcp6" );
+    set_callbacks( Open, Close );
+vlc_module_end();
+
+
+/*****************************************************************************
+ * Local prototypes
+ *****************************************************************************/
+struct access_sys_t
+{
+    int fd;
+};
+
+static ssize_t  Read ( input_thread_t *, byte_t *, size_t );
+
+/*****************************************************************************
+ * Open: open the socket
+ *****************************************************************************/
+static int Open( vlc_object_t *p_this )
+{
+    input_thread_t *p_input = (input_thread_t *)p_this;
+    access_sys_t   *p_sys;
+
+    char           *psz_network;
+    char           *psz_dup = strdup(p_input->psz_name);
+    char           *psz_parser = psz_dup;
+
+    network_socket_t sock;
+    module_t         *p_network;
+
+    vlc_value_t     val;
+
+    /* Select ip version */
+    psz_network = "";
+    if( config_GetInt( p_input, "ipv4" ) )
+    {
+        psz_network = "ipv4";
+    }
+    if( config_GetInt( p_input, "ipv6" ) )
+    {
+        psz_network = "ipv6";
+    }
+    if( *p_input->psz_access )
+    {
+        /* Find out which shortcut was used */
+        if( !strncmp( p_input->psz_access, "tcp6", 5 ) )
+        {
+            psz_network = "ipv6";
+        }
+        else if( !strncmp( p_input->psz_access, "tcp4", 5 ) )
+        {
+            psz_network = "ipv4";
+        }
+    }
+
+    /* Parse server:port */
+    while( *psz_parser && *psz_parser != ':' )
+    {
+        if( *psz_parser == '[' )
+        {
+            /* IPV6 */
+            while( *psz_parser && *psz_parser  != ']' )
+            {
+                psz_parser++;
+            }
+        }
+        psz_parser++;
+    }
+
+    if( *psz_parser != ':' || psz_parser == psz_dup )
+    {
+        msg_Err( p_input, "you have to provide server:port addresse" );
+        free( psz_dup );
+        return VLC_EGENERIC;
+    }
+
+    *psz_parser++ = '\0';
+
+    /* Prepare the network_socket_t structure */
+    sock.i_type = NETWORK_TCP;
+    sock.psz_bind_addr = "";
+    sock.i_bind_port = 0;
+    sock.psz_server_addr = psz_dup;
+    sock.i_server_port = atoi( psz_parser );
+    sock.i_ttl           = 0;
+
+    if( sock.i_server_port <= 0 )
+    {
+        msg_Err( p_input, "invalid port number (%d)", sock.i_server_port );
+        free( psz_dup );
+        return VLC_EGENERIC;
+    }
+
+    /* connecting */
+    msg_Dbg( p_input, "opening server=%s:%d",
+             sock.psz_server_addr, sock.i_server_port );
+    p_input->p_private = (void*)&sock;
+    p_network = module_Need( p_input, "network", psz_network );
+    free( psz_dup );
+    if( p_network == NULL )
+    {
+        return VLC_EGENERIC;
+    }
+    module_Unneed( p_input, p_network );
+
+    p_input->p_access_data = p_sys = malloc( sizeof( access_sys_t ) );
+    p_sys->fd = sock.i_handle;
+
+    p_input->pf_read = Read;
+    p_input->pf_set_program = input_SetProgram;
+    p_input->pf_set_area = NULL;
+    p_input->pf_seek = NULL;
+
+    vlc_mutex_lock( &p_input->stream.stream_lock );
+    p_input->stream.b_pace_control = VLC_TRUE;  /* FIXME ? */
+    p_input->stream.b_seekable = 0;
+    p_input->stream.p_selected_area->i_tell = 0;
+    p_input->stream.i_method = INPUT_METHOD_NETWORK;
+    p_input->i_mtu = 0;
+    vlc_mutex_unlock( &p_input->stream.stream_lock );
+
+    /* Update default_pts to a suitable value for udp access */
+    var_Create( p_input, "tcp-caching", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
+    var_Get( p_input, "tcp-caching", &val );
+    p_input->i_pts_delay = val.i_int * 1000;
+
+    return VLC_SUCCESS;
+}
+
+/*****************************************************************************
+ * Close: free unused data structures
+ *****************************************************************************/
+static void Close( vlc_object_t *p_this )
+{
+    input_thread_t *p_input = (input_thread_t *)p_this;
+    access_sys_t   *p_sys = p_input->p_access_data;
+
+    msg_Info( p_input, "closing TCP target `%s'", p_input->psz_source );
+
+#ifdef UNDER_CE
+    CloseHandle( (HANDLE)p_sys->fd );
+#elif defined( WIN32 )
+    closesocket( p_sys->fd );
+#else
+    close( p_sys->fd );
+#endif
+
+    free( p_sys );
+}
+
+/*****************************************************************************
+ * Read: read on a file descriptor, checking b_die periodically
+ *****************************************************************************/
+static ssize_t Read( input_thread_t * p_input, byte_t * p_buffer, size_t i_len )
+{
+#ifdef UNDER_CE
+    return -1;
+#else
+    access_sys_t   *p_sys = p_input->p_access_data;
+    struct timeval  timeout;
+    fd_set          fds;
+    ssize_t         i_recv;
+    int             i_ret;
+
+    do
+    {
+        if( p_input->b_die || p_input->b_error )
+        {
+            return 0;
+        }
+
+        /* Initialize file descriptor set */
+        FD_ZERO( &fds );
+        FD_SET( p_sys->fd, &fds );
+
+        /* We'll wait 0.5 second if nothing happens */
+        timeout.tv_sec = 0;
+        timeout.tv_usec = 500000;
+    } while( ( i_ret = select( p_sys->fd + 1, &fds, NULL, NULL, &timeout )) == 0 ||
+             ( i_ret < 0 && errno == EINTR ) );
+
+    if( i_ret < 0 )
+    {
+        msg_Err( p_input, "network select error (%s)", strerror(errno) );
+        return -1;
+    }
+
+    if( ( i_recv = recv( p_sys->fd, p_buffer, i_len, 0 ) ) < 0 )
+    {
+        msg_Err( p_input, "recv failed (%s)", strerror(errno) );
+    }
+    return i_recv;
+#endif
+}
+