From e5daac69bbf814759ffe6ab126aacffe2efb340a Mon Sep 17 00:00:00 2001
From: Arnaud Schauly <gitan@videolan.org>
Date: Tue, 3 Dec 2002 16:29:04 +0000
Subject: [PATCH] * Added a session announcement protol module (enabled by
 default). * Added the PLAYLIST_CHECK_INSERT option to the playlist. That
 option checks previously enqueued sessions before enqueing.

---
 configure.ac.in         |   2 +-
 include/vlc/vlc.h       |  11 +-
 modules/misc/Modules.am |   1 +
 modules/misc/sap.c      | 412 ++++++++++++++++++++++++++++++++++++++++
 src/libvlc.c            |  10 +-
 src/libvlc.h            |   7 +-
 src/playlist/playlist.c |  34 +++-
 7 files changed, 466 insertions(+), 11 deletions(-)
 create mode 100644 modules/misc/sap.c

diff --git a/configure.ac.in b/configure.ac.in
index aa8c1f99b6..863e745d30 100644
--- a/configure.ac.in
+++ b/configure.ac.in
@@ -625,7 +625,7 @@ PLUGINS="${PLUGINS} wav araw demuxdump demuxsub"
 dnl
 dnl  Network modules
 dnl
-NETWORK_MODULES="access_udp access_http access_rtp ipv4 access_mms"
+NETWORK_MODULES="access_udp access_http access_rtp ipv4 access_mms sap"
 
 dnl
 dnl  Accelerated modules
diff --git a/include/vlc/vlc.h b/include/vlc/vlc.h
index 769acc894f..7949f0d9cc 100644
--- a/include/vlc/vlc.h
+++ b/include/vlc/vlc.h
@@ -2,7 +2,7 @@
  * vlc.h: global header for vlc
  *****************************************************************************
  * Copyright (C) 1998, 1999, 2000 VideoLAN
- * $Id: vlc.h,v 1.17 2002/10/17 13:15:30 sam Exp $
+ * $Id: vlc.h,v 1.18 2002/12/03 16:29:04 gitan Exp $
  *
  * 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
@@ -74,10 +74,11 @@ typedef union
  *****************************************************************************/
 
 /* Used by playlist_Add */
-#define PLAYLIST_INSERT      0x0001
-#define PLAYLIST_REPLACE     0x0002
-#define PLAYLIST_APPEND      0x0004
-#define PLAYLIST_GO          0x0008
+#define PLAYLIST_INSERT          0x0001
+#define PLAYLIST_REPLACE         0x0002
+#define PLAYLIST_APPEND          0x0004
+#define PLAYLIST_GO              0x0008
+#define PLAYLIST_CHECK_INSERT    0x0010
 
 #define PLAYLIST_END           -666
 
diff --git a/modules/misc/Modules.am b/modules/misc/Modules.am
index b6a1f2b3fc..737f7a35e5 100644
--- a/modules/misc/Modules.am
+++ b/modules/misc/Modules.am
@@ -1,2 +1,3 @@
 SOURCES_gtk_main = modules/misc/gtk_main.c
 SOURCES_gnome_main = modules/misc/gtk_main.c
