/* * Copyright (C) 2002-2003 the xine project * * This file is part of xine, a free video player. * * xine 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. * * xine 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA * * sdp/sdpplin parser. */ #define LOG_MODULE "sdpplin" #define LOG_VERBOSE /* #define LOG */ #include "rmff.h" #include "rtsp.h" #include "sdpplin.h" #include "xineutils.h" /* libavutil from FFmpeg */ #include static char *nl(char *data) { char *nlptr = (data) ? strchr(data,'\n') : NULL; return (nlptr) ? nlptr + 1 : NULL; } static int filter(const char *in, const char *filter, char **out) { int flen=strlen(filter); size_t len; if (!in) return 0; len = (strchr(in,'\n')) ? (size_t)(strchr(in,'\n')-in) : strlen(in); if (!strncmp(in,filter,flen)) { if(in[flen]=='"') flen++; if(in[len-1]==13) len--; if(in[len-1]=='"') len--; xine_buffer_copyin(*out, 0, in+flen, len-flen+1); (*out)[len-flen]=0; return len-flen; } return 0; } static sdpplin_stream_t *sdpplin_parse_stream(char **data) { sdpplin_stream_t *desc = xine_xmalloc(sizeof(sdpplin_stream_t)); char *buf=xine_buffer_init(32); int handled; if (filter(*data, "m=", &buf)) { desc->id = strdup(buf); } else { lprintf("sdpplin: no m= found.\n"); free(desc); xine_buffer_free(buf); return NULL; } *data=nl(*data); while (*data && **data && *data[0]!='m') { handled=0; if(filter(*data,"a=control:streamid=",&buf)) { desc->stream_id=atoi(buf); handled=1; *data=nl(*data); } if(filter(*data,"a=MaxBitRate:integer;",&buf)) { desc->max_bit_rate=atoi(buf); if (!desc->avg_bit_rate) desc->avg_bit_rate=desc->max_bit_rate; handled=1; *data=nl(*data); } if(filter(*data,"a=MaxPacketSize:integer;",&buf)) { desc->max_packet_size=atoi(buf); if (!desc->avg_packet_size) desc->avg_packet_size=desc->max_packet_size; handled=1; *data=nl(*data); } if(filter(*data,"a=StartTime:integer;",&buf)) { desc->start_time=atoi(buf); handled=1; *data=nl(*data); } if(filter(*data,"a=Preroll:integer;",&buf)) { desc->preroll=atoi(buf); handled=1; *data=nl(*data); } if(filter(*data,"a=length:npt=",&buf)) { desc->duration=(uint32_t)(atof(buf)*1000); handled=1; *data=nl(*data); } if(filter(*data,"a=StreamName:string;",&buf)) { desc->stream_name=strdup(buf); desc->stream_name_size=strlen(desc->stream_name); handled=1; *data=nl(*data); } if(filter(*data,"a=mimetype:string;",&buf)) { desc->mime_type=strdup(buf); desc->mime_type_size=strlen(desc->mime_type); handled=1; *data=nl(*data); } if(filter(*data,"a=OpaqueData:buffer;",&buf)) { uint8_t decoded[32]; desc->mlti_data_size = av_base64_decode(decoded, buf, 32); if ( desc->mlti_data_size > 0 ) { desc->mlti_data = malloc(sizeof(char)*desc->mlti_data_size); memcpy(desc->mlti_data, decoded, desc->mlti_data_size); handled=1; *data=nl(*data); lprintf("mlti_data_size: %i\n", desc->mlti_data_size); } } if(filter(*data,"a=ASMRuleBook:string;",&buf)) { desc->asm_rule_book=strdup(buf); handled=1; *data=nl(*data); } if(!handled) { #ifdef LOG int len=strchr(*data,'\n')-(*data); xine_buffer_copyin(buf, 0, *data, len+1); buf[len]=0; printf("libreal: sdpplin: not handled: '%s'\n", buf); #endif *data=nl(*data); } } xine_buffer_free(buf); return desc; } sdpplin_t *sdpplin_parse(char *data) { sdpplin_t *desc = xine_xmalloc(sizeof(sdpplin_t)); sdpplin_stream_t *stream; char *buf=xine_buffer_init(32); int handled; int len; uint8_t decoded[32]; desc->stream = NULL; while (data && *data) { handled=0; if (filter(data, "m=", &buf)) { if ( ! desc->stream ) { fprintf(stderr, "sdpplin.c: stream identifier found before stream count, skipping."); continue; } stream=sdpplin_parse_stream(&data); lprintf("got data for stream id %u\n", stream->stream_id); desc->stream[stream->stream_id]=stream; continue; } if(filter(data,"a=Title:buffer;",&buf)) { len = av_base64_decode(decoded, buf, 32); if ( len > 0 ) { desc->title=strdup(decoded); handled=1; data=nl(data); } } if(filter(data,"a=Author:buffer;",&buf)) { len = av_base64_decode(decoded, buf, 32); if ( len > 0 ) { desc->author=strdup(decoded); handled=1; data=nl(data); } } if(filter(data,"a=Copyright:buffer;",&buf)) { len = av_base64_decode(decoded, buf, 32); if ( len > 0 ) { desc->copyright=strdup(decoded); handled=1; data=nl(data); } } if(filter(data,"a=Abstract:buffer;",&buf)) { len = av_base64_decode(decoded, buf, 32); if ( len > 0 ) { desc->abstract=strdup(decoded); handled=1; data=nl(data); } } if(filter(data,"a=StreamCount:integer;",&buf)) { desc->stream_count=atoi(buf); desc->stream = calloc(desc->stream_count, sizeof(sdpplin_stream_t*)); handled=1; data=nl(data); } if(filter(data,"a=Flags:integer;",&buf)) { desc->flags=atoi(buf); handled=1; data=nl(data); } if(!handled) { #ifdef LOG int len=strchr(data,'\n')-data; xine_buffer_copyin(buf, 0, data, len+1); buf[len]=0; printf("libreal: sdpplin: not handled: '%s'\n", buf); #endif data=nl(data); } } xine_buffer_free(buf); return desc; } void sdpplin_free(sdpplin_t *description) { /* TODO: free strings */ free(description); }