/* GStreamer * Copyright (C) <1999> Erik Walthinsen * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library 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 * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library; if not, write to the * Free Software Foundation, Inc., 59 Temple Place - Suite 330, * Boston, MA 02111-1307, USA. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "gstasfdemux.h" #include "asfheaders.h" /* elementfactory information */ static GstElementDetails gst_asf_demux_details = { "ASF Demuxer", "Codec/Demuxer", "LGPL", "Demultiplexes ASF Streams", VERSION, "Owen Fraser-Green ", "(C) 2002", }; static GstCaps* asf_asf_type_find (GstBuffer *buf, gpointer private); static GstCaps* asf_wma_type_find (GstBuffer *buf, gpointer private); static GstCaps* asf_wax_type_find (GstBuffer *buf, gpointer private); static GstCaps* asf_wmv_type_find (GstBuffer *buf, gpointer private); static GstCaps* asf_wvx_type_find (GstBuffer *buf, gpointer private); static GstCaps* asf_wm_type_find (GstBuffer *buf, gpointer private); /* typefactory for 'asf' */ static GstTypeDefinition asf_type_definitions[] = { { "asfdemux_video/asf", "video/x-ms-asf", ".asf .asx", asf_asf_type_find }, { "asfdemux_video/wma", "video/x-ms-wma", ".wma", asf_wma_type_find }, { "asfdemux_video/wax", "video/x-ms-wax", ".wax", asf_wax_type_find }, { "asfdemux_video/wmv", "video/x-ms-wmv", ".wmv", asf_wmv_type_find }, { "asfdemux_video/wvx", "video/x-ms-wvx", ".wvx", asf_wvx_type_find }, { "asfdemux_video/wm", "video/x-ms-wm", ".wm", asf_wm_type_find }, { NULL, NULL, NULL, NULL } }; static GstCaps* asf_asf_type_find (GstBuffer *buf, gpointer private) { GstCaps *new; new = gst_caps_new ( "asf_type_find", "video/x-ms-asf", gst_props_new ("asfversion", GST_PROPS_INT (1), NULL)); return new; } static GstCaps* asf_wma_type_find (GstBuffer *buf, gpointer private) { GstCaps *new; new = gst_caps_new ( "asf_type_find", "video/x-ms-asf", gst_props_new ("asfversion", GST_PROPS_INT (1), NULL)); return new; } static GstCaps* asf_wax_type_find (GstBuffer *buf, gpointer private) { GstCaps *new; new = gst_caps_new ( "asf_type_find", "video/x-ms-asf", gst_props_new ("asfversion", GST_PROPS_INT (1), NULL)); return new; } static GstCaps* asf_wmv_type_find (GstBuffer *buf, gpointer private) { GstCaps *new; new = gst_caps_new ( "asf_type_find", "video/x-ms-asf", gst_props_new ("asfversion", GST_PROPS_INT (1), NULL)); return new; } static GstCaps* asf_wvx_type_find (GstBuffer *buf, gpointer private) { GstCaps *new; new = gst_caps_new ( "asf_type_find", "video/x-ms-asf", gst_props_new ("asfversion", GST_PROPS_INT (1), NULL)); return new; } static GstCaps* asf_wm_type_find (GstBuffer *buf, gpointer private) { GstCaps *new; new = gst_caps_new ( "asf_type_find", "video/x-ms-asf", gst_props_new ("asfversion", GST_PROPS_INT (1), NULL)); return new; } GST_PAD_TEMPLATE_FACTORY (sink_factory, "sink", GST_PAD_SINK, GST_PAD_ALWAYS, GST_CAPS_NEW ("asf_asf_demux_sink", "video/x-asf", NULL) ); static void gst_asf_demux_class_init (GstASFDemuxClass *klass); static void gst_asf_demux_init (GstASFDemux *asf_demux); static gboolean gst_asf_demux_send_event (GstElement *element, GstEvent *event); static void gst_asf_demux_loop (GstElement *element); static gboolean gst_asf_demux_process_object (GstASFDemux *asf_demux, guint64 *filepos); static void gst_asf_demux_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec); static guint32 gst_asf_demux_identify_guid (GstASFDemux *asf_demux, ASFGuidHash *guids, ASFGuid *guid_raw); static gboolean gst_asf_demux_process_chunk (GstASFDemux *asf_demux, asf_packet_info *packet_info, asf_segment_info *segment_info); static const GstEventMask* gst_asf_demux_get_src_event_mask (GstPad *pad); static gboolean gst_asf_demux_handle_sink_event (GstASFDemux *asf_demux); static gboolean gst_asf_demux_handle_src_event (GstPad *pad, GstEvent *event); static const GstFormat* gst_asf_demux_get_src_formats (GstPad *pad); static const GstQueryType* gst_asf_demux_get_src_query_types (GstPad *pad); static gboolean gst_asf_demux_handle_src_query (GstPad *pad, GstQueryType type, GstFormat *format, gint64 *value); static gboolean gst_asf_demux_add_video_stream (GstASFDemux *asf_demux, asf_stream_video_format *video_format, guint16 id); static gboolean gst_asf_demux_add_audio_stream (GstASFDemux *asf_demux, asf_stream_audio *audio, guint16 id); static gboolean gst_asf_demux_setup_pad (GstASFDemux *asf_demux, GstPad *src_pad, GstCaps *caps_list, guint16 id); static GstElementStateReturn gst_asf_demux_change_state (GstElement *element); static GstPadTemplate *videosrctempl, *audiosrctempl; static GstElementClass *parent_class = NULL; GType asf_demux_get_type (void) { static GType asf_demux_type = 0; if (!asf_demux_type) { static const GTypeInfo asf_demux_info = { sizeof(GstASFDemuxClass), NULL, NULL, (GClassInitFunc)gst_asf_demux_class_init, NULL, NULL, sizeof(GstASFDemux), 0, (GInstanceInitFunc)gst_asf_demux_init, }; asf_demux_type = g_type_register_static(GST_TYPE_ELEMENT, "GstASFDemux", &asf_demux_info, 0); } return asf_demux_type; } static void gst_asf_demux_class_init (GstASFDemuxClass *klass) { GObjectClass *gobject_class; GstElementClass *gstelement_class; gobject_class = (GObjectClass*)klass; gstelement_class = (GstElementClass*)klass; parent_class = g_type_class_ref (GST_TYPE_ELEMENT); gobject_class->get_property = gst_asf_demux_get_property; gstelement_class->change_state = gst_asf_demux_change_state; gstelement_class->send_event = gst_asf_demux_send_event; } static void gst_asf_demux_init (GstASFDemux *asf_demux) { guint i; asf_demux->sinkpad = gst_pad_new_from_template( GST_PAD_TEMPLATE_GET (sink_factory), "sink"); gst_element_add_pad (GST_ELEMENT (asf_demux), asf_demux->sinkpad); gst_element_set_loop_function (GST_ELEMENT (asf_demux), gst_asf_demux_loop); /* We should zero everything to be on the safe side */ for (i = 0; i < GST_ASF_DEMUX_NUM_VIDEO_PADS; i++) { asf_demux->video_pad[i] = NULL; asf_demux->video_PTS[i] = 0; } for (i = 0; i < GST_ASF_DEMUX_NUM_AUDIO_PADS; i++) { asf_demux->audio_pad[i] = NULL; asf_demux->audio_PTS[i] = 0; } asf_demux->num_audio_streams = 0; asf_demux->num_video_streams = 0; asf_demux->num_streams = 0; GST_FLAG_SET (asf_demux, GST_ELEMENT_EVENT_AWARE); } static gboolean gst_asf_demux_send_event (GstElement *element, GstEvent *event) { const GList *pads; pads = gst_element_get_pad_list (element); while (pads) { GstPad *pad = GST_PAD (pads->data); if (GST_PAD_DIRECTION (pad) == GST_PAD_SRC) { /* we ref the event here as we might have to try again if the event * failed on this pad */ gst_event_ref (event); if (gst_asf_demux_handle_src_event (pad, event)) { gst_event_unref (event); return TRUE; } } pads = g_list_next (pads); } gst_event_unref (event); return FALSE; } static void gst_asf_demux_loop (GstElement *element) { GstASFDemux *asf_demux; guint64 filepos = 0; g_return_if_fail (element != NULL); g_return_if_fail (GST_IS_ASF_DEMUX (element)); asf_demux = GST_ASF_DEMUX (element); asf_demux->restart = FALSE; /* this is basically an infinite loop */ while (gst_asf_demux_process_object (asf_demux, &filepos)) { } GST_DEBUG ("Ending loop"); if (!asf_demux->restart) { /* if we exit the loop we are EOS */ gst_pad_event_default (asf_demux->sinkpad, gst_event_new (GST_EVENT_EOS)); } } static guint32 gst_asf_demux_read_var_length (GstASFDemux *asf_demux, guint8 type, guint32 *rsize) { guint32 got_bytes; guint8 *var; guint32 ret = 0; GstByteStream *bs = asf_demux->bs; if (type == 0) { return 0; } got_bytes = gst_bytestream_peek_bytes (bs, &var, 4); while (got_bytes < 4) { guint32 remaining; GstEvent *event; gst_bytestream_get_status (bs, &remaining, &event); gst_event_unref (event); got_bytes = gst_bytestream_peek_bytes (bs, &var, 4); } switch (type) { case 1: ret = GUINT32_FROM_LE(*(guint32 *)var) & 0xff; gst_bytestream_flush (bs, 1); *rsize += 1; break; case 2: ret = GUINT32_FROM_LE(*(guint32 *)var) & 0xffff; gst_bytestream_flush (bs, 2); *rsize += 2; break; case 3: ret = GUINT32_FROM_LE(*(guint32 *)var); gst_bytestream_flush (bs, 4); *rsize += 4; break; } return ret; } static void gst_asf_demux_read_object_header_rest (GstASFDemux *asf_demux, guint8 **buf, guint32 size) { guint32 got_bytes; GstByteStream *bs = asf_demux->bs; do { got_bytes = gst_bytestream_peek_bytes (bs, buf, size); if (got_bytes == size) { gst_bytestream_flush (bs, size); return; } } while (gst_asf_demux_handle_sink_event (asf_demux)); } static gboolean gst_asf_demux_process_file (GstASFDemux *asf_demux, guint64 *filepos, guint64 *obj_size) { asf_obj_file *object; guint8 *ptr; guint64 packets; /* Get the rest of the header's header */ gst_asf_demux_read_object_header_rest (asf_demux, &ptr, 80); object = (asf_obj_file *)ptr; packets = GUINT64_FROM_LE (object->packets_count); asf_demux->packet_size = GUINT32_FROM_LE (object->max_pktsize); asf_demux->play_time = (guint32) GUINT64_FROM_LE (object->play_time) / 10; asf_demux->preroll = GUINT64_FROM_LE (object->preroll); GST_INFO ( "Object is a file with %" G_GUINT64_FORMAT " data packets", packets); return TRUE; } static gboolean gst_asf_demux_process_bitrate_props_object (GstASFDemux *asf_demux, guint64 *filepos, guint64 *obj_size) { guint32 got_bytes; GstBuffer *buf; guint16 num_streams; guint8 stream_id; guint16 i; guint8 *ptr; asf_bitrate_record *bitrate_record; got_bytes = gst_bytestream_read (asf_demux->bs, &buf, 2); num_streams = GUINT16_FROM_LE (*GST_BUFFER_DATA (buf)); GST_INFO ( "Object is a bitrate properties object with %u streams.", num_streams); for (i = 0; i < num_streams; i++) { gst_asf_demux_read_object_header_rest (asf_demux, &ptr, 6); bitrate_record = (asf_bitrate_record *)ptr; stream_id = GUINT16_FROM_LE (bitrate_record->stream_id) & 0x7f; asf_demux->bitrate[stream_id] = GUINT32_FROM_LE (bitrate_record->bitrate); } return TRUE; } static gboolean gst_asf_demux_process_comment (GstASFDemux *asf_demux, guint64 *filepos, guint64 *obj_size) { asf_obj_comment *object; guint16 title_length; guint16 author_length; guint16 copyright_length; guint16 description_length; guint16 rating_length; guint8 *ptr; GstByteStream *bs = asf_demux->bs; GST_INFO ( "Object is a comment."); /* Get the rest of the comment's header */ gst_asf_demux_read_object_header_rest (asf_demux, &ptr, 10); object = (asf_obj_comment *)ptr; title_length = GUINT16_FROM_LE (object->title_length); author_length = GUINT16_FROM_LE (object->author_length); copyright_length = GUINT16_FROM_LE (object->copyright_length); description_length = GUINT16_FROM_LE (object->description_length); rating_length = GUINT16_FROM_LE (object->rating_length); GST_DEBUG ("Comment lengths: title=%d author=%d copyright=%d description=%d rating=%d", title_length, author_length, copyright_length, description_length, rating_length); /* We don't do anything with them at the moment so just skip them */ gst_bytestream_flush (bs, title_length); gst_bytestream_flush (bs, author_length); gst_bytestream_flush (bs, copyright_length); gst_bytestream_flush (bs, description_length); gst_bytestream_flush (bs, rating_length); return TRUE; } static gboolean gst_asf_demux_process_header (GstASFDemux *asf_demux, guint64 *filepos, guint64 *obj_size) { guint32 num_objects; asf_obj_header *object; guint32 i; guint8 *ptr; /* Get the rest of the header's header */ gst_asf_demux_read_object_header_rest (asf_demux, &ptr, 6); object = (asf_obj_header *)ptr; num_objects = GUINT32_FROM_LE (object->num_objects); GST_INFO ( "Object is a header with %u parts", num_objects); /* Loop through the header's objects, processing those */ for (i = 0; i < num_objects; i++) { if (!gst_asf_demux_process_object (asf_demux, filepos)) { return FALSE; } } return TRUE; } static gboolean gst_asf_demux_process_segment (GstASFDemux *asf_demux, asf_packet_info *packet_info) { guint8 *byte; gboolean key_frame; guint32 replic_size; guint8 time_delta; guint32 time_start; guint32 frag_size; guint32 rsize; asf_segment_info segment_info; gst_asf_demux_read_object_header_rest (asf_demux, (guint8**)&byte, 1); rsize = 1; segment_info.stream_number = *byte & 0x7f; key_frame = (*byte & 0x80) >> 7; GST_INFO ( "Processing segment for stream %u", segment_info.stream_number); segment_info.sequence = gst_asf_demux_read_var_length (asf_demux, packet_info->seqtype, &rsize); segment_info.frag_offset = gst_asf_demux_read_var_length (asf_demux, packet_info->fragoffsettype, &rsize); replic_size = gst_asf_demux_read_var_length (asf_demux, packet_info->replicsizetype, &rsize); GST_DEBUG ("sequence = %x, frag_offset = %x, replic_size = %x", segment_info.sequence, segment_info.frag_offset, replic_size); if (replic_size > 1) { asf_replicated_data *replicated_data_header; guint8 **replicated_data = NULL; guint8 *ptr; segment_info.compressed = FALSE; /* It's uncompressed with replic data*/ if (replic_size < 8) { gst_element_error (GST_ELEMENT (asf_demux), "The payload has replicated data but the size is less than 8"); return FALSE; } gst_asf_demux_read_object_header_rest (asf_demux, &ptr, 8); replicated_data_header = (asf_replicated_data *)ptr; segment_info.frag_timestamp = GUINT32_FROM_LE (replicated_data_header->frag_timestamp); segment_info.segment_size = GUINT32_FROM_LE (replicated_data_header->object_size); if (replic_size > 8) { gst_asf_demux_read_object_header_rest (asf_demux, replicated_data, replic_size - 8); } rsize += replic_size; } else { if (replic_size == 1) { /* It's compressed */ segment_info.compressed = TRUE; gst_asf_demux_read_object_header_rest (asf_demux, (guint8**)&byte, 1); rsize++; time_delta = *byte; GST_DEBUG ("time_delta %u", time_delta); } else { segment_info.compressed = FALSE; } time_start = segment_info.frag_offset; segment_info.frag_offset = 0; segment_info.frag_timestamp = asf_demux->timestamp; } GST_DEBUG ("multiple = %u compressed = %u", packet_info->multiple, segment_info.compressed); if (packet_info->multiple) { frag_size = gst_asf_demux_read_var_length (asf_demux, packet_info->segsizetype, &rsize); } else { frag_size = packet_info->size_left - rsize; } packet_info->size_left -= rsize; GST_DEBUG ("size left = %u frag size = %u rsize = %u", packet_info->size_left, frag_size, rsize); if (segment_info.compressed) { while (frag_size > 0) { gst_asf_demux_read_object_header_rest (asf_demux, (guint8**)&byte, 1); packet_info->size_left--; segment_info.chunk_size = *byte; segment_info.segment_size = segment_info.chunk_size; if (segment_info.chunk_size > packet_info->size_left) { gst_element_error (GST_ELEMENT (asf_demux), "Payload chunk overruns packet size."); return FALSE; } gst_asf_demux_process_chunk (asf_demux, packet_info, &segment_info); frag_size -= segment_info.chunk_size + 1; } } else { segment_info.chunk_size = frag_size; gst_asf_demux_process_chunk (asf_demux, packet_info, &segment_info); } return TRUE; } static gboolean gst_asf_demux_process_data (GstASFDemux *asf_demux, guint64 *filepos, guint64 *obj_size) { asf_obj_data *object; asf_obj_data_packet *packet_properties_object; gboolean correction; guint64 packets; guint64 packet; guint8 *buf; guint32 sequence; guint32 packet_length; guint32 *timestamp; guint16 *duration; guint8 segment; guint8 segments; guint8 flags; guint8 property; asf_packet_info packet_info; guint32 rsize; guint8 *ptr; /* Get the rest of the header */ gst_asf_demux_read_object_header_rest (asf_demux, &ptr, 26); object = (asf_obj_data *)ptr; packets = GUINT64_FROM_LE (object->packets); GST_INFO ( "Object is data with %" G_GUINT64_FORMAT " packets", packets); for (packet = 0; packet < packets; packet++) { GST_INFO ( "\n\nProcess packet %" G_GUINT64_FORMAT, packet); gst_asf_demux_read_object_header_rest (asf_demux, (guint8**)&buf, 1); rsize=1; if (*buf & 0x80) { asf_obj_data_correction *correction_object; guint8 *ptr; /* Uses error correction */ correction = TRUE; GST_DEBUG ("Data has error correction (%x)", *buf); gst_asf_demux_read_object_header_rest (asf_demux, &ptr, 2); rsize += 2; correction_object = (asf_obj_data_correction *)ptr; } /* Read the packet flags */ gst_asf_demux_read_object_header_rest (asf_demux, &ptr, 2); rsize += 2; packet_properties_object = (asf_obj_data_packet *)ptr; flags = packet_properties_object->flags; property = packet_properties_object->property; packet_info.multiple = flags & 0x01; sequence = gst_asf_demux_read_var_length (asf_demux, (flags >> 1) & 0x03, &rsize); packet_info.padsize = gst_asf_demux_read_var_length (asf_demux, (flags >> 3) & 0x03, &rsize); packet_length = gst_asf_demux_read_var_length (asf_demux, (flags >> 5) & 0x03, &rsize); if (packet_length == 0) packet_length = asf_demux->packet_size; GST_DEBUG ("Multiple = %u, Sequence = %u, Padsize = %u, Packet length = %u", packet_info.multiple, sequence, packet_info.padsize, packet_length); /* Read the property flags */ packet_info.replicsizetype = property & 0x03; packet_info.fragoffsettype = (property >> 2) & 0x03; packet_info.seqtype = (property >> 4) & 0x03; gst_asf_demux_read_object_header_rest (asf_demux, &ptr, 4); timestamp = (guint32 *)ptr; asf_demux->timestamp = *timestamp; gst_asf_demux_read_object_header_rest (asf_demux, &ptr, 2); duration = (guint16 *)ptr; rsize += 6; GST_DEBUG ("Timestamp = %x, Duration = %x", asf_demux->timestamp, *duration); if (packet_info.multiple) { /* There are multiple payloads */ gst_asf_demux_read_object_header_rest (asf_demux, (guint8**)&buf, 1); rsize++; packet_info.segsizetype = (*buf >> 6) & 0x03; segments = *buf & 0x3f; } else { packet_info.segsizetype = 2; segments = 1; } packet_info.size_left = packet_length - packet_info.padsize - rsize; GST_DEBUG ("rsize: %u size left: %u", rsize, packet_info.size_left); for (segment = 0; segment < segments; segment++) { if (!gst_asf_demux_process_segment (asf_demux, &packet_info)) return FALSE; } /* Skip the padding */ if (packet_info.padsize > 0) gst_bytestream_flush (asf_demux->bs, packet_info.padsize); GST_DEBUG ("Remaining size left: %u", packet_info.size_left); if (packet_info.size_left > 0) gst_bytestream_flush (asf_demux->bs, packet_info.size_left); } return TRUE; } static gboolean gst_asf_demux_process_stream (GstASFDemux *asf_demux, guint64 *filepos, guint64 *obj_size) { asf_obj_stream *object; guint32 stream_id; guint32 correction; asf_stream_audio *audio_object; asf_stream_correction *correction_object; asf_stream_video *video_object; asf_stream_video_format *video_format_object; guint16 size; GstBuffer *buf; guint32 got_bytes; guint16 id; guint8 *ptr; /* Get the rest of the header's header */ gst_asf_demux_read_object_header_rest (asf_demux, &ptr, 54); object = (asf_obj_stream *)ptr; /* Identify the stream type */ stream_id = gst_asf_demux_identify_guid (asf_demux, asf_stream_guids, &(object->type)); correction = gst_asf_demux_identify_guid (asf_demux, asf_correction_guids, &(object->correction)); id = GUINT16_FROM_LE (object->id); switch (stream_id) { case ASF_STREAM_AUDIO: gst_asf_demux_read_object_header_rest (asf_demux, &ptr, 18); audio_object = (asf_stream_audio *)ptr; size = GUINT16_FROM_LE (audio_object->size); GST_INFO ( "Object is an audio stream with %u bytes of additional data.", size); if (!gst_asf_demux_add_audio_stream (asf_demux, audio_object, id)) return FALSE; switch (correction) { case ASF_CORRECTION_ON: GST_INFO ( "Using error correction"); /* Have to read the first byte seperately to avoid endian problems */ got_bytes = gst_bytestream_read (asf_demux->bs, &buf, 1); asf_demux->span = *GST_BUFFER_DATA (buf); gst_asf_demux_read_object_header_rest (asf_demux, &ptr, 7); correction_object = (asf_stream_correction *)ptr; GST_DEBUG ("Descrambling: ps:%d cs:%d ds:%d s:%d sd:%d", GUINT16_FROM_LE(correction_object->packet_size), GUINT16_FROM_LE(correction_object->chunk_size), GUINT16_FROM_LE(correction_object->data_size), asf_demux->span, correction_object->silence_data); if (asf_demux->span > 1) { if (!correction_object->chunk_size || ((correction_object->packet_size / correction_object->chunk_size) <= 1)) /* Disable descrambling */ asf_demux->span = 0; } else { /* Descambling is enabled */ asf_demux->ds_packet_size = correction_object->packet_size; asf_demux->ds_chunk_size = correction_object->chunk_size; } /* Now skip the rest of the silence data */ if (correction_object->data_size > 1) gst_bytestream_flush (asf_demux->bs, correction_object->data_size - 1); break; case ASF_CORRECTION_OFF: break; default: gst_element_error (GST_ELEMENT (asf_demux), "Audio stream using unknown error correction"); return FALSE; } break; case ASF_STREAM_VIDEO: gst_asf_demux_read_object_header_rest (asf_demux, &ptr, 11); video_object = (asf_stream_video *)ptr; size = GUINT16_FROM_LE(video_object->size) - 40; /* Byte order gets * offset by single * byte */ GST_INFO ( "Object is a video stream with %u bytes of additional data.", size); gst_asf_demux_read_object_header_rest (asf_demux, &ptr, 40); video_format_object = (asf_stream_video_format *)ptr; if (!gst_asf_demux_add_video_stream (asf_demux, video_format_object, id)) return FALSE; /* Read any additional information */ if (size) { got_bytes = gst_bytestream_read (asf_demux->bs, &buf, size); /* There is additional data */ while (got_bytes < size) { guint32 remaining; GstEvent *event; gst_bytestream_get_status (asf_demux->bs, &remaining, &event); gst_event_unref (event); got_bytes = gst_bytestream_read (asf_demux->bs, &buf, size); } } break; default: gst_element_error (GST_ELEMENT (asf_demux), "Object is a stream of unrecognised type"); return FALSE; } return TRUE; } static gboolean gst_asf_demux_skip_object (GstASFDemux *asf_demux, guint64 *filepos, guint64 *obj_size) { GstByteStream *bs = asf_demux->bs; GST_INFO ( "Skipping object..."); gst_bytestream_flush (bs, *obj_size - 24); return TRUE; } static inline gboolean gst_asf_demux_read_object_header (GstASFDemux *asf_demux, guint32 *obj_id, guint64 *obj_size) { guint32 got_bytes; ASFGuid *guid; guint64 *size; GstByteStream *bs = asf_demux->bs; guint8 *ptr; /* First get the GUID */ got_bytes = gst_bytestream_peek_bytes (bs, &ptr, sizeof(ASFGuid)); guid = (ASFGuid *)ptr; if (got_bytes < sizeof (ASFGuid)) { guint32 remaining; GstEvent *event; gst_bytestream_get_status (bs, &remaining, &event); gst_event_unref (event); return FALSE; } *obj_id = gst_asf_demux_identify_guid (asf_demux, asf_object_guids, guid); gst_bytestream_flush (bs, sizeof (ASFGuid)); if (*obj_id == ASF_OBJ_UNDEFINED) { GST_INFO ( "Object found with unknown GUID %08x %08x %08x %08x", guid->v1, guid->v2, guid->v3, guid->v4); gst_element_error (GST_ELEMENT (asf_demux), "Could not identify object"); return FALSE; } /* Now get the object size */ got_bytes = gst_bytestream_peek_bytes (bs, &ptr, sizeof (guint64)); size = (guint64 *)ptr; while (got_bytes < sizeof (guint64)) { guint32 remaining; GstEvent *event; gst_bytestream_get_status (bs, &remaining, &event); gst_event_unref (event); got_bytes = gst_bytestream_peek_bytes (bs, &ptr, sizeof (guint64)); size = (guint64 *)ptr; } *obj_size = GUINT64_FROM_LE (*size); gst_bytestream_flush (bs, sizeof (guint64)); return TRUE; } static gboolean gst_asf_demux_process_object (GstASFDemux *asf_demux, guint64 *filepos) { guint32 obj_id; guint64 obj_size; if (!gst_asf_demux_read_object_header (asf_demux, &obj_id, &obj_size)) { g_print (" ***** Error reading object at filepos %" G_GUINT64_FORMAT "\n", *filepos); return FALSE; } GST_INFO ( "Found object %u with size %" G_GUINT64_FORMAT, obj_id, obj_size); switch (obj_id) { case ASF_OBJ_STREAM: return gst_asf_demux_process_stream (asf_demux, filepos, &obj_size); case ASF_OBJ_DATA: gst_asf_demux_process_data (asf_demux, filepos, &obj_size); /* This is the last object */ return FALSE; case ASF_OBJ_FILE: return gst_asf_demux_process_file (asf_demux, filepos, &obj_size); case ASF_OBJ_HEADER: return gst_asf_demux_process_header (asf_demux, filepos, &obj_size); case ASF_OBJ_CONCEAL_NONE: break; case ASF_OBJ_COMMENT: return gst_asf_demux_process_comment (asf_demux, filepos, &obj_size); case ASF_OBJ_CODEC_COMMENT: return gst_asf_demux_skip_object (asf_demux, filepos, &obj_size); case ASF_OBJ_INDEX: return gst_asf_demux_skip_object (asf_demux, filepos, &obj_size); case ASF_OBJ_HEAD1: return gst_asf_demux_skip_object (asf_demux, filepos, &obj_size); case ASF_OBJ_HEAD2: break; case ASF_OBJ_PADDING: return gst_asf_demux_skip_object (asf_demux, filepos, &obj_size); case ASF_OBJ_EXT_CONTENT_DESC: return gst_asf_demux_skip_object (asf_demux, filepos, &obj_size); case ASF_OBJ_BITRATE_PROPS: return gst_asf_demux_process_bitrate_props_object (asf_demux, filepos, &obj_size); case ASF_OBJ_BITRATE_MUTEX: return gst_asf_demux_skip_object (asf_demux, filepos, &obj_size); default: gst_element_error (GST_ELEMENT (asf_demux), "Unknown ASF object"); return FALSE; } return TRUE; } static asf_stream_context * gst_asf_demux_get_stream (GstASFDemux *asf_demux, guint16 id) { guint8 i; asf_stream_context *stream; for (i = 0; i < asf_demux->num_streams; i++) { stream = &asf_demux->stream[i]; if (stream->id == id) { /* We've found the one with the matching id */ return &asf_demux->stream[i]; } } /* Base case if we haven't found one at all */ gst_element_error (GST_ELEMENT (asf_demux), "Segment found for undefined stream: (%d)", id); return NULL; } static inline void gst_asf_demux_descramble_segment (GstASFDemux *asf_demux, asf_segment_info *segment_info, asf_stream_context *stream) { GstBuffer *scrambled_buffer; GstBuffer *descrambled_buffer; GstBuffer *sub_buffer; guint offset; guint off; guint row; guint col; guint idx; /* descrambled_buffer is initialised in the first iteration */ descrambled_buffer = NULL; scrambled_buffer = stream->payload; offset = 0; for (offset = 0; offset < segment_info->segment_size; offset += asf_demux->ds_chunk_size) { off = offset / asf_demux->ds_chunk_size; row = off / asf_demux->span; col = off % asf_demux->span; idx = row + col * asf_demux->ds_packet_size / asf_demux->ds_chunk_size; sub_buffer = gst_buffer_create_sub (scrambled_buffer, idx * asf_demux->ds_chunk_size, asf_demux->ds_chunk_size); if (!offset) { descrambled_buffer = sub_buffer; } else { gst_buffer_merge (descrambled_buffer, sub_buffer); gst_buffer_unref (sub_buffer); } } stream->payload = descrambled_buffer; gst_buffer_unref (scrambled_buffer); } static gboolean gst_asf_demux_process_chunk (GstASFDemux *asf_demux, asf_packet_info *packet_info, asf_segment_info *segment_info) { asf_stream_context *stream; guint32 got_bytes; GstByteStream *bs = asf_demux->bs; GstBuffer *buffer; if (!(stream = gst_asf_demux_get_stream (asf_demux, segment_info->stream_number))) return FALSE; GST_DEBUG ("Processing chunk of size %u (fo = %d)", segment_info->chunk_size, stream->frag_offset); if (stream->frag_offset == 0) { /* new packet */ stream->sequence = segment_info->sequence; asf_demux->pts = segment_info->frag_timestamp - asf_demux->preroll; got_bytes = gst_bytestream_peek (bs, &buffer, segment_info->chunk_size); GST_DEBUG ("BUFFER: Copied stream to buffer (%p - %d)", buffer, GST_BUFFER_REFCOUNT_VALUE(buffer)); stream->payload = buffer; } else { GST_DEBUG ("segment_info->sequence = %d, stream->sequence = %d, segment_info->frag_offset = %d, stream->frag_offset = %d", segment_info->sequence, stream->sequence, segment_info->frag_offset, stream->frag_offset); if (segment_info->sequence == stream->sequence && segment_info->frag_offset == stream->frag_offset) { GstBuffer *new_buffer; /* continuing packet */ GST_INFO ( "A continuation packet"); got_bytes = gst_bytestream_peek (bs, &buffer, segment_info->chunk_size); GST_DEBUG ("Copied stream to buffer (%p - %d)", buffer, GST_BUFFER_REFCOUNT_VALUE(buffer)); new_buffer = gst_buffer_merge (stream->payload, buffer); GST_DEBUG ("BUFFER: Merged new_buffer (%p - %d) from stream->payload(%p - %d) and buffer (%p - %d)", new_buffer, GST_BUFFER_REFCOUNT_VALUE(new_buffer), stream->payload, GST_BUFFER_REFCOUNT_VALUE(stream->payload), buffer, GST_BUFFER_REFCOUNT_VALUE(buffer)); gst_buffer_unref (stream->payload); gst_buffer_unref (buffer); stream->payload = new_buffer; } else { /* cannot continue current packet: free it */ stream->frag_offset = 0; if (segment_info->frag_offset != 0) { /* cannot create new packet */ GST_DEBUG ("BUFFER: Freeing stream->payload (%p)", stream->payload); gst_buffer_free(stream->payload); gst_bytestream_flush (bs, segment_info->chunk_size); packet_info->size_left -= segment_info->chunk_size; return TRUE; } else { /* create new packet */ stream->sequence = segment_info->sequence; } } } stream->frag_offset +=segment_info->chunk_size; GST_DEBUG ("frag_offset = %d segment_size = %d ", stream->frag_offset, segment_info->segment_size); if (stream->frag_offset < segment_info->segment_size) { /* We don't have the whole packet yet */ } else { /* We have the whole packet now so we should push the packet to the src pad now. First though we should check if we need to do descrambling */ if (asf_demux->span > 1) { gst_asf_demux_descramble_segment (asf_demux, segment_info, stream); } if (GST_PAD_IS_USABLE (stream->pad)) { GST_DEBUG ("New buffer is at: %p size: %u", GST_BUFFER_DATA(stream->payload), GST_BUFFER_SIZE(stream->payload)); GST_BUFFER_TIMESTAMP (stream->payload) = (GST_SECOND / 1000) * asf_demux->pts; /*!!! Should handle flush events here? */ GST_DEBUG ("Sending strem %d of size %d", stream->id , segment_info->chunk_size); GST_INFO ( "Pushing pad"); gst_pad_push (stream->pad, stream->payload); } stream->frag_offset = 0; } gst_bytestream_flush (bs, segment_info->chunk_size); packet_info->size_left -= segment_info->chunk_size; GST_DEBUG (" "); return TRUE; } /* * Event stuff */ static gboolean gst_asf_demux_handle_sink_event (GstASFDemux *asf_demux) { guint32 remaining; GstEvent *event; GstEventType type; gst_bytestream_get_status (asf_demux->bs, &remaining, &event); type = event? GST_EVENT_TYPE (event) : GST_EVENT_UNKNOWN; switch (type) { case GST_EVENT_EOS: gst_bytestream_flush (asf_demux->bs, remaining); gst_pad_event_default (asf_demux->sinkpad, event); break; case GST_EVENT_FLUSH: g_warning ("flush event"); break; case GST_EVENT_DISCONTINUOUS: { gint i; GstEvent *discont; for (i = 0; i < asf_demux->num_streams; i++) { asf_stream_context *stream = &asf_demux->stream[i]; if (GST_PAD_IS_USABLE (stream->pad)) { GST_DEBUG ("sending discont on %d %" G_GINT64_FORMAT " + %" G_GINT64_FORMAT " = %" G_GINT64_FORMAT, i, asf_demux->last_seek, stream->delay, asf_demux->last_seek + stream->delay); discont = gst_event_new_discontinuous (FALSE, GST_FORMAT_TIME, asf_demux->last_seek + stream->delay , NULL); gst_pad_push (stream->pad, GST_BUFFER (discont)); } } break; } default: g_warning ("unhandled event %d", type); break; } gst_event_unref (event); return TRUE; } static gboolean gst_asf_demux_handle_src_event (GstPad *pad, GstEvent *event) { GST_DEBUG ("asfdemux: handle_src_event"); return FALSE; } static const GstEventMask* gst_asf_demux_get_src_event_mask (GstPad *pad) { static const GstEventMask masks[] = { { GST_EVENT_SEEK, GST_SEEK_METHOD_SET | GST_SEEK_FLAG_KEY_UNIT }, { 0, } }; return masks; } static const GstFormat* gst_asf_demux_get_src_formats (GstPad *pad) { static const GstFormat formats[] = { GST_FORMAT_TIME, 0 }; return formats; } static const GstQueryType* gst_asf_demux_get_src_query_types (GstPad *pad) { static const GstQueryType types[] = { GST_QUERY_TOTAL, GST_QUERY_POSITION, 0 }; return types; } static gboolean gst_asf_demux_handle_src_query (GstPad *pad, GstQueryType type, GstFormat *format, gint64 *value) { GstASFDemux *asf_demux; gboolean res = TRUE; asf_demux = GST_ASF_DEMUX (gst_pad_get_parent (pad)); switch (type) { case GST_QUERY_TOTAL: switch (*format) { case GST_FORMAT_DEFAULT: *format = GST_FORMAT_TIME; /* fall through */ case GST_FORMAT_TIME: *value = (GST_SECOND / 1000) * asf_demux->pts; break; default: res = FALSE; } break; case GST_QUERY_POSITION: switch (*format) { case GST_FORMAT_DEFAULT: *format = GST_FORMAT_TIME; /* fall through */ case GST_FORMAT_TIME: *value = (GST_SECOND / 1000) * asf_demux->timestamp; break; default: res = FALSE; } default: res = FALSE; break; } return res; } static void gst_asf_demux_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { GstASFDemux *src; g_return_if_fail (GST_IS_ASF_DEMUX (object)); src = GST_ASF_DEMUX (object); switch (prop_id) { default: break; } } static GstElementStateReturn gst_asf_demux_change_state (GstElement *element) { GstASFDemux *asf_demux = GST_ASF_DEMUX (element); gint i; switch (GST_STATE_TRANSITION (element)) { case GST_STATE_READY_TO_PAUSED: asf_demux->bs = gst_bytestream_new (asf_demux->sinkpad); asf_demux->last_seek = 0; break; case GST_STATE_PAUSED_TO_READY: gst_bytestream_destroy (asf_demux->bs); asf_demux->restart = TRUE; for (i = 0 ; i < GST_ASF_DEMUX_NUM_VIDEO_PADS; i++) { asf_demux->video_PTS[i] = 0; } for (i = 0 ; i < GST_ASF_DEMUX_NUM_AUDIO_PADS;i++) { asf_demux->audio_PTS[i] = 0; } break; case GST_STATE_READY_TO_NULL: break; default: break; } return GST_ELEMENT_CLASS (parent_class)->change_state (element); } static guint32 gst_asf_demux_identify_guid (GstASFDemux *asf_demux, ASFGuidHash *guids, ASFGuid *guid_raw) { guint32 i; ASFGuid guid; /* guid_raw is passed in 'as-is' from the file * so we must convert it's endianess */ guid.v1 = GUINT32_FROM_LE (guid_raw->v1); guid.v2 = GUINT32_FROM_LE (guid_raw->v2); guid.v3 = GUINT32_FROM_LE (guid_raw->v3); guid.v4 = GUINT32_FROM_LE (guid_raw->v4); i = 0; while (guids[i].obj_id != ASF_OBJ_UNDEFINED) { if (guids[i].guid.v1 == guid.v1 && guids[i].guid.v2 == guid.v2 && guids[i].guid.v3 == guid.v3 && guids[i].guid.v4 == guid.v4) { return guids[i].obj_id; } i++; } /* The base case if none is found */ return ASF_OBJ_UNDEFINED; } /* * Stream and pad setup code */ #ifdef G_HAVE_ISO_VARARGS #define GST_ASF_AUD_CAPS_NEW(name, mimetype, ...) \ (audio != NULL) ? \ GST_CAPS_NEW (name, \ mimetype, \ "rate", GST_PROPS_INT ( \ GUINT32_FROM_LE (audio->byte_rate)), \ "channels", GST_PROPS_INT ( \ GUINT16_FROM_LE (audio->channels)), \ __VA_ARGS__) \ : \ GST_CAPS_NEW (name, \ mimetype, \ "rate", GST_PROPS_INT_RANGE (8000, 96000), \ "channels", GST_PROPS_INT_RANGE (1, 2), \ __VA_ARGS__) #elif defined(G_HAVE_GNUC_VARARGS) #define GST_ASF_AUD_CAPS_NEW(name, mimetype, props...) \ (audio != NULL) ? \ GST_CAPS_NEW (name, \ mimetype, \ "rate", GST_PROPS_INT ( \ GUINT32_FROM_LE (audio->byte_rate)), \ "channels", GST_PROPS_INT ( \ GUINT16_FROM_LE (audio->channels)), \ ##props) \ : \ GST_CAPS_NEW (name, \ mimetype, \ "rate", GST_PROPS_INT_RANGE (8000, 96000), \ "channels", GST_PROPS_INT_RANGE (1, 2), \ ##props) #endif static GstCaps * gst_asf_demux_audio_caps (guint16 codec_id, asf_stream_audio *audio) { GstCaps *caps = NULL; switch (codec_id) { case GST_RIFF_WAVE_FORMAT_MPEGL3: /* mp3 */ caps = GST_ASF_AUD_CAPS_NEW ("asf_demux_audio_src_mp3", "audio/mpeg", "layer", GST_PROPS_INT (3)); break; case GST_RIFF_WAVE_FORMAT_MPEGL12: /* mp1 or mp2 */ caps = GST_ASF_AUD_CAPS_NEW ("asf_demux_audio_src_mp12", "audio/mpeg", "layer", GST_PROPS_INT (2)); break; case GST_RIFF_WAVE_FORMAT_PCM: /* PCM/wav */ { GstPropsEntry *width, *depth, *signedness; if (audio != NULL) { gint ba = GUINT16_FROM_LE (audio->block_align); gint ch = GUINT16_FROM_LE (audio->channels); gint ws = GUINT16_FROM_LE (audio->word_size); width = gst_props_entry_new ("width", GST_PROPS_INT (ba * 8 / ch)); depth = gst_props_entry_new ("depth", GST_PROPS_INT (ws)); signedness = gst_props_entry_new ("signedness", GST_PROPS_BOOLEAN (ws != 8)); } else { signedness = gst_props_entry_new ("signed", GST_PROPS_LIST ( GST_PROPS_BOOLEAN (TRUE), GST_PROPS_BOOLEAN (FALSE))); width = gst_props_entry_new ("width", GST_PROPS_LIST ( GST_PROPS_INT (8), GST_PROPS_INT (16))); depth = gst_props_entry_new ("depth", GST_PROPS_LIST ( GST_PROPS_INT (8), GST_PROPS_INT (16))); } caps = GST_ASF_AUD_CAPS_NEW ("asf_demux_audio_src_pcm", "audio/x-raw-int", "endianness", GST_PROPS_INT (G_LITTLE_ENDIAN)); gst_props_add_entry (caps->properties, width); gst_props_add_entry (caps->properties, depth); gst_props_add_entry (caps->properties, signedness); } break; case GST_RIFF_WAVE_FORMAT_VORBIS1: /* vorbis mode 1 */ case GST_RIFF_WAVE_FORMAT_VORBIS2: /* vorbis mode 2 */ case GST_RIFF_WAVE_FORMAT_VORBIS3: /* vorbis mode 3 */ case GST_RIFF_WAVE_FORMAT_VORBIS1PLUS: /* vorbis mode 1+ */ case GST_RIFF_WAVE_FORMAT_VORBIS2PLUS: /* vorbis mode 2+ */ case GST_RIFF_WAVE_FORMAT_VORBIS3PLUS: /* vorbis mode 3+ */ caps = GST_ASF_AUD_CAPS_NEW ("asf_demux_audio_src_vorbis", "audio/x-vorbis", NULL); break; case GST_RIFF_WAVE_FORMAT_A52: caps = GST_ASF_AUD_CAPS_NEW ("asf_demux_audio_src_ac3", "audio/x-ac3", NULL); break; case GST_RIFF_WAVE_FORMAT_DIVX_WMAV1: caps = GST_ASF_AUD_CAPS_NEW ("asf_demux_audio_src_wmav1", "audio/x-wma", "wmaversion", GST_PROPS_INT (1)); break; case GST_RIFF_WAVE_FORMAT_DIVX_WMAV2: caps = GST_ASF_AUD_CAPS_NEW ("asf_demux_audio_src_wmav2", "audio/x-wma", "wmaversion", GST_PROPS_INT (2)); break; case GST_RIFF_WAVE_FORMAT_WMAV9: caps = GST_ASF_AUD_CAPS_NEW ("asf_demux_audio_src_wmav9", "audio/x-wma", "wmaversion", GST_PROPS_INT (9)); break; default: g_warning ("asfdemux: unkown audio format 0x%04x", codec_id); break; } return caps; } static gboolean gst_asf_demux_add_audio_stream (GstASFDemux *asf_demux, asf_stream_audio *audio, guint16 id) { GstPad *src_pad; GstCaps *caps; gchar *name = NULL; guint16 size_left = 0; size_left = GUINT16_FROM_LE(audio->size); /* Create the audio pad */ name = g_strdup_printf ("audio_%02d", asf_demux->num_audio_streams); src_pad = gst_pad_new_from_template (audiosrctempl, name); g_free (name); /* Now set up the standard propertis from the header info */ caps = gst_asf_demux_audio_caps (GUINT16_FROM_LE(audio->codec_tag), audio); GST_INFO ("Adding audio stream %u codec %u (0x%x)", asf_demux->num_video_streams, GUINT16_FROM_LE(audio->codec_tag), GUINT16_FROM_LE(audio->codec_tag)); asf_demux->num_audio_streams++; /* Swallow up any left over data */ if (size_left) { g_warning ("asfdemux: Audio header contains %d bytes of surplus data", size_left); gst_bytestream_flush (asf_demux->bs, size_left); } return gst_asf_demux_setup_pad (asf_demux, src_pad, caps, id); } #ifdef G_HAVE_ISO_VARARGS #define GST_ASF_VID_CAPS_NEW(name, mimetype, ...) \ (video != NULL) ? \ GST_CAPS_NEW (name, \ mimetype, \ "width", GST_PROPS_INT (width), \ "height", GST_PROPS_INT (height), \ "framerate", GST_PROPS_FLOAT (0),/* FIXME */ \ __VA_ARGS__) \ : \ GST_CAPS_NEW (name, \ mimetype, \ "width", GST_PROPS_INT_RANGE (16, 4096), \ "height", GST_PROPS_INT_RANGE (16, 4096), \ "framerate", GST_PROPS_FLOAT_RANGE (0, G_MAXFLOAT), \ __VA_ARGS__) #elif defined(G_HAVE_GNUC_VARARGS) #define GST_ASF_VID_CAPS_NEW(name, mimetype, props...) \ (video != NULL) ? \ GST_CAPS_NEW (name, \ mimetype, \ "width", GST_PROPS_INT (width), \ "height", GST_PROPS_INT (height), \ "framerate", GST_PROPS_FLOAT (0),/* FIXME */ \ ##props) \ : \ GST_CAPS_NEW (name, \ mimetype, \ "width", GST_PROPS_INT_RANGE (16, 4096), \ "height", GST_PROPS_INT_RANGE (16, 4096), \ "framerate", GST_PROPS_FLOAT_RANGE (0, G_MAXFLOAT), \ ##props) #endif static GstCaps * gst_asf_demux_video_caps (guint32 codec_fcc, asf_stream_video_format *video) { GstCaps *caps = NULL; gint width = 0, height = 0; if (video != NULL) { width = GUINT32_FROM_LE (video->width); height = GUINT32_FROM_LE (video->height); } switch (codec_fcc) { case GST_MAKE_FOURCC('I','4','2','0'): case GST_MAKE_FOURCC('Y','U','Y','2'): caps = GST_ASF_VID_CAPS_NEW ("asf_demux_video_src_raw", "video/x-raw-yuv", "format", GST_PROPS_FOURCC (codec_fcc)); break; case GST_MAKE_FOURCC('M','J','P','G'): case GST_MAKE_FOURCC('J','P','E','G'): case GST_MAKE_FOURCC('P','I','X','L'): case GST_MAKE_FOURCC('V','I','X','L'): caps = GST_ASF_VID_CAPS_NEW ("asf_demux_video_src_jpeg", "video/x-jpeg", NULL); break; case GST_MAKE_FOURCC('D','V','S','D'): case GST_MAKE_FOURCC('d','v','s','d'): caps = GST_ASF_VID_CAPS_NEW ("asf_demux_video_src_dv", "video/x-dv", "systemstream", GST_PROPS_BOOLEAN (FALSE)); break; case GST_MAKE_FOURCC('W','M','V','1'): caps = GST_ASF_VID_CAPS_NEW ("asf_demux_video_src_wmv1", "video/x-wmv", "wmvversion", GST_PROPS_INT (1)); break; case GST_MAKE_FOURCC('W','M','V','2'): caps = GST_ASF_VID_CAPS_NEW ("asf_demux_video_src_wmv2", "video/x-wmv", "wmvversion", GST_PROPS_INT (2)); break; case GST_MAKE_FOURCC('M','P','G','4'): caps = GST_ASF_VID_CAPS_NEW ("asf_demux_video_src_msmpeg41", "video/x-msmpeg", "msmpegversion", GST_PROPS_INT (41)); break; case GST_MAKE_FOURCC('M','P','4','2'): caps = GST_ASF_VID_CAPS_NEW ("asf_demux_video_src_msmpeg42", "video/x-msmpeg", "msmpegversion", GST_PROPS_INT (42)); break; case GST_MAKE_FOURCC('M','P','4','3'): caps = GST_ASF_VID_CAPS_NEW ("asf_demux_video_src_msmpeg43", "video/x-msmpeg", "msmpegversion", GST_PROPS_INT (43)); break; case GST_MAKE_FOURCC('D','I','V','3'): case GST_MAKE_FOURCC('D','I','V','4'): case GST_MAKE_FOURCC('D','I','V','5'): caps = GST_ASF_VID_CAPS_NEW ("asf_demux_video_src_div345", "video/x-divx", "divxversion", GST_PROPS_INT (3)); break; default: g_warning ("asfdemux: unkown video format " GST_FOURCC_FORMAT "(0x%08x)", GST_FOURCC_ARGS(codec_fcc), codec_fcc); break; } return caps; } static gboolean gst_asf_demux_add_video_stream (GstASFDemux *asf_demux, asf_stream_video_format *video, guint16 id) { GstPad *src_pad; GstCaps *caps; gchar *name = NULL; /* Create the audio pad */ name = g_strdup_printf ("video_%02d", asf_demux->num_video_streams); src_pad = gst_pad_new_from_template (videosrctempl, name); g_free (name); /* Now try some gstreamer formatted MIME types (from gst_avi_demux_strf_vids) */ caps = gst_asf_demux_video_caps (GUINT32_FROM_LE(video->tag), video); GST_INFO ("Adding video stream %u codec " GST_FOURCC_FORMAT " (0x%08x)", asf_demux->num_video_streams, GST_FOURCC_ARGS(GUINT32_FROM_LE(video->tag)), GUINT32_FROM_LE(video->tag)); asf_demux->num_video_streams++; return gst_asf_demux_setup_pad (asf_demux, src_pad, caps, id); } static gboolean gst_asf_demux_setup_pad (GstASFDemux *asf_demux, GstPad *src_pad, GstCaps *caps_list, guint16 id) { asf_stream_context *stream; gst_pad_try_set_caps (src_pad, caps_list); gst_pad_set_formats_function (src_pad, gst_asf_demux_get_src_formats); gst_pad_set_event_mask_function (src_pad, gst_asf_demux_get_src_event_mask); gst_pad_set_event_function (src_pad, gst_asf_demux_handle_src_event); gst_pad_set_query_type_function (src_pad, gst_asf_demux_get_src_query_types); gst_pad_set_query_function (src_pad, gst_asf_demux_handle_src_query); stream = &asf_demux->stream[asf_demux->num_streams]; stream->pad = src_pad; stream->id = id; stream->frag_offset = 0; stream->sequence = 0; stream->delay = 0LL; gst_pad_set_element_private (src_pad, stream); GST_INFO ("Adding pad for stream %u", asf_demux->num_streams); asf_demux->num_streams++; gst_element_add_pad (GST_ELEMENT (asf_demux), src_pad); return TRUE; } static gboolean plugin_init (GModule *module, GstPlugin *plugin) { GstElementFactory *factory; gint i = 0; GstCaps *audcaps = NULL, *vidcaps = NULL, *temp; guint32 vid_list[] = { GST_MAKE_FOURCC('I','4','2','0'), GST_MAKE_FOURCC('Y','U','Y','2'), GST_MAKE_FOURCC('M','J','P','G'), GST_MAKE_FOURCC('D','V','S','D'), GST_MAKE_FOURCC('W','M','V','1'), GST_MAKE_FOURCC('W','M','V','2'), GST_MAKE_FOURCC('M','P','G','4'), GST_MAKE_FOURCC('M','P','4','2'), GST_MAKE_FOURCC('M','P','4','3'), 0 /* end */ }; gint aud_list[] = { GST_RIFF_WAVE_FORMAT_MPEGL3, GST_RIFF_WAVE_FORMAT_MPEGL12, GST_RIFF_WAVE_FORMAT_PCM, GST_RIFF_WAVE_FORMAT_VORBIS1, GST_RIFF_WAVE_FORMAT_A52, GST_RIFF_WAVE_FORMAT_DIVX_WMAV1, GST_RIFF_WAVE_FORMAT_DIVX_WMAV2, GST_RIFF_WAVE_FORMAT_WMAV9, -1 /* end */ }; /* this filter needs bytestream */ if (!gst_library_load ("gstbytestream")) { GST_INFO ("asfdemux: could not load support library: 'gstbytestream'\n"); return FALSE; } /* create an elementfactory for the asf_demux element */ factory = gst_element_factory_new ("asfdemux",GST_TYPE_ASF_DEMUX, &gst_asf_demux_details); while (asf_type_definitions[i].name) { GstTypeFactory *type; type = gst_type_factory_new (&asf_type_definitions[i]); //gst_plugin_add_feature (plugin, GST_PLUGIN_FEATURE (type)); i++; } g_return_val_if_fail (factory != NULL, FALSE); gst_element_factory_set_rank (factory, GST_ELEMENT_RANK_NONE); for (i = 0; aud_list[i] != -1; i++) { temp = gst_asf_demux_audio_caps (aud_list[i], NULL); audcaps = gst_caps_append (audcaps, temp); } audiosrctempl = gst_pad_template_new ("audio_%02d", GST_PAD_SRC, GST_PAD_SOMETIMES, audcaps, NULL); for (i = 0; vid_list[i] != 0; i++) { temp = gst_asf_demux_video_caps (vid_list[i], NULL); vidcaps = gst_caps_append (vidcaps, temp); } videosrctempl = gst_pad_template_new ("video_%02d", GST_PAD_SRC, GST_PAD_SOMETIMES, vidcaps, NULL); gst_element_factory_add_pad_template (factory, audiosrctempl); gst_element_factory_add_pad_template (factory, videosrctempl); gst_element_factory_add_pad_template (factory, GST_PAD_TEMPLATE_GET (sink_factory)); gst_plugin_add_feature (plugin, GST_PLUGIN_FEATURE (factory)); return TRUE; } GstPluginDesc plugin_desc = { GST_VERSION_MAJOR, GST_VERSION_MINOR, "asfdemux", plugin_init };