2001-04-20 17:02:48 +02:00
|
|
|
/*****************************************************************************
|
2002-03-01 01:33:18 +01:00
|
|
|
* network.h: interface to communicate with network plug-ins
|
2001-04-20 17:02:48 +02:00
|
|
|
*****************************************************************************
|
2005-07-09 08:17:09 +02:00
|
|
|
* Copyright (C) 2002-2005 the VideoLAN team
|
* ALL: Better announce system
- The SAP handler now runs in a separate thread.
- RTP sessions can be announced with sap (sdp=sap://,name=...)
TODO: Make this more configurable
- Better SDP generation (the timestamp problem is not resolved)
About this, there is a problem : as, for a RTP session, the URI
is the complete SDP, if the session is recreated, as the URI has
changed, a new item is added to the playlist
- Experimental flow control algorithm :
It does not follow the "Recommended" implementation, as it needs
to count the sessions (to achieve this, we should make this work
together with the SAP listener)
It is disabled by default (use --sap-flow-control to enable).
When it is disabled, sap announcement interval is set by --sap-interval
* src/misc/net.c : created net_ReadNonBlock
* sap.c : Fixed memory problem
2004-04-18 20:21:09 +02:00
|
|
|
* $Id$
|
2001-04-20 17:02:48 +02:00
|
|
|
*
|
2001-12-27 04:47:09 +01:00
|
|
|
* Authors: Christophe Massiot <massiot@via.ecp.fr>
|
2003-11-07 18:44:43 +01:00
|
|
|
* Laurent Aimar <fenrir@via.ecp.fr>
|
2005-05-22 13:24:08 +02:00
|
|
|
* Rémi Denis-Courmont <rem # videolan.org>
|
2001-04-20 17:02:48 +02:00
|
|
|
*
|
|
|
|
* 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
|
2006-01-13 00:10:04 +01:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
|
2001-04-20 17:02:48 +02:00
|
|
|
*****************************************************************************/
|
|
|
|
|
2004-11-07 12:17:37 +01:00
|
|
|
#ifndef __VLC_NETWORK_H
|
|
|
|
# define __VLC_NETWORK_H
|
2005-07-02 17:39:21 +02:00
|
|
|
|
2005-07-30 12:49:52 +02:00
|
|
|
#if defined( WIN32 )
|
|
|
|
# if defined(UNDER_CE) && defined(sockaddr_storage)
|
|
|
|
# undef sockaddr_storage
|
|
|
|
# endif
|
2005-09-22 22:14:20 +02:00
|
|
|
# if defined(UNDER_CE)
|
|
|
|
# define HAVE_STRUCT_ADDRINFO
|
|
|
|
# else
|
2006-11-22 12:30:30 +01:00
|
|
|
# define _NO_OLDNAMES 1
|
2005-09-22 22:14:20 +02:00
|
|
|
# include <io.h>
|
|
|
|
# endif
|
2005-07-04 11:33:43 +02:00
|
|
|
# include <winsock2.h>
|
2005-07-17 23:21:04 +02:00
|
|
|
# include <ws2tcpip.h>
|
2005-09-22 22:14:20 +02:00
|
|
|
# define ENETUNREACH WSAENETUNREACH
|
2006-03-09 23:56:37 +01:00
|
|
|
# define net_errno (WSAGetLastError())
|
|
|
|
extern const char *net_strerror( int val );
|
2005-07-21 23:18:39 +02:00
|
|
|
#else
|
2005-07-30 12:49:52 +02:00
|
|
|
# if HAVE_SYS_SOCKET_H
|
|
|
|
# include <sys/socket.h>
|
|
|
|
# endif
|
|
|
|
# if HAVE_NETINET_IN_H
|
|
|
|
# include <netinet/in.h>
|
|
|
|
# endif
|
|
|
|
# if HAVE_ARPA_INET_H
|
|
|
|
# include <arpa/inet.h>
|
|
|
|
# elif defined( SYS_BEOS )
|
|
|
|
# include <net/netdb.h>
|
|
|
|
# endif
|
2005-07-21 23:18:39 +02:00
|
|
|
# include <netdb.h>
|
2006-03-09 23:56:37 +01:00
|
|
|
# define net_errno errno
|
|
|
|
# define net_strerror strerror
|
2005-07-21 23:18:39 +02:00
|
|
|
#endif
|
|
|
|
|
2005-08-24 21:49:47 +02:00
|
|
|
# ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
# endif
|
2005-07-02 17:39:21 +02:00
|
|
|
|
2004-11-07 12:02:59 +01:00
|
|
|
/* Portable networking layer communication */
|
2006-11-05 20:20:52 +01:00
|
|
|
int net_Socket (vlc_object_t *obj, int family, int socktype, int proto);
|
|
|
|
|
2005-12-04 18:40:59 +01:00
|
|
|
#define net_ConnectTCP(a, b, c) __net_ConnectTCP(VLC_OBJECT(a), b, c)
|
|
|
|
#define net_OpenTCP(a, b, c) __net_ConnectTCP(VLC_OBJECT(a), b, c)
|
|
|
|
VLC_EXPORT( int, __net_ConnectTCP, ( vlc_object_t *p_this, const char *psz_host, int i_port ) );
|
2004-01-05 15:10:58 +01:00
|
|
|
|
2006-11-06 18:30:21 +01:00
|
|
|
int *net_Listen (vlc_object_t *p_this, const char *psz_host, int i_port,
|
|
|
|
int family, int socktype, int protocol);
|
2006-11-06 18:59:01 +01:00
|
|
|
VLC_EXPORT( int, net_ListenSingle, (vlc_object_t *p_this, const char *psz_host, int i_port, int family, int socktype, int protocol) );
|
2006-11-05 16:50:31 +01:00
|
|
|
|
2004-06-06 14:45:32 +02:00
|
|
|
#define net_ListenTCP(a, b, c) __net_ListenTCP(VLC_OBJECT(a), b, c)
|
2005-05-22 13:24:08 +02:00
|
|
|
VLC_EXPORT( int *, __net_ListenTCP, ( vlc_object_t *, const char *, int ) );
|
2004-06-06 14:45:32 +02:00
|
|
|
|
2004-07-25 00:49:28 +02:00
|
|
|
#define net_Accept(a, b, c) __net_Accept(VLC_OBJECT(a), b, c)
|
2005-05-22 13:24:08 +02:00
|
|
|
VLC_EXPORT( int, __net_Accept, ( vlc_object_t *, int *, mtime_t ) );
|
2004-07-25 00:49:28 +02:00
|
|
|
|
2005-12-04 18:40:59 +01:00
|
|
|
#define net_ConnectUDP(a, b, c, d ) __net_ConnectUDP(VLC_OBJECT(a), b, c, d)
|
|
|
|
VLC_EXPORT( int, __net_ConnectUDP, ( vlc_object_t *p_this, const char *psz_host, int i_port, int hlim ) );
|
|
|
|
|
2006-11-06 18:29:07 +01:00
|
|
|
static inline int net_ListenUDP1 (vlc_object_t *obj, const char *host, int port)
|
|
|
|
{
|
|
|
|
return net_ListenSingle (obj, host, port, AF_UNSPEC, SOCK_DGRAM, IPPROTO_UDP);
|
|
|
|
}
|
2006-11-05 20:20:52 +01:00
|
|
|
|
2004-01-21 11:22:31 +01:00
|
|
|
#define net_OpenUDP(a, b, c, d, e ) __net_OpenUDP(VLC_OBJECT(a), b, c, d, e)
|
2005-07-27 17:46:36 +02:00
|
|
|
VLC_EXPORT( int, __net_OpenUDP, ( vlc_object_t *p_this, const char *psz_bind, int i_bind, const char *psz_server, int i_server ) );
|
2004-01-21 11:22:31 +01:00
|
|
|
|
2004-01-05 15:10:58 +01:00
|
|
|
VLC_EXPORT( void, net_Close, ( int fd ) );
|
2005-05-22 13:24:08 +02:00
|
|
|
VLC_EXPORT( void, net_ListenClose, ( int *fd ) );
|
2004-01-05 15:10:58 +01:00
|
|
|
|
2006-09-07 21:49:30 +02:00
|
|
|
VLC_EXPORT( int, net_SetDSCP, ( int fd, uint8_t dscp ) );
|
2006-11-05 20:20:52 +01:00
|
|
|
int net_Subscribe (vlc_object_t *obj, int fd, const struct sockaddr *addr,
|
|
|
|
socklen_t addrlen);
|
2004-01-05 15:10:58 +01:00
|
|
|
|
2004-11-07 12:02:59 +01:00
|
|
|
/* Functions to read from or write to the networking layer */
|
|
|
|
struct virtual_socket_t
|
|
|
|
{
|
|
|
|
void *p_sys;
|
|
|
|
int (*pf_recv) ( void *, void *, int );
|
|
|
|
int (*pf_send) ( void *, const void *, int );
|
|
|
|
};
|
|
|
|
|
|
|
|
#define net_Read(a,b,c,d,e,f) __net_Read(VLC_OBJECT(a),b,c,d,e,f)
|
2006-09-08 20:14:25 +02:00
|
|
|
VLC_EXPORT( int, __net_Read, ( vlc_object_t *p_this, int fd, const v_socket_t *, uint8_t *p_data, int i_data, vlc_bool_t b_retry ) );
|
* ALL: Better announce system
- The SAP handler now runs in a separate thread.
- RTP sessions can be announced with sap (sdp=sap://,name=...)
TODO: Make this more configurable
- Better SDP generation (the timestamp problem is not resolved)
About this, there is a problem : as, for a RTP session, the URI
is the complete SDP, if the session is recreated, as the URI has
changed, a new item is added to the playlist
- Experimental flow control algorithm :
It does not follow the "Recommended" implementation, as it needs
to count the sessions (to achieve this, we should make this work
together with the SAP listener)
It is disabled by default (use --sap-flow-control to enable).
When it is disabled, sap announcement interval is set by --sap-interval
* src/misc/net.c : created net_ReadNonBlock
* sap.c : Fixed memory problem
2004-04-18 20:21:09 +02:00
|
|
|
|
2004-11-07 12:02:59 +01:00
|
|
|
#define net_ReadNonBlock(a,b,c,d,e,f) __net_ReadNonBlock(VLC_OBJECT(a),b,c,d,e,f)
|
2006-09-08 20:14:25 +02:00
|
|
|
VLC_EXPORT( int, __net_ReadNonBlock, ( vlc_object_t *p_this, int fd, const v_socket_t *, uint8_t *p_data, int i_data, mtime_t i_wait ) );
|
2004-11-06 12:13:23 +01:00
|
|
|
|
2004-11-07 12:02:59 +01:00
|
|
|
#define net_Select(a,b,c,d,e,f,g) __net_Select(VLC_OBJECT(a),b,c,d,e,f,g)
|
2006-09-08 20:14:25 +02:00
|
|
|
VLC_EXPORT( int, __net_Select, ( vlc_object_t *p_this, const int *pi_fd, const v_socket_t *const *, int i_fd, uint8_t *p_data, int i_data, mtime_t i_wait ) );
|
2004-11-06 12:13:23 +01:00
|
|
|
|
2004-11-07 12:02:59 +01:00
|
|
|
#define net_Write(a,b,c,d,e) __net_Write(VLC_OBJECT(a),b,c,d,e)
|
2006-09-08 20:14:25 +02:00
|
|
|
VLC_EXPORT( int, __net_Write, ( vlc_object_t *p_this, int fd, const v_socket_t *, const uint8_t *p_data, int i_data ) );
|
2004-01-05 15:10:58 +01:00
|
|
|
|
2004-11-07 12:02:59 +01:00
|
|
|
#define net_Gets(a,b,c) __net_Gets(VLC_OBJECT(a),b,c)
|
2006-09-08 20:14:25 +02:00
|
|
|
VLC_EXPORT( char *, __net_Gets, ( vlc_object_t *p_this, int fd, const v_socket_t * ) );
|
2004-01-05 15:10:58 +01:00
|
|
|
|
2006-09-08 20:14:25 +02:00
|
|
|
VLC_EXPORT( int, net_Printf, ( vlc_object_t *p_this, int fd, const v_socket_t *, const char *psz_fmt, ... ) );
|
2004-01-05 15:10:58 +01:00
|
|
|
|
2004-11-07 12:02:59 +01:00
|
|
|
#define net_vaPrintf(a,b,c,d,e) __net_vaPrintf(VLC_OBJECT(a),b,c,d,e)
|
2006-09-08 20:14:25 +02:00
|
|
|
VLC_EXPORT( int, __net_vaPrintf, ( vlc_object_t *p_this, int fd, const v_socket_t *, const char *psz_fmt, va_list args ) );
|
2004-11-07 12:17:37 +01:00
|
|
|
|
2005-07-13 20:11:18 +02:00
|
|
|
|
2006-11-05 12:46:43 +01:00
|
|
|
#ifndef HAVE_INET_PTON
|
2005-07-28 17:31:10 +02:00
|
|
|
/* only in core, so no need for C++ extern "C" */
|
|
|
|
int inet_pton(int af, const char *src, void *dst);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2005-07-01 16:48:29 +02:00
|
|
|
/*****************************************************************************
|
|
|
|
* net_StopRecv/Send
|
|
|
|
*****************************************************************************
|
|
|
|
* Wrappers for shutdown()
|
|
|
|
*****************************************************************************/
|
|
|
|
#if defined (SHUT_WR)
|
|
|
|
/* the standard way */
|
2005-07-04 16:47:36 +02:00
|
|
|
# define net_StopSend( fd ) (void)shutdown( fd, SHUT_WR )
|
|
|
|
# define net_StopRecv( fd ) (void)shutdown( fd, SHUT_RD )
|
2005-07-01 16:48:29 +02:00
|
|
|
#elif defined (SD_SEND)
|
|
|
|
/* the Microsoft seemingly-purposedly-different-for-the-sake-of-it way */
|
2005-07-04 16:47:36 +02:00
|
|
|
# define net_StopSend( fd ) (void)shutdown( fd, SD_SEND )
|
|
|
|
# define net_StopRecv( fd ) (void)shutdown( fd, SD_RECEIVE )
|
2005-07-01 16:48:29 +02:00
|
|
|
#else
|
2005-08-14 15:15:38 +02:00
|
|
|
# ifndef SYS_BEOS /* R5 just doesn't have a working shutdown() */
|
|
|
|
# warning FIXME: implement shutdown on your platform!
|
|
|
|
# endif
|
2005-07-04 16:47:36 +02:00
|
|
|
# define net_StopSend( fd ) (void)0
|
|
|
|
# define net_StopRecv( fd ) (void)0
|
2005-07-01 16:48:29 +02:00
|
|
|
#endif
|
2005-06-30 20:17:53 +02:00
|
|
|
|
2005-05-22 13:24:08 +02:00
|
|
|
/* Portable network names/addresses resolution layer */
|
|
|
|
|
|
|
|
/* GAI error codes */
|
|
|
|
# ifndef EAI_BADFLAGS
|
|
|
|
# define EAI_BADFLAGS -1
|
|
|
|
# endif
|
|
|
|
# ifndef EAI_NONAME
|
|
|
|
# define EAI_NONAME -2
|
|
|
|
# endif
|
|
|
|
# ifndef EAI_AGAIN
|
|
|
|
# define EAI_AGAIN -3
|
|
|
|
# endif
|
|
|
|
# ifndef EAI_FAIL
|
|
|
|
# define EAI_FAIL -4
|
|
|
|
# endif
|
|
|
|
# ifndef EAI_NODATA
|
|
|
|
# define EAI_NODATA -5
|
|
|
|
# endif
|
|
|
|
# ifndef EAI_FAMILY
|
|
|
|
# define EAI_FAMILY -6
|
|
|
|
# endif
|
|
|
|
# ifndef EAI_SOCKTYPE
|
|
|
|
# define EAI_SOCKTYPE -7
|
|
|
|
# endif
|
|
|
|
# ifndef EAI_SERVICE
|
|
|
|
# define EAI_SERVICE -8
|
|
|
|
# endif
|
|
|
|
# ifndef EAI_ADDRFAMILY
|
|
|
|
# define EAI_ADDRFAMILY -9
|
|
|
|
# endif
|
|
|
|
# ifndef EAI_MEMORY
|
|
|
|
# define EAI_MEMORY -10
|
|
|
|
# endif
|
|
|
|
# ifndef EAI_SYSTEM
|
|
|
|
# define EAI_SYSTEM -11
|
|
|
|
# endif
|
|
|
|
|
|
|
|
|
|
|
|
# ifndef NI_MAXHOST
|
|
|
|
# define NI_MAXHOST 1025
|
|
|
|
# define NI_MAXSERV 32
|
|
|
|
# endif
|
2005-07-27 19:22:28 +02:00
|
|
|
# define NI_MAXNUMERICHOST 64
|
2005-05-22 13:24:08 +02:00
|
|
|
|
|
|
|
# ifndef NI_NUMERICHOST
|
|
|
|
# define NI_NUMERICHOST 0x01
|
|
|
|
# define NI_NUMERICSERV 0x02
|
|
|
|
# define NI_NOFQDN 0x04
|
|
|
|
# define NI_NAMEREQD 0x08
|
|
|
|
# define NI_DGRAM 0x10
|
|
|
|
# endif
|
|
|
|
|
|
|
|
# ifndef HAVE_STRUCT_ADDRINFO
|
|
|
|
struct addrinfo
|
|
|
|
{
|
|
|
|
int ai_flags;
|
|
|
|
int ai_family;
|
|
|
|
int ai_socktype;
|
|
|
|
int ai_protocol;
|
|
|
|
size_t ai_addrlen;
|
|
|
|
struct sockaddr *ai_addr;
|
|
|
|
char *ai_canonname;
|
|
|
|
struct addrinfo *ai_next;
|
|
|
|
};
|
|
|
|
# define AI_PASSIVE 1
|
|
|
|
# define AI_CANONNAME 2
|
|
|
|
# define AI_NUMERICHOST 4
|
|
|
|
# endif /* if !HAVE_STRUCT_ADDRINFO */
|
|
|
|
|
|
|
|
VLC_EXPORT( const char *, vlc_gai_strerror, ( int ) );
|
2005-06-26 12:24:35 +02:00
|
|
|
VLC_EXPORT( int, vlc_getnameinfo, ( const struct sockaddr *, int, char *, int, int *, int ) );
|
2005-06-26 11:12:36 +02:00
|
|
|
VLC_EXPORT( int, vlc_getaddrinfo, ( vlc_object_t *, const char *, int, const struct addrinfo *, struct addrinfo ** ) );
|
2005-05-22 13:24:08 +02:00
|
|
|
VLC_EXPORT( void, vlc_freeaddrinfo, ( struct addrinfo * ) );
|
|
|
|
|
2006-11-05 17:59:47 +01:00
|
|
|
|
|
|
|
static inline vlc_bool_t
|
|
|
|
net_SockAddrIsMulticast (const struct sockaddr *addr, socklen_t len)
|
|
|
|
{
|
|
|
|
switch (addr->sa_family)
|
|
|
|
{
|
|
|
|
#ifdef IN_MULTICAST
|
|
|
|
case AF_INET:
|
|
|
|
{
|
|
|
|
struct sockaddr_in *v4 = (struct sockaddr_in *)addr;
|
|
|
|
if (len < sizeof (*v4))
|
|
|
|
return VLC_FALSE;
|
2006-11-09 19:27:43 +01:00
|
|
|
return IN_MULTICAST (ntohl (v4->sin_addr.s_addr)) != 0;
|
2006-11-05 17:59:47 +01:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef IN6_IS_ADDR_MULTICAST
|
|
|
|
case AF_INET6:
|
|
|
|
{
|
|
|
|
struct sockaddr_in6 *v6 = (struct sockaddr_in6 *)addr;
|
|
|
|
if (len < sizeof (*v6))
|
|
|
|
return VLC_FALSE;
|
|
|
|
return IN6_IS_ADDR_MULTICAST (&v6->sin6_addr) != 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
return VLC_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* net_AddressIsMulticast
|
|
|
|
* @return VLC_FALSE iff the psz_addr does not specify a multicast address,
|
|
|
|
* or the address is not a valid address.
|
|
|
|
*/
|
2005-07-31 21:01:09 +02:00
|
|
|
static inline vlc_bool_t net_AddressIsMulticast( vlc_object_t *p_object, const char *psz_addr )
|
2005-07-21 23:18:39 +02:00
|
|
|
{
|
|
|
|
struct addrinfo hints, *res;
|
|
|
|
|
2006-11-05 17:59:47 +01:00
|
|
|
memset (&hints, 0, sizeof (hints));
|
2005-07-21 23:18:39 +02:00
|
|
|
hints.ai_socktype = SOCK_DGRAM; /* UDP */
|
|
|
|
hints.ai_flags = AI_NUMERICHOST;
|
|
|
|
|
2006-11-05 17:59:47 +01:00
|
|
|
int i = vlc_getaddrinfo (p_object, psz_addr, 0,
|
|
|
|
&hints, &res);
|
|
|
|
if (i)
|
2005-07-21 23:18:39 +02:00
|
|
|
{
|
2006-11-05 17:59:47 +01:00
|
|
|
msg_Err (p_object, "invalid address \"%s\" for net_AddressIsMulticast (%s)",
|
|
|
|
psz_addr, vlc_gai_strerror (i));
|
2005-07-27 21:05:21 +02:00
|
|
|
return VLC_FALSE;
|
2005-07-21 23:18:39 +02:00
|
|
|
}
|
2005-12-15 12:37:45 +01:00
|
|
|
|
2006-11-05 17:59:47 +01:00
|
|
|
vlc_bool_t b = net_SockAddrIsMulticast (res->ai_addr, res->ai_addrlen);
|
|
|
|
vlc_freeaddrinfo (res);
|
|
|
|
return b;
|
2005-07-21 23:18:39 +02:00
|
|
|
}
|
|
|
|
|
2005-07-29 19:28:01 +02:00
|
|
|
static inline int net_GetSockAddress( int fd, char *address, int *port )
|
2005-07-29 17:26:08 +02:00
|
|
|
{
|
|
|
|
struct sockaddr_storage addr;
|
|
|
|
socklen_t addrlen = sizeof( addr );
|
|
|
|
|
2005-10-20 14:52:05 +02:00
|
|
|
return getsockname( fd, (struct sockaddr *)&addr, &addrlen )
|
2005-07-29 19:28:01 +02:00
|
|
|
|| vlc_getnameinfo( (struct sockaddr *)&addr, addrlen, address,
|
|
|
|
NI_MAXNUMERICHOST, port, NI_NUMERICHOST )
|
|
|
|
? VLC_EGENERIC : 0;
|
|
|
|
}
|
2005-07-29 17:26:08 +02:00
|
|
|
|
2005-07-29 19:28:01 +02:00
|
|
|
static inline int net_GetPeerAddress( int fd, char *address, int *port )
|
|
|
|
{
|
|
|
|
struct sockaddr_storage addr;
|
|
|
|
socklen_t addrlen = sizeof( addr );
|
2005-07-29 17:26:08 +02:00
|
|
|
|
2005-07-29 19:28:01 +02:00
|
|
|
return getpeername( fd, (struct sockaddr *)&addr, &addrlen )
|
|
|
|
|| vlc_getnameinfo( (struct sockaddr *)&addr, addrlen, address,
|
|
|
|
NI_MAXNUMERICHOST, port, NI_NUMERICHOST )
|
|
|
|
? VLC_EGENERIC : 0;
|
2005-07-29 17:26:08 +02:00
|
|
|
}
|
|
|
|
|
2005-08-24 21:49:47 +02:00
|
|
|
# ifdef __cplusplus
|
|
|
|
}
|
|
|
|
# endif
|
|
|
|
|
2004-11-07 12:17:37 +01:00
|
|
|
#endif
|