[vlc-devel] [PATCH 13/13] modules/access/rtsp: `rtsp_get_answers` (many fixes)
Filip Roséen
filip at atch.se
Thu Feb 25 10:12:39 CET 2016
`rtsp_get_answers` would crash/have unexpected behavior in the following
scenarios:
- remote server sent more than 256 headers (off-by-one write)
- remote server sent one of the following headers without payload
- Server
- Session
- CSeq
In order to fix the issue a bunch of checks have been introduced, mostly
boiling down checking to see that `sscanf` succeeds in reading data into
the destination variable.
A few warnings (`msg_Warn`) have been introduced to help diagnostic
servers that misbehave.
---
modules/access/rtsp/rtsp.c | 56 +++++++++++++++++++++++++++-------------------
1 file changed, 33 insertions(+), 23 deletions(-)
diff --git a/modules/access/rtsp/rtsp.c b/modules/access/rtsp/rtsp.c
index 3072055..35e4caa 100644
--- a/modules/access/rtsp/rtsp.c
+++ b/modules/access/rtsp/rtsp.c
@@ -215,6 +215,7 @@ static void rtsp_schedule_standard( rtsp_client_t *rtsp )
static int rtsp_get_answers( rtsp_client_t *rtsp )
{
+ access_t *p_access = (access_t*)rtsp->p_userdata;
char *answer = NULL;
unsigned int answer_seq;
char **answer_ptr = rtsp->p_private->answers;
@@ -235,40 +236,46 @@ static int rtsp_get_answers( rtsp_client_t *rtsp )
if( !strncasecmp( answer, "CSeq:", 5 ) )
{
- sscanf( answer, "%*s %u", &answer_seq );
- if( rtsp->p_private->cseq != answer_seq )
- {
- //fprintf( stderr, "warning: Cseq mismatch. got %u, assumed %u",
- // answer_seq, rtsp->p_private->cseq );
-
- rtsp->p_private->cseq = answer_seq;
+ if (sscanf( answer, "%*s %u", &answer_seq ) == 1) {
+ if( rtsp->p_private->cseq != answer_seq )
+ {
+ msg_Warn (p_access, "Cseq mismatch, got %u, assumed %u", answer_seq, rtsp->p_private->cseq);
+ rtsp->p_private->cseq = answer_seq;
+ }
+ } else {
+ msg_Warn (p_access, "remote server sent CSeq without payload, ignoring.");
}
}
if( !strncasecmp( answer, "Server:", 7 ) )
{
char *buf = xmalloc( strlen(answer) );
- sscanf( answer, "%*s %s", buf );
- free( rtsp->p_private->server );
- rtsp->p_private->server = buf;
+ if (sscanf( answer, "%*s %s", buf ) == 1) {
+ free( rtsp->p_private->server );
+ rtsp->p_private->server = buf;
+ } else {
+ msg_Warn(p_access, "remote server sent Server without payload, ignoring.");
+ }
}
if( !strncasecmp( answer, "Session:", 8 ) )
{
char *buf = xmalloc( strlen(answer) );
- sscanf( answer, "%*s %s", buf );
- if( rtsp->p_private->session )
- {
- if( strcmp( buf, rtsp->p_private->session ) )
+ if (sscanf( answer, "%*s %s", buf ) == 1) { // TODO: ignore attributes "Session: ${session-id};${attribute=value...}"
+ if( rtsp->p_private->session )
{
- //fprintf( stderr,
- // "rtsp: warning: setting NEW session: %s\n", buf );
- free( rtsp->p_private->session );
+ if( strcmp( buf, rtsp->p_private->session ) )
+ {
+ msg_Warn (p_access, "setting NEW session: %s", buf);
+ free( rtsp->p_private->session );
+ rtsp->p_private->session = strdup( buf );
+ }
+ }
+ else
+ {
+ msg_Dbg (p_access, "session id: '%s'", buf);
rtsp->p_private->session = strdup( buf );
}
- }
- else
- {
- //fprintf( stderr, "setting session id to: %s\n", buf );
- rtsp->p_private->session = strdup( buf );
+ } else {
+ msg_Warn(p_access, "remote server sent Session without payload, ignoring.");
}
free( buf );
}
@@ -279,7 +286,10 @@ static int rtsp_get_answers( rtsp_client_t *rtsp )
rtsp->p_private->cseq++;
- *answer_ptr = NULL;
+ if (ans_count != MAX_FIELDS) {
+ *answer_ptr = NULL;
+ }
+
rtsp_schedule_standard( rtsp );
return code;
--
2.7.1
More information about the vlc-devel
mailing list