mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-12 10:25:33 +00:00
311 lines
8.7 KiB
C++
311 lines
8.7 KiB
C++
|
/* GStreamer
|
||
|
* Copyright (C) 2018 Joshua M. Doe <oss@nvl.army.mil>
|
||
|
*
|
||
|
* dshowdeviceprovider.cpp: DirectShow device probing
|
||
|
*
|
||
|
* 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.
|
||
|
*/
|
||
|
|
||
|
#ifdef HAVE_CONFIG_H
|
||
|
#include "config.h"
|
||
|
#endif
|
||
|
|
||
|
#include "gstdshow.h"
|
||
|
#include "gstdshowvideosrc.h"
|
||
|
#include "dshowdeviceprovider.h"
|
||
|
|
||
|
#include <gst/gst.h>
|
||
|
|
||
|
#include <windows.h>
|
||
|
|
||
|
GST_DEBUG_CATEGORY_EXTERN (dshowsrcwrapper_debug);
|
||
|
#define GST_CAT_DEFAULT dshowsrcwrapper_debug
|
||
|
|
||
|
|
||
|
static GstDevice *gst_dshow_device_new (guint id,
|
||
|
const gchar * device_name, GstCaps * caps, const gchar * device_path,
|
||
|
GstDshowDeviceType type);
|
||
|
|
||
|
G_DEFINE_TYPE (GstDshowDeviceProvider, gst_dshow_device_provider,
|
||
|
GST_TYPE_DEVICE_PROVIDER);
|
||
|
|
||
|
static void gst_dshow_device_provider_dispose (GObject * gobject);
|
||
|
|
||
|
static GList *gst_dshow_device_provider_probe (GstDeviceProvider * provider);
|
||
|
static gboolean gst_dshow_device_provider_start (GstDeviceProvider * provider);
|
||
|
static void gst_dshow_device_provider_stop (GstDeviceProvider * provider);
|
||
|
|
||
|
static void
|
||
|
gst_dshow_device_provider_class_init (GstDshowDeviceProviderClass * klass)
|
||
|
{
|
||
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
||
|
GstDeviceProviderClass *dm_class = GST_DEVICE_PROVIDER_CLASS (klass);
|
||
|
|
||
|
gobject_class->dispose = gst_dshow_device_provider_dispose;
|
||
|
|
||
|
dm_class->probe = gst_dshow_device_provider_probe;
|
||
|
dm_class->start = gst_dshow_device_provider_start;
|
||
|
dm_class->stop = gst_dshow_device_provider_stop;
|
||
|
|
||
|
gst_device_provider_class_set_static_metadata (dm_class,
|
||
|
"DirectShow Device Provider", "Source/Audio/Video",
|
||
|
"List and provide DirectShow source devices",
|
||
|
"Руслан Ижбулатов <lrn1986@gmail.com>");
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gst_dshow_device_provider_init (GstDshowDeviceProvider * self)
|
||
|
{
|
||
|
CoInitializeEx (NULL, COINIT_MULTITHREADED);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gst_dshow_device_provider_dispose (GObject * gobject)
|
||
|
{
|
||
|
CoUninitialize ();
|
||
|
}
|
||
|
|
||
|
static GstDevice *
|
||
|
new_video_source (const DshowDeviceEntry * info)
|
||
|
{
|
||
|
g_assert (info && info->device != NULL);
|
||
|
|
||
|
return gst_dshow_device_new (info->device_index, info->device_name,
|
||
|
info->caps, info->device, GST_DSHOW_DEVICE_TYPE_VIDEO_SOURCE);
|
||
|
}
|
||
|
|
||
|
static GList *
|
||
|
gst_dshow_device_provider_probe (GstDeviceProvider * provider)
|
||
|
{
|
||
|
/*GstDshowDeviceProvider *self = GST_DSHOW_DEVICE_PROVIDER (provider); */
|
||
|
GList *devices, *cur;
|
||
|
GList *result;
|
||
|
|
||
|
result = NULL;
|
||
|
|
||
|
devices = gst_dshow_enumerate_devices (&CLSID_VideoInputDeviceCategory, TRUE);
|
||
|
if (devices == NULL)
|
||
|
return result;
|
||
|
|
||
|
/* TODO: try and sort camera first like ksvideosrc? */
|
||
|
|
||
|
for (cur = devices; cur != NULL; cur = cur->next) {
|
||
|
GstDevice *source;
|
||
|
DshowDeviceEntry *entry = (DshowDeviceEntry *) cur->data;
|
||
|
|
||
|
source = new_video_source (entry);
|
||
|
if (source)
|
||
|
result = g_list_prepend (result, gst_object_ref_sink (source));
|
||
|
}
|
||
|
|
||
|
result = g_list_reverse (result);
|
||
|
|
||
|
gst_dshow_device_list_free (devices);
|
||
|
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
static gboolean
|
||
|
gst_dshow_device_provider_start (GstDeviceProvider * provider)
|
||
|
{
|
||
|
GList *devs;
|
||
|
GList *dev;
|
||
|
GstDshowDeviceProvider *self = GST_DSHOW_DEVICE_PROVIDER (provider);
|
||
|
|
||
|
devs = gst_dshow_device_provider_probe (provider);
|
||
|
for (dev = devs; dev; dev = dev->next) {
|
||
|
if (dev->data)
|
||
|
gst_device_provider_device_add (provider, (GstDevice *) dev->data);
|
||
|
}
|
||
|
g_list_free (devs);
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gst_dshow_device_provider_stop (GstDeviceProvider * provider)
|
||
|
{
|
||
|
|
||
|
}
|
||
|
|
||
|
enum
|
||
|
{
|
||
|
PROP_0,
|
||
|
PROP_DEVICE,
|
||
|
PROP_DEVICE_NAME,
|
||
|
PROP_DEVICE_INDEX
|
||
|
};
|
||
|
|
||
|
G_DEFINE_TYPE (GstDshowDevice, gst_dshow_device, GST_TYPE_DEVICE);
|
||
|
|
||
|
static void gst_dshow_device_get_property (GObject * object, guint prop_id,
|
||
|
GValue * value, GParamSpec * pspec);
|
||
|
static void gst_dshow_device_set_property (GObject * object, guint prop_id,
|
||
|
const GValue * value, GParamSpec * pspec);
|
||
|
static void gst_dshow_device_finalize (GObject * object);
|
||
|
static GstElement *gst_dshow_device_create_element (GstDevice * device,
|
||
|
const gchar * name);
|
||
|
|
||
|
static void
|
||
|
gst_dshow_device_class_init (GstDshowDeviceClass * klass)
|
||
|
{
|
||
|
GstDeviceClass *dev_class = GST_DEVICE_CLASS (klass);
|
||
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
||
|
|
||
|
dev_class->create_element = gst_dshow_device_create_element;
|
||
|
|
||
|
object_class->get_property = gst_dshow_device_get_property;
|
||
|
object_class->set_property = gst_dshow_device_set_property;
|
||
|
object_class->finalize = gst_dshow_device_finalize;
|
||
|
|
||
|
g_object_class_install_property (object_class, PROP_DEVICE,
|
||
|
g_param_spec_string ("device", "Device",
|
||
|
"DirectShow device path (@..classID/name)", "",
|
||
|
(GParamFlags) (G_PARAM_STATIC_STRINGS | G_PARAM_READWRITE |
|
||
|
G_PARAM_CONSTRUCT_ONLY)));
|
||
|
|
||
|
g_object_class_install_property (object_class, PROP_DEVICE_NAME,
|
||
|
g_param_spec_string ("device-name", "Device name",
|
||
|
"Human-readable name of the audio/video device", "",
|
||
|
(GParamFlags) (G_PARAM_STATIC_STRINGS | G_PARAM_READWRITE |
|
||
|
G_PARAM_CONSTRUCT_ONLY)));
|
||
|
|
||
|
g_object_class_install_property (object_class, PROP_DEVICE_INDEX,
|
||
|
g_param_spec_int ("device-index", "Device index",
|
||
|
"Index of the enumerated audio/video device", 0, G_MAXINT, 0,
|
||
|
(GParamFlags) (G_PARAM_STATIC_STRINGS | G_PARAM_READWRITE |
|
||
|
G_PARAM_CONSTRUCT_ONLY)));
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gst_dshow_device_init (GstDshowDevice * device)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gst_dshow_device_finalize (GObject * object)
|
||
|
{
|
||
|
GstDshowDevice *device = GST_DSHOW_DEVICE (object);
|
||
|
|
||
|
g_free (device->device);
|
||
|
g_free (device->device_name);
|
||
|
|
||
|
G_OBJECT_CLASS (gst_dshow_device_parent_class)->finalize (object);
|
||
|
}
|
||
|
|
||
|
static GstElement *
|
||
|
gst_dshow_device_create_element (GstDevice * device, const gchar * name)
|
||
|
{
|
||
|
GstDshowDevice *dev = GST_DSHOW_DEVICE (device);
|
||
|
GstElement *elem;
|
||
|
|
||
|
elem = gst_element_factory_make (dev->element, name);
|
||
|
g_object_set (elem, "device", dev->device, NULL);
|
||
|
g_object_set (elem, "device-name", dev->device_name, NULL);
|
||
|
|
||
|
return elem;
|
||
|
}
|
||
|
|
||
|
|
||
|
static GstDevice *
|
||
|
gst_dshow_device_new (guint device_index, const gchar * device_name,
|
||
|
GstCaps * caps, const gchar * device_path, GstDshowDeviceType type)
|
||
|
{
|
||
|
GstDshowDevice *gstdev;
|
||
|
const gchar *element = NULL;
|
||
|
const gchar *klass = NULL;
|
||
|
|
||
|
g_return_val_if_fail (device_name, NULL);
|
||
|
g_return_val_if_fail (device_path, NULL);
|
||
|
g_return_val_if_fail (caps, NULL);
|
||
|
|
||
|
switch (type) {
|
||
|
case GST_DSHOW_DEVICE_TYPE_VIDEO_SOURCE:
|
||
|
element = "dshowvideosrc";
|
||
|
klass = "Video/Source";
|
||
|
break;
|
||
|
case GST_DSHOW_DEVICE_TYPE_AUDIO_SOURCE:
|
||
|
element = "dshowaudiosrc";
|
||
|
klass = "Audio/Source";
|
||
|
break;
|
||
|
default:
|
||
|
g_assert_not_reached ();
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
/* set props of parent device class */
|
||
|
gstdev = (GstDshowDevice *) g_object_new (GST_TYPE_DSHOW_DEVICE,
|
||
|
"display-name", device_name, "caps", caps, "device-class", klass, NULL);
|
||
|
|
||
|
gstdev->type = type;
|
||
|
gstdev->device = g_strdup (device_path);
|
||
|
gstdev->device_name = g_strdup (device_name);
|
||
|
gstdev->device_index = device_index;
|
||
|
gstdev->element = element;
|
||
|
|
||
|
return GST_DEVICE (gstdev);
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
gst_dshow_device_get_property (GObject * object, guint prop_id,
|
||
|
GValue * value, GParamSpec * pspec)
|
||
|
{
|
||
|
GstDshowDevice *device;
|
||
|
|
||
|
device = GST_DSHOW_DEVICE_CAST (object);
|
||
|
|
||
|
switch (prop_id) {
|
||
|
case PROP_DEVICE:
|
||
|
g_value_set_string (value, device->device);
|
||
|
break;
|
||
|
case PROP_DEVICE_NAME:
|
||
|
g_value_set_string (value, device->device_name);
|
||
|
break;
|
||
|
case PROP_DEVICE_INDEX:
|
||
|
g_value_set_int (value, device->device_index);
|
||
|
break;
|
||
|
default:
|
||
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
gst_dshow_device_set_property (GObject * object, guint prop_id,
|
||
|
const GValue * value, GParamSpec * pspec)
|
||
|
{
|
||
|
GstDshowDevice *device;
|
||
|
|
||
|
device = GST_DSHOW_DEVICE_CAST (object);
|
||
|
|
||
|
switch (prop_id) {
|
||
|
case PROP_DEVICE:
|
||
|
device->device = g_value_dup_string (value);
|
||
|
break;
|
||
|
case PROP_DEVICE_NAME:
|
||
|
device->device_name = g_value_dup_string (value);
|
||
|
break;
|
||
|
case PROP_DEVICE_INDEX:
|
||
|
device->device_index = g_value_get_int (value);
|
||
|
break;
|
||
|
default:
|
||
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||
|
break;
|
||
|
}
|
||
|
}
|