mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-26 00:58:12 +00:00
baseaudiodecoder: replace context helper structure by various
_get/_set
This commit is contained in:
parent
a39a66dd4b
commit
d71e427c49
2 changed files with 259 additions and 71 deletions
|
@ -36,7 +36,7 @@
|
||||||
* <listitem><para>
|
* <listitem><para>
|
||||||
* Initially, GstBaseAudioDecoder calls @start when the decoder element
|
* Initially, GstBaseAudioDecoder calls @start when the decoder element
|
||||||
* is activated, which allows subclass to perform any global setup.
|
* is activated, which allows subclass to perform any global setup.
|
||||||
* Base class context parameters can already be set according to subclass
|
* Base class (context) parameters can already be set according to subclass
|
||||||
* capabilities (or possibly upon receive more information in subsequent
|
* capabilities (or possibly upon receive more information in subsequent
|
||||||
* @set_format).
|
* @set_format).
|
||||||
* </para></listitem>
|
* </para></listitem>
|
||||||
|
@ -52,9 +52,8 @@
|
||||||
* </itemizedlist>
|
* </itemizedlist>
|
||||||
* </listitem>
|
* </listitem>
|
||||||
* As of configuration stage, and throughout processing, GstBaseAudioDecoder
|
* As of configuration stage, and throughout processing, GstBaseAudioDecoder
|
||||||
* provides a GstBaseAudioDecoderContext that provides required context,
|
* provides various (context) parameters, e.g. describing the format of
|
||||||
* e.g. describing the format of output audio data
|
* output audio data (valid when output caps have been caps) or current parsing state.
|
||||||
* (valid when output caps have been caps) or current parsing state.
|
|
||||||
* Conversely, subclass can and should configure context to inform
|
* Conversely, subclass can and should configure context to inform
|
||||||
* base class of its expectation w.r.t. buffer handling.
|
* base class of its expectation w.r.t. buffer handling.
|
||||||
* <listitem>
|
* <listitem>
|
||||||
|
@ -180,6 +179,28 @@ enum
|
||||||
#define DEFAULT_TOLERANCE 0
|
#define DEFAULT_TOLERANCE 0
|
||||||
#define DEFAULT_PLC FALSE
|
#define DEFAULT_PLC FALSE
|
||||||
|
|
||||||
|
typedef struct _GstBaseAudioDecoderContext
|
||||||
|
{
|
||||||
|
/* input */
|
||||||
|
/* (output) audio format */
|
||||||
|
GstAudioFormatInfo info;
|
||||||
|
|
||||||
|
/* parsing state */
|
||||||
|
gboolean eos;
|
||||||
|
gboolean sync;
|
||||||
|
|
||||||
|
/* misc */
|
||||||
|
gint delay;
|
||||||
|
|
||||||
|
/* output */
|
||||||
|
gboolean do_plc;
|
||||||
|
gboolean do_byte_time;
|
||||||
|
gint max_errors;
|
||||||
|
/* MT-protected (with LOCK) */
|
||||||
|
GstClockTime min_latency;
|
||||||
|
GstClockTime max_latency;
|
||||||
|
} GstBaseAudioDecoderContext;
|
||||||
|
|
||||||
struct _GstBaseAudioDecoderPrivate
|
struct _GstBaseAudioDecoderPrivate
|
||||||
{
|
{
|
||||||
/* activation status */
|
/* activation status */
|
||||||
|
@ -365,7 +386,6 @@ gst_base_audio_decoder_init (GstBaseAudioDecoder * dec,
|
||||||
dec->priv->adapter = gst_adapter_new ();
|
dec->priv->adapter = gst_adapter_new ();
|
||||||
dec->priv->adapter_out = gst_adapter_new ();
|
dec->priv->adapter_out = gst_adapter_new ();
|
||||||
g_queue_init (&dec->priv->frames);
|
g_queue_init (&dec->priv->frames);
|
||||||
dec->ctx = &dec->priv->ctx;
|
|
||||||
|
|
||||||
/* property default */
|
/* property default */
|
||||||
dec->latency = DEFAULT_LATENCY;
|
dec->latency = DEFAULT_LATENCY;
|
||||||
|
@ -391,8 +411,8 @@ gst_base_audio_decoder_reset (GstBaseAudioDecoder * dec, gboolean full)
|
||||||
dec->priv->error_count = 0;
|
dec->priv->error_count = 0;
|
||||||
gst_base_audio_decoder_clear_queues (dec);
|
gst_base_audio_decoder_clear_queues (dec);
|
||||||
|
|
||||||
g_free (dec->ctx->state.channel_pos);
|
g_free (dec->priv->ctx.info.channel_pos);
|
||||||
memset (dec->ctx, 0, sizeof (dec->ctx));
|
memset (&dec->priv->ctx, 0, sizeof (dec->priv->ctx));
|
||||||
|
|
||||||
if (dec->priv->taglist) {
|
if (dec->priv->taglist) {
|
||||||
gst_tag_list_free (dec->priv->taglist);
|
gst_tag_list_free (dec->priv->taglist);
|
||||||
|
@ -446,7 +466,7 @@ gst_base_audio_decoder_src_setcaps (GstPad * pad, GstCaps * caps)
|
||||||
gboolean res = TRUE, changed;
|
gboolean res = TRUE, changed;
|
||||||
|
|
||||||
dec = GST_BASE_AUDIO_DECODER (gst_pad_get_parent (pad));
|
dec = GST_BASE_AUDIO_DECODER (gst_pad_get_parent (pad));
|
||||||
state = &dec->ctx->state;
|
state = &dec->priv->ctx.info;
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (dec, "setting src caps %" GST_PTR_FORMAT, caps);
|
GST_DEBUG_OBJECT (dec, "setting src caps %" GST_PTR_FORMAT, caps);
|
||||||
|
|
||||||
|
@ -536,13 +556,13 @@ gst_base_audio_decoder_output (GstBaseAudioDecoder * dec, GstBuffer * buf)
|
||||||
|
|
||||||
klass = GST_BASE_AUDIO_DECODER_GET_CLASS (dec);
|
klass = GST_BASE_AUDIO_DECODER_GET_CLASS (dec);
|
||||||
priv = dec->priv;
|
priv = dec->priv;
|
||||||
ctx = dec->ctx;
|
ctx = &dec->priv->ctx;
|
||||||
|
|
||||||
if (G_UNLIKELY (priv->agg < 0))
|
if (G_UNLIKELY (priv->agg < 0))
|
||||||
gst_base_audio_decoder_setup (dec);
|
gst_base_audio_decoder_setup (dec);
|
||||||
|
|
||||||
if (G_LIKELY (buf)) {
|
if (G_LIKELY (buf)) {
|
||||||
g_return_val_if_fail (ctx->state.bpf != 0, GST_FLOW_ERROR);
|
g_return_val_if_fail (ctx->info.bpf != 0, GST_FLOW_ERROR);
|
||||||
|
|
||||||
GST_LOG_OBJECT (dec, "output buffer of size %d with ts %" GST_TIME_FORMAT
|
GST_LOG_OBJECT (dec, "output buffer of size %d with ts %" GST_TIME_FORMAT
|
||||||
", duration %" GST_TIME_FORMAT, GST_BUFFER_SIZE (buf),
|
", duration %" GST_TIME_FORMAT, GST_BUFFER_SIZE (buf),
|
||||||
|
@ -550,8 +570,8 @@ gst_base_audio_decoder_output (GstBaseAudioDecoder * dec, GstBuffer * buf)
|
||||||
GST_TIME_ARGS (GST_BUFFER_DURATION (buf)));
|
GST_TIME_ARGS (GST_BUFFER_DURATION (buf)));
|
||||||
|
|
||||||
/* clip buffer */
|
/* clip buffer */
|
||||||
buf = gst_audio_buffer_clip (buf, &dec->segment, ctx->state.rate,
|
buf = gst_audio_buffer_clip (buf, &dec->segment, ctx->info.rate,
|
||||||
ctx->state.bpf);
|
ctx->info.bpf);
|
||||||
if (G_UNLIKELY (!buf)) {
|
if (G_UNLIKELY (!buf)) {
|
||||||
GST_DEBUG_OBJECT (dec, "no data after clipping to segment");
|
GST_DEBUG_OBJECT (dec, "no data after clipping to segment");
|
||||||
} else {
|
} else {
|
||||||
|
@ -684,18 +704,18 @@ gst_base_audio_decoder_finish_frame (GstBaseAudioDecoder * dec, GstBuffer * buf,
|
||||||
g_return_val_if_fail (frames != 0, GST_FLOW_ERROR);
|
g_return_val_if_fail (frames != 0, GST_FLOW_ERROR);
|
||||||
|
|
||||||
priv = dec->priv;
|
priv = dec->priv;
|
||||||
ctx = dec->ctx;
|
ctx = &dec->priv->ctx;
|
||||||
|
|
||||||
GST_LOG_OBJECT (dec, "accepting %d bytes == %d samples for %d frames",
|
GST_LOG_OBJECT (dec, "accepting %d bytes == %d samples for %d frames",
|
||||||
buf ? GST_BUFFER_SIZE (buf) : -1,
|
buf ? GST_BUFFER_SIZE (buf) : -1,
|
||||||
buf ? GST_BUFFER_SIZE (buf) / ctx->state.bpf : -1, frames);
|
buf ? GST_BUFFER_SIZE (buf) / ctx->info.bpf : -1, frames);
|
||||||
|
|
||||||
/* output shoud be whole number of sample frames */
|
/* output shoud be whole number of sample frames */
|
||||||
if (G_LIKELY (buf && ctx->state.bpf)) {
|
if (G_LIKELY (buf && ctx->info.bpf)) {
|
||||||
if (GST_BUFFER_SIZE (buf) % ctx->state.bpf)
|
if (GST_BUFFER_SIZE (buf) % ctx->info.bpf)
|
||||||
goto wrong_buffer;
|
goto wrong_buffer;
|
||||||
/* per channel least */
|
/* per channel least */
|
||||||
samples = GST_BUFFER_SIZE (buf) / ctx->state.bpf;
|
samples = GST_BUFFER_SIZE (buf) / ctx->info.bpf;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* frame and ts book-keeping */
|
/* frame and ts book-keeping */
|
||||||
|
@ -722,7 +742,7 @@ gst_base_audio_decoder_finish_frame (GstBaseAudioDecoder * dec, GstBuffer * buf,
|
||||||
|
|
||||||
while (priv->frames.length && frames) {
|
while (priv->frames.length && frames) {
|
||||||
gst_buffer_unref (g_queue_pop_head (&priv->frames));
|
gst_buffer_unref (g_queue_pop_head (&priv->frames));
|
||||||
dec->ctx->delay = dec->priv->frames.length;
|
dec->priv->ctx.delay = dec->priv->frames.length;
|
||||||
frames--;
|
frames--;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -742,7 +762,7 @@ gst_base_audio_decoder_finish_frame (GstBaseAudioDecoder * dec, GstBuffer * buf,
|
||||||
|
|
||||||
g_assert (GST_CLOCK_TIME_IS_VALID (priv->base_ts));
|
g_assert (GST_CLOCK_TIME_IS_VALID (priv->base_ts));
|
||||||
next_ts = priv->base_ts +
|
next_ts = priv->base_ts +
|
||||||
gst_util_uint64_scale (samples, GST_SECOND, ctx->state.rate);
|
gst_util_uint64_scale (samples, GST_SECOND, ctx->info.rate);
|
||||||
GST_LOG_OBJECT (dec, "buffer is %d samples past base_ts %" GST_TIME_FORMAT
|
GST_LOG_OBJECT (dec, "buffer is %d samples past base_ts %" GST_TIME_FORMAT
|
||||||
", expected ts %" GST_TIME_FORMAT, samples,
|
", expected ts %" GST_TIME_FORMAT, samples,
|
||||||
GST_TIME_ARGS (priv->base_ts), GST_TIME_ARGS (next_ts));
|
GST_TIME_ARGS (priv->base_ts), GST_TIME_ARGS (next_ts));
|
||||||
|
@ -778,14 +798,14 @@ gst_base_audio_decoder_finish_frame (GstBaseAudioDecoder * dec, GstBuffer * buf,
|
||||||
if (G_LIKELY (GST_CLOCK_TIME_IS_VALID (priv->base_ts))) {
|
if (G_LIKELY (GST_CLOCK_TIME_IS_VALID (priv->base_ts))) {
|
||||||
GST_BUFFER_TIMESTAMP (buf) =
|
GST_BUFFER_TIMESTAMP (buf) =
|
||||||
priv->base_ts +
|
priv->base_ts +
|
||||||
GST_FRAMES_TO_CLOCK_TIME (priv->samples, ctx->state.rate);
|
GST_FRAMES_TO_CLOCK_TIME (priv->samples, ctx->info.rate);
|
||||||
GST_BUFFER_DURATION (buf) = priv->base_ts +
|
GST_BUFFER_DURATION (buf) = priv->base_ts +
|
||||||
GST_FRAMES_TO_CLOCK_TIME (priv->samples + samples, ctx->state.rate) -
|
GST_FRAMES_TO_CLOCK_TIME (priv->samples + samples, ctx->info.rate) -
|
||||||
GST_BUFFER_TIMESTAMP (buf);
|
GST_BUFFER_TIMESTAMP (buf);
|
||||||
} else {
|
} else {
|
||||||
GST_BUFFER_TIMESTAMP (buf) = GST_CLOCK_TIME_NONE;
|
GST_BUFFER_TIMESTAMP (buf) = GST_CLOCK_TIME_NONE;
|
||||||
GST_BUFFER_DURATION (buf) =
|
GST_BUFFER_DURATION (buf) =
|
||||||
GST_FRAMES_TO_CLOCK_TIME (samples, ctx->state.rate);
|
GST_FRAMES_TO_CLOCK_TIME (samples, ctx->info.rate);
|
||||||
}
|
}
|
||||||
priv->samples += samples;
|
priv->samples += samples;
|
||||||
priv->samples_out += samples;
|
priv->samples_out += samples;
|
||||||
|
@ -802,7 +822,7 @@ wrong_buffer:
|
||||||
{
|
{
|
||||||
GST_ELEMENT_ERROR (dec, STREAM, ENCODE, (NULL),
|
GST_ELEMENT_ERROR (dec, STREAM, ENCODE, (NULL),
|
||||||
("buffer size %d not a multiple of %d", GST_BUFFER_SIZE (buf),
|
("buffer size %d not a multiple of %d", GST_BUFFER_SIZE (buf),
|
||||||
ctx->state.bpf));
|
ctx->info.bpf));
|
||||||
gst_buffer_unref (buf);
|
gst_buffer_unref (buf);
|
||||||
return GST_FLOW_ERROR;
|
return GST_FLOW_ERROR;
|
||||||
}
|
}
|
||||||
|
@ -827,7 +847,7 @@ gst_base_audio_decoder_handle_frame (GstBaseAudioDecoder * dec,
|
||||||
GST_BUFFER_SIZE (buffer),
|
GST_BUFFER_SIZE (buffer),
|
||||||
GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (buffer)));
|
GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (buffer)));
|
||||||
g_queue_push_tail (&dec->priv->frames, buffer);
|
g_queue_push_tail (&dec->priv->frames, buffer);
|
||||||
dec->ctx->delay = dec->priv->frames.length;
|
dec->priv->ctx.delay = dec->priv->frames.length;
|
||||||
dec->priv->bytes_in += GST_BUFFER_SIZE (buffer);
|
dec->priv->bytes_in += GST_BUFFER_SIZE (buffer);
|
||||||
} else {
|
} else {
|
||||||
GST_LOG_OBJECT (dec, "providing subclass with NULL frame");
|
GST_LOG_OBJECT (dec, "providing subclass with NULL frame");
|
||||||
|
@ -852,7 +872,7 @@ gst_base_audio_decoder_push_buffers (GstBaseAudioDecoder * dec, gboolean force)
|
||||||
|
|
||||||
klass = GST_BASE_AUDIO_DECODER_GET_CLASS (dec);
|
klass = GST_BASE_AUDIO_DECODER_GET_CLASS (dec);
|
||||||
priv = dec->priv;
|
priv = dec->priv;
|
||||||
ctx = dec->ctx;
|
ctx = &dec->priv->ctx;
|
||||||
|
|
||||||
g_return_val_if_fail (klass->handle_frame != NULL, GST_FLOW_ERROR);
|
g_return_val_if_fail (klass->handle_frame != NULL, GST_FLOW_ERROR);
|
||||||
|
|
||||||
|
@ -1253,8 +1273,8 @@ gst_base_audio_decoder_chain (GstPad * pad, GstBuffer * buffer)
|
||||||
static inline gboolean
|
static inline gboolean
|
||||||
gst_base_audio_decoder_do_byte (GstBaseAudioDecoder * dec)
|
gst_base_audio_decoder_do_byte (GstBaseAudioDecoder * dec)
|
||||||
{
|
{
|
||||||
return dec->ctx->do_byte_time && dec->ctx->state.bpf &&
|
return dec->priv->ctx.do_byte_time && dec->priv->ctx.info.bpf &&
|
||||||
dec->ctx->state.rate <= dec->priv->samples_out;
|
dec->priv->ctx.info.rate <= dec->priv->samples_out;
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
|
@ -1289,7 +1309,7 @@ gst_base_audio_decoder_sink_eventfunc (GstBaseAudioDecoder * dec,
|
||||||
/* handle newsegment resulting from legacy simple seeking */
|
/* handle newsegment resulting from legacy simple seeking */
|
||||||
/* note that we need to convert this whether or not enough data
|
/* note that we need to convert this whether or not enough data
|
||||||
* to handle initial newsegment */
|
* to handle initial newsegment */
|
||||||
if (dec->ctx->do_byte_time &&
|
if (dec->priv->ctx.do_byte_time &&
|
||||||
gst_pad_query_convert (dec->sinkpad, GST_FORMAT_BYTES, start,
|
gst_pad_query_convert (dec->sinkpad, GST_FORMAT_BYTES, start,
|
||||||
&dformat, &start)) {
|
&dformat, &start)) {
|
||||||
/* best attempt convert */
|
/* best attempt convert */
|
||||||
|
@ -1318,8 +1338,9 @@ gst_base_audio_decoder_sink_eventfunc (GstBaseAudioDecoder * dec,
|
||||||
* some concealment data */
|
* some concealment data */
|
||||||
GST_DEBUG_OBJECT (dec,
|
GST_DEBUG_OBJECT (dec,
|
||||||
"segment update: plc %d, do_plc %d, last_stop %" GST_TIME_FORMAT,
|
"segment update: plc %d, do_plc %d, last_stop %" GST_TIME_FORMAT,
|
||||||
dec->plc, dec->ctx->do_plc, GST_TIME_ARGS (dec->segment.last_stop));
|
dec->plc, dec->priv->ctx.do_plc,
|
||||||
if (dec->plc && dec->ctx->do_plc && dec->segment.rate > 0.0 &&
|
GST_TIME_ARGS (dec->segment.last_stop));
|
||||||
|
if (dec->plc && dec->priv->ctx.do_plc && dec->segment.rate > 0.0 &&
|
||||||
dec->segment.last_stop < start) {
|
dec->segment.last_stop < start) {
|
||||||
GstBaseAudioDecoderClass *klass;
|
GstBaseAudioDecoderClass *klass;
|
||||||
GstBuffer *buf;
|
GstBuffer *buf;
|
||||||
|
@ -1552,7 +1573,7 @@ gst_base_audio_decoder_sink_query (GstPad * pad, GstQuery * query)
|
||||||
gint64 src_val, dest_val;
|
gint64 src_val, dest_val;
|
||||||
|
|
||||||
gst_query_parse_convert (query, &src_fmt, &src_val, &dest_fmt, &dest_val);
|
gst_query_parse_convert (query, &src_fmt, &src_val, &dest_fmt, &dest_val);
|
||||||
if (!(res = gst_base_audio_encoded_audio_convert (&dec->ctx->state,
|
if (!(res = gst_base_audio_encoded_audio_convert (&dec->priv->ctx.info,
|
||||||
dec->priv->bytes_in, dec->priv->samples_out,
|
dec->priv->bytes_in, dec->priv->samples_out,
|
||||||
src_fmt, src_val, &dest_fmt, &dest_val)))
|
src_fmt, src_val, &dest_fmt, &dest_val)))
|
||||||
goto error;
|
goto error;
|
||||||
|
@ -1670,7 +1691,7 @@ gst_base_audio_decoder_src_query (GstPad * pad, GstQuery * query)
|
||||||
gint64 src_val, dest_val;
|
gint64 src_val, dest_val;
|
||||||
|
|
||||||
gst_query_parse_convert (query, &src_fmt, &src_val, &dest_fmt, &dest_val);
|
gst_query_parse_convert (query, &src_fmt, &src_val, &dest_fmt, &dest_val);
|
||||||
if (!(res = gst_base_audio_raw_audio_convert (&dec->ctx->state,
|
if (!(res = gst_base_audio_raw_audio_convert (&dec->priv->ctx.info,
|
||||||
src_fmt, src_val, &dest_fmt, &dest_val)))
|
src_fmt, src_val, &dest_fmt, &dest_val)))
|
||||||
break;
|
break;
|
||||||
gst_query_set_convert (query, src_fmt, src_val, dest_fmt, dest_val);
|
gst_query_set_convert (query, src_fmt, src_val, dest_fmt, dest_val);
|
||||||
|
@ -1690,9 +1711,9 @@ gst_base_audio_decoder_src_query (GstPad * pad, GstQuery * query)
|
||||||
GST_OBJECT_LOCK (dec);
|
GST_OBJECT_LOCK (dec);
|
||||||
/* add our latency */
|
/* add our latency */
|
||||||
if (min_latency != -1)
|
if (min_latency != -1)
|
||||||
min_latency += dec->ctx->min_latency;
|
min_latency += dec->priv->ctx.min_latency;
|
||||||
if (max_latency != -1)
|
if (max_latency != -1)
|
||||||
max_latency += dec->ctx->max_latency;
|
max_latency += dec->priv->ctx.max_latency;
|
||||||
GST_OBJECT_UNLOCK (dec);
|
GST_OBJECT_UNLOCK (dec);
|
||||||
|
|
||||||
gst_query_set_latency (query, live, min_latency, max_latency);
|
gst_query_set_latency (query, live, min_latency, max_latency);
|
||||||
|
@ -1868,7 +1889,7 @@ _gst_base_audio_decoder_error (GstBaseAudioDecoder * dec, gint weight,
|
||||||
GST_WARNING_OBJECT (dec, "error: %s", dbg);
|
GST_WARNING_OBJECT (dec, "error: %s", dbg);
|
||||||
dec->priv->error_count += weight;
|
dec->priv->error_count += weight;
|
||||||
dec->priv->discont = TRUE;
|
dec->priv->discont = TRUE;
|
||||||
if (dec->ctx->max_errors < dec->priv->error_count) {
|
if (dec->priv->ctx.max_errors < dec->priv->error_count) {
|
||||||
gst_element_message_full (GST_ELEMENT (dec), GST_MESSAGE_ERROR,
|
gst_element_message_full (GST_ELEMENT (dec), GST_MESSAGE_ERROR,
|
||||||
domain, code, txt, dbg, file, function, line);
|
domain, code, txt, dbg, file, function, line);
|
||||||
return GST_FLOW_ERROR;
|
return GST_FLOW_ERROR;
|
||||||
|
@ -1876,3 +1897,182 @@ _gst_base_audio_decoder_error (GstBaseAudioDecoder * dec, gint weight,
|
||||||
return GST_FLOW_OK;
|
return GST_FLOW_OK;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* gst_base_audio_decoder_get_info:
|
||||||
|
* @dec: a #GstBaseAudioDecoder
|
||||||
|
*
|
||||||
|
* Returns: a #AudioFormatInfo describing input audio format
|
||||||
|
*/
|
||||||
|
GstAudioFormatInfo *
|
||||||
|
gst_base_audio_decoder_get_info (GstBaseAudioDecoder * dec)
|
||||||
|
{
|
||||||
|
g_return_val_if_fail (GST_IS_BASE_AUDIO_DECODER (dec), NULL);
|
||||||
|
|
||||||
|
return &dec->priv->ctx.info;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* gst_base_audio_decoder_set_plc_aware:
|
||||||
|
* @dec: a #GstBaseAudioDecoder
|
||||||
|
* @plc: new plc state
|
||||||
|
*
|
||||||
|
* Indicates whether or not subclass handles packet loss concealment (plc).
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
gst_base_audio_decoder_set_plc_aware (GstBaseAudioDecoder * dec, gboolean plc)
|
||||||
|
{
|
||||||
|
g_return_if_fail (GST_IS_BASE_AUDIO_DECODER (dec));
|
||||||
|
|
||||||
|
dec->priv->ctx.do_plc = plc;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* gst_base_audio_decoder_get_plc_aware:
|
||||||
|
* @dec: a #GstBaseAudioDecoder
|
||||||
|
*
|
||||||
|
* Returns: currently configured plc handling
|
||||||
|
*/
|
||||||
|
gint
|
||||||
|
gst_base_audio_decoder_get_plc_aware (GstBaseAudioDecoder * dec)
|
||||||
|
{
|
||||||
|
g_return_val_if_fail (GST_IS_BASE_AUDIO_DECODER (dec), 0);
|
||||||
|
|
||||||
|
return dec->priv->ctx.do_plc;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* gst_base_audio_decoder_set_byte_time:
|
||||||
|
* @dec: a #GstBaseAudioDecoder
|
||||||
|
* @enabled: whether to enable byte to time conversion
|
||||||
|
*
|
||||||
|
* Allows baseclass to perform byte to time estimated conversion.
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
gst_base_audio_decoder_set_byte_time (GstBaseAudioDecoder * dec,
|
||||||
|
gboolean enabled)
|
||||||
|
{
|
||||||
|
g_return_if_fail (GST_IS_BASE_AUDIO_DECODER (dec));
|
||||||
|
|
||||||
|
dec->priv->ctx.do_byte_time = enabled;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* gst_base_audio_decoder_get_byte_time:
|
||||||
|
* @dec: a #GstBaseAudioDecoder
|
||||||
|
*
|
||||||
|
* Returns: currently configured byte to time conversion setting
|
||||||
|
*/
|
||||||
|
gint
|
||||||
|
gst_base_audio_decoder_get_byte_time (GstBaseAudioDecoder * dec)
|
||||||
|
{
|
||||||
|
g_return_val_if_fail (GST_IS_BASE_AUDIO_DECODER (dec), 0);
|
||||||
|
|
||||||
|
return dec->priv->ctx.do_byte_time;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* gst_base_audio_decoder_get_delay:
|
||||||
|
* @dec: a #GstBaseAudioDecoder
|
||||||
|
*
|
||||||
|
* Returns: currently configured decoder delay
|
||||||
|
*/
|
||||||
|
gint
|
||||||
|
gst_base_audio_decoder_get_delay (GstBaseAudioDecoder * dec)
|
||||||
|
{
|
||||||
|
g_return_val_if_fail (GST_IS_BASE_AUDIO_DECODER (dec), 0);
|
||||||
|
|
||||||
|
return dec->priv->ctx.delay;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* gst_base_audio_decoder_set_max_errors:
|
||||||
|
* @dec: a #GstBaseAudioDecoder
|
||||||
|
* @num: max tolerated errors
|
||||||
|
*
|
||||||
|
* Sets numbers of tolerated decoder errors, where a tolerated one is then only
|
||||||
|
* warned about, but more than tolerated will lead to fatal error.
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
gst_base_audio_decoder_set_max_errors (GstBaseAudioDecoder * enc, gint num)
|
||||||
|
{
|
||||||
|
g_return_if_fail (GST_IS_BASE_AUDIO_DECODER (enc));
|
||||||
|
|
||||||
|
enc->priv->ctx.max_errors = num;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* gst_base_audio_decoder_get_max_errors:
|
||||||
|
* @dec: a #GstBaseAudioDecoder
|
||||||
|
*
|
||||||
|
* Returns: currently configured decoder tolerated error count.
|
||||||
|
*/
|
||||||
|
gint
|
||||||
|
gst_base_audio_decoder_get_max_errors (GstBaseAudioDecoder * dec)
|
||||||
|
{
|
||||||
|
g_return_val_if_fail (GST_IS_BASE_AUDIO_DECODER (dec), 0);
|
||||||
|
|
||||||
|
return dec->priv->ctx.max_errors;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* gst_base_audio_decoder_set_latency:
|
||||||
|
* @dec: a #GstBaseAudioDecoder
|
||||||
|
* @min: minimum latency
|
||||||
|
* @max: maximum latency
|
||||||
|
*
|
||||||
|
* Sets decoder latency.
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
gst_base_audio_decoder_set_latency (GstBaseAudioDecoder * dec,
|
||||||
|
GstClockTime min, GstClockTime max)
|
||||||
|
{
|
||||||
|
g_return_if_fail (GST_IS_BASE_AUDIO_DECODER (dec));
|
||||||
|
|
||||||
|
GST_OBJECT_LOCK (dec);
|
||||||
|
dec->priv->ctx.min_latency = min;
|
||||||
|
dec->priv->ctx.max_latency = max;
|
||||||
|
GST_OBJECT_UNLOCK (dec);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* gst_base_audio_decoder_get_latency:
|
||||||
|
* @dec: a #GstBaseAudioDecoder
|
||||||
|
* @min: a pointer to storage to hold minimum latency
|
||||||
|
* @max: a pointer to storage to hold maximum latency
|
||||||
|
*
|
||||||
|
* Returns currently configured latency.
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
gst_base_audio_decoder_get_latency (GstBaseAudioDecoder * dec,
|
||||||
|
GstClockTime * min, GstClockTime * max)
|
||||||
|
{
|
||||||
|
g_return_if_fail (GST_IS_BASE_AUDIO_DECODER (dec));
|
||||||
|
|
||||||
|
GST_OBJECT_LOCK (dec);
|
||||||
|
if (min)
|
||||||
|
*min = dec->priv->ctx.min_latency;
|
||||||
|
if (max)
|
||||||
|
*max = dec->priv->ctx.max_latency;
|
||||||
|
GST_OBJECT_UNLOCK (dec);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* gst_base_audio_decoder_get_parse_state:
|
||||||
|
* @dec: a #GstBaseAudioDecoder
|
||||||
|
* @min: a pointer to storage to hold current sync state
|
||||||
|
* @max: a pointer to storage to hold current eos state
|
||||||
|
*
|
||||||
|
* Return current parsing (sync and eos) state.
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
gst_base_audio_decoder_get_parse_state (GstBaseAudioDecoder * dec,
|
||||||
|
gboolean * sync, gboolean * eos)
|
||||||
|
{
|
||||||
|
g_return_if_fail (GST_IS_BASE_AUDIO_DECODER (dec));
|
||||||
|
|
||||||
|
if (sync)
|
||||||
|
*sync = dec->priv->ctx.sync;
|
||||||
|
if (eos)
|
||||||
|
*eos = dec->priv->ctx.eos;
|
||||||
|
}
|
||||||
|
|
|
@ -81,7 +81,6 @@ typedef struct _GstBaseAudioDecoder GstBaseAudioDecoder;
|
||||||
typedef struct _GstBaseAudioDecoderClass GstBaseAudioDecoderClass;
|
typedef struct _GstBaseAudioDecoderClass GstBaseAudioDecoderClass;
|
||||||
|
|
||||||
typedef struct _GstBaseAudioDecoderPrivate GstBaseAudioDecoderPrivate;
|
typedef struct _GstBaseAudioDecoderPrivate GstBaseAudioDecoderPrivate;
|
||||||
typedef struct _GstBaseAudioDecoderContext GstBaseAudioDecoderContext;
|
|
||||||
|
|
||||||
/* do not use this one, use macro below */
|
/* do not use this one, use macro below */
|
||||||
GstFlowReturn _gst_base_audio_decoder_error (GstBaseAudioDecoder *dec, gint weight,
|
GstFlowReturn _gst_base_audio_decoder_error (GstBaseAudioDecoder *dec, gint weight,
|
||||||
|
@ -121,40 +120,6 @@ G_STMT_START { \
|
||||||
GST_FUNCTION, __LINE__); \
|
GST_FUNCTION, __LINE__); \
|
||||||
} G_STMT_END
|
} G_STMT_END
|
||||||
|
|
||||||
/**
|
|
||||||
* GstBaseAudioDecoderContext:
|
|
||||||
* @state: a #GstAudioState describing input audio format
|
|
||||||
* @eos: no (immediate) subsequent data in stream
|
|
||||||
* @sync: stream parsing in sync
|
|
||||||
* @delay: number of frames pending decoding (typically at least 1 for current)
|
|
||||||
* @do_plc: whether subclass is prepared to handle (packet) loss concealment
|
|
||||||
* @min_latency: min latency of element
|
|
||||||
* @max_latency: max latency of element
|
|
||||||
* @lookahead: decoder lookahead (in units of input rate samples)
|
|
||||||
*
|
|
||||||
* Transparent #GstBaseAudioEncoderContext data structure.
|
|
||||||
*/
|
|
||||||
struct _GstBaseAudioDecoderContext {
|
|
||||||
/* input */
|
|
||||||
/* (output) audio format */
|
|
||||||
GstAudioFormatInfo state;
|
|
||||||
|
|
||||||
/* parsing state */
|
|
||||||
gboolean eos;
|
|
||||||
gboolean sync;
|
|
||||||
|
|
||||||
/* misc */
|
|
||||||
gint delay;
|
|
||||||
|
|
||||||
/* output */
|
|
||||||
gboolean do_plc;
|
|
||||||
gboolean do_byte_time;
|
|
||||||
gint max_errors;
|
|
||||||
/* MT-protected (with LOCK) */
|
|
||||||
GstClockTime min_latency;
|
|
||||||
GstClockTime max_latency;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* GstBaseAudioDecoder:
|
* GstBaseAudioDecoder:
|
||||||
*
|
*
|
||||||
|
@ -171,7 +136,6 @@ struct _GstBaseAudioDecoder
|
||||||
|
|
||||||
/* MT-protected (with STREAM_LOCK) */
|
/* MT-protected (with STREAM_LOCK) */
|
||||||
GstSegment segment;
|
GstSegment segment;
|
||||||
GstBaseAudioDecoderContext *ctx;
|
|
||||||
|
|
||||||
/* properties */
|
/* properties */
|
||||||
GstClockTime latency;
|
GstClockTime latency;
|
||||||
|
@ -254,6 +218,30 @@ struct _GstBaseAudioDecoderClass
|
||||||
GstFlowReturn gst_base_audio_decoder_finish_frame (GstBaseAudioDecoder * dec,
|
GstFlowReturn gst_base_audio_decoder_finish_frame (GstBaseAudioDecoder * dec,
|
||||||
GstBuffer * buf, gint frames);
|
GstBuffer * buf, gint frames);
|
||||||
|
|
||||||
|
GstAudioFormatInfo *gst_base_audio_decoder_get_info (GstBaseAudioDecoder * dec);
|
||||||
|
|
||||||
|
void gst_base_audio_decoder_set_plc_aware (GstBaseAudioDecoder * dec,
|
||||||
|
gboolean plc);
|
||||||
|
gint gst_base_audio_decoder_get_plc_aware (GstBaseAudioDecoder * dec);
|
||||||
|
|
||||||
|
void gst_base_audio_decoder_set_byte_time (GstBaseAudioDecoder * dec,
|
||||||
|
gboolean enabled);
|
||||||
|
gint gst_base_audio_decoder_get_byte_time (GstBaseAudioDecoder * dec);
|
||||||
|
|
||||||
|
gint gst_base_audio_decoder_get_delay (GstBaseAudioDecoder * dec);
|
||||||
|
|
||||||
|
void gst_base_audio_decoder_set_max_errors (GstBaseAudioDecoder * enc,
|
||||||
|
gint num);
|
||||||
|
gint gst_base_audio_decoder_get_max_errors (GstBaseAudioDecoder * dec);
|
||||||
|
|
||||||
|
void gst_base_audio_decoder_set_latency (GstBaseAudioDecoder * dec,
|
||||||
|
GstClockTime min, GstClockTime max);
|
||||||
|
void gst_base_audio_decoder_get_latency (GstBaseAudioDecoder * dec,
|
||||||
|
GstClockTime * min, GstClockTime * max);
|
||||||
|
|
||||||
|
void gst_base_audio_decoder_get_parse_state (GstBaseAudioDecoder * dec,
|
||||||
|
gboolean * sync, gboolean * eos);
|
||||||
|
|
||||||
GType gst_base_audio_decoder_get_type (void);
|
GType gst_base_audio_decoder_get_type (void);
|
||||||
|
|
||||||
G_END_DECLS
|
G_END_DECLS
|
||||||
|
|
Loading…
Reference in a new issue