+SOURCES_sap = modules/misc/sap.c
diff --git a/modules/misc/sap.c b/modules/misc/sap.c
new file mode 100644
index 0000000000..e4cd566eab
--- /dev/null
+++ b/modules/misc/sap.c
@@ -0,0 +1,412 @@
+/*****************************************************************************
+ * sap.c :  SAP interface module
+ *****************************************************************************
+ * Copyright (C) 2001 VideoLAN
+ * $Id: sap.c,v 1.1 2002/12/03 16:29:04 gitan Exp $
+ *
+ * Authors: Arnaud Schauly <gitan@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>                                      /* malloc(), free() */
+#include <string.h>
+
+#include <errno.h>                                                 /* ENOMEM */
+#include <stdio.h>
+#include <ctype.h>
+#include <signal.h>
+
+#include <vlc/vlc.h>
+#include <vlc/intf.h>
+#include <vlc/vout.h>
+
+#ifdef HAVE_UNISTD_H
+#    include <unistd.h>
+#endif
+
+#ifdef HAVE_SYS_TIME_H
+#    include <sys/time.h>
+#endif
+#include <sys/types.h>
+
+
+#ifdef WIN32
+#   include <winsock2.h>
+#   include <ws2tcpip.h>
+#   ifndef IN_MULTICAST
+#       define IN_MULTICAST(a) IN_CLASSD(a)
+#   endif
+#else
+#   include <sys/socket.h>
+#   include <netinet/in.h>
+#   include <arpa/inet.h>
+#endif
+
+#include "network.h"
+
+#define MAX_LINE_LENGTH 256
+
+#define HELLO_PORT 9875  // SAP is aloways on that port
+#define HELLO_GROUP "239.255.255.255"   // FIX ME !!!!!!!!!!!!!!!
+#define ADD_SESSION 1;
+
+/*****************************************************************************
+ * Local prototypes
+ *****************************************************************************/
+
+
+
+static int  Activate     ( vlc_object_t * );
+static void Run          ( intf_thread_t *p_intf );
+static int Kill          ( intf_thread_t * );
+        
+static ssize_t NetRead    ( intf_thread_t*, int , byte_t *, size_t );
+
+typedef struct  sess_descr_s {
+/*    char *psz_version; // v field (protocol version) */
+    char *psz_origin; /* o field (username sess-id sess-version 
+                         nettype addrtype addr*/
+    char *psz_sessionname; 
+    char *psz_information; 
+    char *psz_uri; 
+    char *psz_emails;
+    char *psz_phone;  
+    char *psz_time; /* t start-time stop-time */
+    char *psz_repeat; /* r repeat-interval typed-time */
+    char *psz_attribute; 
+    char *psz_media; /* m media port protocol */
+} sess_descr_t;  
+/* All this informations are not useful yet.  */
+
+static int parse_sap ( char ** );
+static int packet_handle ( char **, intf_thread_t * );
+
+static sess_descr_t *  parse_sdp(char *  psz_pct) ;
+
+
+
+
+/*****************************************************************************
+ * Module descriptor
+ *****************************************************************************/
+vlc_module_begin();
+    add_category_hint( N_("SAP"), NULL );
+        add_string( "sap-addr", NULL, NULL, 
+                     "SAP multicast address", "SAP multicast address" );
+    set_description( _("SAP interface module") );
+    set_capability( "interface", 0 );
+    set_callbacks( Activate, NULL);
+vlc_module_end();
+
+/*****************************************************************************
+ * Activate: initialize and create stuff
+ *****************************************************************************/
+static int Activate( vlc_object_t *p_this )
+{
+    intf_thread_t *p_intf = (intf_thread_t*)p_this;
+
+    p_intf->pf_run = Run;
+
+    return VLC_SUCCESS;
+}
+
+/*****************************************************************************
+ * Run: sap thread
+ *****************************************************************************
+ * Listens to SAP packets, and sends them to packet_handle
+ *****************************************************************************/
+
+static void Run( intf_thread_t *p_intf )
+{
+    char *psz_addr;
+    char *psz_network = NULL;
+    struct sockaddr_in addr;
+    struct ip_mreq mreq;
+    int fd,addrlen;
+    char *psz_buf;
+
+    module_t            *p_network;
+    network_socket_t    socket_desc;
+    
+    
+    if( !(psz_addr = config_GetPsz( p_intf, "sap-addr" ) ) )
+    {
+        psz_addr = strdup( HELLO_GROUP );
+    }
+    
+    /* Prepare the network_socket_t structure */
+    socket_desc.i_type = NETWORK_UDP;
+    socket_desc.psz_bind_addr = psz_addr;
+    socket_desc.i_bind_port   = HELLO_PORT;
+    socket_desc.psz_server_addr   = "";
+    socket_desc.i_server_port     = 0;
+    p_intf->p_private = (void*) &socket_desc;
+
+    psz_network = "ipv4";
+    
+    if( !( p_network = module_Need( p_intf, "network", psz_network ) ) )
+    {
+        msg_Err( p_intf, "failed to open a connection (udp)" );
+        return;
+    }
+    module_Unneed( p_intf, p_network ); 
+
+    fd = socket_desc.i_handle;
+    
+    psz_buf = malloc( 1500 ); // FIXME!!
+
+    /* read SAP packets */
+    while( !p_intf->b_die )
+    {
+        int i_read;
+        
+        addrlen=sizeof(addr);
+    
+    
+        memset(psz_buf, 0, 1500);
+        
+        i_read = NetRead( p_intf, fd, psz_buf, 1500 );
+        
+        if( i_read < 0 )
+        {
+            msg_Err( p_intf, "argggg" );
+        }
+        if( i_read == 0 )
+        {
+            continue;
+        }
+
+
+        msg_Dbg( p_intf, "New packet arrived" );
+        packet_handle( &psz_buf, p_intf  );
+                                   
+    }
+    free( psz_buf );
+
+}
+
+/********************************************************************
+ * Kill 
+ *******************************************************************
+ * Kills the SAP interface. 
+ ********************************************************************/
+static int Kill( intf_thread_t *p_intf )
+{
+
+    p_intf->b_die = VLC_TRUE;
+
+    return VLC_SUCCESS;
+}
+
+ 
+/********************************************************************
+ * parse_sap : Takes care of the SAP headers
+ ********************************************************************
+ * checks if the packet has the true headers ; 
+ ********************************************************************/
+
+static int parse_sap( char **  ppsz_sa_packet ) {  /* Dummy Parser : does nothing !*/
+/*   int j;
+   
+   for (j=0;j<255;j++) {
+       fprintf(stderr, "%c",(*ppsz_sa_packet)[j]);
+   }*/
+   
+   return ADD_SESSION; //Add this packet
+}
+
+/***********************************************************************
+ * packet_handle : handle the received packet and enques the 
+ * the understated session
+ * ******************************************************************/
+
+static int packet_handle( char **  ppsz_packet, intf_thread_t * p_intf )  {
+    int j=0;
+    sess_descr_t * sd;
+    playlist_t *p_playlist; 
+    char *  psz_enqueue;
+    char *  psz_udp = "udp://@\0";
+    int i;
+    
+    j=parse_sap( ppsz_packet ); 
+    
+    if(j != 0) {
+        msg_Dbg( p_intf, "SAP packet parsed");
+        sd = parse_sdp( * ppsz_packet ); 
+        
+        i = strlen( psz_udp ) + strlen( sd->psz_uri )+1 ;
+        psz_enqueue = malloc ( i * sizeof (char) );
+        memset( psz_enqueue, '\0',i );
+        strcat ( psz_enqueue,psz_udp );
+        strcat ( psz_enqueue,sd->psz_uri );
+        
+        p_playlist = vlc_object_find( p_intf, VLC_OBJECT_PLAYLIST, FIND_ANYWHERE );
+        playlist_Add ( p_playlist, psz_enqueue, PLAYLIST_CHECK_INSERT, PLAYLIST_END); 
+        msg_Dbg(p_intf,"SAP packet : End of treatment");
+        vlc_object_release( p_playlist );
+        
+        return 1;
+    }
+    free ( sd );
+    return 0; // Invalid Packet 
+}
+        
+
+
+
+/******************************************************
+ * parse_sap : SDP parsing
+ * ****************************************************
+ * Make a sess_descr_t with a psz
+ ******************************************************/
+
+static sess_descr_t *  parse_sdp(char *  psz_pct) 
+{
+    int j,k;
+    char **  ppsz_fill;
+    sess_descr_t *  sd;
+    
+    sd = malloc( sizeof(sess_descr_t) );
+    for (j=0 ; j < 255 ; j++) 
+    {
+       if (psz_pct[j] == '=') 
+       {           
+           switch(psz_pct[(j-1)]) {
+/*               case ('v') : {
+//               ppsz_fill = & sd->psz_version;
+//               break;
+            } */
+            case ('o') : {
+               ppsz_fill = & sd->psz_origin;
+               break;
+            }
+            case ('s') : {
+               ppsz_fill = & sd->psz_sessionname;
+               break;
+            }
+            case ('i') : {
+               ppsz_fill = & sd->psz_information;
+               break;
+            }
+            case ('u') : {
+               ppsz_fill = & sd->psz_uri;
+               break;
+            }
+            case ('e') : {
+               ppsz_fill = & sd->psz_emails;
+               break;
+            }
+            case ('p') : {
+               ppsz_fill = & sd->psz_phone;
+               break;
+            }
+            case ('t') : {
+               ppsz_fill = & sd->psz_time;
+               break;
+            }
+            case ('r') : {
+               ppsz_fill = & sd->psz_repeat;
+               break;
+            }
+            case ('a') : {
+               ppsz_fill = & sd->psz_attribute;
+               break;
+            }
+            case ('m') : {
+               ppsz_fill = & sd->psz_media;
+               break;
+            }
+
+            default : { 
+               fprintf( stderr,"Warning : Ignored field \"%c\" \n",psz_pct[j-1] );
+               ppsz_fill = NULL;
+            }
+
+
+         } 
+      k=0;j++;
+      while (psz_pct[j] != '\n'&& psz_pct[j] != EOF) {
+         k++; j++;
+      }
+      j--;
+      if( ppsz_fill != NULL )
+      {
+         *ppsz_fill= malloc( sizeof(char) * (k+1) );
+         memccpy(*ppsz_fill, &(psz_pct[j-k+1]),'\n',  k ); 
+         (*ppsz_fill)[k]='\0';
+      }
+      ppsz_fill = NULL;
+
+      } // if
+   } //for
+
+   return sd;
+}
+
+
+
+/*****************************************************************************
+ * Read: read on a file descriptor, checking b_die periodically
+ *****************************************************************************
+ * Taken from udp.c
+ ******************************************************************************/
+static ssize_t NetRead( intf_thread_t *p_intf, 
+                        int i_handle, byte_t *p_buffer, size_t i_len)
+{
+#ifdef UNDER_CE
+    return -1;
+    
+#else
+    struct timeval  timeout;
+    fd_set          fds;
+    int             i_ret;
+    
+    /* Initialize file descriptor set */
+    FD_ZERO( &fds );
+    FD_SET( i_handle, &fds );
+    
+    /* We'll wait 0.5 second if nothing happens */
+    timeout.tv_sec = 0;
+    timeout.tv_usec = 500000;
+    
+    /* Find if some data is available */
+    i_ret = select( i_handle + 1, &fds,
+    NULL, NULL, &timeout );
+    
+    if( i_ret == -1 && errno != EINTR )
+    {
+        msg_Err( p_intf, "network select error (%s)", strerror(errno) );
+    }
+    else if( i_ret > 0 )
+    {
+        ssize_t i_recv = recv( i_handle, p_buffer, i_len, 0 );
+    
+        if( i_recv < 0 )
+        {
+           msg_Err( p_intf, "recv failed (%s)", strerror(errno) );
+        }
+    
+        return i_recv;
+    }
+    
+    return 0;
+
+#endif
+}
+
diff --git a/src/libvlc.c b/src/libvlc.c
index 9a267eef97..2d6679aa60 100644
--- a/src/libvlc.c
+++ b/src/libvlc.c
@@ -2,7 +2,7 @@
  * libvlc.c: main libvlc source
  *****************************************************************************
  * Copyright (C) 1998-2002 VideoLAN
