• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2008 Ole André Vadla Ravnås <ole.andre.ravnas@tandberg.com>
3  *
4  * This library is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU Library General Public
6  * License as published by the Free Software Foundation; either
7  * version 2 of the License, or (at your option) any later version.
8  *
9  * This library is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * Library General Public License for more details.
13  *
14  * You should have received a copy of the GNU Library General Public
15  * License along with this library; if not, write to the
16  * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
17  * Boston, MA 02110-1301, USA.
18  */
19 
20 #include "gstksvideodevice.h"
21 
22 #include "gstksclock.h"
23 #include "kshelpers.h"
24 #include "ksvideohelpers.h"
25 
26 #define READ_TIMEOUT           (10 * 1000)
27 #define MJPEG_MAX_PADDING      128
28 #define MAX_OUTSTANDING_FRAMES 128
29 
30 #define KS_BUFFER_ALIGNMENT    4096
31 
32 #define DEFAULT_DEVICE_PATH NULL
33 
34 GST_DEBUG_CATEGORY_EXTERN (gst_ks_debug);
35 #define GST_CAT_DEFAULT gst_ks_debug
36 
37 #define GST_DEBUG_IS_ENABLED()  \
38     (gst_debug_category_get_threshold (GST_CAT_DEFAULT) >= GST_LEVEL_DEBUG)
39 #define UNREF_BUFFER(b)         \
40   G_STMT_START {                \
41     if (*(b) != NULL) {         \
42       gst_buffer_unref (*(b));  \
43       *(b) = NULL;              \
44     }                           \
45   } G_STMT_END
46 
47 enum
48 {
49   PROP_0,
50   PROP_CLOCK,
51   PROP_DEVICE_PATH,
52 };
53 
54 typedef struct
55 {
56   KSSTREAM_HEADER header;
57   KS_FRAME_INFO frame_info;
58 } KSSTREAM_READ_PARAMS;
59 
60 typedef struct
61 {
62   KSSTREAM_READ_PARAMS params;
63   GstBuffer *buf;
64   OVERLAPPED overlapped;
65 } ReadRequest;
66 
67 struct _GstKsVideoDevicePrivate
68 {
69   gboolean open;
70   KSSTATE state;
71 
72   GstKsClock *clock;
73   gchar *dev_path;
74   HANDLE filter_handle;
75   GList *media_types;
76   GstCaps *cached_caps;
77   HANDLE cancel_event;
78 
79   KsVideoMediaType *cur_media_type;
80   GstCaps *cur_fixed_caps;
81   guint width;
82   guint height;
83   guint fps_n;
84   guint fps_d;
85   guint8 *rgb_swap_buf;
86   gboolean is_muxed;
87 
88   HANDLE pin_handle;
89 
90   gboolean requests_submitted;
91   gulong num_requests;
92   GArray *requests;
93   GArray *request_events;
94   GstBuffer *spare_buffers[2];
95   GstClockTime last_timestamp;
96 };
97 
98 #define GST_KS_VIDEO_DEVICE_GET_PRIVATE(o) ((o)->priv)
99 
100 static void gst_ks_video_device_dispose (GObject * object);
101 static void gst_ks_video_device_get_property (GObject * object, guint prop_id,
102     GValue * value, GParamSpec * pspec);
103 static void gst_ks_video_device_set_property (GObject * object, guint prop_id,
104     const GValue * value, GParamSpec * pspec);
105 
106 static void gst_ks_video_device_reset_caps (GstKsVideoDevice * self);
107 static guint gst_ks_video_device_get_frame_size (GstKsVideoDevice * self);
108 
109 G_DEFINE_TYPE_WITH_PRIVATE (GstKsVideoDevice, gst_ks_video_device,
110     G_TYPE_OBJECT);
111 
112 static GstKsVideoDeviceClass *parent_class = NULL;
113 
114 static void
gst_ks_video_device_class_init(GstKsVideoDeviceClass * klass)115 gst_ks_video_device_class_init (GstKsVideoDeviceClass * klass)
116 {
117   GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
118 
119   parent_class = g_type_class_peek_parent (klass);
120 
121   gobject_class->dispose = gst_ks_video_device_dispose;
122   gobject_class->get_property = gst_ks_video_device_get_property;
123   gobject_class->set_property = gst_ks_video_device_set_property;
124 
125   g_object_class_install_property (gobject_class, PROP_CLOCK,
126       g_param_spec_object ("clock", "Clock to use",
127           "Clock to use", GST_TYPE_KS_CLOCK,
128           G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS));
129 
130   g_object_class_install_property (gobject_class, PROP_DEVICE_PATH,
131       g_param_spec_string ("device-path", "Device Path",
132           "The device path", DEFAULT_DEVICE_PATH,
133           G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS));
134 }
135 
136 static void
gst_ks_video_device_init(GstKsVideoDevice * self)137 gst_ks_video_device_init (GstKsVideoDevice * self)
138 {
139   GstKsVideoDevicePrivate *priv;
140 
141   self->priv = gst_ks_video_device_get_instance_private (self);
142 
143   priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
144   priv->open = FALSE;
145   priv->state = KSSTATE_STOP;
146 }
147 
148 static void
gst_ks_video_device_dispose(GObject * object)149 gst_ks_video_device_dispose (GObject * object)
150 {
151   GstKsVideoDevice *self = GST_KS_VIDEO_DEVICE (object);
152   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
153 
154   gst_ks_video_device_reset_caps (self);
155   gst_ks_video_device_close (self);
156 
157   if (priv->clock != NULL) {
158     g_object_unref (priv->clock);
159     priv->clock = NULL;
160   }
161 
162   G_OBJECT_CLASS (parent_class)->dispose (object);
163 }
164 
165 static void
gst_ks_video_device_get_property(GObject * object,guint prop_id,GValue * value,GParamSpec * pspec)166 gst_ks_video_device_get_property (GObject * object, guint prop_id,
167     GValue * value, GParamSpec * pspec)
168 {
169   GstKsVideoDevice *self = GST_KS_VIDEO_DEVICE (object);
170   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
171 
172   switch (prop_id) {
173     case PROP_CLOCK:
174       g_value_set_object (value, priv->clock);
175       break;
176     case PROP_DEVICE_PATH:
177       g_value_set_string (value, priv->dev_path);
178       break;
179     default:
180       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
181       break;
182   }
183 }
184 
185 static void
gst_ks_video_device_set_property(GObject * object,guint prop_id,const GValue * value,GParamSpec * pspec)186 gst_ks_video_device_set_property (GObject * object, guint prop_id,
187     const GValue * value, GParamSpec * pspec)
188 {
189   GstKsVideoDevice *self = GST_KS_VIDEO_DEVICE (object);
190   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
191 
192   switch (prop_id) {
193     case PROP_CLOCK:
194       if (priv->clock != NULL)
195         g_object_unref (priv->clock);
196       priv->clock = g_value_dup_object (value);
197       break;
198     case PROP_DEVICE_PATH:
199       g_free (priv->dev_path);
200       priv->dev_path = g_value_dup_string (value);
201       break;
202     default:
203       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
204       break;
205   }
206 }
207 
208 static void
gst_ks_video_device_parse_win32_error(const gchar * func_name,DWORD error_code,gulong * ret_error_code,gchar ** ret_error_str)209 gst_ks_video_device_parse_win32_error (const gchar * func_name,
210     DWORD error_code, gulong * ret_error_code, gchar ** ret_error_str)
211 {
212   if (ret_error_code != NULL)
213     *ret_error_code = error_code;
214 
215   if (ret_error_str != NULL) {
216     GString *message;
217     gchar buf[1480];
218     DWORD result;
219 
220     message = g_string_sized_new (1600);
221     g_string_append_printf (message, "%s returned ", func_name);
222 
223     result =
224         FormatMessage (FORMAT_MESSAGE_FROM_SYSTEM |
225         FORMAT_MESSAGE_IGNORE_INSERTS, NULL, error_code, 0, buf, sizeof (buf),
226         NULL);
227     if (result != 0) {
228       g_string_append_printf (message, "0x%08x: %s", (guint) error_code,
229           g_strchomp (buf));
230     } else {
231       DWORD format_error_code = GetLastError ();
232 
233       g_string_append_printf (message,
234           "<0x%08x (FormatMessage error code: %s)>", (guint) error_code,
235           (format_error_code == ERROR_MR_MID_NOT_FOUND)
236           ? "no system error message found"
237           : "failed to retrieve system error message");
238     }
239 
240     *ret_error_str = message->str;
241     g_string_free (message, FALSE);
242   }
243 }
244 
245 static void
gst_ks_video_device_clear_buffers(GstKsVideoDevice * self)246 gst_ks_video_device_clear_buffers (GstKsVideoDevice * self)
247 {
248   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
249   guint i;
250 
251   if (priv->requests == NULL)
252     return;
253 
254   /* Join any pending requests */
255   for (i = 0; i < priv->num_requests; i++) {
256     ReadRequest *req = &g_array_index (priv->requests, ReadRequest, i);
257     HANDLE ev = g_array_index (priv->request_events, HANDLE, i);
258     DWORD n;
259 
260     if (!GetOverlappedResult (priv->pin_handle, &req->overlapped, &n, FALSE)) {
261       if (WaitForSingleObject (ev, 1000) == WAIT_OBJECT_0)
262         GetOverlappedResult (priv->pin_handle, &req->overlapped, &n, FALSE);
263     }
264   }
265 
266   /* Clean up */
267   for (i = 0; i < G_N_ELEMENTS (priv->spare_buffers); i++) {
268     gst_buffer_unref (priv->spare_buffers[i]);
269     priv->spare_buffers[i] = NULL;
270   }
271 
272   for (i = 0; i < priv->requests->len; i++) {
273     ReadRequest *req = &g_array_index (priv->requests, ReadRequest, i);
274     HANDLE ev = g_array_index (priv->request_events, HANDLE, i);
275 
276     gst_buffer_unref (req->buf);
277 
278     if (ev)
279       CloseHandle (ev);
280   }
281 
282   g_array_free (priv->requests, TRUE);
283   priv->requests = NULL;
284 
285   g_array_free (priv->request_events, TRUE);
286   priv->request_events = NULL;
287 }
288 
289 static void
gst_ks_video_device_prepare_buffers(GstKsVideoDevice * self)290 gst_ks_video_device_prepare_buffers (GstKsVideoDevice * self)
291 {
292   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
293   guint i;
294   guint frame_size;
295 
296   g_assert (priv->cur_media_type != NULL);
297 
298   gst_ks_video_device_clear_buffers (self);
299 
300   priv->requests = g_array_sized_new (FALSE, TRUE, sizeof (ReadRequest),
301       priv->num_requests);
302   priv->request_events = g_array_sized_new (FALSE, TRUE, sizeof (HANDLE),
303       priv->num_requests + 1);
304 
305   frame_size = gst_ks_video_device_get_frame_size (self);
306 
307   for (i = 0; i < G_N_ELEMENTS (priv->spare_buffers); i++) {
308     priv->spare_buffers[i] = self->allocfunc (frame_size, KS_BUFFER_ALIGNMENT,
309         self->allocfunc_data);
310   }
311 
312   for (i = 0; i < priv->num_requests; i++) {
313     ReadRequest req;
314     memset (&req, '0', sizeof (ReadRequest));
315 
316     req.buf = self->allocfunc (frame_size, KS_BUFFER_ALIGNMENT,
317         self->allocfunc_data);
318 
319     req.overlapped.hEvent = CreateEvent (NULL, TRUE, FALSE, NULL);
320 
321     g_array_append_val (priv->requests, req);
322     g_array_append_val (priv->request_events, req.overlapped.hEvent);
323   }
324 
325   g_array_append_val (priv->request_events, priv->cancel_event);
326 
327   /*
328    * REVISIT: Could probably remove this later, for now it's here to help
329    *          track down the case where we capture old frames. This has been
330    *          observed with UVC cameras, presumably with some system load.
331    */
332   priv->last_timestamp = GST_CLOCK_TIME_NONE;
333 }
334 
335 static void
gst_ks_video_device_dump_supported_property_sets(GstKsVideoDevice * self,const gchar * obj_name,const GUID * propsets,gulong propsets_len)336 gst_ks_video_device_dump_supported_property_sets (GstKsVideoDevice * self,
337     const gchar * obj_name, const GUID * propsets, gulong propsets_len)
338 {
339   guint i;
340 
341   GST_DEBUG ("%s supports %lu property set%s", obj_name, propsets_len,
342       (propsets_len != 1) ? "s" : "");
343 
344   for (i = 0; i < propsets_len; i++) {
345     gchar *propset_name = ks_property_set_to_string (&propsets[i]);
346     GST_DEBUG ("[%d] %s", i, propset_name);
347     g_free (propset_name);
348   }
349 }
350 
351 GstKsVideoDevice *
gst_ks_video_device_new(const gchar * device_path,GstKsClock * clock,GstKsAllocFunction allocfunc,gpointer allocfunc_data)352 gst_ks_video_device_new (const gchar * device_path, GstKsClock * clock,
353     GstKsAllocFunction allocfunc, gpointer allocfunc_data)
354 {
355   GstKsVideoDevice *device;
356 
357   device = g_object_new (GST_TYPE_KS_VIDEO_DEVICE,
358       "device-path", device_path, "clock", clock, NULL);
359   device->allocfunc = allocfunc;
360   device->allocfunc_data = allocfunc_data;
361 
362   return device;
363 }
364 
365 gboolean
gst_ks_video_device_open(GstKsVideoDevice * self)366 gst_ks_video_device_open (GstKsVideoDevice * self)
367 {
368   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
369   GUID *propsets = NULL;
370   gulong propsets_len;
371   GList *cur;
372 
373   g_assert (!priv->open);
374   g_assert (priv->dev_path != NULL);
375 
376   /*
377    * Open the filter.
378    */
379   priv->filter_handle = CreateFile (priv->dev_path,
380       GENERIC_READ | GENERIC_WRITE, 0, NULL, OPEN_EXISTING,
381       FILE_ATTRIBUTE_NORMAL | FILE_FLAG_OVERLAPPED, NULL);
382   if (!ks_is_valid_handle (priv->filter_handle))
383     goto error;
384 
385   /*
386    * Query the filter for supported property sets.
387    */
388   if (ks_object_get_supported_property_sets (priv->filter_handle, &propsets,
389           &propsets_len)) {
390     gst_ks_video_device_dump_supported_property_sets (self, "filter",
391         propsets, propsets_len);
392     g_free (propsets);
393   } else {
394     GST_DEBUG ("failed to query filter for supported property sets");
395   }
396 
397   /*
398    * Probe for supported media types.
399    */
400   priv->media_types = ks_video_probe_filter_for_caps (priv->filter_handle);
401   priv->cached_caps = gst_caps_new_empty ();
402 
403   for (cur = priv->media_types; cur != NULL; cur = cur->next) {
404     KsVideoMediaType *media_type = cur->data;
405 
406     gst_caps_append (priv->cached_caps,
407         gst_caps_copy (media_type->translated_caps));
408 
409 #if 1
410     {
411       gchar *str;
412       str = gst_caps_to_string (media_type->translated_caps);
413       GST_DEBUG ("pin[%d]: found media type: %s", media_type->pin_id, str);
414       g_free (str);
415     }
416 #endif
417   }
418 
419   priv->cancel_event = CreateEvent (NULL, TRUE, FALSE, NULL);
420 
421   priv->open = TRUE;
422 
423   return TRUE;
424 
425 error:
426   g_free (priv->dev_path);
427   priv->dev_path = NULL;
428 
429   return FALSE;
430 }
431 
432 void
gst_ks_video_device_close(GstKsVideoDevice * self)433 gst_ks_video_device_close (GstKsVideoDevice * self)
434 {
435   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
436   GList *cur;
437 
438   gst_ks_video_device_reset_caps (self);
439 
440   g_free (priv->dev_path);
441   priv->dev_path = NULL;
442 
443   if (ks_is_valid_handle (priv->filter_handle)) {
444     CloseHandle (priv->filter_handle);
445     priv->filter_handle = INVALID_HANDLE_VALUE;
446   }
447 
448   for (cur = priv->media_types; cur != NULL; cur = cur->next) {
449     KsVideoMediaType *mt = cur->data;
450     ks_video_media_type_free (mt);
451   }
452 
453   if (priv->media_types != NULL) {
454     g_list_free (priv->media_types);
455     priv->media_types = NULL;
456   }
457 
458   if (priv->cached_caps != NULL) {
459     gst_caps_unref (priv->cached_caps);
460     priv->cached_caps = NULL;
461   }
462 
463   if (ks_is_valid_handle (priv->cancel_event))
464     CloseHandle (priv->cancel_event);
465   priv->cancel_event = INVALID_HANDLE_VALUE;
466 
467   priv->open = FALSE;
468 }
469 
470 GstCaps *
gst_ks_video_device_get_available_caps(GstKsVideoDevice * self)471 gst_ks_video_device_get_available_caps (GstKsVideoDevice * self)
472 {
473   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
474 
475   g_assert (priv->open);
476 
477   return gst_caps_ref (priv->cached_caps);
478 }
479 
480 gboolean
gst_ks_video_device_has_caps(GstKsVideoDevice * self)481 gst_ks_video_device_has_caps (GstKsVideoDevice * self)
482 {
483   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
484 
485   return (priv->cur_media_type != NULL) ? TRUE : FALSE;
486 }
487 
488 static HANDLE
gst_ks_video_device_create_pin(GstKsVideoDevice * self,KsVideoMediaType * media_type,gulong * num_outstanding)489 gst_ks_video_device_create_pin (GstKsVideoDevice * self,
490     KsVideoMediaType * media_type, gulong * num_outstanding)
491 {
492   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
493 
494   HANDLE pin_handle = INVALID_HANDLE_VALUE;
495   KSPIN_CONNECT *pin_conn = NULL;
496   DWORD ret;
497   guint retry_count;
498 
499   GUID *propsets = NULL;
500   gulong propsets_len;
501   gboolean supports_mem_transport = FALSE;
502 
503   KSALLOCATOR_FRAMING *framing = NULL;
504   gulong framing_size = sizeof (KSALLOCATOR_FRAMING);
505   KSALLOCATOR_FRAMING_EX *framing_ex = NULL;
506   gulong alignment;
507 
508   DWORD mem_transport;
509 
510   /*
511    * Instantiate the pin.
512    */
513   pin_conn = ks_video_create_pin_conn_from_media_type (media_type);
514 
515   for (retry_count = 0; retry_count != 5; retry_count++) {
516 
517     GST_DEBUG ("calling KsCreatePin with pin_id = %d", media_type->pin_id);
518 
519     ret = KsCreatePin (priv->filter_handle, pin_conn, GENERIC_READ,
520         &pin_handle);
521     if (ret != ERROR_NOT_READY)
522       break;
523 
524     /* wait and retry, like the reference implementation does */
525     if (WaitForSingleObject (priv->cancel_event, 1000) == WAIT_OBJECT_0)
526       goto cancelled;
527   }
528 
529   if (ret != ERROR_SUCCESS)
530     goto error_create_pin;
531 
532   GST_DEBUG ("KsCreatePin succeeded, pin %p created", pin_handle);
533 
534   g_free (pin_conn);
535   pin_conn = NULL;
536 
537   /*
538    * Query the pin for supported property sets.
539    */
540   if (ks_object_get_supported_property_sets (pin_handle, &propsets,
541           &propsets_len)) {
542     guint i;
543 
544     gst_ks_video_device_dump_supported_property_sets (self, "pin", propsets,
545         propsets_len);
546 
547     for (i = 0; i < propsets_len; i++) {
548       if (IsEqualGUID (&propsets[i], &KSPROPSETID_MemoryTransport))
549         supports_mem_transport = TRUE;
550     }
551 
552     g_free (propsets);
553   } else {
554     GST_DEBUG ("failed to query pin for supported property sets");
555   }
556 
557   /*
558    * Figure out how many simultaneous requests it prefers.
559    *
560    * This is really important as it depends on the driver and the device.
561    * Doing too few will result in poor capture performance, whilst doing too
562    * many will make some drivers crash really horribly and leave you with a
563    * BSOD. I've experienced the latter with older Logitech drivers.
564    */
565   *num_outstanding = 0;
566   alignment = 0;
567 
568   if (ks_object_get_property (pin_handle, KSPROPSETID_Connection,
569           KSPROPERTY_CONNECTION_ALLOCATORFRAMING_EX, (void *) &framing_ex, NULL,
570           NULL)) {
571     if (framing_ex->CountItems >= 1) {
572       *num_outstanding = framing_ex->FramingItem[0].Frames;
573       alignment = framing_ex->FramingItem[0].FileAlignment;
574     } else {
575       GST_DEBUG ("ignoring empty ALLOCATORFRAMING_EX");
576     }
577   } else {
578     GST_DEBUG ("query for ALLOCATORFRAMING_EX failed, trying "
579         "ALLOCATORFRAMING");
580 
581     if (ks_object_get_property (pin_handle, KSPROPSETID_Connection,
582             KSPROPERTY_CONNECTION_ALLOCATORFRAMING, (void *) &framing,
583             &framing_size, NULL)) {
584       *num_outstanding = framing->Frames;
585       alignment = framing->FileAlignment;
586     } else {
587       GST_DEBUG ("query for ALLOCATORFRAMING failed");
588     }
589   }
590 
591   GST_DEBUG ("num_outstanding: %lu alignment: 0x%08x", *num_outstanding,
592       (guint) alignment);
593 
594   if (*num_outstanding == 0 || *num_outstanding > MAX_OUTSTANDING_FRAMES) {
595     GST_DEBUG ("setting number of allowable outstanding frames to 1");
596     *num_outstanding = 1;
597   }
598 
599   g_free (framing);
600   framing = NULL;
601   g_free (framing_ex);
602   framing_ex = NULL;
603 
604   /*
605    * TODO: We also need to respect alignment, but for now we just assume
606    *       that allocfunc provides the appropriate alignment...
607    */
608 
609   /* Set the memory transport to use. */
610   if (supports_mem_transport) {
611     mem_transport = 0;          /* REVISIT: use the constant here */
612     if (!ks_object_set_property (pin_handle, KSPROPSETID_MemoryTransport,
613             KSPROPERTY_MEMORY_TRANSPORT, &mem_transport,
614             sizeof (mem_transport), NULL)) {
615       GST_DEBUG ("failed to set memory transport, sticking with the default");
616     }
617   }
618 
619   /*
620    * Override the clock if we have one and the pin doesn't have any either.
621    */
622   if (priv->clock != NULL) {
623     HANDLE *cur_clock_handle = NULL;
624     gulong cur_clock_handle_size = sizeof (HANDLE);
625 
626     if (ks_object_get_property (pin_handle, KSPROPSETID_Stream,
627             KSPROPERTY_STREAM_MASTERCLOCK, (gpointer *) & cur_clock_handle,
628             &cur_clock_handle_size, NULL)) {
629       GST_DEBUG ("current master clock handle: %p", *cur_clock_handle);
630       CloseHandle (*cur_clock_handle);
631       g_free (cur_clock_handle);
632     } else {
633       HANDLE new_clock_handle = gst_ks_clock_get_handle (priv->clock);
634 
635       if (ks_object_set_property (pin_handle, KSPROPSETID_Stream,
636               KSPROPERTY_STREAM_MASTERCLOCK, &new_clock_handle,
637               sizeof (new_clock_handle), NULL)) {
638         gst_ks_clock_prepare (priv->clock);
639       } else {
640         GST_WARNING ("failed to set pin's master clock");
641       }
642     }
643   }
644 
645   return pin_handle;
646 
647   /* ERRORS */
648 error_create_pin:
649   {
650     gchar *str;
651 
652     gst_ks_video_device_parse_win32_error ("KsCreatePin", ret, NULL, &str);
653     GST_ERROR ("%s", str);
654     g_free (str);
655 
656     goto beach;
657   }
658 cancelled:
659 beach:
660   {
661     g_free (framing);
662     g_free (framing_ex);
663     if (ks_is_valid_handle (pin_handle))
664       CloseHandle (pin_handle);
665     g_free (pin_conn);
666 
667     return INVALID_HANDLE_VALUE;
668   }
669 }
670 
671 static void
gst_ks_video_device_close_current_pin(GstKsVideoDevice * self)672 gst_ks_video_device_close_current_pin (GstKsVideoDevice * self)
673 {
674   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
675 
676   if (!ks_is_valid_handle (priv->pin_handle))
677     return;
678 
679   gst_ks_video_device_set_state (self, KSSTATE_STOP, NULL);
680 
681   CloseHandle (priv->pin_handle);
682   priv->pin_handle = INVALID_HANDLE_VALUE;
683 }
684 
685 static void
gst_ks_video_device_reset_caps(GstKsVideoDevice * self)686 gst_ks_video_device_reset_caps (GstKsVideoDevice * self)
687 {
688   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
689 
690   gst_ks_video_device_close_current_pin (self);
691 
692   ks_video_media_type_free (priv->cur_media_type);
693   priv->cur_media_type = NULL;
694 
695   priv->width = priv->height = priv->fps_n = priv->fps_d = 0;
696 
697   g_free (priv->rgb_swap_buf);
698   priv->rgb_swap_buf = NULL;
699 
700   if (priv->cur_fixed_caps != NULL) {
701     gst_caps_unref (priv->cur_fixed_caps);
702     priv->cur_fixed_caps = NULL;
703   }
704 }
705 
706 gboolean
gst_ks_video_device_set_caps(GstKsVideoDevice * self,GstCaps * caps)707 gst_ks_video_device_set_caps (GstKsVideoDevice * self, GstCaps * caps)
708 {
709   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
710   GList *cur;
711   GstStructure *s;
712 
713   /* State to be committed on success */
714   KsVideoMediaType *media_type = NULL;
715   gint width, height, fps_n, fps_d;
716   HANDLE pin_handle = INVALID_HANDLE_VALUE;
717 
718   /* Reset? */
719   if (caps == NULL) {
720     gst_ks_video_device_reset_caps (self);
721     return TRUE;
722   }
723 
724   /* Validate the caps */
725   if (!gst_caps_is_subset (caps, priv->cached_caps)) {
726     gchar *string_caps = gst_caps_to_string (caps);
727     gchar *string_c_caps = gst_caps_to_string (priv->cached_caps);
728 
729     GST_ERROR ("caps (%s) is not a subset of device caps (%s)",
730         string_caps, string_c_caps);
731 
732     g_free (string_caps);
733     g_free (string_c_caps);
734 
735     goto error;
736   }
737 
738   for (cur = priv->media_types; cur != NULL; cur = cur->next) {
739     KsVideoMediaType *mt = cur->data;
740 
741     if (gst_caps_is_subset (caps, mt->translated_caps)) {
742       media_type = ks_video_media_type_dup (mt);
743       break;
744     }
745   }
746 
747   if (media_type == NULL)
748     goto error;
749 
750   s = gst_caps_get_structure (caps, 0);
751   if (!gst_structure_get_int (s, "width", &width) ||
752       !gst_structure_get_int (s, "height", &height) ||
753       !gst_structure_get_fraction (s, "framerate", &fps_n, &fps_d)) {
754     gst_structure_get_boolean (s, "systemstream", &priv->is_muxed);
755     if (!priv->is_muxed) {
756       GST_ERROR ("Failed to get width/height/fps");
757       goto error;
758     }
759   } else {
760     if (!ks_video_fixate_media_type (media_type->range,
761             media_type->format, width, height, fps_n, fps_d))
762       goto error;
763   }
764 
765   if (priv->cur_media_type != NULL) {
766     if (media_type->format_size == priv->cur_media_type->format_size &&
767         memcmp (media_type->format, priv->cur_media_type->format,
768             priv->cur_media_type->format_size) == 0) {
769       GST_DEBUG ("%s: re-using existing pin", G_STRFUNC);
770       goto same_caps;
771     } else {
772       GST_DEBUG ("%s: re-creating pin", G_STRFUNC);
773     }
774   }
775 
776   gst_ks_video_device_close_current_pin (self);
777 
778   pin_handle = gst_ks_video_device_create_pin (self, media_type,
779       &priv->num_requests);
780   if (!ks_is_valid_handle (pin_handle)) {
781     /* Re-create the old pin */
782     if (priv->cur_media_type != NULL)
783       priv->pin_handle = gst_ks_video_device_create_pin (self,
784           priv->cur_media_type, &priv->num_requests);
785     goto error;
786   }
787 
788   /* Commit state: no turning back past this */
789   gst_ks_video_device_reset_caps (self);
790 
791   priv->cur_media_type = media_type;
792   priv->width = width;
793   priv->height = height;
794   priv->fps_n = fps_n;
795   priv->fps_d = fps_d;
796 
797   if (media_type->is_rgb)
798     priv->rgb_swap_buf = g_malloc (media_type->sample_size / priv->height);
799   else
800     priv->rgb_swap_buf = NULL;
801 
802   priv->pin_handle = pin_handle;
803 
804   priv->cur_fixed_caps = gst_caps_copy (caps);
805 
806   return TRUE;
807 
808 error:
809   {
810     ks_video_media_type_free (media_type);
811     return FALSE;
812   }
813 same_caps:
814   {
815     ks_video_media_type_free (media_type);
816     return TRUE;
817   }
818 }
819 
820 gboolean
gst_ks_video_device_set_state(GstKsVideoDevice * self,KSSTATE state,gulong * error_code)821 gst_ks_video_device_set_state (GstKsVideoDevice * self, KSSTATE state,
822     gulong * error_code)
823 {
824   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
825   KSSTATE initial_state;
826   gint addend;
827 
828   g_assert (priv->cur_media_type != NULL);
829 
830   if (state == priv->state)
831     return TRUE;
832 
833   initial_state = priv->state;
834   addend = (state > priv->state) ? 1 : -1;
835 
836   GST_DEBUG ("Initiating pin state change from %s to %s",
837       ks_state_to_string (priv->state), ks_state_to_string (state));
838 
839   while (priv->state != state) {
840     KSSTATE next_state = priv->state + addend;
841 
842     /* Skip the ACQUIRE step on the way down like DirectShow does */
843     if (addend < 0 && next_state == KSSTATE_ACQUIRE)
844       next_state = KSSTATE_STOP;
845 
846     GST_DEBUG ("Changing pin state from %s to %s",
847         ks_state_to_string (priv->state), ks_state_to_string (next_state));
848 
849     if (ks_object_set_connection_state (priv->pin_handle, next_state,
850             error_code)) {
851       priv->state = next_state;
852 
853       GST_DEBUG ("Changed pin state to %s", ks_state_to_string (priv->state));
854 
855       if (priv->state == KSSTATE_PAUSE && addend > 0)
856         gst_ks_video_device_prepare_buffers (self);
857       else if (priv->state == KSSTATE_STOP && addend < 0)
858         gst_ks_video_device_clear_buffers (self);
859     } else {
860       GST_WARNING ("Failed to change pin state to %s",
861           ks_state_to_string (next_state));
862 
863       return FALSE;
864     }
865   }
866 
867   GST_DEBUG ("Finished pin state change from %s to %s",
868       ks_state_to_string (initial_state), ks_state_to_string (state));
869 
870   return TRUE;
871 }
872 
873 static guint
gst_ks_video_device_get_frame_size(GstKsVideoDevice * self)874 gst_ks_video_device_get_frame_size (GstKsVideoDevice * self)
875 {
876   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
877 
878   g_assert (priv->cur_media_type != NULL);
879 
880   return priv->cur_media_type->sample_size;
881 }
882 
883 GstClockTime
gst_ks_video_device_get_duration(GstKsVideoDevice * self)884 gst_ks_video_device_get_duration (GstKsVideoDevice * self)
885 {
886   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
887 
888   g_assert (priv->cur_media_type != NULL);
889 
890   return gst_util_uint64_scale_int (GST_SECOND, priv->fps_d, priv->fps_n);
891 }
892 
893 gboolean
gst_ks_video_device_get_latency(GstKsVideoDevice * self,GstClockTime * min_latency,GstClockTime * max_latency)894 gst_ks_video_device_get_latency (GstKsVideoDevice * self,
895     GstClockTime * min_latency, GstClockTime * max_latency)
896 {
897   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
898 
899   if (priv->cur_media_type == NULL)
900     return FALSE;
901 
902   *min_latency =
903       gst_util_uint64_scale_int (GST_SECOND, priv->fps_d, priv->fps_n);
904   *max_latency = *min_latency;
905 
906   return TRUE;
907 }
908 
909 static gboolean
gst_ks_read_request_pick_buffer(GstKsVideoDevice * self,ReadRequest * req)910 gst_ks_read_request_pick_buffer (GstKsVideoDevice * self, ReadRequest * req)
911 {
912   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
913   gboolean buffer_found = FALSE;
914   guint i;
915 
916   buffer_found = gst_buffer_is_writable (req->buf)
917       && gst_buffer_is_all_memory_writable (req->buf);
918 
919   for (i = 0; !buffer_found && i < G_N_ELEMENTS (priv->spare_buffers); i++) {
920     if (gst_buffer_is_writable (priv->spare_buffers[i])
921         && gst_buffer_is_all_memory_writable (priv->spare_buffers[i])) {
922       GstBuffer *hold;
923 
924       hold = req->buf;
925       req->buf = priv->spare_buffers[i];
926       priv->spare_buffers[i] = hold;
927 
928       buffer_found = TRUE;
929     }
930   }
931 
932   if (!buffer_found) {
933     gst_buffer_unref (req->buf);
934     req->buf = self->allocfunc (gst_ks_video_device_get_frame_size (self),
935         KS_BUFFER_ALIGNMENT, self->allocfunc_data);
936   }
937 
938   if (req->buf != NULL) {
939     GST_BUFFER_FLAGS (req->buf) = 0;
940     return TRUE;
941   } else {
942     return FALSE;
943   }
944 }
945 
946 static gboolean
gst_ks_video_device_request_frame(GstKsVideoDevice * self,ReadRequest * req,gulong * error_code,gchar ** error_str)947 gst_ks_video_device_request_frame (GstKsVideoDevice * self, ReadRequest * req,
948     gulong * error_code, gchar ** error_str)
949 {
950   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
951   HANDLE event;
952   KSSTREAM_READ_PARAMS *params;
953   BOOL success;
954   DWORD bytes_returned = 0;
955   GstMapInfo info;
956 
957   if (!gst_ks_read_request_pick_buffer (self, req))
958     goto error_pick_buffer;
959 
960   /* Reset the OVERLAPPED structure */
961   event = req->overlapped.hEvent;
962   memset (&req->overlapped, 0, sizeof (OVERLAPPED));
963   req->overlapped.hEvent = event;
964 
965   /* Fill out KSSTREAM_HEADER and KS_FRAME_INFO */
966   params = &req->params;
967   memset (params, 0, sizeof (KSSTREAM_READ_PARAMS));
968 
969   if (!gst_buffer_map (req->buf, &info, GST_MAP_WRITE))
970     goto map_failed;
971 
972   params->header.Size = sizeof (KSSTREAM_HEADER);
973   if (!priv->is_muxed) {
974     params->header.Size += sizeof (KS_FRAME_INFO);
975   }
976   params->header.PresentationTime.Numerator = 1;
977   params->header.PresentationTime.Denominator = 1;
978   params->header.FrameExtent = gst_ks_video_device_get_frame_size (self);
979   params->header.Data = info.data;
980   params->frame_info.ExtendedHeaderSize = sizeof (KS_FRAME_INFO);
981 
982   success = DeviceIoControl (priv->pin_handle, IOCTL_KS_READ_STREAM, NULL, 0,
983       params, params->header.Size, &bytes_returned, &req->overlapped);
984   gst_buffer_unmap (req->buf, &info);
985   if (!success && GetLastError () != ERROR_IO_PENDING)
986     goto error_ioctl;
987 
988   return TRUE;
989 
990   /* ERRORS */
991 error_pick_buffer:
992   {
993     if (error_code != NULL)
994       *error_code = 0;
995     if (error_str != NULL)
996       *error_str = NULL;
997     return FALSE;
998   }
999 error_ioctl:
1000   {
1001     gst_ks_video_device_parse_win32_error ("DeviceIoControl", GetLastError (),
1002         error_code, error_str);
1003     return FALSE;
1004   }
1005 map_failed:
1006   {
1007     return FALSE;
1008   }
1009 }
1010 
1011 GstFlowReturn
gst_ks_video_device_read_frame(GstKsVideoDevice * self,GstBuffer ** buf,GstClockTime * presentation_time,gulong * error_code,gchar ** error_str)1012 gst_ks_video_device_read_frame (GstKsVideoDevice * self, GstBuffer ** buf,
1013     GstClockTime * presentation_time, gulong * error_code, gchar ** error_str)
1014 {
1015   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
1016   guint req_idx;
1017   DWORD wait_ret;
1018   BOOL success;
1019   DWORD bytes_returned;
1020 
1021   g_assert (priv->cur_media_type != NULL);
1022 
1023   /* First time we're called, submit the requests. */
1024   if (G_UNLIKELY (!priv->requests_submitted)) {
1025     priv->requests_submitted = TRUE;
1026 
1027     for (req_idx = 0; req_idx < priv->num_requests; req_idx++) {
1028       ReadRequest *req = &g_array_index (priv->requests, ReadRequest, req_idx);
1029 
1030       if (!gst_ks_video_device_request_frame (self, req, error_code, error_str))
1031         goto error_request_failed;
1032     }
1033   }
1034 
1035   *buf = NULL;
1036 
1037   do {
1038     /* Wait for either a request to complete, a cancel or a timeout */
1039     wait_ret = WaitForMultipleObjects (priv->request_events->len,
1040         (HANDLE *) priv->request_events->data, FALSE, READ_TIMEOUT);
1041     if (wait_ret == WAIT_TIMEOUT)
1042       goto error_timeout;
1043     else if (wait_ret == WAIT_FAILED)
1044       goto error_wait;
1045 
1046     /* Stopped? */
1047     if (WaitForSingleObject (priv->cancel_event, 0) == WAIT_OBJECT_0)
1048       goto error_cancel;
1049 
1050     /* Find the last ReadRequest that finished and get the result, immediately
1051      * re-issuing each request that has completed. */
1052     for (req_idx = wait_ret - WAIT_OBJECT_0;
1053         req_idx < priv->num_requests; req_idx++) {
1054       ReadRequest *req = &g_array_index (priv->requests, ReadRequest, req_idx);
1055 
1056       /*
1057        * Completed? WaitForMultipleObjects() returns the lowest index if
1058        * multiple objects are in the signaled state, and we know that requests
1059        * are processed one by one so there's no point in looking further once
1060        * we've found the first that's non-signaled.
1061        */
1062       if (WaitForSingleObject (req->overlapped.hEvent, 0) != WAIT_OBJECT_0)
1063         break;
1064 
1065       success = GetOverlappedResult (priv->pin_handle, &req->overlapped,
1066           &bytes_returned, TRUE);
1067 
1068       ResetEvent (req->overlapped.hEvent);
1069 
1070       if (success) {
1071         KSSTREAM_HEADER *hdr = &req->params.header;
1072         KS_FRAME_INFO *frame_info = &req->params.frame_info;
1073         GstClockTime timestamp = GST_CLOCK_TIME_NONE;
1074         GstClockTime duration = GST_CLOCK_TIME_NONE;
1075 
1076         if (hdr->OptionsFlags & KSSTREAM_HEADER_OPTIONSF_TIMEVALID)
1077           timestamp = hdr->PresentationTime.Time * 100;
1078 
1079         if (hdr->OptionsFlags & KSSTREAM_HEADER_OPTIONSF_DURATIONVALID)
1080           duration = hdr->Duration * 100;
1081 
1082         UNREF_BUFFER (buf);
1083 
1084         if (G_LIKELY (hdr->DataUsed != 0)) {
1085           /* Assume it's a good frame */
1086           gst_buffer_set_size (req->buf, hdr->DataUsed);
1087           *buf = gst_buffer_ref (req->buf);
1088         }
1089 
1090         if (G_LIKELY (presentation_time != NULL))
1091           *presentation_time = timestamp;
1092 
1093         if (G_UNLIKELY (GST_DEBUG_IS_ENABLED ())) {
1094           gchar *options_flags_str =
1095               ks_options_flags_to_string (hdr->OptionsFlags);
1096 
1097           GST_DEBUG ("PictureNumber=%" G_GUINT64_FORMAT ", DropCount=%"
1098               G_GUINT64_FORMAT ", PresentationTime=%" GST_TIME_FORMAT
1099               ", Duration=%" GST_TIME_FORMAT ", OptionsFlags=%s: %lu bytes",
1100               frame_info->PictureNumber, frame_info->DropCount,
1101               GST_TIME_ARGS (timestamp), GST_TIME_ARGS (duration),
1102               options_flags_str, hdr->DataUsed);
1103 
1104           g_free (options_flags_str);
1105         }
1106 
1107         /* Protect against old frames. This should never happen, see previous
1108          * comment on last_timestamp. */
1109         if (G_LIKELY (GST_CLOCK_TIME_IS_VALID (timestamp))) {
1110           if (G_UNLIKELY (GST_CLOCK_TIME_IS_VALID (priv->last_timestamp) &&
1111                   timestamp < priv->last_timestamp)) {
1112             GST_INFO ("got an old frame (last_timestamp=%" GST_TIME_FORMAT
1113                 ", timestamp=%" GST_TIME_FORMAT ")",
1114                 GST_TIME_ARGS (priv->last_timestamp),
1115                 GST_TIME_ARGS (timestamp));
1116             UNREF_BUFFER (buf);
1117           } else {
1118             priv->last_timestamp = timestamp;
1119           }
1120         }
1121       } else if (GetLastError () != ERROR_OPERATION_ABORTED) {
1122         goto error_get_result;
1123       }
1124 
1125       /* Submit a new request immediately */
1126       if (!gst_ks_video_device_request_frame (self, req, error_code, error_str))
1127         goto error_request_failed;
1128     }
1129   } while (*buf == NULL);
1130 
1131   return GST_FLOW_OK;
1132 
1133   /* ERRORS */
1134 error_request_failed:
1135   {
1136     UNREF_BUFFER (buf);
1137 
1138     return GST_FLOW_ERROR;
1139   }
1140 error_timeout:
1141   {
1142     GST_DEBUG ("IOCTL_KS_READ_STREAM timed out");
1143 
1144     if (error_code != NULL)
1145       *error_code = 0;
1146     if (error_str != NULL)
1147       *error_str = NULL;
1148 
1149     return GST_FLOW_CUSTOM_ERROR;
1150   }
1151 error_wait:
1152   {
1153     gst_ks_video_device_parse_win32_error ("WaitForMultipleObjects",
1154         GetLastError (), error_code, error_str);
1155 
1156     return GST_FLOW_ERROR;
1157   }
1158 error_cancel:
1159   {
1160     if (error_code != NULL)
1161       *error_code = 0;
1162     if (error_str != NULL)
1163       *error_str = NULL;
1164 
1165     return GST_FLOW_FLUSHING;
1166   }
1167 error_get_result:
1168   {
1169     gst_ks_video_device_parse_win32_error ("GetOverlappedResult",
1170         GetLastError (), error_code, error_str);
1171 
1172     return GST_FLOW_ERROR;
1173   }
1174 }
1175 
1176 gboolean
gst_ks_video_device_postprocess_frame(GstKsVideoDevice * self,GstBuffer ** bufptr)1177 gst_ks_video_device_postprocess_frame (GstKsVideoDevice * self,
1178     GstBuffer ** bufptr)
1179 {
1180   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
1181   GstBuffer *buf = *bufptr;
1182 
1183   /* If it's RGB we need to flip the image */
1184   if (priv->rgb_swap_buf != NULL) {
1185     GstMapInfo info;
1186     gint stride, line;
1187     guint8 *dst, *src;
1188 
1189     /* Need to make the buffer writable because
1190      * the pseudo-bufferpool of requests keeps a ref */
1191     buf = gst_buffer_make_writable (buf);
1192 
1193     if (!gst_buffer_map (buf, &info, GST_MAP_READWRITE))
1194       return FALSE;
1195 
1196     stride = info.size / priv->height;
1197     dst = info.data;
1198     src = info.data + info.size - stride;
1199 
1200     for (line = 0; line < priv->height / 2; line++) {
1201       memcpy (priv->rgb_swap_buf, dst, stride);
1202 
1203       memcpy (dst, src, stride);
1204       memcpy (src, priv->rgb_swap_buf, stride);
1205 
1206       dst += stride;
1207       src -= stride;
1208     }
1209 
1210     gst_buffer_unmap (buf, &info);
1211   }
1212   *bufptr = buf;
1213 
1214   return TRUE;
1215 }
1216 
1217 void
gst_ks_video_device_cancel(GstKsVideoDevice * self)1218 gst_ks_video_device_cancel (GstKsVideoDevice * self)
1219 {
1220   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
1221 
1222   SetEvent (priv->cancel_event);
1223 }
1224 
1225 void
gst_ks_video_device_cancel_stop(GstKsVideoDevice * self)1226 gst_ks_video_device_cancel_stop (GstKsVideoDevice * self)
1227 {
1228   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
1229 
1230   ResetEvent (priv->cancel_event);
1231 }
1232 
1233 gboolean
gst_ks_video_device_stream_is_muxed(GstKsVideoDevice * self)1234 gst_ks_video_device_stream_is_muxed (GstKsVideoDevice * self)
1235 {
1236   GstKsVideoDevicePrivate *priv = GST_KS_VIDEO_DEVICE_GET_PRIVATE (self);
1237 
1238   return priv->is_muxed;
1239 }
1240