mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-01 13:08:49 +00:00
dacf8eaa18
Original commit message from CVS: Patch by: j^ <j at bootlab dot org> * ext/amrwb/gstamrwbdec.c: * ext/amrwb/gstamrwbenc.c: * ext/amrwb/gstamrwbparse.c: * ext/arts/gst_arts.c: * ext/artsd/gstartsdsink.c: * ext/audiofile/gstafparse.c: * ext/audiofile/gstafsink.c: * ext/audiofile/gstafsrc.c: * ext/cdaudio/gstcdaudio.c: * ext/directfb/dfbvideosink.c: * ext/divx/gstdivxdec.c: * ext/divx/gstdivxenc.c: * ext/dts/gstdtsdec.c: (gst_dtsdec_base_init): * ext/faac/gstfaac.c: (gst_faac_base_init): * ext/faad/gstfaad.c: * ext/gsm/gstgsmdec.c: * ext/gsm/gstgsmenc.c: * ext/hermes/gsthermescolorspace.c: * ext/ivorbis/vorbisfile.c: * ext/lcs/gstcolorspace.c: * ext/libfame/gstlibfame.c: * ext/libmms/gstmms.c: (gst_mms_base_init): * ext/musicbrainz/gsttrm.c: (gst_musicbrainz_base_init): * ext/nas/nassink.c: (gst_nassink_base_init): * ext/neon/gstneonhttpsrc.c: * ext/polyp/polypsink.c: (gst_polypsink_base_init): * ext/sdl/sdlaudiosink.c: * ext/sdl/sdlvideosink.c: * ext/shout/gstshout.c: * ext/snapshot/gstsnapshot.c: * ext/sndfile/gstsf.c: * ext/tarkin/gsttarkindec.c: * ext/tarkin/gsttarkinenc.c: * ext/theora/theoradec.c: * ext/wavpack/gstwavpackdec.c: (gst_wavpack_dec_base_init): * ext/wavpack/gstwavpackparse.c: (gst_wavpack_parse_base_init): * ext/xvid/gstxviddec.c: * ext/xvid/gstxvidenc.c: * gst/cdxaparse/gstcdxaparse.c: (gst_cdxa_parse_base_init): * gst/cdxaparse/gstcdxastrip.c: (gst_cdxastrip_base_init): * gst/chart/gstchart.c: * gst/equalizer/gstiirequalizer.c: (gst_iir_equalizer_base_init): * gst/festival/gstfestival.c: * gst/filter/gstiir.c: * gst/filter/gstlpwsinc.c: * gst/freeze/gstfreeze.c: * gst/games/gstpuzzle.c: (gst_puzzle_base_init): * gst/mixmatrix/mixmatrix.c: * gst/mpeg1sys/gstmpeg1systemencode.c: * gst/mpeg1videoparse/gstmp1videoparse.c: * gst/mpeg2sub/gstmpeg2subt.c: * gst/mpegaudioparse/gstmpegaudioparse.c: * gst/multifilesink/gstmultifilesink.c: * gst/overlay/gstoverlay.c: * gst/passthrough/gstpassthrough.c: * gst/playondemand/gstplayondemand.c: * gst/qtdemux/qtdemux.c: * gst/rtjpeg/gstrtjpegdec.c: * gst/rtjpeg/gstrtjpegenc.c: * gst/smooth/gstsmooth.c: * gst/tta/gstttadec.c: (gst_tta_dec_base_init): * gst/tta/gstttaparse.c: (gst_tta_parse_base_init): * gst/videocrop/gstvideocrop.c: * gst/videodrop/gstvideodrop.c: * gst/virtualdub/gstxsharpen.c: * gst/xingheader/gstxingmux.c: (gst_xing_mux_base_init): * gst/y4m/gsty4mencode.c: Unify the long descriptions in the plugin details (#337263).
703 lines
18 KiB
C
703 lines
18 KiB
C
/*
|
|
* This sink plugin works, but has some room for improvements:
|
|
*
|
|
* - Export polypaudio's stream clock through gstreamer's API
|
|
* - Add support for querying latency information
|
|
* - Add a source for polypaudio
|
|
*
|
|
* Lennart Poettering, 2004
|
|
*/
|
|
|
|
#include <pthread.h>
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
|
|
#include <polyp/polyplib-error.h>
|
|
#include <polyp/mainloop.h>
|
|
|
|
#include "polypsink.h"
|
|
|
|
GST_DEBUG_CATEGORY_EXTERN (polyp_debug);
|
|
#define GST_CAT_DEFAULT polyp_debug
|
|
|
|
enum
|
|
{
|
|
ARG_0,
|
|
ARG_SERVER,
|
|
ARG_SINK,
|
|
};
|
|
|
|
static GstElementClass *parent_class = NULL;
|
|
|
|
static void create_stream (GstPolypSink * polypsink);
|
|
static void destroy_stream (GstPolypSink * polypsink);
|
|
|
|
static void create_context (GstPolypSink * polypsink);
|
|
static void destroy_context (GstPolypSink * polypsink);
|
|
|
|
static void
|
|
gst_polypsink_base_init (gpointer g_class)
|
|
{
|
|
|
|
static GstStaticPadTemplate pad_template = GST_STATIC_PAD_TEMPLATE ("sink",
|
|
GST_PAD_SINK,
|
|
GST_PAD_ALWAYS,
|
|
GST_STATIC_CAPS ("audio/x-raw-int, "
|
|
"endianness = (int) { LITTLE_ENDIAN , BIG_ENDIAN }, "
|
|
"signed = (boolean) TRUE, "
|
|
"width = (int) 16, "
|
|
"depth = (int) 16, "
|
|
"rate = (int) [ 1, MAX ], " "channels = (int) [ 1, 16 ]"
|
|
#if 0
|
|
";audio/x-raw-float, "
|
|
"endianness = (int) { LITTLE_ENDIAN, BIG_ENDIAN }, "
|
|
"width = (int) 32, "
|
|
"rate = (int) [ 1, MAX ], "
|
|
"channels = (int) [ 1, 16 ];"
|
|
"audio/x-raw-int, "
|
|
"signed = (boolean) FALSE, "
|
|
"width = (int) 8, "
|
|
"depth = (int) 8, "
|
|
"rate = (int) [ 1, MAX ], " "channels = (int) [ 1, 16 ]"
|
|
#endif
|
|
)
|
|
);
|
|
|
|
static const GstElementDetails details =
|
|
GST_ELEMENT_DETAILS ("Polypaudio audio sink",
|
|
"Sink/Audio",
|
|
"Plays audio to a Polypaudio server",
|
|
"Lennart Poettering");
|
|
|
|
GstElementClass *element_class = GST_ELEMENT_CLASS (g_class);
|
|
|
|
gst_element_class_add_pad_template (element_class,
|
|
gst_static_pad_template_get (&pad_template));
|
|
gst_element_class_set_details (element_class, &details);
|
|
}
|
|
|
|
static void
|
|
gst_polypsink_set_property (GObject * object, guint prop_id,
|
|
const GValue * value, GParamSpec * pspec)
|
|
{
|
|
GstPolypSink *polypsink;
|
|
|
|
g_return_if_fail (GST_IS_POLYPSINK (object));
|
|
polypsink = GST_POLYPSINK (object);
|
|
|
|
switch (prop_id) {
|
|
case ARG_SERVER:
|
|
g_free (polypsink->server);
|
|
polypsink->server = g_strdup (g_value_get_string (value));
|
|
break;
|
|
|
|
case ARG_SINK:
|
|
g_free (polypsink->sink);
|
|
polypsink->sink = g_strdup (g_value_get_string (value));
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_polypsink_get_property (GObject * object, guint prop_id, GValue * value,
|
|
GParamSpec * pspec)
|
|
{
|
|
GstPolypSink *polypsink;
|
|
|
|
g_return_if_fail (GST_IS_POLYPSINK (object));
|
|
polypsink = GST_POLYPSINK (object);
|
|
|
|
switch (prop_id) {
|
|
case ARG_SERVER:
|
|
g_value_set_string (value, polypsink->server);
|
|
break;
|
|
|
|
case ARG_SINK:
|
|
g_value_set_string (value, polypsink->sink);
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static GstStateChangeReturn
|
|
gst_polypsink_change_state (GstElement * element, GstStateChange transition)
|
|
{
|
|
GstPolypSink *polypsink;
|
|
|
|
polypsink = GST_POLYPSINK (element);
|
|
|
|
switch (transition) {
|
|
|
|
case GST_STATE_CHANGE_NULL_TO_READY:
|
|
create_context (polypsink);
|
|
break;
|
|
|
|
case GST_STATE_CHANGE_READY_TO_NULL:
|
|
destroy_context (polypsink);
|
|
break;
|
|
|
|
case GST_STATE_CHANGE_READY_TO_PAUSED:
|
|
|
|
create_stream (polypsink);
|
|
|
|
if (polypsink->stream
|
|
&& pa_stream_get_state (polypsink->stream) == PA_STREAM_READY)
|
|
pa_operation_unref (pa_stream_cork (polypsink->stream, 1, NULL, NULL));
|
|
break;
|
|
|
|
case GST_STATE_CHANGE_PLAYING_TO_PAUSED:
|
|
|
|
if (polypsink->stream
|
|
&& pa_stream_get_state (polypsink->stream) == PA_STREAM_READY)
|
|
pa_operation_unref (pa_stream_cork (polypsink->stream, 1, NULL, NULL));
|
|
|
|
break;
|
|
|
|
case GST_STATE_CHANGE_PAUSED_TO_PLAYING:
|
|
|
|
create_stream (polypsink);
|
|
|
|
if (polypsink->stream
|
|
&& pa_stream_get_state (polypsink->stream) == PA_STREAM_READY)
|
|
pa_operation_unref (pa_stream_cork (polypsink->stream, 0, NULL, NULL));
|
|
|
|
break;
|
|
|
|
case GST_STATE_CHANGE_PAUSED_TO_READY:
|
|
|
|
destroy_stream (polypsink);
|
|
break;
|
|
}
|
|
|
|
if (GST_ELEMENT_CLASS (parent_class)->change_state)
|
|
return GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
|
|
|
|
return GST_STATE_CHANGE_SUCCESS;
|
|
}
|
|
|
|
|
|
static void
|
|
do_write (GstPolypSink * polypsink, size_t length)
|
|
{
|
|
size_t l;
|
|
|
|
if (!polypsink->buffer)
|
|
return;
|
|
|
|
g_assert (polypsink->buffer_index < GST_BUFFER_SIZE (polypsink->buffer));
|
|
l = GST_BUFFER_SIZE (polypsink->buffer) - polypsink->buffer_index;
|
|
|
|
if (l > length)
|
|
l = length;
|
|
|
|
pa_stream_write (polypsink->stream,
|
|
GST_BUFFER_DATA (polypsink->buffer) + polypsink->buffer_index, l, NULL,
|
|
0);
|
|
polypsink->buffer_index += l;
|
|
|
|
if (polypsink->buffer_index >= GST_BUFFER_SIZE (polypsink->buffer)) {
|
|
gst_buffer_unref (polypsink->buffer);
|
|
polypsink->buffer = NULL;
|
|
polypsink->buffer_index = 0;
|
|
}
|
|
}
|
|
|
|
static void
|
|
stream_write_callback (struct pa_stream *s, size_t length, void *userdata)
|
|
{
|
|
GstPolypSink *polypsink = userdata;
|
|
|
|
g_assert (s && length && polypsink);
|
|
|
|
do_write (polypsink, length);
|
|
}
|
|
|
|
static void
|
|
stream_state_callback (struct pa_stream *s, void *userdata)
|
|
{
|
|
GstPolypSink *polypsink = userdata;
|
|
|
|
g_assert (s && polypsink);
|
|
|
|
switch (pa_stream_get_state (s)) {
|
|
case PA_STREAM_DISCONNECTED:
|
|
case PA_STREAM_CREATING:
|
|
break;
|
|
|
|
case PA_STREAM_READY:
|
|
break;
|
|
|
|
case PA_STREAM_FAILED:
|
|
GST_ELEMENT_ERROR (GST_ELEMENT (polypsink), RESOURCE, BUSY,
|
|
("Stream creation failed: %s",
|
|
pa_strerror (pa_context_errno (pa_stream_get_context (s)))),
|
|
(NULL));
|
|
|
|
/* Pass over */
|
|
case PA_STREAM_TERMINATED:
|
|
default:
|
|
polypsink->mainloop_api->quit (polypsink->mainloop_api, 1);
|
|
destroy_context (polypsink);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
context_state_callback (struct pa_context *c, void *userdata)
|
|
{
|
|
GstPolypSink *polypsink = userdata;
|
|
|
|
g_assert (c && polypsink);
|
|
|
|
switch (pa_context_get_state (c)) {
|
|
case PA_CONTEXT_UNCONNECTED:
|
|
case PA_CONTEXT_CONNECTING:
|
|
case PA_CONTEXT_AUTHORIZING:
|
|
case PA_CONTEXT_SETTING_NAME:
|
|
break;
|
|
|
|
case PA_CONTEXT_READY:{
|
|
GstState state;
|
|
|
|
g_assert (!polypsink->stream);
|
|
|
|
state = gst_element_get_state (GST_ELEMENT (polypsink));
|
|
if (state == GST_STATE_PAUSED || state == GST_STATE_PLAYING)
|
|
create_stream (polypsink);
|
|
|
|
break;
|
|
}
|
|
|
|
case PA_CONTEXT_FAILED:
|
|
GST_ELEMENT_ERROR (GST_ELEMENT (polypsink), RESOURCE, BUSY,
|
|
("Connection failed: %s", pa_strerror (pa_context_errno (c))),
|
|
(NULL));
|
|
|
|
/* Pass over */
|
|
case PA_CONTEXT_TERMINATED:
|
|
default:
|
|
polypsink->mainloop_api->quit (polypsink->mainloop_api, 1);
|
|
destroy_context (polypsink);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
create_stream (GstPolypSink * polypsink)
|
|
{
|
|
char t[256];
|
|
|
|
g_assert (polypsink);
|
|
|
|
if (polypsink->stream)
|
|
return;
|
|
|
|
if (!polypsink->context) {
|
|
create_context (polypsink);
|
|
return;
|
|
}
|
|
|
|
if (!polypsink->negotiated)
|
|
return;
|
|
|
|
if (pa_context_get_state (polypsink->context) != PA_CONTEXT_READY)
|
|
return;
|
|
|
|
pa_sample_spec_snprint (t, sizeof (t), &polypsink->sample_spec);
|
|
|
|
polypsink->stream =
|
|
pa_stream_new (polypsink->context, "gstreamer output",
|
|
&polypsink->sample_spec);
|
|
g_assert (polypsink->stream);
|
|
|
|
pa_stream_set_state_callback (polypsink->stream, stream_state_callback,
|
|
polypsink);
|
|
pa_stream_set_write_callback (polypsink->stream, stream_write_callback,
|
|
polypsink);
|
|
pa_stream_connect_playback (polypsink->stream, NULL, NULL,
|
|
PA_STREAM_INTERPOLATE_LATENCY, PA_VOLUME_NORM);
|
|
}
|
|
|
|
static void
|
|
create_context (GstPolypSink * polypsink)
|
|
{
|
|
g_assert (polypsink);
|
|
|
|
if (polypsink->context)
|
|
return;
|
|
|
|
polypsink->context = pa_context_new (polypsink->mainloop_api, "gstreamer");
|
|
g_assert (polypsink->context);
|
|
|
|
pa_context_set_state_callback (polypsink->context, context_state_callback,
|
|
polypsink);
|
|
if (polypsink->server && polypsink->server[0]) {
|
|
pa_context_connect (polypsink->context, polypsink->server, 1, NULL);
|
|
} else {
|
|
pa_context_connect (polypsink->context, NULL, 1, NULL);
|
|
}
|
|
}
|
|
|
|
static void
|
|
destroy_stream (GstPolypSink * polypsink)
|
|
{
|
|
g_assert (polypsink);
|
|
|
|
if (polypsink->stream) {
|
|
struct pa_stream *s = polypsink->stream;
|
|
|
|
polypsink->stream = NULL;
|
|
pa_stream_set_state_callback (s, NULL, NULL);
|
|
pa_stream_set_write_callback (s, NULL, NULL);
|
|
pa_stream_unref (s);
|
|
}
|
|
}
|
|
|
|
static void
|
|
destroy_context (GstPolypSink * polypsink)
|
|
{
|
|
destroy_stream (polypsink);
|
|
|
|
if (polypsink->context) {
|
|
struct pa_context *c = polypsink->context;
|
|
|
|
polypsink->context = NULL;
|
|
pa_context_set_state_callback (c, NULL, NULL);
|
|
pa_context_unref (c);
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_polypsink_chain (GstPad * pad, GstData * data)
|
|
{
|
|
GstPolypSink *polypsink = GST_POLYPSINK (gst_pad_get_parent (pad));
|
|
|
|
g_assert (!polypsink->buffer);
|
|
|
|
if (GST_IS_EVENT (data)) {
|
|
GstEvent *event = GST_EVENT (data);
|
|
|
|
switch (GST_EVENT_TYPE (event)) {
|
|
case GST_EVENT_EOS:
|
|
if (polypsink->stream) {
|
|
struct pa_operation *o;
|
|
|
|
pa_operation_unref (pa_stream_cork (polypsink->stream, 0, NULL,
|
|
NULL));
|
|
o = pa_stream_drain (polypsink->stream, NULL, NULL);
|
|
|
|
/* drain now */
|
|
while (pa_operation_get_state (o) == PA_OPERATION_RUNNING) {
|
|
if (pa_mainloop_iterate (polypsink->mainloop, 1, NULL) < 0)
|
|
return;
|
|
}
|
|
|
|
pa_operation_unref (o);
|
|
}
|
|
|
|
break;
|
|
case GST_EVENT_FLUSH:
|
|
if (polypsink->stream)
|
|
pa_operation_unref (pa_stream_flush (polypsink->stream, NULL, NULL));
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
gst_pad_event_default (polypsink->sinkpad, event);
|
|
} else {
|
|
size_t l;
|
|
|
|
polypsink->buffer = GST_BUFFER (data);
|
|
polypsink->buffer_index = 0;
|
|
polypsink->counter += GST_BUFFER_SIZE (polypsink->buffer);
|
|
|
|
if (polypsink->stream
|
|
&& (l = pa_stream_writable_size (polypsink->stream)) > 0)
|
|
do_write (polypsink, l);
|
|
}
|
|
|
|
while (polypsink->context && (pa_context_is_pending (polypsink->context)
|
|
|| polypsink->buffer)) {
|
|
if (pa_mainloop_iterate (polypsink->mainloop, 1, NULL) < 0)
|
|
return;
|
|
}
|
|
|
|
}
|
|
|
|
#if 0
|
|
static void
|
|
stream_get_latency_callback (struct pa_stream *s,
|
|
const struct pa_latency_info *i, void *userdata)
|
|
{
|
|
GstPolypSink *polypsink = (GstPolypSink *) userdata;
|
|
|
|
polypsink->latency = i->buffer_usec + i->sink_usec;
|
|
}
|
|
|
|
static GstClockTime
|
|
gst_polypsink_get_time (GstClock * clock, gpointer data)
|
|
{
|
|
struct pa_operation *o;
|
|
GstPolypSink *polypsink = GST_POLYPSINK (data);
|
|
GstClockTime r, l;
|
|
|
|
if (!polypsink->stream
|
|
|| pa_stream_get_state (polypsink->stream) != PA_STREAM_READY)
|
|
return 0;
|
|
|
|
polypsink->latency = 0;
|
|
|
|
o = pa_stream_get_latency (polypsink_ > stream, latency_func, polypsink);
|
|
g_assert (o);
|
|
|
|
while (pa_operation_get_state (o) == PA_OPERATION_RUNNING) {
|
|
if (pa_mainloop_iterate (polypsink->mainloop, 1, NULL) < 0)
|
|
return;
|
|
}
|
|
|
|
r = ((GstClockTime) polypsink->counter /
|
|
pa_frame_size (&polypsink->sample_spec)) * GST_SECOND /
|
|
polypsink->sample_spec.rate;
|
|
l = polypsink->latency * GST_USECOND;
|
|
|
|
return r > l ? r - l : 0;
|
|
}
|
|
|
|
static GstClock *
|
|
gst_polypsink_get_clock (GstElement * element)
|
|
{
|
|
GstPolypSink *polypsink = GST_POLYPSINK (element);
|
|
|
|
return GST_CLOCK (polypsink->provided_clock);
|
|
}
|
|
|
|
static void
|
|
gst_polypsink_set_clock (GstElement * element, GstClock * clock)
|
|
{
|
|
GstPolypSink *polypsink = GST_POLYPSINK (element);
|
|
|
|
polypsink->clock = clock;
|
|
}
|
|
|
|
#endif
|
|
|
|
static GstPadLinkReturn
|
|
gst_polypsink_link (GstPad * pad, const GstCaps * caps)
|
|
{
|
|
int depth = 16, endianness = 1234;
|
|
gboolean sign = TRUE;
|
|
GstPolypSink *polypsink;
|
|
GstStructure *structure;
|
|
const char *n;
|
|
char t[256];
|
|
GstState state;
|
|
int n_channels;
|
|
|
|
polypsink = GST_POLYPSINK (gst_pad_get_parent (pad));
|
|
|
|
structure = gst_caps_get_structure (caps, 0);
|
|
|
|
if (!(gst_structure_get_int (structure, "depth", &depth)))
|
|
gst_structure_get_int (structure, "width", &depth);
|
|
|
|
gst_structure_get_int (structure, "endianness", &endianness);
|
|
gst_structure_get_boolean (structure, "signed", &sign);
|
|
|
|
n = gst_structure_get_name (structure);
|
|
|
|
if (depth == 16 && endianness == 1234 && sign
|
|
&& !strcmp (n, "audio/x-raw-int"))
|
|
polypsink->sample_spec.format = PA_SAMPLE_S16LE;
|
|
else if (depth == 16 && endianness == 4321 && sign
|
|
&& !strcmp (n, "audio/x-raw-int"))
|
|
polypsink->sample_spec.format = PA_SAMPLE_S16BE;
|
|
else if (depth == 8 && !sign && !strcmp (n, "audio/x-raw-int"))
|
|
polypsink->sample_spec.format = PA_SAMPLE_U8;
|
|
else if (depth == 32 && endianness == 1234
|
|
&& !strcmp (n, "audio/x-raw-float"))
|
|
polypsink->sample_spec.format = PA_SAMPLE_FLOAT32LE;
|
|
else if (depth == 32 && endianness == 4321
|
|
&& !strcmp (n, "audio/x-raw-float"))
|
|
polypsink->sample_spec.format = PA_SAMPLE_FLOAT32LE;
|
|
else {
|
|
GST_DEBUG ("unrecognized format, refusing link");
|
|
return GST_PAD_LINK_REFUSED;
|
|
}
|
|
|
|
GST_DEBUG ("using format %d", polypsink->sample_spec.format);
|
|
|
|
polypsink->sample_spec.rate = 44100;
|
|
polypsink->sample_spec.channels = 2;
|
|
|
|
pa_sample_spec_snprint (t, sizeof (t), &polypsink->sample_spec);
|
|
|
|
gst_structure_get_int (structure, "channels", &n_channels);
|
|
polypsink->sample_spec.channels = n_channels;
|
|
gst_structure_get_int (structure, "rate", &polypsink->sample_spec.rate);
|
|
|
|
pa_sample_spec_snprint (t, sizeof (t), &polypsink->sample_spec);
|
|
GST_DEBUG ("using format %s", t);
|
|
|
|
if (!pa_sample_spec_valid (&polypsink->sample_spec)) {
|
|
GST_DEBUG ("invalid format, refusing link");
|
|
return GST_PAD_LINK_REFUSED;
|
|
}
|
|
|
|
polypsink->negotiated = 1;
|
|
|
|
destroy_stream (polypsink);
|
|
|
|
state = gst_element_get_state (GST_ELEMENT (polypsink));
|
|
if (state == GST_STATE_PAUSED || state == GST_STATE_PLAYING)
|
|
create_stream (polypsink);
|
|
|
|
return GST_PAD_LINK_OK;
|
|
}
|
|
|
|
static GstCaps *
|
|
gst_polypsink_sink_fixate (GstPad * pad, const GstCaps * caps)
|
|
{
|
|
GstCaps *newcaps;
|
|
GstStructure *structure;
|
|
|
|
newcaps =
|
|
gst_caps_new_full (gst_structure_copy (gst_caps_get_structure (caps, 0)),
|
|
NULL);
|
|
structure = gst_caps_get_structure (newcaps, 0);
|
|
|
|
if (gst_structure_fixate_field_nearest_int (structure, "rate", 44100) ||
|
|
gst_structure_fixate_field_nearest_int (structure, "depth", 16) ||
|
|
gst_structure_fixate_field_nearest_int (structure, "width", 16) ||
|
|
gst_structure_fixate_field_nearest_int (structure, "channels", 2))
|
|
return newcaps;
|
|
|
|
gst_caps_free (newcaps);
|
|
return NULL;
|
|
}
|
|
|
|
static void
|
|
gst_polypsink_init (GTypeInstance * instance, gpointer g_class)
|
|
{
|
|
GstPolypSink *polypsink = GST_POLYPSINK (instance);
|
|
|
|
polypsink->sinkpad =
|
|
gst_pad_new_from_template (gst_element_class_get_pad_template
|
|
(GST_ELEMENT_GET_CLASS (instance), "sink"), "sink");
|
|
|
|
gst_element_add_pad (GST_ELEMENT (polypsink), polypsink->sinkpad);
|
|
gst_pad_set_chain_function (polypsink->sinkpad, gst_polypsink_chain);
|
|
gst_pad_set_link_function (polypsink->sinkpad, gst_polypsink_link);
|
|
gst_pad_set_fixate_function (polypsink->sinkpad, gst_polypsink_sink_fixate);
|
|
|
|
/* GST_OBJECT_FLAG_SET(polypsink, GST_ELEMENT_THREAD_SUGGESTED); */
|
|
GST_OBJECT_FLAG_SET (polypsink, GST_ELEMENT_EVENT_AWARE);
|
|
|
|
polypsink->context = NULL;
|
|
polypsink->stream = NULL;
|
|
|
|
polypsink->mainloop = pa_mainloop_new ();
|
|
g_assert (polypsink->mainloop);
|
|
|
|
polypsink->mainloop_api = pa_mainloop_get_api (polypsink->mainloop);
|
|
|
|
polypsink->sample_spec.rate = 0;
|
|
polypsink->sample_spec.channels = 0;
|
|
polypsink->sample_spec.format = 0;
|
|
|
|
polypsink->negotiated = 0;
|
|
|
|
polypsink->buffer = NULL;
|
|
polypsink->buffer_index = 0;
|
|
|
|
polypsink->latency = 0;
|
|
polypsink->counter = 0;
|
|
}
|
|
|
|
static void
|
|
gst_polypsink_dispose (GObject * object)
|
|
{
|
|
GstPolypSink *polypsink = GST_POLYPSINK (object);
|
|
|
|
/* gst_object_unparent(GST_OBJECT(polypsink->provided_clock)); */
|
|
|
|
|
|
destroy_context (polypsink);
|
|
|
|
if (polypsink->buffer)
|
|
gst_buffer_unref (polypsink->buffer);
|
|
|
|
|
|
g_free (polypsink->server);
|
|
g_free (polypsink->sink);
|
|
|
|
pa_mainloop_free (polypsink->mainloop);
|
|
|
|
G_OBJECT_CLASS (parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
gst_polypsink_class_init (gpointer g_class, gpointer class_data)
|
|
{
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (g_class);
|
|
GstElementClass *gstelement_class = GST_ELEMENT_CLASS (g_class);
|
|
|
|
parent_class = g_type_class_peek_parent (g_class);
|
|
|
|
g_object_class_install_property (gobject_class, ARG_SERVER,
|
|
g_param_spec_string ("server", "server", "server", NULL,
|
|
G_PARAM_READWRITE));
|
|
g_object_class_install_property (gobject_class, ARG_SINK,
|
|
g_param_spec_string ("sink", "sink", "sink", NULL, G_PARAM_READWRITE));
|
|
|
|
gobject_class->set_property = gst_polypsink_set_property;
|
|
gobject_class->get_property = gst_polypsink_get_property;
|
|
gobject_class->dispose = gst_polypsink_dispose;
|
|
|
|
gstelement_class->change_state = gst_polypsink_change_state;
|
|
/* gstelement_class->set_clock = gst_polypsink_set_clock; */
|
|
/* gstelement_class->get_clock = gst_polypsink_get_clock; */
|
|
}
|
|
|
|
|
|
gboolean
|
|
gst_polypsink_factory_init (GstPlugin * plugin)
|
|
{
|
|
return gst_element_register (plugin, "polypsink", GST_RANK_NONE,
|
|
GST_TYPE_POLYPSINK);
|
|
}
|
|
|
|
GType
|
|
gst_polypsink_get_type (void)
|
|
{
|
|
static GType polypsink_type = 0;
|
|
|
|
if (!polypsink_type) {
|
|
|
|
static const GTypeInfo polypsink_info = {
|
|
sizeof (GstPolypSinkClass),
|
|
gst_polypsink_base_init,
|
|
NULL,
|
|
gst_polypsink_class_init,
|
|
NULL,
|
|
NULL,
|
|
sizeof (GstPolypSink),
|
|
0,
|
|
gst_polypsink_init,
|
|
};
|
|
|
|
polypsink_type =
|
|
g_type_register_static (GST_TYPE_ELEMENT, "GstPolypSink",
|
|
&polypsink_info, 0);
|
|
}
|
|
|
|
return polypsink_type;
|
|
}
|