- * $Id: libvlc.c,v 1.48 2002/11/28 17:35:00 sam Exp $
+ * $Id: libvlc.c,v 1.49 2002/12/03 16:29:04 gitan Exp $
  *
  * Authors: Vincent Seguin <seguin@via.ecp.fr>
  *          Samuel Hocevar <sam@zoy.org>
@@ -755,7 +755,15 @@ int VLC_Play( int i_object )
     {
         return VLC_ENOOBJ;
     }
+    
+    /* add pseudo sap interface; non blocking */
+    if( config_GetInt( p_vlc, "sap" ) )
+    {
+        msg_Dbg( p_vlc, "adding sap interface" );
+        VLC_AddIntf( 0, "sap", VLC_FALSE );
+    }
 
+    
     p_playlist = vlc_object_find( p_vlc, VLC_OBJECT_PLAYLIST, FIND_CHILD );
 
     if( !p_playlist )
diff --git a/src/libvlc.h b/src/libvlc.h
index ca72062675..cfbb9be627 100644
--- a/src/libvlc.h
+++ b/src/libvlc.h
@@ -2,7 +2,7 @@
  * libvlc.h: main libvlc header
  *****************************************************************************
  * Copyright (C) 1998-2002 VideoLAN
- * $Id: libvlc.h,v 1.24 2002/12/03 12:59:21 sam Exp $
+ * $Id: libvlc.h,v 1.25 2002/12/03 16:29:04 gitan Exp $
  *
  * Authors: Vincent Seguin <seguin@via.ecp.fr>
  *          Samuel Hocevar <sam@zoy.org>
