mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-12 02:15:31 +00:00
0e4e351b83
Provides the following elements: qtkitvideosrc: OS X video source relying on the QTKit API. Comes with hard-coded caps as the API does not provide any way of querying for formats supported by the hardware. Hasn't been tested a lot, but seems to work. miovideosrc: OS X video source which uses the undocumented/private CoreMediaIOServices API, which is also the one used by iChat. Present on latest version of Leopard and all versions of Snow Leopard. Has been tested extensively with built-in cameras and TANDBERG's PrecisionHD USB camera. vtenc, vtdec: Generic codec wrappers which make use of the undocumented/ private VideoToolbox API on OS X and iOS. List of codecs are currently hard-coded to H.264 for vtenc, and H.264 + JPEG for vtdec. Can easily be expanded by adding new entries to the lists, but haven't yet had time to do that. Should probably also implement probing as available codecs depend on the OS and its version, and there doesn't seem to be any way to enumerate the available codecs. vth264decbin, vth264encbin: Wrapper bins to make it easier to use vtdec_h264/vtenc_h264 in live scenarios. iphonecamerasrc: iPhone camera source relying on the undocumented/private Celestial API. Tested on iOS 3.1 running on an iPhone 3GS. Stops working after a few minutes, presumably because of a resource leak. Needs some love. Note that the iOS parts haven't yet been ported to iOS 4.x.
846 lines
24 KiB
C
846 lines
24 KiB
C
/*
|
|
* Copyright (C) 2009 Ole André Vadla Ravnås <oravnas@cisco.com>
|
|
* 2009 Knut Inge Hvidsten <knut.inge.hvidsten@tandberg.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., 59 Temple Place - Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
#include "miovideodevice.h"
|
|
|
|
#include <gst/video/video.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
GST_DEBUG_CATEGORY_EXTERN (gst_mio_video_src_debug);
|
|
#define GST_CAT_DEFAULT gst_mio_video_src_debug
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_CONTEXT,
|
|
PROP_HANDLE,
|
|
PROP_UID,
|
|
PROP_NAME,
|
|
PROP_TRANSPORT
|
|
};
|
|
|
|
G_DEFINE_TYPE (GstMIOVideoDevice, gst_mio_video_device, G_TYPE_OBJECT);
|
|
|
|
typedef struct _GstMIOVideoFormat GstMIOVideoFormat;
|
|
typedef struct _GstMIOSetFormatCtx GstMIOSetFormatCtx;
|
|
typedef struct _GstMIOFindRateCtx GstMIOFindRateCtx;
|
|
|
|
struct _GstMIOVideoFormat
|
|
{
|
|
TundraObjectID stream;
|
|
FigFormatDescription *desc;
|
|
|
|
UInt32 type;
|
|
FigVideoDimensions dim;
|
|
};
|
|
|
|
struct _GstMIOSetFormatCtx
|
|
{
|
|
UInt32 format;
|
|
gint width, height;
|
|
gint fps_n, fps_d;
|
|
gboolean success;
|
|
};
|
|
|
|
struct _GstMIOFindRateCtx
|
|
{
|
|
gdouble needle;
|
|
gdouble closest_match;
|
|
gboolean success;
|
|
};
|
|
|
|
static void gst_mio_video_device_collect_format (GstMIOVideoDevice * self,
|
|
GstMIOVideoFormat * format, gpointer user_data);
|
|
static GstStructure *gst_mio_video_device_format_basics_to_structure
|
|
(GstMIOVideoDevice * self, GstMIOVideoFormat * format);
|
|
static gboolean gst_mio_video_device_add_framerates_to_structure
|
|
(GstMIOVideoDevice * self, GstMIOVideoFormat * format, GstStructure * s);
|
|
static void gst_mio_video_device_add_pixel_aspect_to_structure
|
|
(GstMIOVideoDevice * self, GstMIOVideoFormat * format, GstStructure * s);
|
|
|
|
static void gst_mio_video_device_append_framerate (GstMIOVideoDevice * self,
|
|
GstMIOVideoFormat * format, TundraFramerate * rate, gpointer user_data);
|
|
static void gst_mio_video_device_framerate_to_fraction_value
|
|
(TundraFramerate * rate, GValue * fract);
|
|
static gdouble gst_mio_video_device_round_to_whole_hundreths (gdouble value);
|
|
static void gst_mio_video_device_guess_pixel_aspect_ratio
|
|
(gint width, gint height, gint * par_width, gint * par_height);
|
|
|
|
static void gst_mio_video_device_activate_matching_format
|
|
(GstMIOVideoDevice * self, GstMIOVideoFormat * format, gpointer user_data);
|
|
static void gst_mio_video_device_find_closest_framerate
|
|
(GstMIOVideoDevice * self, GstMIOVideoFormat * format,
|
|
TundraFramerate * rate, gpointer user_data);
|
|
|
|
typedef void (*GstMIOVideoDeviceEachFormatFunc) (GstMIOVideoDevice * self,
|
|
GstMIOVideoFormat * format, gpointer user_data);
|
|
typedef void (*GstMIOVideoDeviceEachFramerateFunc) (GstMIOVideoDevice * self,
|
|
GstMIOVideoFormat * format, TundraFramerate * rate, gpointer user_data);
|
|
static void gst_mio_video_device_formats_foreach (GstMIOVideoDevice * self,
|
|
GstMIOVideoDeviceEachFormatFunc func, gpointer user_data);
|
|
static void gst_mio_video_device_format_framerates_foreach
|
|
(GstMIOVideoDevice * self, GstMIOVideoFormat * format,
|
|
GstMIOVideoDeviceEachFramerateFunc func, gpointer user_data);
|
|
|
|
static gint gst_mio_video_device_compare (GstMIOVideoDevice * a,
|
|
GstMIOVideoDevice * b);
|
|
static gint gst_mio_video_device_calculate_score (GstMIOVideoDevice * device);
|
|
|
|
static void
|
|
gst_mio_video_device_init (GstMIOVideoDevice * self)
|
|
{
|
|
}
|
|
|
|
static void
|
|
gst_mio_video_device_dispose (GObject * object)
|
|
{
|
|
GstMIOVideoDevice *self = GST_MIO_VIDEO_DEVICE_CAST (object);
|
|
|
|
if (self->cached_caps != NULL) {
|
|
gst_caps_unref (self->cached_caps);
|
|
self->cached_caps = NULL;
|
|
}
|
|
|
|
G_OBJECT_CLASS (gst_mio_video_device_parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
gst_mio_video_device_finalize (GObject * object)
|
|
{
|
|
GstMIOVideoDevice *self = GST_MIO_VIDEO_DEVICE_CAST (object);
|
|
|
|
g_free (self->cached_uid);
|
|
g_free (self->cached_name);
|
|
|
|
G_OBJECT_CLASS (gst_mio_video_device_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
gst_mio_video_device_get_property (GObject * object, guint prop_id,
|
|
GValue * value, GParamSpec * pspec)
|
|
{
|
|
GstMIOVideoDevice *self = GST_MIO_VIDEO_DEVICE (object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_CONTEXT:
|
|
g_value_set_pointer (value, self->ctx);
|
|
break;
|
|
case PROP_HANDLE:
|
|
g_value_set_int (value, gst_mio_video_device_get_handle (self));
|
|
break;
|
|
case PROP_UID:
|
|
g_value_set_string (value, gst_mio_video_device_get_uid (self));
|
|
break;
|
|
case PROP_NAME:
|
|
g_value_set_string (value, gst_mio_video_device_get_name (self));
|
|
break;
|
|
case PROP_TRANSPORT:
|
|
g_value_set_uint (value, gst_mio_video_device_get_transport_type (self));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_mio_video_device_set_property (GObject * object, guint prop_id,
|
|
const GValue * value, GParamSpec * pspec)
|
|
{
|
|
GstMIOVideoDevice *self = GST_MIO_VIDEO_DEVICE (object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_CONTEXT:
|
|
self->ctx = g_value_get_pointer (value);
|
|
break;
|
|
case PROP_HANDLE:
|
|
self->handle = g_value_get_int (value);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
TundraObjectID
|
|
gst_mio_video_device_get_handle (GstMIOVideoDevice * self)
|
|
{
|
|
return self->handle;
|
|
}
|
|
|
|
const gchar *
|
|
gst_mio_video_device_get_uid (GstMIOVideoDevice * self)
|
|
{
|
|
if (self->cached_uid == NULL) {
|
|
TundraTargetSpec pspec = { 0, };
|
|
|
|
pspec.name = kTundraObjectPropertyUID;
|
|
pspec.scope = kTundraScopeGlobal;
|
|
self->cached_uid =
|
|
gst_mio_object_get_string (self->handle, &pspec, self->ctx->mio);
|
|
}
|
|
|
|
return self->cached_uid;
|
|
}
|
|
|
|
const gchar *
|
|
gst_mio_video_device_get_name (GstMIOVideoDevice * self)
|
|
{
|
|
if (self->cached_name == NULL) {
|
|
TundraTargetSpec pspec = { 0, };
|
|
|
|
pspec.name = kTundraObjectPropertyName;
|
|
pspec.scope = kTundraScopeGlobal;
|
|
self->cached_name =
|
|
gst_mio_object_get_string (self->handle, &pspec, self->ctx->mio);
|
|
}
|
|
|
|
return self->cached_name;
|
|
}
|
|
|
|
TundraDeviceTransportType
|
|
gst_mio_video_device_get_transport_type (GstMIOVideoDevice * self)
|
|
{
|
|
if (self->cached_transport == kTundraDeviceTransportInvalid) {
|
|
TundraTargetSpec pspec = { 0, };
|
|
|
|
pspec.name = kTundraDevicePropertyTransportType;
|
|
pspec.scope = kTundraScopeGlobal;
|
|
self->cached_transport =
|
|
gst_mio_object_get_uint32 (self->handle, &pspec, self->ctx->mio);
|
|
}
|
|
|
|
return self->cached_transport;
|
|
}
|
|
|
|
gboolean
|
|
gst_mio_video_device_open (GstMIOVideoDevice * self)
|
|
{
|
|
/* nothing for now */
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
gst_mio_video_device_close (GstMIOVideoDevice * self)
|
|
{
|
|
/* nothing for now */
|
|
}
|
|
|
|
GstCaps *
|
|
gst_mio_video_device_get_available_caps (GstMIOVideoDevice * self)
|
|
{
|
|
if (self->cached_caps == NULL) {
|
|
GstCaps *caps;
|
|
|
|
caps = gst_caps_new_empty ();
|
|
gst_mio_video_device_formats_foreach (self,
|
|
gst_mio_video_device_collect_format, caps);
|
|
|
|
self->cached_caps = caps;
|
|
}
|
|
|
|
return self->cached_caps;
|
|
}
|
|
|
|
static void
|
|
gst_mio_video_device_collect_format (GstMIOVideoDevice * self,
|
|
GstMIOVideoFormat * format, gpointer user_data)
|
|
{
|
|
GstCaps *caps = user_data;
|
|
GstStructure *s;
|
|
|
|
s = gst_mio_video_device_format_basics_to_structure (self, format);
|
|
if (s == NULL)
|
|
goto unsupported_format;
|
|
|
|
if (!gst_mio_video_device_add_framerates_to_structure (self, format, s))
|
|
goto no_framerates;
|
|
|
|
gst_mio_video_device_add_pixel_aspect_to_structure (self, format, s);
|
|
|
|
gst_caps_append_structure (caps, s);
|
|
|
|
return;
|
|
|
|
/* ERRORS */
|
|
unsupported_format:
|
|
{
|
|
gchar *fcc;
|
|
|
|
fcc = gst_mio_fourcc_to_string (format->type);
|
|
GST_WARNING ("skipping unsupported format %s", fcc);
|
|
g_free (fcc);
|
|
|
|
return;
|
|
}
|
|
no_framerates:
|
|
{
|
|
GST_WARNING ("no framerates?");
|
|
|
|
gst_structure_free (s);
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
static GstStructure *
|
|
gst_mio_video_device_format_basics_to_structure (GstMIOVideoDevice * self,
|
|
GstMIOVideoFormat * format)
|
|
{
|
|
GstStructure *s;
|
|
|
|
switch (format->type) {
|
|
case kCVPixelFormatType_422YpCbCr8:
|
|
case kCVPixelFormatType_422YpCbCr8Deprecated:
|
|
{
|
|
guint fcc;
|
|
|
|
if (format->type == kCVPixelFormatType_422YpCbCr8)
|
|
fcc = GST_MAKE_FOURCC ('U', 'Y', 'V', 'Y');
|
|
else
|
|
fcc = GST_MAKE_FOURCC ('Y', 'U', 'Y', '2');
|
|
|
|
s = gst_structure_new ("video/x-raw-yuv",
|
|
"format", GST_TYPE_FOURCC, fcc,
|
|
"width", G_TYPE_INT, format->dim.width,
|
|
"height", G_TYPE_INT, format->dim.height, NULL);
|
|
break;
|
|
}
|
|
case kFigVideoCodecType_JPEG_OpenDML:
|
|
{
|
|
s = gst_structure_new ("image/jpeg",
|
|
"width", G_TYPE_INT, format->dim.width,
|
|
"height", G_TYPE_INT, format->dim.height, NULL);
|
|
break;
|
|
}
|
|
default:
|
|
s = NULL;
|
|
break;
|
|
}
|
|
|
|
return s;
|
|
}
|
|
|
|
static gboolean
|
|
gst_mio_video_device_add_framerates_to_structure (GstMIOVideoDevice * self,
|
|
GstMIOVideoFormat * format, GstStructure * s)
|
|
{
|
|
GValue rates = { 0, };
|
|
const GValue *rates_value;
|
|
|
|
g_value_init (&rates, GST_TYPE_LIST);
|
|
|
|
gst_mio_video_device_format_framerates_foreach (self, format,
|
|
gst_mio_video_device_append_framerate, &rates);
|
|
if (gst_value_list_get_size (&rates) == 0)
|
|
goto no_framerates;
|
|
|
|
if (gst_value_list_get_size (&rates) > 1)
|
|
rates_value = &rates;
|
|
else
|
|
rates_value = gst_value_list_get_value (&rates, 0);
|
|
gst_structure_set_value (s, "framerate", rates_value);
|
|
|
|
g_value_unset (&rates);
|
|
|
|
return TRUE;
|
|
|
|
/* ERRORS */
|
|
no_framerates:
|
|
{
|
|
g_value_unset (&rates);
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_mio_video_device_add_pixel_aspect_to_structure (GstMIOVideoDevice * self,
|
|
GstMIOVideoFormat * format, GstStructure * s)
|
|
{
|
|
gint par_width, par_height;
|
|
|
|
gst_mio_video_device_guess_pixel_aspect_ratio
|
|
(format->dim.width, format->dim.height, &par_width, &par_height);
|
|
|
|
gst_structure_set (s, "pixel-aspect-ratio",
|
|
GST_TYPE_FRACTION, par_width, par_height, NULL);
|
|
}
|
|
|
|
static void
|
|
gst_mio_video_device_append_framerate (GstMIOVideoDevice * self,
|
|
GstMIOVideoFormat * format, TundraFramerate * rate, gpointer user_data)
|
|
{
|
|
GValue *rates = user_data;
|
|
GValue value = { 0, };
|
|
|
|
g_value_init (&value, GST_TYPE_FRACTION);
|
|
gst_mio_video_device_framerate_to_fraction_value (rate, &value);
|
|
gst_value_list_append_value (rates, &value);
|
|
g_value_unset (&value);
|
|
}
|
|
|
|
static void
|
|
gst_mio_video_device_framerate_to_fraction_value (TundraFramerate * rate,
|
|
GValue * fract)
|
|
{
|
|
gdouble rounded;
|
|
gint n, d;
|
|
|
|
rounded = gst_mio_video_device_round_to_whole_hundreths (rate->value);
|
|
gst_util_double_to_fraction (rounded, &n, &d);
|
|
gst_value_set_fraction (fract, n, d);
|
|
}
|
|
|
|
static gdouble
|
|
gst_mio_video_device_round_to_whole_hundreths (gdouble value)
|
|
{
|
|
gdouble m, x, y, z;
|
|
|
|
m = 0.01;
|
|
x = value;
|
|
y = floor ((x / m) + 0.5);
|
|
z = y * m;
|
|
|
|
return z;
|
|
}
|
|
|
|
static void
|
|
gst_mio_video_device_guess_pixel_aspect_ratio (gint width, gint height,
|
|
gint * par_width, gint * par_height)
|
|
{
|
|
/*
|
|
* As we dont have access to the actual pixel aspect, we will try to do a
|
|
* best-effort guess. The guess is based on most sensors being either 4/3
|
|
* or 16/9, and most pixel aspects being close to 1/1.
|
|
*/
|
|
|
|
if (width == 768 && height == 448) { /* special case for w448p */
|
|
*par_width = 28;
|
|
*par_height = 27;
|
|
} else {
|
|
if (((gdouble) width / (gdouble) height) < 1.2778) {
|
|
*par_width = 12;
|
|
*par_height = 11;
|
|
} else {
|
|
*par_width = 1;
|
|
*par_height = 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
gboolean
|
|
gst_mio_video_device_set_caps (GstMIOVideoDevice * self, GstCaps * caps)
|
|
{
|
|
GstVideoFormat format;
|
|
GstMIOSetFormatCtx ctx = { 0, };
|
|
|
|
if (gst_video_format_parse_caps (caps, &format, &ctx.width, &ctx.height)) {
|
|
if (format == GST_VIDEO_FORMAT_UYVY)
|
|
ctx.format = kCVPixelFormatType_422YpCbCr8;
|
|
else if (format == GST_VIDEO_FORMAT_YUY2)
|
|
ctx.format = kCVPixelFormatType_422YpCbCr8Deprecated;
|
|
else
|
|
g_assert_not_reached ();
|
|
} else {
|
|
GstStructure *s;
|
|
|
|
s = gst_caps_get_structure (caps, 0);
|
|
g_assert (gst_structure_has_name (s, "image/jpeg"));
|
|
gst_structure_get_int (s, "width", &ctx.width);
|
|
gst_structure_get_int (s, "height", &ctx.height);
|
|
|
|
ctx.format = kFigVideoCodecType_JPEG_OpenDML;
|
|
}
|
|
|
|
gst_video_parse_caps_framerate (caps, &ctx.fps_n, &ctx.fps_d);
|
|
|
|
gst_mio_video_device_formats_foreach (self,
|
|
gst_mio_video_device_activate_matching_format, &ctx);
|
|
|
|
return ctx.success;
|
|
}
|
|
|
|
static void
|
|
gst_mio_video_device_activate_matching_format (GstMIOVideoDevice * self,
|
|
GstMIOVideoFormat * format, gpointer user_data)
|
|
{
|
|
GstMIOSetFormatCtx *ctx = user_data;
|
|
GstMIOFindRateCtx find_ctx;
|
|
TundraTargetSpec spec = { 0, };
|
|
TundraStatus status;
|
|
|
|
if (format->type != ctx->format)
|
|
return;
|
|
else if (format->dim.width != ctx->width)
|
|
return;
|
|
else if (format->dim.height != ctx->height)
|
|
return;
|
|
|
|
find_ctx.needle = (gdouble) ctx->fps_n / (gdouble) ctx->fps_d;
|
|
find_ctx.closest_match = 0.0;
|
|
find_ctx.success = FALSE;
|
|
gst_mio_video_device_format_framerates_foreach (self, format,
|
|
gst_mio_video_device_find_closest_framerate, &find_ctx);
|
|
if (!find_ctx.success)
|
|
goto no_matching_framerate_found;
|
|
|
|
spec.scope = kTundraScopeInput;
|
|
|
|
spec.name = kTundraStreamPropertyFormatDescription;
|
|
status = self->ctx->mio->TundraObjectSetPropertyData (format->stream, &spec,
|
|
NULL, NULL, sizeof (format->desc), &format->desc);
|
|
if (status != kTundraSuccess)
|
|
goto failed_to_set_format;
|
|
|
|
spec.name = kTundraStreamPropertyFrameRate;
|
|
status = self->ctx->mio->TundraObjectSetPropertyData (format->stream, &spec,
|
|
NULL, NULL, sizeof (find_ctx.closest_match), &find_ctx.closest_match);
|
|
if (status != kTundraSuccess)
|
|
goto failed_to_set_framerate;
|
|
|
|
self->selected_format = format->desc;
|
|
self->selected_fps_n = ctx->fps_n;
|
|
self->selected_fps_d = ctx->fps_d;
|
|
|
|
ctx->success = TRUE;
|
|
return;
|
|
|
|
/* ERRORS */
|
|
no_matching_framerate_found:
|
|
{
|
|
GST_ERROR ("no matching framerate found");
|
|
return;
|
|
}
|
|
failed_to_set_format:
|
|
{
|
|
GST_ERROR ("failed to set format: 0x%08x", status);
|
|
return;
|
|
}
|
|
failed_to_set_framerate:
|
|
{
|
|
GST_ERROR ("failed to set framerate: 0x%08x", status);
|
|
return;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_mio_video_device_find_closest_framerate (GstMIOVideoDevice * self,
|
|
GstMIOVideoFormat * format, TundraFramerate * rate, gpointer user_data)
|
|
{
|
|
GstMIOFindRateCtx *ctx = user_data;
|
|
|
|
if (fabs (rate->value - ctx->needle) <= 0.1) {
|
|
ctx->closest_match = rate->value;
|
|
ctx->success = TRUE;
|
|
}
|
|
}
|
|
|
|
FigFormatDescription *
|
|
gst_mio_video_device_get_selected_format (GstMIOVideoDevice * self)
|
|
{
|
|
return self->selected_format;
|
|
}
|
|
|
|
GstClockTime
|
|
gst_mio_video_device_get_duration (GstMIOVideoDevice * self)
|
|
{
|
|
return gst_util_uint64_scale_int (GST_SECOND,
|
|
self->selected_fps_d, self->selected_fps_n);
|
|
}
|
|
|
|
static void
|
|
gst_mio_video_device_formats_foreach (GstMIOVideoDevice * self,
|
|
GstMIOVideoDeviceEachFormatFunc func, gpointer user_data)
|
|
{
|
|
GstCMApi *cm = self->ctx->cm;
|
|
GstMIOApi *mio = self->ctx->mio;
|
|
TundraTargetSpec spec = { 0, };
|
|
GArray *streams;
|
|
guint stream_idx;
|
|
|
|
spec.name = kTundraDevicePropertyStreams;
|
|
spec.scope = kTundraScopeInput;
|
|
streams = gst_mio_object_get_array (self->handle, &spec,
|
|
sizeof (TundraObjectID), mio);
|
|
|
|
/* TODO: We only consider the first stream for now */
|
|
for (stream_idx = 0; stream_idx != MIN (streams->len, 1); stream_idx++) {
|
|
TundraObjectID stream;
|
|
CFArrayRef formats;
|
|
CFIndex num_formats, fmt_idx;
|
|
|
|
stream = g_array_index (streams, TundraObjectID, stream_idx);
|
|
|
|
spec.name = kTundraStreamPropertyFormatDescriptions;
|
|
spec.scope = kTundraScopeInput;
|
|
|
|
formats = gst_mio_object_get_pointer (stream, &spec, mio);
|
|
num_formats = CFArrayGetCount (formats);
|
|
|
|
for (fmt_idx = 0; fmt_idx != num_formats; fmt_idx++) {
|
|
GstMIOVideoFormat fmt;
|
|
|
|
fmt.stream = stream;
|
|
fmt.desc = (FigFormatDescription *)
|
|
CFArrayGetValueAtIndex (formats, fmt_idx);
|
|
if (cm->FigFormatDescriptionGetMediaType (fmt.desc) != kFigMediaTypeVideo)
|
|
continue;
|
|
fmt.type = cm->FigFormatDescriptionGetMediaSubType (fmt.desc);
|
|
fmt.dim = cm->FigVideoFormatDescriptionGetDimensions (fmt.desc);
|
|
|
|
func (self, &fmt, user_data);
|
|
}
|
|
}
|
|
|
|
g_array_free (streams, TRUE);
|
|
}
|
|
|
|
static void
|
|
gst_mio_video_device_format_framerates_foreach (GstMIOVideoDevice * self,
|
|
GstMIOVideoFormat * format, GstMIOVideoDeviceEachFramerateFunc func,
|
|
gpointer user_data)
|
|
{
|
|
TundraTargetSpec spec = { 0, };
|
|
GArray *rates;
|
|
guint rate_idx;
|
|
|
|
spec.name = kTundraStreamPropertyFrameRates;
|
|
spec.scope = kTundraScopeInput;
|
|
rates = gst_mio_object_get_array_full (format->stream, &spec,
|
|
sizeof (format->desc), &format->desc, sizeof (TundraFramerate),
|
|
self->ctx->mio);
|
|
|
|
for (rate_idx = 0; rate_idx != rates->len; rate_idx++) {
|
|
TundraFramerate *rate;
|
|
|
|
rate = &g_array_index (rates, TundraFramerate, rate_idx);
|
|
|
|
func (self, format, rate, user_data);
|
|
}
|
|
|
|
g_array_free (rates, TRUE);
|
|
}
|
|
|
|
void
|
|
gst_mio_video_device_print_debug_info (GstMIOVideoDevice * self)
|
|
{
|
|
GstCMApi *cm = self->ctx->cm;
|
|
GstMIOApi *mio = self->ctx->mio;
|
|
TundraTargetSpec spec = { 0, };
|
|
gchar *str;
|
|
GArray *streams;
|
|
guint stream_idx;
|
|
|
|
g_print ("Device %p with handle %d\n", self, self->handle);
|
|
|
|
spec.scope = kTundraScopeGlobal;
|
|
|
|
spec.name = kTundraObjectPropertyClass;
|
|
str = gst_mio_object_get_fourcc (self->handle, &spec, mio);
|
|
g_print (" Class: '%s'\n", str);
|
|
g_free (str);
|
|
|
|
spec.name = kTundraObjectPropertyCreator;
|
|
str = gst_mio_object_get_string (self->handle, &spec, mio);
|
|
g_print (" Creator: \"%s\"\n", str);
|
|
g_free (str);
|
|
|
|
spec.name = kTundraDevicePropertyModelUID;
|
|
str = gst_mio_object_get_string (self->handle, &spec, mio);
|
|
g_print (" Model UID: \"%s\"\n", str);
|
|
g_free (str);
|
|
|
|
spec.name = kTundraDevicePropertyTransportType;
|
|
str = gst_mio_object_get_fourcc (self->handle, &spec, mio);
|
|
g_print (" Transport Type: '%s'\n", str);
|
|
g_free (str);
|
|
|
|
g_print (" Streams:\n");
|
|
spec.name = kTundraDevicePropertyStreams;
|
|
spec.scope = kTundraScopeInput;
|
|
streams = gst_mio_object_get_array (self->handle, &spec,
|
|
sizeof (TundraObjectID), mio);
|
|
for (stream_idx = 0; stream_idx != streams->len; stream_idx++) {
|
|
TundraObjectID stream;
|
|
CFArrayRef formats;
|
|
CFIndex num_formats, fmt_idx;
|
|
|
|
stream = g_array_index (streams, TundraObjectID, stream_idx);
|
|
|
|
g_print (" stream[%u] = %d\n", stream_idx, stream);
|
|
|
|
spec.scope = kTundraScopeInput;
|
|
spec.name = kTundraStreamPropertyFormatDescriptions;
|
|
|
|
formats = gst_mio_object_get_pointer (stream, &spec, mio);
|
|
num_formats = CFArrayGetCount (formats);
|
|
|
|
g_print (" <%u formats>\n", (guint) num_formats);
|
|
|
|
for (fmt_idx = 0; fmt_idx != num_formats; fmt_idx++) {
|
|
const FigFormatDescription *fmt;
|
|
gchar *media_type;
|
|
gchar *media_sub_type;
|
|
FigVideoDimensions dim;
|
|
GArray *rates;
|
|
guint rate_idx;
|
|
|
|
fmt = CFArrayGetValueAtIndex (formats, fmt_idx);
|
|
media_type = gst_mio_fourcc_to_string
|
|
(cm->FigFormatDescriptionGetMediaType (fmt));
|
|
media_sub_type = gst_mio_fourcc_to_string
|
|
(cm->FigFormatDescriptionGetMediaSubType (fmt));
|
|
dim = cm->FigVideoFormatDescriptionGetDimensions (fmt);
|
|
|
|
g_print (" format[%u]: MediaType='%s' MediaSubType='%s' %ux%u\n",
|
|
(guint) fmt_idx, media_type, media_sub_type,
|
|
(guint) dim.width, (guint) dim.height);
|
|
|
|
spec.name = kTundraStreamPropertyFrameRates;
|
|
rates = gst_mio_object_get_array_full (stream, &spec, sizeof (fmt), &fmt,
|
|
sizeof (TundraFramerate), mio);
|
|
for (rate_idx = 0; rate_idx != rates->len; rate_idx++) {
|
|
TundraFramerate *rate;
|
|
|
|
rate = &g_array_index (rates, TundraFramerate, rate_idx);
|
|
g_print (" %f\n", rate->value);
|
|
}
|
|
g_array_free (rates, TRUE);
|
|
|
|
g_free (media_sub_type);
|
|
g_free (media_type);
|
|
}
|
|
}
|
|
|
|
g_array_free (streams, TRUE);
|
|
}
|
|
|
|
static void
|
|
gst_mio_video_device_class_init (GstMIOVideoDeviceClass * klass)
|
|
{
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
|
|
gobject_class->dispose = gst_mio_video_device_dispose;
|
|
gobject_class->finalize = gst_mio_video_device_finalize;
|
|
gobject_class->get_property = gst_mio_video_device_get_property;
|
|
gobject_class->set_property = gst_mio_video_device_set_property;
|
|
|
|
g_object_class_install_property (gobject_class, PROP_CONTEXT,
|
|
g_param_spec_pointer ("context", "CoreMedia Context",
|
|
"CoreMedia context to use",
|
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS));
|
|
g_object_class_install_property (gobject_class, PROP_HANDLE,
|
|
g_param_spec_int ("handle", "Handle",
|
|
"MIO handle of this video capture device",
|
|
G_MININT, G_MAXINT, -1,
|
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS));
|
|
g_object_class_install_property (gobject_class, PROP_UID,
|
|
g_param_spec_string ("uid", "Unique ID",
|
|
"Unique ID of this video capture device", NULL,
|
|
G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
|
g_object_class_install_property (gobject_class, PROP_NAME,
|
|
g_param_spec_string ("name", "Device Name",
|
|
"Name of this video capture device", NULL,
|
|
G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
|
g_object_class_install_property (gobject_class, PROP_TRANSPORT,
|
|
g_param_spec_uint ("transport", "Transport",
|
|
"Transport type of this video capture device",
|
|
0, G_MAXUINT, 0, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
|
}
|
|
|
|
GList *
|
|
gst_mio_video_device_list_create (GstCoreMediaCtx * ctx)
|
|
{
|
|
GList *devices = NULL;
|
|
TundraTargetSpec pspec = { 0, };
|
|
GArray *handles;
|
|
guint handle_idx;
|
|
|
|
pspec.name = kTundraSystemPropertyDevices;
|
|
pspec.scope = kTundraScopeGlobal;
|
|
handles = gst_mio_object_get_array (TUNDRA_SYSTEM_OBJECT_ID, &pspec,
|
|
sizeof (TundraObjectID), ctx->mio);
|
|
if (handles == NULL)
|
|
goto beach;
|
|
|
|
for (handle_idx = 0; handle_idx != handles->len; handle_idx++) {
|
|
TundraObjectID handle;
|
|
GstMIOVideoDevice *device;
|
|
|
|
handle = g_array_index (handles, TundraObjectID, handle_idx);
|
|
device = g_object_new (GST_TYPE_MIO_VIDEO_DEVICE,
|
|
"context", ctx, "handle", handle, NULL);
|
|
|
|
/* TODO: Skip screen input devices for now */
|
|
if (gst_mio_video_device_get_transport_type (device) !=
|
|
kTundraDeviceTransportScreen) {
|
|
devices = g_list_prepend (devices, device);
|
|
} else {
|
|
g_object_unref (device);
|
|
}
|
|
}
|
|
|
|
devices = g_list_sort (devices, (GCompareFunc) gst_mio_video_device_compare);
|
|
|
|
g_array_free (handles, TRUE);
|
|
|
|
beach:
|
|
return devices;
|
|
}
|
|
|
|
void
|
|
gst_mio_video_device_list_destroy (GList * devices)
|
|
{
|
|
g_list_foreach (devices, (GFunc) g_object_unref, NULL);
|
|
g_list_free (devices);
|
|
}
|
|
|
|
static gint
|
|
gst_mio_video_device_compare (GstMIOVideoDevice * a, GstMIOVideoDevice * b)
|
|
{
|
|
gint score_a, score_b;
|
|
|
|
score_a = gst_mio_video_device_calculate_score (a);
|
|
score_b = gst_mio_video_device_calculate_score (b);
|
|
|
|
if (score_a > score_b)
|
|
return -1;
|
|
else if (score_a < score_b)
|
|
return 1;
|
|
|
|
return g_ascii_strcasecmp (gst_mio_video_device_get_name (a),
|
|
gst_mio_video_device_get_name (b));
|
|
}
|
|
|
|
static gint
|
|
gst_mio_video_device_calculate_score (GstMIOVideoDevice * device)
|
|
{
|
|
switch (gst_mio_video_device_get_transport_type (device)) {
|
|
case kTundraDeviceTransportScreen:
|
|
return 0;
|
|
case kTundraDeviceTransportBuiltin:
|
|
return 1;
|
|
case kTundraDeviceTransportUSB:
|
|
return 2;
|
|
default:
|
|
return 3;
|
|
}
|
|
}
|