udp.c 7.79 KB
Newer Older
Christophe Massiot's avatar
Christophe Massiot committed
1 2 3 4
/*****************************************************************************
 * udp.c: raw UDP access plug-in
 *****************************************************************************
 * Copyright (C) 2001, 2002 VideoLAN
5
 * $Id: udp.c,v 1.12 2002/07/31 20:56:50 sam Exp $
Christophe Massiot's avatar
Christophe Massiot committed
6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33
 *
 * Authors: Christophe Massiot <massiot@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 <sys/types.h>
#include <sys/stat.h>
#include <string.h>
#include <errno.h>
#include <fcntl.h>

34 35
#include <vlc/vlc.h>
#include <vlc/input.h>
Christophe Massiot's avatar
Christophe Massiot committed
36 37 38 39 40 41 42 43 44 45 46 47

#ifdef HAVE_UNISTD_H
#   include <unistd.h>
#elif defined( _MSC_VER ) && defined( _WIN32 )
#   include <io.h>
#endif

#include "network.h"

/*****************************************************************************
 * Local prototypes
 *****************************************************************************/
48
static int  Open       ( vlc_object_t * );
Christophe Massiot's avatar
Christophe Massiot committed
49 50

/*****************************************************************************
51
 * Module descriptor
Christophe Massiot's avatar
Christophe Massiot committed
52
 *****************************************************************************/
53 54 55 56 57 58 59 60
vlc_module_begin();
    set_description( _("raw UDP access module") );
    set_capability( "access", 0 );
    add_shortcut( "udpstream" );
    add_shortcut( "udp4" );
    add_shortcut( "udp6" );
    set_callbacks( Open, __input_FDNetworkClose );
vlc_module_end();
Christophe Massiot's avatar
Christophe Massiot committed
61 62

/*****************************************************************************
63
 * Open: open the socket
Christophe Massiot's avatar
Christophe Massiot committed
64
 *****************************************************************************/