@@ -358,6 +358,10 @@
     "\n  vlc:quit                       quit VLC" \
     "\n")
 
+#define SAP_TEXT N_( "Session Announcement Protocol support" )
+#define SAP_LONGTEXT N_( "Session Announcement Protocol support" )
+
+
 /*
  * Quick usage guide for the configuration options:
  *
@@ -475,6 +479,7 @@ vlc_module_begin();
 
     /* Misc options */
     add_category_hint( N_("Miscellaneous"), NULL );
+    add_bool( "sap", 1, NULL, SAP_TEXT, SAP_LONGTEXT );
     add_module( "memcpy", "memcpy", NULL, NULL, MEMCPY_TEXT, MEMCPY_LONGTEXT );
     add_module( "access", "access", NULL, NULL, ACCESS_TEXT, ACCESS_LONGTEXT );
     add_module( "demux", "demux", NULL, NULL, DEMUX_TEXT, DEMUX_LONGTEXT );
diff --git a/src/playlist/playlist.c b/src/playlist/playlist.c
index e5c15c1118..5345b93808 100644
--- a/src/playlist/playlist.c
+++ b/src/playlist/playlist.c
@@ -2,7 +2,7 @@
  * playlist.c : Playlist management functions
  *****************************************************************************
  * Copyright (C) 1999-2001 VideoLAN
