Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
V
vlc
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Redmine
Redmine
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Operations
Operations
Metrics
Environments
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
videolan
vlc
Commits
14fb31d9
Commit
14fb31d9
authored
Feb 11, 2007
by
Rémi Denis-Courmont
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Improve SAP parser
parent
60a679f5
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
446 additions
and
342 deletions
+446
-342
modules/services_discovery/sap.c
modules/services_discovery/sap.c
+446
-342
No files found.
modules/services_discovery/sap.c
View file @
14fb31d9
...
@@ -2,9 +2,11 @@
...
@@ -2,9 +2,11 @@
* sap.c : SAP interface module
* sap.c : SAP interface module
*****************************************************************************
*****************************************************************************
* Copyright (C) 2004-2005 the VideoLAN team
* Copyright (C) 2004-2005 the VideoLAN team
* Copyright © 2007 Rémi Denis-Courmont
* $Id$
* $Id$
*
*
* Authors: Clément Stenac <zorglub@videolan.org>
* Authors: Clément Stenac <zorglub@videolan.org>
* Rémi Denis-Courmont
*
*
* This program is free software; you can redistribute it and/or modify
* 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
* it under the terms of the GNU General Public License as published by
...
@@ -164,41 +166,52 @@ typedef struct sdp_t sdp_t;
...
@@ -164,41 +166,52 @@ typedef struct sdp_t sdp_t;
typedef
struct
attribute_t
attribute_t
;
typedef
struct
attribute_t
attribute_t
;
typedef
struct
sap_announce_t
sap_announce_t
;
typedef
struct
sap_announce_t
sap_announce_t
;
struct
sdp_media_t
{
struct
sdp_t
*
parent
;
char
*
fmt
;
struct
sockaddr_storage
addr
;
socklen_t
addrlen
;
unsigned
n_addr
;
int
i_attributes
;
attribute_t
**
pp_attributes
;
};
/* The structure that contains sdp information */
/* The structure that contains sdp information */
struct
sdp_t
struct
sdp_t
{
{
char
*
psz_sdp
;
char
*
psz_sdp
;
/* o field */
char
username
[
64
];
uint64_t
session_id
;
uint64_t
session_version
;
unsigned
orig_ip_version
;
char
orig_host
[
1024
];
/* s= field */
/* s= field */
char
*
psz_sessionname
;
char
*
psz_sessionname
;
/* Raw m= and c= fields */
/* old cruft */
char
*
psz_connection
;
char
*
psz_media
;
/* o field */
char
*
psz_username
;
char
*
psz_network_type
;
char
*
psz_address_type
;
char
*
psz_address
;
int64_t
i_session_id
;
/* "computed" URI */
/* "computed" URI */
char
*
psz_uri
;
char
*
psz_uri
;
int
i_in
;
/* IP version */
int
i_media
;
int
i_media_type
;
int
i_media_type
;
/* a= global attributes */
int
i_attributes
;
int
i_attributes
;
attribute_t
**
pp_attributes
;
attribute_t
**
pp_attributes
;
/* medias (well, we only support one atm) */
unsigned
mediac
;
struct
sdp_media_t
mediav
[
1
];
};
};
struct
attribute_t
struct
attribute_t
{
{
c
har
*
psz_field
;
c
onst
char
*
value
;
char
*
psz_value
;
char
name
[
0
]
;
};
};
struct
sap_announce_t
struct
sap_announce_t
...
@@ -261,7 +274,10 @@ struct demux_sys_t
...
@@ -261,7 +274,10 @@ struct demux_sys_t
static
int
RemoveAnnounce
(
services_discovery_t
*
p_sd
,
sap_announce_t
*
p_announce
);
static
int
RemoveAnnounce
(
services_discovery_t
*
p_sd
,
sap_announce_t
*
p_announce
);
/* Helper functions */
/* Helper functions */
static
char
*
GetAttribute
(
sdp_t
*
p_sdp
,
const
char
*
psz_search
);
static
inline
attribute_t
*
MakeAttribute
(
const
char
*
str
);
static
const
char
*
GetAttribute
(
attribute_t
**
tab
,
unsigned
n
,
const
char
*
name
);
static
inline
void
FreeAttribute
(
attribute_t
*
a
);
static
vlc_bool_t
IsSameSession
(
sdp_t
*
p_sdp1
,
sdp_t
*
p_sdp2
);
static
vlc_bool_t
IsSameSession
(
sdp_t
*
p_sdp1
,
sdp_t
*
p_sdp2
);
static
int
InitSocket
(
services_discovery_t
*
p_sd
,
const
char
*
psz_address
,
int
i_port
);
static
int
InitSocket
(
services_discovery_t
*
p_sd
,
const
char
*
psz_address
,
int
i_port
);
static
int
Decompress
(
const
unsigned
char
*
psz_src
,
unsigned
char
**
_dst
,
int
i_len
);
static
int
Decompress
(
const
unsigned
char
*
psz_src
,
unsigned
char
**
_dst
,
int
i_len
);
...
@@ -377,7 +393,7 @@ static int OpenDemux( vlc_object_t *p_this )
...
@@ -377,7 +393,7 @@ static int OpenDemux( vlc_object_t *p_this )
goto
error
;
goto
error
;
}
}
if
(
p_sdp
->
i_media
>
1
)
if
(
p_sdp
->
mediac
>
1
)
{
{
goto
error
;
goto
error
;
}
}
...
@@ -571,7 +587,7 @@ static int Demux( demux_t *p_demux )
...
@@ -571,7 +587,7 @@ static int Demux( demux_t *p_demux )
FREENULL
(
p_parent_input
->
psz_uri
);
FREENULL
(
p_parent_input
->
psz_uri
);
p_parent_input
->
psz_uri
=
strdup
(
p_sdp
->
psz_uri
);
p_parent_input
->
psz_uri
=
strdup
(
p_sdp
->
psz_uri
);
FREENULL
(
p_parent_input
->
psz_name
);
FREENULL
(
p_parent_input
->
psz_name
);
p_parent_input
->
psz_name
=
strdup
(
EnsureUTF8
(
p_sdp
->
psz_sessionname
)
);
p_parent_input
->
psz_name
=
strdup
(
p_sdp
->
psz_sessionname
);
p_parent_input
->
i_type
=
ITEM_TYPE_NET
;
p_parent_input
->
i_type
=
ITEM_TYPE_NET
;
if
(
p_playlist
->
status
.
p_item
&&
if
(
p_playlist
->
status
.
p_item
&&
...
@@ -702,7 +718,7 @@ static int ParseSAP( services_discovery_t *p_sd, const uint8_t *buf,
...
@@ -702,7 +718,7 @@ static int ParseSAP( services_discovery_t *p_sd, const uint8_t *buf,
}
}
/* Multi-media or no-parse -> pass to LIVE.COM */
/* Multi-media or no-parse -> pass to LIVE.COM */
if
(
p_sdp
->
i_media
>
1
||
(
p_sdp
->
i_media_type
!=
14
&&
if
(
p_sdp
->
mediac
>
1
||
(
p_sdp
->
i_media_type
!=
14
&&
p_sdp
->
i_media_type
!=
32
&&
p_sdp
->
i_media_type
!=
32
&&
p_sdp
->
i_media_type
!=
33
)
||
p_sdp
->
i_media_type
!=
33
)
||
p_sd
->
p_sys
->
b_parse
==
VLC_FALSE
)
p_sd
->
p_sys
->
b_parse
==
VLC_FALSE
)
...
@@ -731,11 +747,6 @@ static int ParseSAP( services_discovery_t *p_sd, const uint8_t *buf,
...
@@ -731,11 +747,6 @@ static int ParseSAP( services_discovery_t *p_sd, const uint8_t *buf,
return
VLC_SUCCESS
;
return
VLC_SUCCESS
;
}
}
}
}
/* Add item */
if
(
p_sdp
->
i_media
>
1
)
{
msg_Dbg
(
p_sd
,
"passing to liveMedia"
);
}
CreateAnnounce
(
p_sd
,
i_hash
,
p_sdp
);
CreateAnnounce
(
p_sd
,
i_hash
,
p_sdp
);
...
@@ -748,7 +759,7 @@ sap_announce_t *CreateAnnounce( services_discovery_t *p_sd, uint16_t i_hash,
...
@@ -748,7 +759,7 @@ sap_announce_t *CreateAnnounce( services_discovery_t *p_sd, uint16_t i_hash,
{
{
input_item_t
*
p_input
;
input_item_t
*
p_input
;
playlist_item_t
*
p_item
,
*
p_child
;
playlist_item_t
*
p_item
,
*
p_child
;
char
*
psz_value
;
c
onst
c
har
*
psz_value
;
sap_announce_t
*
p_sap
=
(
sap_announce_t
*
)
malloc
(
sap_announce_t
*
p_sap
=
(
sap_announce_t
*
)
malloc
(
sizeof
(
sap_announce_t
)
);
sizeof
(
sap_announce_t
)
);
services_discovery_sys_t
*
p_sys
;
services_discovery_sys_t
*
p_sys
;
...
@@ -757,7 +768,6 @@ sap_announce_t *CreateAnnounce( services_discovery_t *p_sd, uint16_t i_hash,
...
@@ -757,7 +768,6 @@ sap_announce_t *CreateAnnounce( services_discovery_t *p_sd, uint16_t i_hash,
p_sys
=
p_sd
->
p_sys
;
p_sys
=
p_sd
->
p_sys
;
EnsureUTF8
(
p_sdp
->
psz_sessionname
);
p_sap
->
i_last
=
mdate
();
p_sap
->
i_last
=
mdate
();
p_sap
->
i_hash
=
i_hash
;
p_sap
->
i_hash
=
i_hash
;
p_sap
->
p_sdp
=
p_sdp
;
p_sap
->
p_sdp
=
p_sdp
;
...
@@ -777,26 +787,24 @@ sap_announce_t *CreateAnnounce( services_discovery_t *p_sd, uint16_t i_hash,
...
@@ -777,26 +787,24 @@ sap_announce_t *CreateAnnounce( services_discovery_t *p_sd, uint16_t i_hash,
if
(
p_sys
->
b_timeshift
)
if
(
p_sys
->
b_timeshift
)
input_ItemAddOption
(
p_input
,
":access-filter=timeshift"
);
input_ItemAddOption
(
p_input
,
":access-filter=timeshift"
);
psz_value
=
GetAttribute
(
p_sap
->
p_sdp
,
"tool"
);
psz_value
=
GetAttribute
(
p_sap
->
p_sdp
->
pp_attributes
,
p_sap
->
p_sdp
->
i_attributes
,
"tool"
);
if
(
psz_value
!=
NULL
)
if
(
psz_value
!=
NULL
)
{
{
input_ItemAddInfo
(
p_input
,
_
(
"Session"
),
_
(
"Tool"
),
psz_value
);
input_ItemAddInfo
(
p_input
,
_
(
"Session"
),
_
(
"Tool"
),
psz_value
);
}
}
if
(
strcmp
(
p_sdp
->
psz_
username
,
"-"
)
)
if
(
strcmp
(
p_sdp
->
username
,
"-"
)
)
{
{
input_ItemAddInfo
(
p_input
,
_
(
"Session"
),
input_ItemAddInfo
(
p_input
,
_
(
"Session"
),
_
(
"User"
),
p_sdp
->
psz_
username
);
_
(
"User"
),
p_sdp
->
username
);
}
}
/* Handle group */
/* Handle group */
psz_value
=
GetAttribute
(
p_sap
->
p_sdp
,
"x-plgroup"
);
psz_value
=
GetAttribute
(
p_sap
->
p_sdp
->
pp_attributes
,
p_sap
->
p_sdp
->
i_attributes
,
"x-plgroup"
);
if
(
psz_value
==
NULL
)
if
(
psz_value
==
NULL
)
psz_value
=
GetAttribute
(
p_sap
->
p_sdp
,
"plgroup"
);
psz_value
=
GetAttribute
(
p_sap
->
p_sdp
->
pp_attributes
,
p_sap
->
p_sdp
->
i_attributes
,
"plgroup"
);
if
(
psz_value
!=
NULL
)
if
(
psz_value
!=
NULL
)
{
{
EnsureUTF8
(
psz_value
);
p_child
=
playlist_ChildSearchName
(
p_sys
->
p_node_cat
,
psz_value
);
p_child
=
playlist_ChildSearchName
(
p_sys
->
p_node_cat
,
psz_value
);
if
(
p_child
==
NULL
)
if
(
p_child
==
NULL
)
...
@@ -828,163 +836,91 @@ sap_announce_t *CreateAnnounce( services_discovery_t *p_sd, uint16_t i_hash,
...
@@ -828,163 +836,91 @@ sap_announce_t *CreateAnnounce( services_discovery_t *p_sd, uint16_t i_hash,
return
p_sap
;
return
p_sap
;
}
}
static
char
*
GetAttribute
(
sdp_t
*
p_sdp
,
const
char
*
psz_search
)
{
int
i
;
for
(
i
=
0
;
i
<
p_sdp
->
i_attributes
;
i
++
)
{
if
(
!
strncmp
(
p_sdp
->
pp_attributes
[
i
]
->
psz_field
,
psz_search
,
strlen
(
p_sdp
->
pp_attributes
[
i
]
->
psz_field
)
)
)
{
return
p_sdp
->
pp_attributes
[
i
]
->
psz_value
;
}
}
return
NULL
;
}
/* Fill p_sdp->psz_uri */
/* Fill p_sdp->psz_uri */
static
int
ParseConnection
(
vlc_object_t
*
p_obj
,
sdp_t
*
p_sdp
)
static
int
ParseConnection
(
vlc_object_t
*
p_obj
,
sdp_t
*
p_sdp
)
{
{
char
*
psz_eof
=
NULL
;
if
(
p_sdp
->
mediac
!=
1
)
char
*
psz_parse
=
NULL
;
return
VLC_EGENERIC
;
char
psz_uri
[
1026
];
char
*
psz_proto
=
NULL
;
int
i_port
=
0
;
/* Parse c= field */
if
(
p_sdp
->
psz_connection
)
{
char
hostname
[
1024
];
int
ipv
;
/*
* NOTE: we ignore the TTL parameter on-purpose, as some SAP
* advertisers don't include it (and it is utterly useless).
*/
if
(
sscanf
(
p_sdp
->
psz_connection
,
"IN IP%d %1023[^/]"
,
&
ipv
,
hostname
)
!=
2
)
{
msg_Warn
(
p_obj
,
"unable to parse c field:
\"
%s
\"
"
,
p_sdp
->
psz_connection
);
return
VLC_EGENERIC
;
}
switch
(
ipv
)
char
psz_uri
[
1026
];
{
const
char
*
host
;
case
4
:
int
port
;
case
6
:
break
;
default:
psz_uri
[
0
]
=
'['
;
msg_Warn
(
p_obj
,
"unknown IP version %d"
,
ipv
);
if
(
vlc_getnameinfo
((
struct
sockaddr
*
)
&
(
p_sdp
->
mediav
[
0
].
addr
),
return
VLC_EGENERIC
;
p_sdp
->
mediav
[
0
].
addrlen
,
psz_uri
+
1
,
}
sizeof
(
psz_uri
)
-
2
,
&
port
,
NI_NUMERICHOST
))
return
VLC_EGENERIC
;
if
(
strchr
(
hostname
,
':'
)
!=
NULL
)
if
(
strchr
(
psz_uri
+
1
,
':'
)
)
sprintf
(
psz_uri
,
"[%s]"
,
hostname
);
{
else
host
=
psz_uri
;
strcpy
(
psz_uri
,
hostname
)
;
psz_uri
[
strlen
(
psz_uri
)]
=
']'
;
}
}
else
host
=
psz_uri
+
1
;
/* Parse m= field */
/* Parse m= field */
if
(
p_sdp
->
psz_media
)
char
*
sdp_proto
=
strdup
(
p_sdp
->
mediav
[
0
].
fmt
);
{
if
(
sdp_proto
==
NULL
)
psz_parse
=
p_sdp
->
psz_media
;
return
VLC_ENOMEM
;
psz_eof
=
strchr
(
psz_parse
,
' '
);
if
(
psz_eof
)
char
*
subtype
=
strchr
(
sdp_proto
,
' '
);
{
if
(
sdp_proto
==
NULL
)
*
psz_eof
=
'\0'
;
{
msg_Dbg
(
p_obj
,
"missing SDP media subtype: %s"
,
sdp_proto
);
/*
p_sdp
->
i_media_type
=
0
;
* That's ugly. We should go through every media, and make sure
* at least one of them is audio or video. In the mean time, I
* need to accept data too.
*/
if
(
strncmp
(
psz_parse
,
"audio"
,
5
)
&&
strncmp
(
psz_parse
,
"video"
,
5
)
&&
strncmp
(
psz_parse
,
"data"
,
4
)
)
{
msg_Warn
(
p_obj
,
"unhandled media type
\"
%s
\"
"
,
psz_parse
);
return
VLC_EGENERIC
;
}
psz_parse
=
psz_eof
+
1
;
}
else
{
msg_Warn
(
p_obj
,
"unable to parse m field (1)"
);
return
VLC_EGENERIC
;
}
psz_eof
=
strchr
(
psz_parse
,
' '
);
if
(
psz_eof
)
{
*
psz_eof
=
'\0'
;
/* FIXME : multiple port ! */
i_port
=
atoi
(
psz_parse
);
if
(
i_port
<=
0
||
i_port
>=
65536
)
{
msg_Warn
(
p_obj
,
"invalid transport port %i"
,
i_port
);
}
psz_parse
=
psz_eof
+
1
;
}
else
{
msg_Warn
(
p_obj
,
"unable to parse m field (2)"
);
return
VLC_EGENERIC
;
}
psz_eof
=
strchr
(
psz_parse
,
' '
);
if
(
psz_eof
)
{
*
psz_eof
=
'\0'
;
psz_proto
=
strdup
(
psz_parse
);
psz_parse
=
psz_eof
+
1
;
p_sdp
->
i_media_type
=
atoi
(
psz_parse
);
}
else
{
msg_Dbg
(
p_obj
,
"incorrect m field, %s"
,
p_sdp
->
psz_media
);
p_sdp
->
i_media_type
=
33
;
psz_proto
=
strdup
(
psz_parse
);
}
}
}
else
if
(
psz_proto
&&
!
strncmp
(
psz_proto
,
"RTP/AVP"
,
7
)
)
{
{
free
(
psz_proto
)
;
*
subtype
++
=
'\0'
;
p
sz_proto
=
strdup
(
"rtp"
);
p
_sdp
->
i_media_type
=
atoi
(
subtype
);
}
}
if
(
psz_proto
&&
!
strncasecmp
(
psz_proto
,
"UDP"
,
3
)
)
if
(
p_sdp
->
i_media_type
==
0
)
p_sdp
->
i_media_type
=
33
;
static
const
char
proto_match
[]
=
"udp
\0
"
"udp
\0
"
"RTP/AVP
\0
"
"rtp
\0
"
"UDPLite/RTP/AVP
\0
"
"udplite
\0
"
"DCCP/RTP/AVP
\0
"
"dccp
\0
"
"TCP/RTP/AVP
\0
"
"tcp
\0
"
"
\0
"
;
const
char
*
vlc_proto
=
NULL
;
for
(
const
char
*
proto
=
proto_match
;
*
proto
;)
{
{
free
(
psz_proto
);
if
(
strcasecmp
(
proto
,
sdp_proto
))
psz_proto
=
strdup
(
"udp"
);
{
vlc_proto
=
proto
+
strlen
(
proto
)
+
1
;
break
;
}
proto
+=
strlen
(
proto
)
+
1
;
proto
+=
strlen
(
proto
)
+
1
;
}
}
if
(
i_port
==
0
)
free
(
sdp_proto
);
if
(
vlc_proto
==
NULL
)
{
{
i_port
=
1234
;
msg_Dbg
(
p_obj
,
"unknown SDP media protocol: %s"
,
p_sdp
->
mediav
[
0
].
fmt
);
return
VLC_EGENERIC
;
}
}
/* handle SSM case */
/* handle SSM case */
psz_parse
=
GetAttribute
(
p_sdp
,
"source-filter"
);
const
char
*
sfilter
=
GetAttribute
(
p_sdp
->
mediav
[
0
].
pp_attributes
,
p_sdp
->
mediav
[
0
].
i_attributes
,
"source-filter"
);
if
(
sfilter
==
NULL
)
sfilter
=
GetAttribute
(
p_sdp
->
pp_attributes
,
p_sdp
->
i_attributes
,
"source-filter"
);
char
psz_source
[
258
]
=
""
;
char
psz_source
[
258
]
=
""
;
if
(
psz_parse
!=
NULL
)
if
(
sfilter
!=
NULL
)
{
{
char
psz_source_ip
[
256
];
char
psz_source_ip
[
256
];
if
(
sscanf
(
psz_parse
,
" incl IN IP%*c %*s %255s "
,
psz_source_ip
)
==
1
)
if
(
sscanf
(
sfilter
,
" incl IN IP%*c %*s %255s "
,
psz_source_ip
)
==
1
)
{
{
if
(
strchr
(
psz_source_ip
,
':'
)
!=
NULL
)
if
(
strchr
(
psz_source_ip
,
':'
)
!=
NULL
)
sprintf
(
psz_source
,
"[%s]"
,
psz_source_ip
);
sprintf
(
psz_source
,
"[%s]"
,
psz_source_ip
);
...
@@ -993,13 +929,62 @@ static int ParseConnection( vlc_object_t *p_obj, sdp_t *p_sdp )
...
@@ -993,13 +929,62 @@ static int ParseConnection( vlc_object_t *p_obj, sdp_t *p_sdp )
}
}
}
}
asprintf
(
&
p_sdp
->
psz_uri
,
"%s://%s@%s:%i"
,
psz
_proto
,
psz_source
,
asprintf
(
&
p_sdp
->
psz_uri
,
"%s://%s@%s:%i"
,
vlc
_proto
,
psz_source
,
psz_uri
,
i_
port
);
host
,
port
);
FREENULL
(
psz_proto
);
return
VLC_SUCCESS
;
return
VLC_SUCCESS
;
}
}
static
int
ParseSDPConnection
(
const
char
*
str
,
struct
sockaddr_storage
*
addr
,
socklen_t
*
addrlen
,
unsigned
*
number
)
{
char
host
[
60
];
unsigned
fam
,
n1
,
n2
;
int
res
=
sscanf
(
str
,
"IN IP%u %59[^/]/%u/%u"
,
&
fam
,
host
,
&
n1
,
&
n2
);
if
(
res
<
2
)
return
-
1
;
switch
(
fam
)
{
#ifdef AF_INET6
case
6
:
addr
->
ss_family
=
AF_INET6
;
# ifdef HAVE_SA_LEN
addr
->
ss_len
=
# endif
*
addrlen
=
sizeof
(
struct
sockaddr_in6
);
if
(
inet_pton
(
AF_INET6
,
host
,
&
((
struct
sockaddr_in6
*
)
addr
)
->
sin6_addr
)
<=
0
)
return
-
1
;
*
number
=
(
res
>=
3
)
?
n1
:
1
;
break
;
#endif
case
4
:
addr
->
ss_family
=
AF_INET
;
# ifdef HAVE_SA_LEN
addr
->
ss_len
=
# endif
*
addrlen
=
sizeof
(
struct
sockaddr_in
);
if
(
inet_pton
(
AF_INET
,
host
,
&
((
struct
sockaddr_in
*
)
addr
)
->
sin_addr
)
<=
0
)
return
-
1
;
*
number
=
(
res
>=
4
)
?
n2
:
1
;
break
;
default:
return
-
1
;
}
return
0
;
}
/***********************************************************************
/***********************************************************************
* ParseSDP : SDP parsing
* ParseSDP : SDP parsing
* *********************************************************************
* *********************************************************************
...
@@ -1007,187 +992,285 @@ static int ParseConnection( vlc_object_t *p_obj, sdp_t *p_sdp )
...
@@ -1007,187 +992,285 @@ static int ParseConnection( vlc_object_t *p_obj, sdp_t *p_sdp )
***********************************************************************/
***********************************************************************/
static
sdp_t
*
ParseSDP
(
vlc_object_t
*
p_obj
,
const
char
*
psz_sdp
)
static
sdp_t
*
ParseSDP
(
vlc_object_t
*
p_obj
,
const
char
*
psz_sdp
)
{
{
sdp_t
*
p_sdp
;
vlc_bool_t
b_invalid
=
VLC_FALSE
;
vlc_bool_t
b_end
=
VLC_FALSE
;
if
(
psz_sdp
==
NULL
)
if
(
psz_sdp
==
NULL
)
{
return
NULL
;
return
NULL
;
}
if
(
psz_sdp
[
0
]
!=
'v'
||
psz_sdp
[
1
]
!=
'='
)
{
msg_Warn
(
p_obj
,
"bad packet"
);
return
NULL
;
}
p_sdp
=
(
sdp_t
*
)
malloc
(
sizeof
(
sdp_t
)
);
sdp_t
*
p_sdp
=
calloc
(
1
,
sizeof
(
*
p_sdp
)
);
if
(
p_sdp
==
NULL
)
if
(
p_sdp
==
NULL
)
return
NULL
;
return
NULL
;
/* init to 0 */
char
expect
=
'V'
;
memset
(
p_sdp
,
0
,
sizeof
(
sdp_t
)
);
struct
sockaddr_storage
glob_addr
;
memset
(
&
glob_addr
,
0
,
sizeof
(
glob_addr
));
socklen_t
glob_len
=
0
;
unsigned
glob_count
=
1
;
p_sdp
->
psz_sdp
=
strdup
(
psz_sdp
);
/* TODO: use iconv and charset attribute instead of EnsureUTF8 */
if
(
p_sdp
->
psz_sdp
==
NULL
)
while
(
*
psz_sdp
)
{
{
free
(
p_sdp
);
/* Extract one line */
return
NULL
;
char
*
eol
=
strchr
(
psz_sdp
,
'\n'
);
}
size_t
linelen
=
eol
?
(
size_t
)(
eol
-
psz_sdp
)
:
strlen
(
psz_sdp
);
char
line
[
linelen
+
1
];
while
(
*
psz_sdp
!=
'\0'
&&
b_end
==
VLC_FALSE
)
memcpy
(
line
,
psz_sdp
,
linelen
);
{
line
[
linelen
]
=
'\0'
;
char
*
psz_eol
=
NULL
;
char
*
psz_eof
=
NULL
;
psz_sdp
+=
linelen
+
1
;
char
*
psz_parse
=
NULL
;
char
*
psz_sess_id
=
NULL
;
/* Remove carriage return if present */
eol
=
strchr
(
line
,
'\r'
);
while
(
*
psz_sdp
==
'\r'
||
*
psz_sdp
==
'\n'
||
if
(
eol
!=
NULL
)
*
psz_sdp
==
' '
||
*
psz_sdp
==
'\t'
)
{
{
psz_sdp
++
;
linelen
=
eol
-
line
;
line
[
linelen
]
=
'\0'
;
}
}
if
(
(
psz_eol
=
strchr
(
psz_sdp
,
'\n'
)
)
==
NULL
)
/* Validate line */
{
char
cat
=
line
[
0
],
*
data
=
line
+
2
;
psz_eol
=
(
char
*
)
psz_sdp
+
strlen
(
psz_sdp
);
if
(
!
cat
||
(
strchr
(
"vosiuepcbtrzkam"
,
cat
)
==
NULL
))
b_end
=
VLC_TRUE
;
}
if
(
psz_eol
>
psz_sdp
&&
*
(
psz_eol
-
1
)
==
'\r'
)
{
{
psz_eol
--
;
/* MUST ignore SDP with unknown line type */
msg_Dbg
(
p_obj
,
"unknown SDP line type: 0x%02x"
,
(
int
)
cat
);
goto
error
;
}
}
if
(
line
[
1
]
!=
'='
)
if
(
psz_eol
<=
psz_sdp
)
{
{
break
;
msg_Dbg
(
p_obj
,
"invalid SDP line: %s"
,
line
);
goto
error
;
}
}
*
psz_eol
++
=
'\0'
;
/* no space allowed between fields */
assert
(
linelen
>=
2
);
if
(
psz_sdp
[
1
]
!=
'='
)
{
msg_Warn
(
p_obj
,
"invalid packet"
)
;
FreeSDP
(
p_sdp
);
p_sdp
=
NULL
;
return
NULL
;
}
/* Now parse each line */
/* SDP parsing state machine
switch
(
psz_sdp
[
0
]
)
* We INTERNALLY use uppercase for session, lowercase for media
*/
switch
(
expect
)
{
{
case
(
'v'
):
/* Session description */
break
;
case
'V'
:
case
(
's'
):
expect
=
'O'
;
p_sdp
->
psz_sessionname
=
strdup
(
&
psz_sdp
[
2
]
);
if
(
cat
!=
'v'
)
{
msg_Dbg
(
p_obj
,
"missing SDP version"
);
goto
error
;
}
if
(
strcmp
(
data
,
"0"
))
{
msg_Dbg
(
p_obj
,
"unknown SDP version: %s"
,
data
);
goto
error
;
}
break
;
break
;
case
(
'o'
):
case
'O'
:
{
{
int
i_field
=
0
;
expect
=
'S'
;
/* o field is <username> <session id> <version>
if
(
cat
!=
'o'
)
* <network type> <address type> <address> */
{
msg_Dbg
(
p_obj
,
"missing SDP originator"
);
#define GET_FIELD( store ) \
goto
error
;
psz_eof = strchr( psz_parse, ' ' ); \
}
if( psz_eof ) \
{ \
*psz_eof=0; store = strdup( psz_parse ); \
} \
else \
{ \
if( i_field != 5 ) \
{ \
b_invalid = VLC_TRUE; break; \
} \
else \
{ \
store = strdup( psz_parse ); \
} \
}; \
psz_parse = psz_eof + 1; i_field++;
psz_parse
=
(
char
*
)
&
psz_sdp
[
2
];
GET_FIELD
(
p_sdp
->
psz_username
);
GET_FIELD
(
psz_sess_id
);
p_sdp
->
i_session_id
=
atoll
(
psz_sess_id
);
FREENULL
(
psz_sess_id
);
GET_FIELD
(
psz_sess_id
);
FREENULL
(
psz_sess_id
);
GET_FIELD
(
p_sdp
->
psz_network_type
);
GET_FIELD
(
p_sdp
->
psz_address_type
);
GET_FIELD
(
p_sdp
->
psz_address
);
if
((
sscanf
(
data
,
"%63s "
I64Fu
" "
I64Fu
" IN IP%u %1023s"
,
p_sdp
->
username
,
&
p_sdp
->
session_id
,
&
p_sdp
->
session_version
,
&
p_sdp
->
orig_ip_version
,
p_sdp
->
orig_host
)
!=
5
)
||
((
p_sdp
->
orig_ip_version
!=
4
)
&&
(
p_sdp
->
orig_ip_version
!=
6
)))
{
msg_Dbg
(
p_obj
,
"SDP origin not supported: %s
\n
"
,
data
);
/* Or maybe out-of-range, but this looks suspicious */
return
NULL
;
}
EnsureUTF8
(
p_sdp
->
orig_host
);
break
;
break
;
}
}
case
(
'i'
):
case
(
'u'
):
case
(
'e'
):
case
(
'p'
):
case
(
't'
):
case
(
'r'
):
break
;
case
(
'a'
):
/* attribute */
{
char
*
psz_eon
=
strchr
(
&
psz_sdp
[
2
],
':'
);
attribute_t
*
p_attr
=
malloc
(
sizeof
(
attribute_t
)
);
/* Attribute with value */
case
'S'
:
if
(
psz_eon
)
{
expect
=
'I'
;
if
((
cat
!=
's'
)
||
!*
data
)
{
{
*
psz_eon
++
=
'\0'
;
/* MUST be present AND non-empty */
msg_Dbg
(
p_obj
,
"missing SDP session name"
);
goto
error
;
}
assert
(
p_sdp
->
psz_sessionname
==
NULL
);
// no memleak here
p_sdp
->
psz_sessionname
=
strdup
(
data
);
EnsureUTF8
(
p_sdp
->
psz_sessionname
);
if
(
p_sdp
->
psz_sessionname
==
NULL
)
goto
error
;
break
;
}
p_attr
->
psz_field
=
strdup
(
&
psz_sdp
[
2
]
);
case
'I'
:
p_attr
->
psz_value
=
strdup
(
psz_eon
);
expect
=
'U'
;
if
(
cat
==
'i'
)
break
;
case
'U'
:
expect
=
'E'
;
if
(
cat
==
'u'
)
break
;
case
'E'
:
expect
=
'E'
;
if
(
cat
==
'e'
)
break
;
case
'P'
:
expect
=
'P'
;
if
(
cat
==
'p'
)
break
;
case
'C'
:
expect
=
'B'
;
if
(
cat
==
'c'
)
{
if
(
ParseSDPConnection
(
data
,
&
glob_addr
,
&
glob_len
,
&
glob_count
))
{
msg_Dbg
(
p_obj
,
"SDP connection infos not supported: "
"%s"
,
data
);
goto
error
;
}
break
;
}
}
else
/* Attribute without value */
case
'B'
:
assert
(
expect
==
'B'
);
if
(
cat
==
'b'
)
break
;
case
'T'
:
expect
=
'R'
;
if
(
cat
!=
't'
)
{
{
p_attr
->
psz_field
=
strdup
(
&
psz_sdp
[
2
]
);
msg_Dbg
(
p_obj
,
"missing SDP time description"
);
p_attr
->
psz_value
=
NULL
;
goto
error
;
}
}
TAB_APPEND
(
p_sdp
->
i_attributes
,
p_sdp
->
pp_attributes
,
p_attr
);
break
;
break
;
}
case
(
'm'
):
/* Media announcement */
case
'R'
:
if
((
cat
==
't'
)
||
(
cat
==
'r'
))
break
;
case
'Z'
:
expect
=
'K'
;
if
(
cat
==
'z'
)
break
;
case
'K'
:
expect
=
'A'
;
if
(
cat
==
'k'
)
break
;
case
'A'
:
//expect = 'A';
if
(
cat
==
'a'
)
{
attribute_t
*
p_attr
=
MakeAttribute
(
data
);
TAB_APPEND
(
p_sdp
->
i_attributes
,
p_sdp
->
pp_attributes
,
p_attr
);
break
;
}
/* Media description */
case
'm'
:
{
{
/* If we have several medias, we pass the announcement to
expect
=
'i'
;
* LIVE.COM, so just count them */
if
(
cat
!=
'm'
)
p_sdp
->
i_media
++
;
if
(
p_sdp
->
i_media
==
1
)
{
{
p_sdp
->
psz_media
=
strdup
(
&
psz_sdp
[
2
]
);
msg_Dbg
(
p_obj
,
"missing SDP media description"
);
goto
error
;
}
}
struct
sdp_media_t
*
m
=
p_sdp
->
mediav
+
p_sdp
->
mediac
;
memcpy
(
&
m
->
addr
,
&
glob_addr
,
m
->
addrlen
=
glob_len
);
m
->
n_addr
=
glob_count
;
/* TODO: remember media type (if we need multiple medias) */
data
=
strchr
(
data
,
' '
);
if
(
data
==
NULL
)
{
msg_Dbg
(
p_obj
,
"missing SDP media port"
);
goto
error
;
}
int
port
=
atoi
(
data
);
if
(
port
<=
0
||
port
>=
65536
)
{
msg_Dbg
(
p_obj
,
"invalid transport port %d"
,
port
);
goto
error
;
}
net_SetPort
((
struct
sockaddr
*
)
&
m
->
addr
,
htons
(
port
));
data
=
strchr
(
data
,
' '
);
if
(
data
==
NULL
)
{
msg_Dbg
(
p_obj
,
"missing SDP media format"
);
goto
error
;
}
m
->
fmt
=
strdup
(
data
);
if
(
m
->
fmt
==
NULL
)
goto
error
;
p_sdp
->
mediac
++
;
break
;
break
;
}
}
case
'i'
:
expect
=
'c'
;
if
(
cat
==
'i'
)
break
;
case
'c'
:
expect
=
'b'
;
if
(
cat
==
'c'
)
{
struct
sdp_media_t
*
m
=
p_sdp
->
mediav
+
p_sdp
->
mediac
;
if
(
ParseSDPConnection
(
data
,
&
m
->
addr
,
&
m
->
addrlen
,
&
m
->
n_addr
))
{
msg_Dbg
(
p_obj
,
"SDP connection infos not supported: "
"%s"
,
data
);
goto
error
;
}
break
;
}
case
'b'
:
expect
=
'b'
;
if
(
cat
==
'b'
)
break
;
case
'k'
:
expect
=
'a'
;
if
(
cat
==
'k'
)
break
;
case
'a'
:
assert
(
expect
==
'a'
);
if
(
cat
==
'a'
)
{
attribute_t
*
p_attr
=
MakeAttribute
(
data
);
if
(
p_attr
==
NULL
)
goto
error
;
case
(
'c'
):
TAB_APPEND
(
p_sdp
->
mediav
[
p_sdp
->
mediac
-
1
].
i_attributes
,
{
p_sdp
->
mediav
[
p_sdp
->
mediac
-
1
].
pp_attributes
,
p_attr
);
if
(
p_sdp
->
psz_connection
!=
NULL
)
// FIXME
break
;
break
;
}
p_sdp
->
psz_connection
=
strdup
(
&
psz_sdp
[
2
]
);
if
(
cat
==
'm'
)
{
/* TODO */
msg_Dbg
(
p_obj
,
"multi-media SDP not implemented -> live555"
);
goto
error
;
}
if
(
cat
!=
'm'
)
{
msg_Dbg
(
p_obj
,
"unexpected SDP line: 0x%02x"
,
(
int
)
cat
);
goto
error
;
}
break
;
break
;
}
default:
default:
break
;
msg_Err
(
p_obj
,
"*** BUG in SDP parser! ***"
);
goto
error
;
}
}
if
(
b_invalid
)
{
FreeSDP
(
p_sdp
);
p_sdp
=
NULL
;
return
NULL
;
}
psz_sdp
=
psz_eol
;
}
}
return
p_sdp
;
return
p_sdp
;
error:
FreeSDP
(
p_sdp
);
return
NULL
;
}
}
static
int
InitSocket
(
services_discovery_t
*
p_sd
,
const
char
*
psz_address
,
static
int
InitSocket
(
services_discovery_t
*
p_sd
,
const
char
*
psz_address
,
...
@@ -1253,27 +1336,22 @@ static int Decompress( const unsigned char *psz_src, unsigned char **_dst, int i
...
@@ -1253,27 +1336,22 @@ static int Decompress( const unsigned char *psz_src, unsigned char **_dst, int i
static
void
FreeSDP
(
sdp_t
*
p_sdp
)
static
void
FreeSDP
(
sdp_t
*
p_sdp
)
{
{
int
i
;
FREENULL
(
p_sdp
->
psz_sdp
);
FREENULL
(
p_sdp
->
psz_sdp
);
FREENULL
(
p_sdp
->
psz_sessionname
);
FREENULL
(
p_sdp
->
psz_sessionname
);
FREENULL
(
p_sdp
->
psz_connection
);
FREENULL
(
p_sdp
->
psz_media
);
FREENULL
(
p_sdp
->
psz_uri
);
FREENULL
(
p_sdp
->
psz_uri
);
FREENULL
(
p_sdp
->
psz_username
);
FREENULL
(
p_sdp
->
psz_network_type
);
FREENULL
(
p_sdp
->
psz_address
);
for
(
unsigned
j
=
0
;
j
<
p_sdp
->
mediac
;
j
++
)
FREENULL
(
p_sdp
->
psz_address_type
);
for
(
i
=
p_sdp
->
i_attributes
-
1
;
i
>=
0
;
i
--
)
{
{
struct
attribute_t
*
p_attr
=
p_sdp
->
pp_attributes
[
i
];
free
(
p_sdp
->
mediav
[
j
].
fmt
);
FREENULL
(
p_sdp
->
pp_attributes
[
i
]
->
psz_field
);
for
(
int
i
=
0
;
i
<
p_sdp
->
mediav
[
j
].
i_attributes
;
i
++
)
FREENULL
(
p_sdp
->
pp_attributes
[
i
]
->
psz_value
);
FreeAttribute
(
p_sdp
->
mediav
[
j
].
pp_attributes
[
i
]);
REMOVE_ELEM
(
p_sdp
->
pp_attributes
,
p_sdp
->
i_attributes
,
i
);
FREENULL
(
p_attr
);
}
}
FREENULL
(
p_sdp
);
for
(
int
i
=
0
;
i
<
p_sdp
->
i_attributes
;
i
++
)
FreeAttribute
(
p_sdp
->
pp_attributes
[
i
]);
free
(
p_sdp
->
pp_attributes
);
free
(
p_sdp
);
}
}
static
int
RemoveAnnounce
(
services_discovery_t
*
p_sd
,
static
int
RemoveAnnounce
(
services_discovery_t
*
p_sd
,
...
@@ -1308,27 +1386,53 @@ static int RemoveAnnounce( services_discovery_t *p_sd,
...
@@ -1308,27 +1386,53 @@ static int RemoveAnnounce( services_discovery_t *p_sd,
static
vlc_bool_t
IsSameSession
(
sdp_t
*
p_sdp1
,
sdp_t
*
p_sdp2
)
static
vlc_bool_t
IsSameSession
(
sdp_t
*
p_sdp1
,
sdp_t
*
p_sdp2
)
{
{
/* A session is identified by
/* A session is identified by
* username, session_id, network type, address type and address */
* - username,
if
(
p_sdp1
->
psz_username
&&
p_sdp2
->
psz_username
&&
* - session_id,
p_sdp1
->
psz_network_type
&&
p_sdp2
->
psz_network_type
&&
* - network type (which is always IN),
p_sdp1
->
psz_address_type
&&
p_sdp2
->
psz_address_type
&&
* - address type (currently, this means IP version),
p_sdp1
->
psz_address
&&
p_sdp2
->
psz_address
)
* - and hostname.
*/
if
(
strcmp
(
p_sdp1
->
username
,
p_sdp2
->
username
)
||
(
p_sdp1
->
session_id
!=
p_sdp2
->
session_id
)
||
(
p_sdp1
->
orig_ip_version
!=
p_sdp2
->
orig_ip_version
)
||
strcmp
(
p_sdp1
->
orig_host
,
p_sdp2
->
orig_host
))
return
VLC_FALSE
;
return
VLC_TRUE
;
}
static
inline
attribute_t
*
MakeAttribute
(
const
char
*
str
)
{
attribute_t
*
a
=
malloc
(
sizeof
(
*
a
)
+
strlen
(
str
)
+
1
);
if
(
a
==
NULL
)
return
NULL
;
strcpy
(
a
->
name
,
str
);
EnsureUTF8
(
a
->
name
);
char
*
value
=
strchr
(
a
->
name
,
':'
);
if
(
value
!=
NULL
)
{
{
if
(
!
strcmp
(
p_sdp1
->
psz_username
,
p_sdp2
->
psz_username
)
&&
*
value
++
=
'\0'
;
!
strcmp
(
p_sdp1
->
psz_network_type
,
p_sdp2
->
psz_network_type
)
&&
a
->
value
=
value
;
!
strcmp
(
p_sdp1
->
psz_address_type
,
p_sdp2
->
psz_address_type
)
&&
!
strcmp
(
p_sdp1
->
psz_address
,
p_sdp2
->
psz_address
)
&&
p_sdp1
->
i_session_id
==
p_sdp2
->
i_session_id
)
{
return
VLC_TRUE
;
}
else
{
return
VLC_FALSE
;
}
}
}
else
else
{
a
->
value
=
""
;
return
VLC_FALSE
;
return
a
;
}
}
static
const
char
*
GetAttribute
(
attribute_t
**
tab
,
unsigned
n
,
const
char
*
name
)
{
for
(
unsigned
i
=
0
;
i
<
n
;
i
++
)
if
(
strcasecmp
(
tab
[
i
]
->
name
,
name
)
==
0
)
return
tab
[
i
]
->
value
;
return
NULL
;
}
static
inline
void
FreeAttribute
(
attribute_t
*
a
)
{
free
(
a
);
}
}
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment