/* * Copyright (C) 2009 Ole André Vadla Ravnås <oleavr@soundrop.com> * * 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. */ /* FIXME 1.x: suppress warnings for deprecated API such as GStaticRecMutex * with newer GLib versions (>= 2.31.0) */ #define GLIB_DISABLE_DEPRECATION_WARNINGS #include "miovideosrc.h" #include "coremediabuffer.h" #include <gst/interfaces/propertyprobe.h> #include <gst/video/video.h> #include <CoreVideo/CVHostTime.h> #define DEFAULT_DEVICE_INDEX -1 #define FRAME_QUEUE_SIZE 2 #define FRAME_QUEUE_LOCK(instance) g_mutex_lock (instance->qlock) #define FRAME_QUEUE_UNLOCK(instance) g_mutex_unlock (instance->qlock) #define FRAME_QUEUE_WAIT(instance) \ g_cond_wait (instance->qcond, instance->qlock) #define FRAME_QUEUE_NOTIFY(instance) g_cond_signal (instance->qcond) #define GST_MIO_REQUIRED_APIS \ (GST_API_CORE_VIDEO | GST_API_CORE_MEDIA | GST_API_MIO) GST_DEBUG_CATEGORY (gst_mio_video_src_debug); #define GST_CAT_DEFAULT gst_mio_video_src_debug static GstStaticPadTemplate src_template = GST_STATIC_PAD_TEMPLATE ("src", GST_PAD_SRC, GST_PAD_ALWAYS, GST_STATIC_CAPS (GST_VIDEO_CAPS_YUV ("UYVY") ";" GST_VIDEO_CAPS_YUV ("YUY2") ";" "image/jpeg, " "width = " GST_VIDEO_SIZE_RANGE ", " "height = " GST_VIDEO_SIZE_RANGE ", " "framerate = " GST_VIDEO_FPS_RANGE ";") ); enum { PROP_0, PROP_DEVICE_UID, PROP_DEVICE_NAME, PROP_DEVICE_INDEX }; typedef gboolean (*GstMIOCallback) (GstMIOVideoSrc * self, gpointer data); #define GST_MIO_CALLBACK(cb) ((GstMIOCallback) (cb)) static gboolean gst_mio_video_src_open_device (GstMIOVideoSrc * self); static void gst_mio_video_src_close_device (GstMIOVideoSrc * self); static gboolean gst_mio_video_src_build_capture_graph_for (GstMIOVideoSrc * self, GstMIOVideoDevice * device); static TundraStatus gst_mio_video_src_configure_output_node (GstMIOVideoSrc * self, TundraGraph * graph, guint node_id); static void gst_mio_video_src_start_dispatcher (GstMIOVideoSrc * self); static void gst_mio_video_src_stop_dispatcher (GstMIOVideoSrc * self); static gpointer gst_mio_video_src_dispatcher_thread (gpointer data); static gboolean gst_mio_video_src_perform (GstMIOVideoSrc * self, GstMIOCallback cb, gpointer data); static gboolean gst_mio_video_src_perform_proxy (gpointer data); static void gst_mio_video_src_probe_interface_init (gpointer g_iface, gpointer iface_data); static void gst_mio_video_src_init_interfaces (GType type); GST_BOILERPLATE_FULL (GstMIOVideoSrc, gst_mio_video_src, GstPushSrc, GST_TYPE_PUSH_SRC, gst_mio_video_src_init_interfaces); static void gst_mio_video_src_init (GstMIOVideoSrc * self, GstMIOVideoSrcClass * gclass) { GstBaseSrc *base_src = GST_BASE_SRC_CAST (self); guint64 host_freq; gst_base_src_set_live (base_src, TRUE); gst_base_src_set_format (base_src, GST_FORMAT_TIME); host_freq = gst_gdouble_to_guint64 (CVGetHostClockFrequency ()); if (host_freq <= GST_SECOND) { self->cv_ratio_n = GST_SECOND / host_freq; self->cv_ratio_d = 1; } else { self->cv_ratio_n = 1; self->cv_ratio_d = host_freq / GST_SECOND; } self->queue = g_queue_new (); self->qlock = g_mutex_new (); self->qcond = g_cond_new (); } static void gst_mio_video_src_finalize (GObject * object) { GstMIOVideoSrc *self = GST_MIO_VIDEO_SRC_CAST (object); g_cond_free (self->qcond); g_mutex_free (self->qlock); g_queue_free (self->queue); G_OBJECT_CLASS (parent_class)->finalize (object); } static void gst_mio_video_src_get_property (GObject * object, guint prop_id, GValue * value, GParamSpec * pspec) { GstMIOVideoSrc *self = GST_MIO_VIDEO_SRC_CAST (object); switch (prop_id) { case PROP_DEVICE_UID: g_value_set_string (value, self->device_uid); break; case PROP_DEVICE_NAME: g_value_set_string (value, self->device_name); break; case PROP_DEVICE_INDEX: g_value_set_int (value, self->device_index); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void gst_mio_video_src_set_property (GObject * object, guint prop_id, const GValue * value, GParamSpec * pspec) { GstMIOVideoSrc *self = GST_MIO_VIDEO_SRC_CAST (object); switch (prop_id) { case PROP_DEVICE_UID: g_free (self->device_uid); self->device_uid = g_value_dup_string (value); break; case PROP_DEVICE_NAME: g_free (self->device_name); self->device_name = g_value_dup_string (value); break; case PROP_DEVICE_INDEX: self->device_index = g_value_get_int (value); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static GstStateChangeReturn gst_mio_video_src_change_state (GstElement * element, GstStateChange transition) { GstMIOVideoSrc *self = GST_MIO_VIDEO_SRC_CAST (element); GstStateChangeReturn ret; switch (transition) { case GST_STATE_CHANGE_NULL_TO_READY: gst_mio_video_src_start_dispatcher (self); if (!gst_mio_video_src_perform (self, GST_MIO_CALLBACK (gst_mio_video_src_open_device), NULL)) { goto open_failed; } break; default: break; } ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition); switch (transition) { case GST_STATE_CHANGE_READY_TO_NULL: gst_mio_video_src_perform (self, GST_MIO_CALLBACK (gst_mio_video_src_close_device), NULL); gst_mio_video_src_stop_dispatcher (self); break; default: break; } return ret; /* ERRORS */ open_failed: { gst_mio_video_src_stop_dispatcher (self); return GST_STATE_CHANGE_FAILURE; } } static GstCaps * gst_mio_video_src_get_caps (GstBaseSrc * basesrc) { GstMIOVideoSrc *self = GST_MIO_VIDEO_SRC_CAST (basesrc); GstCaps *result; if (self->device != NULL) { result = gst_caps_ref (gst_mio_video_device_get_available_caps (self->device)); } else { result = NULL; /* BaseSrc will return template caps */ } return result; } static gboolean gst_mio_video_src_do_set_caps (GstMIOVideoSrc * self, GstCaps * caps) { TundraStatus status; if (self->device == NULL) goto no_device; if (!gst_mio_video_device_set_caps (self->device, caps)) goto invalid_format; if (!gst_mio_video_src_build_capture_graph_for (self, self->device)) goto graph_build_error; status = self->ctx->mio->TundraGraphInitialize (self->graph); if (status != kTundraSuccess) goto graph_init_error; status = self->ctx->mio->TundraGraphStart (self->graph); if (status != kTundraSuccess) goto graph_start_error; return TRUE; /* ERRORS */ no_device: { GST_ELEMENT_ERROR (self, RESOURCE, FAILED, ("no device"), (NULL)); return FALSE; } invalid_format: { GST_ELEMENT_ERROR (self, RESOURCE, FAILED, ("invalid format"), (NULL)); return FALSE; } graph_build_error: { GST_ELEMENT_ERROR (self, RESOURCE, FAILED, ("failed to build capture graph"), (NULL)); return FALSE; } graph_init_error: { GST_ELEMENT_ERROR (self, RESOURCE, FAILED, ("failed to initialize capture graph: %08x", status), (NULL)); return FALSE; } graph_start_error: { GST_ELEMENT_ERROR (self, RESOURCE, FAILED, ("failed to start capture graph: %08x", status), (NULL)); return FALSE; } } static gboolean gst_mio_video_src_set_caps (GstBaseSrc * basesrc, GstCaps * caps) { GstMIOVideoSrc *self = GST_MIO_VIDEO_SRC_CAST (basesrc); { gchar *str; str = gst_caps_to_string (caps); GST_DEBUG_OBJECT (self, "caps: %s", str); g_free (str); } return gst_mio_video_src_perform (self, GST_MIO_CALLBACK (gst_mio_video_src_do_set_caps), caps); } static gboolean gst_mio_video_src_start (GstBaseSrc * basesrc) { GstMIOVideoSrc *self = GST_MIO_VIDEO_SRC_CAST (basesrc); self->running = TRUE; self->prev_offset = GST_BUFFER_OFFSET_NONE; self->prev_format = NULL; return TRUE; } static gboolean gst_mio_video_src_do_stop (GstMIOVideoSrc * self) { TundraStatus status; if (self->graph == NULL) goto nothing_to_stop; status = self->ctx->mio->TundraGraphStop (self->graph); if (status != kTundraSuccess) goto graph_failed_to_stop; while (!g_queue_is_empty (self->queue)) gst_buffer_unref (g_queue_pop_head (self->queue)); self->ctx->cm->FigFormatDescriptionRelease (self->prev_format); self->prev_format = NULL; return TRUE; nothing_to_stop: return TRUE; graph_failed_to_stop: GST_WARNING_OBJECT (self, "failed to stop capture graph: %d", status); return FALSE; } static gboolean gst_mio_video_src_stop (GstBaseSrc * basesrc) { GstMIOVideoSrc *self = GST_MIO_VIDEO_SRC_CAST (basesrc); return gst_mio_video_src_perform (self, GST_MIO_CALLBACK (gst_mio_video_src_do_stop), NULL); } static gboolean gst_mio_video_src_query (GstBaseSrc * basesrc, GstQuery * query) { GstMIOVideoSrc *self = GST_MIO_VIDEO_SRC_CAST (basesrc); gboolean result = FALSE; switch (GST_QUERY_TYPE (query)) { case GST_QUERY_LATENCY:{ GstClockTime min_latency, max_latency; if (self->device == NULL) goto beach; if (gst_mio_video_device_get_selected_format (self->device) == NULL) goto beach; min_latency = max_latency = gst_mio_video_device_get_duration (self->device); GST_DEBUG_OBJECT (self, "reporting latency of min %" GST_TIME_FORMAT " max %" GST_TIME_FORMAT, GST_TIME_ARGS (min_latency), GST_TIME_ARGS (max_latency)); gst_query_set_latency (query, TRUE, min_latency, max_latency); result = TRUE; break; } default: result = GST_BASE_SRC_CLASS (parent_class)->query (basesrc, query); break; } beach: return result; } static gboolean gst_mio_video_src_unlock (GstBaseSrc * basesrc) { GstMIOVideoSrc *self = GST_MIO_VIDEO_SRC_CAST (basesrc); FRAME_QUEUE_LOCK (self); self->running = FALSE; FRAME_QUEUE_NOTIFY (self); FRAME_QUEUE_UNLOCK (self); return TRUE; } static gboolean gst_mio_video_src_unlock_stop (GstBaseSrc * basesrc) { return TRUE; } static GstFlowReturn gst_mio_video_src_create (GstPushSrc * pushsrc, GstBuffer ** buf) { GstMIOVideoSrc *self = GST_MIO_VIDEO_SRC_CAST (pushsrc); GstCMApi *cm = self->ctx->cm; CMFormatDescriptionRef format; FRAME_QUEUE_LOCK (self); while (self->running && g_queue_is_empty (self->queue)) FRAME_QUEUE_WAIT (self); *buf = g_queue_pop_tail (self->queue); FRAME_QUEUE_UNLOCK (self); if (G_UNLIKELY (!self->running)) goto shutting_down; format = cm->CMSampleBufferGetFormatDescription (GST_CORE_MEDIA_BUFFER (*buf)->sample_buf); if (self->prev_format != NULL && !cm->CMFormatDescriptionEqual (format, self->prev_format)) { goto unexpected_format; } cm->FigFormatDescriptionRelease (self->prev_format); self->prev_format = cm->FigFormatDescriptionRetain (format); if (self->prev_offset == GST_BUFFER_OFFSET_NONE || GST_BUFFER_OFFSET (*buf) - self->prev_offset != 1) { GST_BUFFER_FLAG_SET (*buf, GST_BUFFER_FLAG_DISCONT); } self->prev_offset = GST_BUFFER_OFFSET (*buf); return GST_FLOW_OK; /* ERRORS */ shutting_down: { if (*buf != NULL) { gst_buffer_unref (*buf); *buf = NULL; } return GST_FLOW_FLUSHING; } unexpected_format: { GST_ELEMENT_ERROR (self, RESOURCE, READ, ("capture format changed unexpectedly"), ("another application likely reconfigured the device")); if (*buf != NULL) { gst_buffer_unref (*buf); *buf = NULL; } return GST_FLOW_ERROR; } } static gboolean gst_mio_video_src_open_device (GstMIOVideoSrc * self) { GError *error = NULL; GList *devices = NULL, *walk; guint device_idx; self->ctx = gst_core_media_ctx_new (GST_API_CORE_VIDEO | GST_API_CORE_MEDIA | GST_API_MIO, &error); if (error != NULL) goto api_error; devices = gst_mio_video_device_list_create (self->ctx); if (devices == NULL) goto no_devices; for (walk = devices, device_idx = 0; walk != NULL; walk = walk->next) { GstMIOVideoDevice *device = walk->data; gboolean match; if (self->device_uid != NULL) { match = g_ascii_strcasecmp (gst_mio_video_device_get_uid (device), self->device_uid) == 0; } else if (self->device_name != NULL) { match = g_ascii_strcasecmp (gst_mio_video_device_get_name (device), self->device_name) == 0; } else if (self->device_index >= 0) { match = device_idx == self->device_index; } else { match = TRUE; /* pick the first entry */ } if (self->device != NULL) match = FALSE; GST_DEBUG_OBJECT (self, "%c device[%u] = handle: %d name: '%s' uid: '%s'", (match) ? '*' : '-', device_idx, gst_mio_video_device_get_handle (device), gst_mio_video_device_get_name (device), gst_mio_video_device_get_uid (device)); /*gst_mio_video_device_print_debug_info (device); */ if (match) self->device = g_object_ref (device); device_idx++; } if (self->device == NULL) goto no_such_device; if (!gst_mio_video_device_open (self->device)) goto device_busy_or_gone; gst_mio_video_device_list_destroy (devices); return TRUE; /* ERRORS */ api_error: { GST_ELEMENT_ERROR (self, RESOURCE, FAILED, ("API error"), ("%s", error->message)); g_clear_error (&error); goto any_error; } no_devices: { GST_ELEMENT_ERROR (self, RESOURCE, NOT_FOUND, ("no video capture devices found"), (NULL)); goto any_error; } no_such_device: { GST_ELEMENT_ERROR (self, RESOURCE, NOT_FOUND, ("specified video capture device not found"), (NULL)); goto any_error; } device_busy_or_gone: { GST_ELEMENT_ERROR (self, RESOURCE, BUSY, ("failed to start capture (device already in use or gone)"), (NULL)); goto any_error; } any_error: { if (devices != NULL) { gst_mio_video_device_list_destroy (devices); } if (self->ctx != NULL) { g_object_unref (self->ctx); self->ctx = NULL; } return FALSE; } } static void gst_mio_video_src_close_device (GstMIOVideoSrc * self) { self->ctx->mio->TundraGraphUninitialize (self->graph); self->ctx->mio->TundraGraphRelease (self->graph); self->graph = NULL; gst_mio_video_device_close (self->device); g_object_unref (self->device); self->device = NULL; g_object_unref (self->ctx); self->ctx = NULL; } #define CHECK_TUNDRA_ERROR(fn) \ if (status != kTundraSuccess) { \ last_function_name = fn; \ goto tundra_error; \ } static gboolean gst_mio_video_src_build_capture_graph_for (GstMIOVideoSrc * self, GstMIOVideoDevice * device) { GstMIOApi *mio = self->ctx->mio; const gchar *last_function_name; TundraGraph *graph = NULL; TundraTargetSpec spec = { 0, }; TundraUnitID input_node = -1; gpointer input_info; TundraObjectID device_handle; TundraUnitID sync_node = -1; guint8 is_master; guint sync_direction; TundraUnitID output_node = -1; TundraStatus status; const gint node_id_input = 1; const gint node_id_sync = 22; const gint node_id_output = 16; /* * Graph */ status = mio->TundraGraphCreate (kCFAllocatorDefault, &graph); CHECK_TUNDRA_ERROR ("TundraGraphCreate"); /* * Node: input */ spec.name = kTundraUnitInput; spec.scope = kTundraScopeDAL; spec.vendor = kTundraVendorApple; status = mio->TundraGraphCreateNode (graph, node_id_input, 0, 0, &spec, 0, &input_node); CHECK_TUNDRA_ERROR ("TundraGraphCreateNode(input)"); /* store node info for setting clock provider */ input_info = NULL; status = mio->TundraGraphGetNodeInfo (graph, input_node, 0, 0, 0, 0, &input_info); CHECK_TUNDRA_ERROR ("TundraGraphGetNodeInfo(input)"); /* set device handle */ device_handle = gst_mio_video_device_get_handle (device); status = mio->TundraGraphSetProperty (graph, node_id_input, 0, kTundraInputPropertyDeviceID, 0, 0, &device_handle, sizeof (device_handle)); CHECK_TUNDRA_ERROR ("TundraGraphSetProperty(input, DeviceID)"); /* * Node: sync */ spec.name = kTundraUnitSync; spec.scope = kTundraScopeVSyn; status = mio->TundraGraphCreateNode (graph, node_id_sync, 0, 0, &spec, 0, &sync_node); CHECK_TUNDRA_ERROR ("TundraGraphCreateNode(sync)"); status = mio->TundraGraphSetProperty (graph, node_id_sync, 0, kTundraSyncPropertyClockProvider, 0, 0, &input_info, sizeof (input_info)); CHECK_TUNDRA_ERROR ("TundraGraphSetProperty(sync, ClockProvider)"); is_master = TRUE; status = mio->TundraGraphSetProperty (graph, node_id_sync, 0, kTundraSyncPropertyMasterSynchronizer, 0, 0, &is_master, sizeof (is_master)); CHECK_TUNDRA_ERROR ("TundraGraphSetProperty(sync, MasterSynchronizer)"); sync_direction = 0; status = mio->TundraGraphSetProperty (graph, node_id_sync, 0, kTundraSyncPropertySynchronizationDirection, 0, 0, &sync_direction, sizeof (sync_direction)); CHECK_TUNDRA_ERROR ("TundraGraphSetProperty(sync, SynchronizationDirection)"); /* * Node: output */ spec.name = kTundraUnitOutput; spec.scope = kTundraScope2PRC; status = mio->TundraGraphCreateNode (graph, node_id_output, 0, 0, &spec, 0, &output_node); CHECK_TUNDRA_ERROR ("TundraGraphCreateNode(output)"); status = gst_mio_video_src_configure_output_node (self, graph, node_id_output); CHECK_TUNDRA_ERROR ("TundraGraphSetProperty(output, Delegate)"); /* * Connect the nodes */ status = mio->TundraGraphConnectNodeInput (graph, input_node, 0, sync_node, 0); CHECK_TUNDRA_ERROR ("TundraGraphConnectNodeInput(input, sync)"); status = mio->TundraGraphConnectNodeInput (graph, sync_node, 0, output_node, 0); CHECK_TUNDRA_ERROR ("TundraGraphConnectNodeInput(sync, output)"); self->graph = graph; return TRUE; tundra_error: { GST_ELEMENT_ERROR (self, RESOURCE, FAILED, ("%s failed (status=%d)", last_function_name, (gint) status), (NULL)); goto any_error; } any_error: { mio->TundraGraphRelease (graph); return FALSE; } } static GstClockTime gst_mio_video_src_get_timestamp (GstMIOVideoSrc * self, CMSampleBufferRef sbuf) { GstClock *clock; GstClockTime base_time; GstClockTime timestamp; GST_OBJECT_LOCK (self); if ((clock = GST_ELEMENT_CLOCK (self)) != NULL) { gst_object_ref (clock); } base_time = GST_ELEMENT_CAST (self)->base_time; GST_OBJECT_UNLOCK (self); if (G_UNLIKELY (clock == NULL)) goto no_clock; timestamp = GST_CLOCK_TIME_NONE; /* * If the current clock is GstSystemClock, we know that it's using the * CoreAudio/CoreVideo clock. As such we may use the timestamp attached * to the CMSampleBuffer. */ if (G_TYPE_FROM_INSTANCE (clock) == GST_TYPE_SYSTEM_CLOCK) { CFNumberRef number; UInt64 ht; number = self->ctx->cm->CMGetAttachment (sbuf, *self->ctx->mio->kTundraSampleBufferAttachmentKey_HostTime, NULL); if (number != NULL && CFNumberGetValue (number, kCFNumberSInt64Type, &ht)) { timestamp = gst_util_uint64_scale_int (ht, self->cv_ratio_n, self->cv_ratio_d); } } if (!GST_CLOCK_TIME_IS_VALID (timestamp)) { timestamp = gst_clock_get_time (clock); } if (timestamp > base_time) timestamp -= base_time; else timestamp = 0; gst_object_unref (clock); return timestamp; no_clock: return GST_CLOCK_TIME_NONE; } static TundraStatus gst_mio_video_src_output_render (gpointer instance, gpointer unk1, gpointer unk2, gpointer unk3, CMSampleBufferRef sample_buf) { GstMIOVideoSrc *self = GST_MIO_VIDEO_SRC_CAST (instance); GstBuffer *buf; CFNumberRef number; UInt32 seq; buf = gst_core_media_buffer_new (self->ctx, sample_buf); if (G_UNLIKELY (buf == NULL)) goto buffer_creation_failed; number = self->ctx->cm->CMGetAttachment (sample_buf, *self->ctx->mio->kTundraSampleBufferAttachmentKey_SequenceNumber, NULL); if (number != NULL && CFNumberGetValue (number, kCFNumberSInt32Type, &seq)) { GST_BUFFER_OFFSET (buf) = seq; GST_BUFFER_OFFSET_END (buf) = seq + 1; } GST_BUFFER_TIMESTAMP (buf) = gst_mio_video_src_get_timestamp (self, sample_buf); if (GST_BUFFER_TIMESTAMP_IS_VALID (buf)) { GST_BUFFER_DURATION (buf) = gst_mio_video_device_get_duration (self->device); } FRAME_QUEUE_LOCK (self); if (g_queue_get_length (self->queue) == FRAME_QUEUE_SIZE) gst_buffer_unref (g_queue_pop_tail (self->queue)); g_queue_push_head (self->queue, buf); FRAME_QUEUE_NOTIFY (self); FRAME_QUEUE_UNLOCK (self); return kTundraSuccess; buffer_creation_failed: GST_WARNING_OBJECT (instance, "failed to create buffer"); return kTundraSuccess; } static TundraStatus gst_mio_video_src_output_initialize (gpointer instance) { GST_DEBUG_OBJECT (instance, "%s", G_STRFUNC); return kTundraSuccess; } static TundraStatus gst_mio_video_src_output_uninitialize (gpointer instance) { GST_DEBUG_OBJECT (instance, "%s", G_STRFUNC); return kTundraSuccess; } static TundraStatus gst_mio_video_src_output_start (gpointer instance) { GST_DEBUG_OBJECT (instance, "%s", G_STRFUNC); return kTundraSuccess; } static TundraStatus gst_mio_video_src_output_stop (gpointer instance) { return kTundraSuccess; } static TundraStatus gst_mio_video_src_output_reset (gpointer instance) { GST_DEBUG_OBJECT (instance, "%s", G_STRFUNC); return kTundraSuccess; } static TundraStatus gst_mio_video_src_output_deallocate (gpointer instance) { GST_DEBUG_OBJECT (instance, "%s", G_STRFUNC); return kTundraSuccess; } static gboolean gst_mio_video_src_output_can_render_now (gpointer instance, guint * unk) { if (unk != NULL) *unk = 0; return TRUE; } static CFArrayRef gst_mio_video_src_output_available_formats (gpointer instance, gboolean ensure_only) { GstMIOVideoSrc *self = GST_MIO_VIDEO_SRC (instance); CMFormatDescriptionRef format_desc; GST_DEBUG_OBJECT (self, "%s: ensure_only=%d", G_STRFUNC, ensure_only); if (ensure_only) return NULL; g_assert (self->device != NULL); format_desc = gst_mio_video_device_get_selected_format (self->device); g_assert (format_desc != NULL); return CFArrayCreate (kCFAllocatorDefault, (const void **) &format_desc, 1, &kCFTypeArrayCallBacks); } static TundraStatus gst_mio_video_src_output_copy_clock (gpointer instance) { GST_DEBUG_OBJECT (instance, "%s", G_STRFUNC); return kTundraSuccess; } static TundraStatus gst_mio_video_src_output_get_property_info (gpointer instance, guint prop_id) { GST_DEBUG_OBJECT (instance, "%s: prop_id=%u", G_STRFUNC, prop_id); if (prop_id == kTundraInputUnitProperty_SourcePath) return kTundraSuccess; return kTundraNotSupported; } static TundraStatus gst_mio_video_src_output_get_property (gpointer instance, guint prop_id) { GST_DEBUG_OBJECT (instance, "%s", G_STRFUNC); if (prop_id == kTundraInputUnitProperty_SourcePath) return kTundraSuccess; return kTundraNotSupported; } static TundraStatus gst_mio_video_src_output_set_property (gpointer instance, guint prop_id) { GST_DEBUG_OBJECT (instance, "%s: prop_id=%u", G_STRFUNC, prop_id); if (prop_id == kTundraInputUnitProperty_SourcePath) return kTundraSuccess; return kTundraNotSupported; } static TundraStatus gst_mio_video_src_configure_output_node (GstMIOVideoSrc * self, TundraGraph * graph, guint node_id) { TundraStatus status; TundraOutputDelegate d = { 0, }; d.unk1 = 2; d.instance = self; d.Render = gst_mio_video_src_output_render; d.Initialize = gst_mio_video_src_output_initialize; d.Uninitialize = gst_mio_video_src_output_uninitialize; d.Start = gst_mio_video_src_output_start; d.Stop = gst_mio_video_src_output_stop; d.Reset = gst_mio_video_src_output_reset; d.Deallocate = gst_mio_video_src_output_deallocate; d.CanRenderNow = gst_mio_video_src_output_can_render_now; d.AvailableFormats = gst_mio_video_src_output_available_formats; d.CopyClock = gst_mio_video_src_output_copy_clock; d.GetPropertyInfo = gst_mio_video_src_output_get_property_info; d.GetProperty = gst_mio_video_src_output_get_property; d.SetProperty = gst_mio_video_src_output_set_property; status = self->ctx->mio->TundraGraphSetProperty (graph, node_id, 0, kTundraOutputPropertyDelegate, 0, 0, &d, sizeof (d)); return status; } static void gst_mio_video_src_start_dispatcher (GstMIOVideoSrc * self) { g_assert (self->dispatcher_ctx == NULL && self->dispatcher_loop == NULL); g_assert (self->dispatcher_thread == NULL); self->dispatcher_ctx = g_main_context_new (); self->dispatcher_loop = g_main_loop_new (self->dispatcher_ctx, TRUE); self->dispatcher_thread = g_thread_create (gst_mio_video_src_dispatcher_thread, self, TRUE, NULL); } static void gst_mio_video_src_stop_dispatcher (GstMIOVideoSrc * self) { g_assert (self->dispatcher_ctx != NULL && self->dispatcher_loop != NULL); g_assert (self->dispatcher_thread != NULL); g_main_loop_quit (self->dispatcher_loop); g_thread_join (self->dispatcher_thread); self->dispatcher_thread = NULL; g_main_loop_unref (self->dispatcher_loop); self->dispatcher_loop = NULL; g_main_context_unref (self->dispatcher_ctx); self->dispatcher_ctx = NULL; } static gpointer gst_mio_video_src_dispatcher_thread (gpointer data) { GstMIOVideoSrc *self = data; g_main_loop_run (self->dispatcher_loop); return NULL; } typedef struct { GstMIOVideoSrc *self; GstMIOCallback callback; gpointer data; gboolean result; GMutex *mutex; GCond *cond; gboolean finished; } GstMIOPerformCtx; static gboolean gst_mio_video_src_perform (GstMIOVideoSrc * self, GstMIOCallback cb, gpointer data) { GstMIOPerformCtx ctx; GSource *source; ctx.self = self; ctx.callback = cb; ctx.data = data; ctx.result = FALSE; ctx.mutex = g_mutex_new (); ctx.cond = g_cond_new (); ctx.finished = FALSE; source = g_idle_source_new (); g_source_set_callback (source, gst_mio_video_src_perform_proxy, &ctx, NULL); g_source_attach (source, self->dispatcher_ctx); g_mutex_lock (ctx.mutex); while (!ctx.finished) g_cond_wait (ctx.cond, ctx.mutex); g_mutex_unlock (ctx.mutex); g_source_destroy (source); g_source_unref (source); g_cond_free (ctx.cond); g_mutex_free (ctx.mutex); return ctx.result; } static gboolean gst_mio_video_src_perform_proxy (gpointer data) { GstMIOPerformCtx *ctx = data; ctx->result = ctx->callback (ctx->self, ctx->data); g_mutex_lock (ctx->mutex); ctx->finished = TRUE; g_cond_signal (ctx->cond); g_mutex_unlock (ctx->mutex); return FALSE; } static const GList * gst_mio_video_src_probe_get_properties (GstPropertyProbe * probe) { static gsize init_value = 0; if (g_once_init_enter (&init_value)) { GObjectClass *klass; GList *props = NULL; klass = G_OBJECT_GET_CLASS (probe); props = g_list_append (props, g_object_class_find_property (klass, "device-uid")); props = g_list_append (props, g_object_class_find_property (klass, "device-name")); props = g_list_append (props, g_object_class_find_property (klass, "device-index")); g_once_init_leave (&init_value, GPOINTER_TO_SIZE (props)); } return GSIZE_TO_POINTER (init_value); } static GValueArray * gst_mio_video_src_probe_get_values (GstPropertyProbe * probe, guint prop_id, const GParamSpec * pspec) { GValueArray *values; GstCoreMediaCtx *ctx = NULL; GError *error = NULL; GList *devices = NULL, *walk; guint device_idx; values = g_value_array_new (3); ctx = gst_core_media_ctx_new (GST_MIO_REQUIRED_APIS, &error); if (error != NULL) goto beach; devices = gst_mio_video_device_list_create (ctx); if (devices == NULL) goto beach; for (walk = devices, device_idx = 0; walk != NULL; walk = walk->next) { GstMIOVideoDevice *device = walk->data; GValue value = { 0, }; switch (prop_id) { case PROP_DEVICE_UID: case PROP_DEVICE_NAME: { const gchar *str; if (prop_id == PROP_DEVICE_UID) str = gst_mio_video_device_get_uid (device); else str = gst_mio_video_device_get_name (device); g_value_init (&value, G_TYPE_STRING); g_value_set_string (&value, str); break; } case PROP_DEVICE_INDEX: { g_value_init (&value, G_TYPE_INT); g_value_set_int (&value, device_idx); break; } default: G_OBJECT_WARN_INVALID_PROPERTY_ID (probe, prop_id, pspec); goto beach; } g_value_array_append (values, &value); g_value_unset (&value); device_idx++; } beach: if (devices != NULL) gst_mio_video_device_list_destroy (devices); if (ctx != NULL) g_object_unref (ctx); g_clear_error (&error); return values; } static void gst_mio_video_src_base_init (gpointer gclass) { GstElementClass *element_class = GST_ELEMENT_CLASS (gclass); gst_element_class_set_metadata (element_class, "Video Source (MIO)", "Source/Video", "Reads frames from a Mac OS X MIO device", "Ole André Vadla Ravnås <oleavr@soundrop.com>"); gst_element_class_add_pad_template (element_class, gst_static_pad_template_get (&src_template)); } static void gst_mio_video_src_class_init (GstMIOVideoSrcClass * klass) { GObjectClass *gobject_class = G_OBJECT_CLASS (klass); GstElementClass *gstelement_class = GST_ELEMENT_CLASS (klass); GstBaseSrcClass *gstbasesrc_class = GST_BASE_SRC_CLASS (klass); GstPushSrcClass *gstpushsrc_class = GST_PUSH_SRC_CLASS (klass); gobject_class->finalize = gst_mio_video_src_finalize; gobject_class->get_property = gst_mio_video_src_get_property; gobject_class->set_property = gst_mio_video_src_set_property; gstelement_class->change_state = gst_mio_video_src_change_state; gstbasesrc_class->get_caps = gst_mio_video_src_get_caps; gstbasesrc_class->set_caps = gst_mio_video_src_set_caps; gstbasesrc_class->start = gst_mio_video_src_start; gstbasesrc_class->stop = gst_mio_video_src_stop; gstbasesrc_class->query = gst_mio_video_src_query; gstbasesrc_class->unlock = gst_mio_video_src_unlock; gstbasesrc_class->unlock_stop = gst_mio_video_src_unlock_stop; gstpushsrc_class->create = gst_mio_video_src_create; g_object_class_install_property (gobject_class, PROP_DEVICE_UID, g_param_spec_string ("device-uid", "Device UID", "Unique ID of the desired device", NULL, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); g_object_class_install_property (gobject_class, PROP_DEVICE_NAME, g_param_spec_string ("device-name", "Device Name", "Name of the desired device", NULL, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); g_object_class_install_property (gobject_class, PROP_DEVICE_INDEX, g_param_spec_int ("device-index", "Device Index", "Zero-based device index of the desired device", -1, G_MAXINT, DEFAULT_DEVICE_INDEX, G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_STATIC_STRINGS)); GST_DEBUG_CATEGORY_INIT (gst_mio_video_src_debug, "miovideosrc", 0, "Mac OS X CoreMedia video source"); } static void gst_mio_video_src_init_interfaces (GType type) { static const GInterfaceInfo probe_info = { gst_mio_video_src_probe_interface_init, NULL, NULL }; g_type_add_interface_static (type, GST_TYPE_PROPERTY_PROBE, &probe_info); } static void gst_mio_video_src_probe_interface_init (gpointer g_iface, gpointer iface_data) { GstPropertyProbeInterface *iface = g_iface; iface->get_properties = gst_mio_video_src_probe_get_properties; iface->get_values = gst_mio_video_src_probe_get_values; }