65
static int Open( vlc_object_t *p_this )
Christophe Massiot's avatar
Christophe Massiot committed
66
{
67
    input_thread_t *    p_input = (input_thread_t *)p_this;
Christophe Massiot's avatar
Christophe Massiot committed
68
    input_socket_t *    p_access_data;
69
    module_t *          p_network;
Sam Hocevar's avatar
 
Sam Hocevar committed
70
    char *              psz_network = "";
Christophe Massiot's avatar
Christophe Massiot committed
71 72
    char *              psz_name = strdup(p_input->psz_name);
    char *              psz_parser = psz_name;
Sam Hocevar's avatar
 
Sam Hocevar committed
73 74 75 76
    char *              psz_server_addr = "";
    char *              psz_server_port = "";
    char *              psz_bind_addr = "";
    char *              psz_bind_port = "";
Christophe Massiot's avatar
Christophe Massiot committed
77 78 79
    int                 i_bind_port = 0, i_server_port = 0;
    network_socket_t    socket_desc;

80
    if( config_GetInt( p_input, "ipv4" ) )
81 82 83
    {
        psz_network = "ipv4";
    }
84
    if( config_GetInt( p_input, "ipv6" ) )
85 86 87 88
    {
        psz_network = "ipv6";
    }

Sam Hocevar's avatar
 
Sam Hocevar committed
89
    if( *p_input->psz_access )
Christophe Massiot's avatar
Christophe Massiot committed
90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111
    {
        /* Find out which shortcut was used */
        if( !strncmp( p_input->psz_access, "udp6", 5 ) )
        {
            psz_network = "ipv6";
        }
        else if( !strncmp( p_input->psz_access, "udp4", 5 ) )
        {
            psz_network = "ipv4";
        }
    }

    /* Parse psz_name syntax :
     * [serveraddr[:serverport]][@[bindaddr]:[bindport]] */

    if( *psz_parser && *psz_parser != '@' )
    {
        /* Found server */
        psz_server_addr = psz_parser;

        while( *psz_parser && *psz_parser != ':' && *psz_parser != '@' )
        {
112 113 114 115 116 117 118 119
            if( *psz_parser == '[' )
            {
                /* IPv6 address */
                while( *psz_parser && *psz_parser != ']' )
                {
                    psz_parser++;
                }
            }
Christophe Massiot's avatar
Christophe Massiot committed
120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139
            psz_parser++;
        }

        if( *psz_parser == ':' )
        {
            /* Found server port */
            *psz_parser = '\0'; /* Terminate server name */
            psz_parser++;
            psz_server_port = psz_parser;

            while( *psz_parser && *psz_parser != '@' )
            {
                psz_parser++;
            }
        }
    }

    if( *psz_parser == '@' )
    {
        /* Found bind address or bind port */
140 141
        *psz_parser = '\0'; /* Terminate server port or name if necessary */
        psz_parser++;
Christophe Massiot's avatar
Christophe Massiot committed
142 143 144 145 146 147 148 149

        if( *psz_parser && *psz_parser != ':' )
        {
            /* Found bind address */
            psz_bind_addr = psz_parser;

            while( *psz_parser && *psz_parser != ':' )
            {
150 151 152 153 154 155 156 157
                if( *psz_parser == '[' )
                {
                    /* IPv6 address */
                    while( *psz_parser && *psz_parser != ']' )
                    {
                        psz_parser++;
                    }
                }
Christophe Massiot's avatar
Christophe Massiot committed
158 159 160 161 162 163 164 165 166 167 168 169 170 171 172
                psz_parser++;
            }
        }

        if( *psz_parser == ':' )
        {
            /* Found bind port */
            *psz_parser = '\0'; /* Terminate bind address if necessary */
            psz_parser++;

            psz_bind_port = psz_parser;
        }
    }

    /* Convert ports format */
Sam Hocevar's avatar
 
Sam Hocevar committed
173
    if( *psz_server_port )
Christophe Massiot's avatar
Christophe Massiot committed
174 175 176 177
    {
        i_server_port = strtol( psz_server_port, &psz_parser, 10 );
        if( *psz_parser )
        {
178
            msg_Err( p_input, "cannot parse server port near %s", psz_parser );
Christophe Massiot's avatar
Christophe Massiot committed
179
            free(psz_name);
Christophe Massiot's avatar
Christophe Massiot committed
180 181 182 183
            return( -1 );
        }
    }

Sam Hocevar's avatar
 
Sam Hocevar committed
184
    if( *psz_bind_port )
Christophe Massiot's avatar
Christophe Massiot committed
185 186 187 188
    {
        i_bind_port = strtol( psz_bind_port, &psz_parser, 10 );
        if( *psz_parser )
        {
189
            msg_Err( p_input, "cannot parse bind port near %s", psz_parser );
Christophe Massiot's avatar
Christophe Massiot committed
190
            free(psz_name);
Christophe Massiot's avatar
Christophe Massiot committed
191 192 193 194
            return( -1 );
        }
    }

195 196 197 198 199
    p_input->pf_read = input_FDNetworkRead;
    p_input->pf_set_program = input_SetProgram;
    p_input->pf_set_area = NULL;
    p_input->pf_seek = NULL;

Christophe Massiot's avatar
Christophe Massiot committed
200 201 202 203 204 205
    vlc_mutex_lock( &p_input->stream.stream_lock );
    p_input->stream.b_pace_control = 0;
    p_input->stream.b_seekable = 0;
    p_input->stream.p_selected_area->i_tell = 0;
    p_input->stream.i_method = INPUT_METHOD_NETWORK;
    vlc_mutex_unlock( &p_input->stream.stream_lock );
206 207 208

    if( *psz_server_addr || i_server_port )
    {
209 210 211 212 213 214
        msg_Err( p_input, "this UDP syntax is deprecated; the server argument will be");
        msg_Err( p_input, "ignored (%s:%d). If you wanted to enter a multicast address",
                          psz_server_addr, i_server_port);
        msg_Err( p_input, "or local port, type : %s:@%s:%d",
                          *p_input->psz_access ? p_input->psz_access : "udp",
                          psz_server_addr, i_server_port );
215 216 217 218

        i_server_port = 0;
        psz_server_addr = "";
    }
Christophe Massiot's avatar
Christophe Massiot committed
219
 
220 221
    msg_Dbg( p_input, "opening server=%s:%d local=%s:%d",
             psz_server_addr, i_server_port, psz_bind_addr, i_bind_port );
Christophe Massiot's avatar
Christophe Massiot committed
222 223 224 225 226 227 228 229 230

    /* Prepare the network_socket_t structure */
    socket_desc.i_type = NETWORK_UDP;
    socket_desc.psz_bind_addr = psz_bind_addr;
    socket_desc.i_bind_port = i_bind_port;
    socket_desc.psz_server_addr = psz_server_addr;
    socket_desc.i_server_port = i_server_port;

    /* Find an appropriate network module */
231 232
    p_input->p_private = (void*) &socket_desc;
    p_network = module_Need( p_input, "network", psz_network );
Christophe Massiot's avatar
Christophe Massiot committed
233
    free(psz_name);
Christophe Massiot's avatar
Christophe Massiot committed
234 235 236 237
    if( p_network == NULL )
    {
        return( -1 );
    }
238
    module_Unneed( p_input, p_network );
Christophe Massiot's avatar
Christophe Massiot committed
239 240 241 242
    
    p_access_data = p_input->p_access_data = malloc( sizeof(input_socket_t) );
    if( p_access_data == NULL )
    {
243
        msg_Err( p_input, "out of memory" );
Christophe Massiot's avatar
Christophe Massiot committed
244 245 246 247 248 249 250 251
        return( -1 );
    }

    p_access_data->i_handle = socket_desc.i_handle;
    p_input->i_mtu = socket_desc.i_mtu;

    return( 0 );
}