- * $Id: playlist.c,v 1.23 2002/11/21 15:51:57 gbazin Exp $
+ * $Id: playlist.c,v 1.24 2002/12/03 16:29:04 gitan Exp $
  *
  * Authors: Samuel Hocevar <sam@zoy.org>
  *
@@ -105,8 +105,37 @@ int playlist_Add( playlist_t *p_playlist, const char * psz_target,
 {
     playlist_item_t *p_item;
 
-    msg_Dbg( p_playlist, "adding playlist item � %s �", psz_target );
 
+    vlc_mutex_lock( &p_playlist->object_lock );
+
+    /*
+     * CHECK_INSERT : checks if the item is already enqued before 
+     * enqueing it
+     */
+    if ( i_mode & PLAYLIST_CHECK_INSERT )
+    {
+         int j;
+       
+         if ( p_playlist->pp_items )
+         {
+             for ( j = 0; j < p_playlist->i_size; j++ )
+             {
+                 if ( !strcmp( p_playlist->pp_items[j]->psz_name, psz_target ) )
+                 {
+                      msg_Dbg( p_playlist, "item � %s � already enqued", 
+                                        psz_target );
+                      vlc_mutex_unlock( &p_playlist->object_lock );
+                      return 0;   
+                 }
+             }
+         }
+         i_mode &= ~PLAYLIST_CHECK_INSERT;
+         i_mode |= PLAYLIST_APPEND;
+    }
+
+    
+    msg_Dbg( p_playlist, "adding playlist item � %s �", psz_target );
+    
     /* Create the new playlist item */
     p_item = malloc( sizeof( playlist_item_t ) );
     if( p_item == NULL )
@@ -119,7 +148,6 @@ int playlist_Add( playlist_t *p_playlist, const char * psz_target,
     p_item->i_status = 0;
     p_item->b_autodeletion = VLC_FALSE;
 
-    vlc_mutex_lock( &p_playlist->object_lock );
 
     /* Do a few boundary checks and allocate space for the item */
     if( i_pos == PLAYLIST_END )
-- 
2.25.4