mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-28 10:08:17 +00:00
9ee60482b2
Support reverse playback for ASF format. Parse packets and queue the payloads, then push the payload buffer to decoder in reverse order. Video buffers are pushed from KeyFrame to next Keyframe. https://bugzilla.gnome.org/show_bug.cgi?id=757341
256 lines
8.3 KiB
C
256 lines
8.3 KiB
C
/* GStreamer
|
|
* Copyright (C) <1999> Erik Walthinsen <omega@cse.ogi.edu>
|
|
*
|
|
* 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., 51 Franklin St, Fifth Floor,
|
|
* Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
|
|
#ifndef __ASF_DEMUX_H__
|
|
#define __ASF_DEMUX_H__
|
|
|
|
#include <gst/gst.h>
|
|
#include <gst/base/gstadapter.h>
|
|
#include <gst/base/gstflowcombiner.h>
|
|
|
|
#include "asfheaders.h"
|
|
|
|
G_BEGIN_DECLS
|
|
|
|
#define GST_TYPE_ASF_DEMUX \
|
|
(gst_asf_demux_get_type())
|
|
#define GST_ASF_DEMUX(obj) \
|
|
(G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_ASF_DEMUX,GstASFDemux))
|
|
#define GST_ASF_DEMUX_CLASS(klass) \
|
|
(G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_ASF_DEMUX,GstASFDemuxClass))
|
|
#define GST_IS_ASF_DEMUX(obj) \
|
|
(G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_ASF_DEMUX))
|
|
#define GST_IS_ASF_DEMUX_CLASS(klass) \
|
|
(G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_ASF_DEMUX))
|
|
|
|
GST_DEBUG_CATEGORY_EXTERN (asfdemux_dbg);
|
|
#define GST_CAT_DEFAULT asfdemux_dbg
|
|
|
|
typedef struct _GstASFDemux GstASFDemux;
|
|
typedef struct _GstASFDemuxClass GstASFDemuxClass;
|
|
typedef enum _GstASF3DMode GstASF3DMode;
|
|
|
|
typedef struct {
|
|
guint32 packet;
|
|
guint16 count;
|
|
} AsfSimpleIndexEntry;
|
|
|
|
typedef struct {
|
|
AsfPayloadExtensionID id : 16; /* extension ID; the :16 makes sure the
|
|
* struct gets packed into 4 bytes */
|
|
guint16 len; /* save this so we can skip unknown IDs */
|
|
} AsfPayloadExtension;
|
|
|
|
/**
|
|
* 3D Types for Media play
|
|
*/
|
|
enum _GstASF3DMode
|
|
{
|
|
GST_ASF_3D_NONE = 0x00,
|
|
|
|
//added, interim format - half
|
|
GST_ASF_3D_SIDE_BY_SIDE_HALF_LR = 0x01,
|
|
GST_ASF_3D_SIDE_BY_SIDE_HALF_RL = 0x02,
|
|
GST_ASF_3D_TOP_AND_BOTTOM_HALF_LR = 0x03,
|
|
GST_ASF_3D_TOP_AND_BOTTOM_HALF_RL = 0x04,
|
|
GST_ASF_3D_DUAL_STREAM = 0x0D, /**< Full format*/
|
|
};
|
|
|
|
typedef struct
|
|
{
|
|
gboolean valid; /* TRUE if structure is valid/filled */
|
|
|
|
GstClockTime start_time;
|
|
GstClockTime end_time;
|
|
GstClockTime avg_time_per_frame;
|
|
guint32 data_bitrate;
|
|
guint32 buffer_size;
|
|
guint32 intial_buf_fullness;
|
|
guint32 data_bitrate2;
|
|
guint32 buffer_size2;
|
|
guint32 intial_buf_fullness2;
|
|
guint32 max_obj_size;
|
|
guint32 flags;
|
|
guint16 lang_idx;
|
|
|
|
/* may be NULL if there are no extensions; otherwise, terminated by
|
|
* an AsfPayloadExtension record with len 0 */
|
|
AsfPayloadExtension *payload_extensions;
|
|
|
|
/* missing: stream names */
|
|
} AsfStreamExtProps;
|
|
|
|
typedef struct
|
|
{
|
|
AsfStreamType type;
|
|
|
|
gboolean active; /* if the stream has been activated (pad added) */
|
|
|
|
GstPad *pad;
|
|
guint16 id;
|
|
|
|
/* video-only */
|
|
gboolean is_video;
|
|
gboolean fps_known;
|
|
|
|
GstCaps *caps;
|
|
|
|
GstBuffer *streamheader;
|
|
|
|
GstTagList *pending_tags;
|
|
|
|
gboolean discont;
|
|
gboolean first_buffer;
|
|
|
|
/* Descrambler settings */
|
|
guint8 span;
|
|
guint16 ds_packet_size;
|
|
guint16 ds_chunk_size;
|
|
guint16 ds_data_size;
|
|
|
|
/* for new parsing code */
|
|
GArray *payloads; /* pending payloads */
|
|
|
|
/* Video stream PAR & interlacing */
|
|
guint8 par_x;
|
|
guint8 par_y;
|
|
gboolean interlaced;
|
|
|
|
/* For reverse playback */
|
|
gboolean reverse_kf_ready; /* Found complete KF payload*/
|
|
GArray *payloads_rev; /* Temp queue for storing multiple payloads of packet*/
|
|
gint kf_pos; /* KF position in payload queue. Payloads from this pos will be pushed */
|
|
|
|
/* extended stream properties (optional) */
|
|
AsfStreamExtProps ext_props;
|
|
|
|
gboolean inspect_payload;
|
|
} AsfStream;
|
|
|
|
typedef enum {
|
|
GST_ASF_DEMUX_STATE_HEADER,
|
|
GST_ASF_DEMUX_STATE_DATA,
|
|
GST_ASF_DEMUX_STATE_INDEX
|
|
} GstASFDemuxState;
|
|
|
|
#define GST_ASF_DEMUX_IS_REVERSE_PLAYBACK(seg) (seg.rate < 0.0? TRUE:FALSE)
|
|
|
|
#define GST_ASF_DEMUX_NUM_VIDEO_PADS 16
|
|
#define GST_ASF_DEMUX_NUM_AUDIO_PADS 32
|
|
#define GST_ASF_DEMUX_NUM_STREAMS 32
|
|
#define GST_ASF_DEMUX_NUM_STREAM_IDS 127
|
|
|
|
struct _GstASFDemux {
|
|
GstElement element;
|
|
|
|
GstPad *sinkpad;
|
|
|
|
gboolean have_group_id;
|
|
guint group_id;
|
|
|
|
GstAdapter *adapter;
|
|
GstTagList *taglist;
|
|
GstASFDemuxState state;
|
|
|
|
/* byte offset where the asf starts, which might not be zero on chained
|
|
* asfs, index_offset and data_offset already are 'offseted' by base_offset */
|
|
guint64 base_offset;
|
|
|
|
guint64 index_offset; /* byte offset where index might be, or 0 */
|
|
guint64 data_offset; /* byte offset where packets start */
|
|
guint64 data_size; /* total size of packet data in bytes, or 0 */
|
|
guint64 num_packets; /* total number of data packets, or 0 */
|
|
gint64 packet; /* current packet */
|
|
guint speed_packets; /* Known number of packets to get in one go*/
|
|
|
|
gchar **languages;
|
|
guint num_languages;
|
|
|
|
GstCaps *metadata; /* metadata, for delayed parsing; one
|
|
* structure ('stream-N') per stream */
|
|
GstStructure *global_metadata; /* metadata which isn't specific to one stream */
|
|
GSList *ext_stream_props; /* for delayed processing (buffers) */
|
|
GSList *mut_ex_streams; /* mutually exclusive streams */
|
|
|
|
guint32 num_audio_streams;
|
|
guint32 num_video_streams;
|
|
guint32 num_streams;
|
|
AsfStream stream[GST_ASF_DEMUX_NUM_STREAMS];
|
|
gboolean activated_streams;
|
|
GstFlowCombiner *flowcombiner;
|
|
|
|
/* for chained asf handling, we need to hold the old asf streams until
|
|
* we detect the new ones */
|
|
AsfStream old_stream[GST_ASF_DEMUX_NUM_STREAMS];
|
|
gboolean old_num_streams;
|
|
|
|
GstClockTime first_ts; /* smallest timestamp found */
|
|
|
|
guint32 packet_size;
|
|
guint64 play_time;
|
|
|
|
guint64 preroll;
|
|
|
|
gboolean seekable;
|
|
gboolean broadcast;
|
|
|
|
GstSegment segment; /* configured play segment */
|
|
gboolean keyunit_sync;
|
|
|
|
gboolean need_newsegment; /* do we need to send a new-segment event? */
|
|
guint32 segment_seqnum; /* if the new segment must have this seqnum */
|
|
GstClockTime segment_ts; /* streaming; timestamp for segment start */
|
|
GstSegment in_segment; /* streaming; upstream segment info */
|
|
GstClockTime in_gap; /* streaming; upstream initial segment gap for interpolation */
|
|
gboolean segment_running; /* if we've started the current segment */
|
|
gboolean streaming; /* TRUE if we are operating chain-based */
|
|
GstClockTime latency;
|
|
|
|
/* for debugging only */
|
|
gchar *objpath;
|
|
|
|
/* simple index, if available */
|
|
GstClockTime sidx_interval; /* interval between entries in ns */
|
|
guint sidx_num_entries; /* number of index entries */
|
|
AsfSimpleIndexEntry *sidx_entries; /* packet number for each entry */
|
|
|
|
GSList *other_streams; /* remember streams that are in header but have unknown type */
|
|
|
|
/* For reverse playback */
|
|
gboolean seek_to_cur_pos; /* Search packets till we reach 'seek' time */
|
|
gboolean multiple_payloads; /* Whether packet has multiple payloads */
|
|
|
|
/* parsing 3D */
|
|
GstASF3DMode asf_3D_mode;
|
|
};
|
|
|
|
struct _GstASFDemuxClass {
|
|
GstElementClass parent_class;
|
|
};
|
|
|
|
GType gst_asf_demux_get_type (void);
|
|
|
|
AsfStream * gst_asf_demux_get_stream (GstASFDemux * demux, guint16 id);
|
|
|
|
gboolean gst_asf_demux_is_unknown_stream(GstASFDemux *demux, guint stream_num);
|
|
|
|
G_END_DECLS
|
|
|
|
#endif /* __ASF_DEMUX_H__ */
|