FreeRDP
Loading...
Searching...
No Matches
camera_v4l.c
1
20#include <errno.h>
21#include <fcntl.h>
22#include <poll.h>
23#include <sys/ioctl.h>
24#include <sys/mman.h>
25
26/* v4l includes */
27#include <linux/videodev2.h>
28
29#include "camera_v4l.h"
30#include "uvc_h264.h"
31
32#define TAG CHANNELS_TAG("rdpecam-v4l.client")
33
34#define CAM_V4L2_BUFFERS_COUNT 4
35#define CAM_V4L2_CAPTURE_THREAD_SLEEP_MS 1000
36
37#define CAM_V4L2_FRAMERATE_NUMERATOR_DEFAULT 30
38#define CAM_V4L2_FRAMERATE_DENOMINATOR_DEFAULT 1
39
40typedef struct
41{
42 ICamHal iHal;
43
44 wHashTable* streams; /* Index: deviceId, Value: CamV4lStream */
45
46} CamV4lHal;
47
48static CamV4lStream* cam_v4l_stream_create(const char* deviceId, size_t streamIndex);
49static void cam_v4l_stream_free(void* obj);
50static void cam_v4l_stream_close_device(CamV4lStream* stream);
51static CAM_ERROR_CODE cam_v4l_stream_stop(CamV4lStream* stream);
52
58static const char* cam_v4l_get_fourcc_str(unsigned int fourcc, char* buffer, size_t size)
59{
60 if (size < 5)
61 return nullptr;
62
63 buffer[0] = (char)(fourcc & 0xFF);
64 buffer[1] = (char)((fourcc >> 8) & 0xFF);
65 buffer[2] = (char)((fourcc >> 16) & 0xFF);
66 buffer[3] = (char)((fourcc >> 24) & 0xFF);
67 buffer[4] = '\0';
68 return buffer;
69}
70
76static UINT32 ecamToV4L2PixFormat(CAM_MEDIA_FORMAT ecamFormat)
77{
78 switch (ecamFormat)
79 {
80 case CAM_MEDIA_FORMAT_H264:
81 return V4L2_PIX_FMT_H264;
82 case CAM_MEDIA_FORMAT_MJPG:
83 return V4L2_PIX_FMT_MJPEG;
84 case CAM_MEDIA_FORMAT_YUY2:
85 return V4L2_PIX_FMT_YUYV;
86 case CAM_MEDIA_FORMAT_NV12:
87 return V4L2_PIX_FMT_NV12;
88 case CAM_MEDIA_FORMAT_I420:
89 return V4L2_PIX_FMT_YUV420;
90 case CAM_MEDIA_FORMAT_RGB24:
91 return V4L2_PIX_FMT_RGB24;
92 case CAM_MEDIA_FORMAT_RGB32:
93 return V4L2_PIX_FMT_RGB32;
94 default:
95 WLog_ERR(TAG, "Unsupported CAM_MEDIA_FORMAT %u", ecamFormat);
96 return 0;
97 }
98}
99
105static BOOL cam_v4l_format_supported(int fd, UINT32 format)
106{
107 struct v4l2_fmtdesc fmtdesc = WINPR_C_ARRAY_INIT;
108 fmtdesc.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
109
110 for (fmtdesc.index = 0; ioctl(fd, VIDIOC_ENUM_FMT, &fmtdesc) == 0; fmtdesc.index++)
111 {
112 if (fmtdesc.pixelformat == format)
113 return TRUE;
114 }
115 return FALSE;
116}
117
123static int cam_v4l_open_device(const char* deviceId, int flags)
124{
125 char device[20] = WINPR_C_ARRAY_INIT;
126 int fd = -1;
127 struct v4l2_capability cap = WINPR_C_ARRAY_INIT;
128
129 if (!deviceId)
130 return -1;
131
132 if (0 == strncmp(deviceId, "/dev/video", 10))
133 return open(deviceId, flags);
134
135 for (UINT n = 0; n < 64; n++)
136 {
137 (void)_snprintf(device, sizeof(device), "/dev/video%" PRIu32, n);
138 if ((fd = open(device, flags)) == -1)
139 continue;
140
141 /* query device capabilities and make sure this is a video capture device */
142 if (ioctl(fd, VIDIOC_QUERYCAP, &cap) < 0 || !(cap.device_caps & V4L2_CAP_VIDEO_CAPTURE))
143 {
144 close(fd);
145 continue;
146 }
147
148 if (cap.bus_info[0] != 0 && 0 == strcmp((const char*)cap.bus_info, deviceId))
149 return fd;
150
151 close(fd);
152 }
153
154 return fd;
155}
156
157static BOOL cam_v4l_activate(ICamHal* ihal, const char* deviceId, CAM_ERROR_CODE* errorCode)
158{
159 WINPR_UNUSED(ihal);
160 WINPR_UNUSED(deviceId);
161
162 *errorCode = CAM_ERROR_CODE_None;
163 return TRUE;
164}
165
166static BOOL cam_v4l_deactivate(ICamHal* ihal, const char* deviceId, CAM_ERROR_CODE* errorCode)
167{
168 WINPR_UNUSED(ihal);
169 WINPR_UNUSED(deviceId);
170
171 *errorCode = CAM_ERROR_CODE_None;
172 return TRUE;
173}
174
181static INT16 cam_v4l_get_media_type_descriptions(ICamHal* ihal, const char* deviceId,
182 size_t streamIndex,
183 const CAM_MEDIA_FORMAT_INFO* supportedFormats,
184 size_t nSupportedFormats,
185 CAM_MEDIA_TYPE_DESCRIPTION* mediaTypes,
186 size_t* nMediaTypes)
187{
188 CamV4lHal* hal = (CamV4lHal*)ihal;
189 size_t maxMediaTypes = *nMediaTypes;
190 size_t nTypes = 0;
191 BOOL formatFound = FALSE;
192
193 CamV4lStream* stream = (CamV4lStream*)HashTable_GetItemValue(hal->streams, deviceId);
194
195 if (!stream)
196 {
197 stream = cam_v4l_stream_create(deviceId, streamIndex);
198 if (!stream)
199 return CAM_ERROR_CODE_OutOfMemory;
200
201 if (!HashTable_Insert(hal->streams, deviceId, stream))
202 {
203 cam_v4l_stream_free(stream);
204 return CAM_ERROR_CODE_UnexpectedError;
205 }
206 }
207
208 int fd = cam_v4l_open_device(deviceId, O_RDONLY);
209 if (fd == -1)
210 {
211 WLog_ERR(TAG, "Unable to open device %s", deviceId);
212 return -1;
213 }
214
215 size_t formatIndex = 0;
216 for (; formatIndex < nSupportedFormats; formatIndex++)
217 {
218 const UINT32 pixelFormat = ecamToV4L2PixFormat(supportedFormats[formatIndex].inputFormat);
219
220 WINPR_ASSERT(pixelFormat != 0);
221 struct v4l2_frmsizeenum frmsize = WINPR_C_ARRAY_INIT;
222
223 if (!cam_v4l_format_supported(fd, pixelFormat))
224 continue;
225
226 frmsize.pixel_format = pixelFormat;
227 for (frmsize.index = 0; ioctl(fd, VIDIOC_ENUM_FRAMESIZES, &frmsize) == 0; frmsize.index++)
228 {
229 struct v4l2_frmivalenum frmival = WINPR_C_ARRAY_INIT;
230
231 if (frmsize.type != V4L2_FRMSIZE_TYPE_DISCRETE)
232 break; /* don't support size types other than discrete */
233
234 formatFound = TRUE;
235 mediaTypes->Width = frmsize.discrete.width;
236 mediaTypes->Height = frmsize.discrete.height;
237 mediaTypes->Format = supportedFormats[formatIndex].inputFormat;
238
239 /* query frame rate (1st is highest fps supported) */
240 frmival.index = 0;
241 frmival.pixel_format = pixelFormat;
242 frmival.width = frmsize.discrete.width;
243 frmival.height = frmsize.discrete.height;
244 if (ioctl(fd, VIDIOC_ENUM_FRAMEINTERVALS, &frmival) == 0 &&
245 frmival.type == V4L2_FRMIVAL_TYPE_DISCRETE)
246 {
247 /* inverse of a fraction */
248 mediaTypes->FrameRateNumerator = frmival.discrete.denominator;
249 mediaTypes->FrameRateDenominator = frmival.discrete.numerator;
250 }
251 else
252 {
253 WLog_DBG(TAG, "VIDIOC_ENUM_FRAMEINTERVALS failed, using default framerate");
254 mediaTypes->FrameRateNumerator = CAM_V4L2_FRAMERATE_NUMERATOR_DEFAULT;
255 mediaTypes->FrameRateDenominator = CAM_V4L2_FRAMERATE_DENOMINATOR_DEFAULT;
256 }
257
258 mediaTypes->PixelAspectRatioNumerator = mediaTypes->PixelAspectRatioDenominator = 1;
259
260 char fourccstr[5] = WINPR_C_ARRAY_INIT;
261 WLog_DBG(TAG, "Camera format: %s, width: %u, height: %u, fps: %u/%u",
262 cam_v4l_get_fourcc_str(pixelFormat, fourccstr, ARRAYSIZE(fourccstr)),
263 mediaTypes->Width, mediaTypes->Height, mediaTypes->FrameRateNumerator,
264 mediaTypes->FrameRateDenominator);
265
266 mediaTypes++;
267 nTypes++;
268
269 if (nTypes == maxMediaTypes)
270 {
271 WLog_ERR(TAG, "Media types reached buffer maximum %" PRIuz "", maxMediaTypes);
272 goto error;
273 }
274 }
275
276 if (formatFound)
277 {
278 /* we are interested in 1st supported format only, with all supported sizes */
279 break;
280 }
281 }
282
283error:
284
285 *nMediaTypes = nTypes;
286 close(fd);
287 if (formatIndex > INT16_MAX)
288 return -1;
289 return (INT16)formatIndex;
290}
291
297static UINT cam_v4l_enumerate(WINPR_ATTR_UNUSED ICamHal* ihal, ICamHalEnumCallback callback,
298 CameraPlugin* ecam, GENERIC_CHANNEL_CALLBACK* hchannel)
299{
300 UINT count = 0;
301
302 for (UINT n = 0; n < 64; n++)
303 {
304 char device[20] = WINPR_C_ARRAY_INIT;
305 struct v4l2_capability cap = WINPR_C_ARRAY_INIT;
306 (void)_snprintf(device, sizeof(device), "/dev/video%" PRIu32, n);
307 int fd = open(device, O_RDONLY);
308 if (fd == -1)
309 continue;
310
311 /* query device capabilities and make sure this is a video capture device */
312 if (ioctl(fd, VIDIOC_QUERYCAP, &cap) < 0 || !(cap.device_caps & V4L2_CAP_VIDEO_CAPTURE))
313 {
314 close(fd);
315 continue;
316 }
317 count++;
318
319 const char* deviceName = (char*)cap.card;
320 const char* deviceId = device;
321 if (cap.bus_info[0] != 0) /* may not be available in all drivers */
322 deviceId = (char*)cap.bus_info;
323
324 IFCALL(callback, ecam, hchannel, deviceId, deviceName);
325
326 close(fd);
327 }
328
329 return count;
330}
331
332static void cam_v4l_stream_free_buffers(CamV4lStream* stream)
333{
334 if (!stream || !stream->buffers)
335 return;
336
337 /* unmap buffers */
338 for (size_t i = 0; i < stream->nBuffers; i++)
339 {
340 if (stream->buffers[i].length && stream->buffers[i].start != MAP_FAILED)
341 {
342 munmap(stream->buffers[i].start, stream->buffers[i].length);
343 }
344 }
345
346 free(stream->buffers);
347 stream->buffers = nullptr;
348 stream->nBuffers = 0;
349}
350
356static size_t cam_v4l_stream_alloc_buffers(CamV4lStream* stream)
357{
358 struct v4l2_requestbuffers rbuffer = WINPR_C_ARRAY_INIT;
359
360 rbuffer.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
361 rbuffer.memory = V4L2_MEMORY_MMAP;
362 rbuffer.count = CAM_V4L2_BUFFERS_COUNT;
363
364 if (ioctl(stream->fd, VIDIOC_REQBUFS, &rbuffer) < 0 || rbuffer.count == 0)
365 {
366 char buffer[64] = WINPR_C_ARRAY_INIT;
367 WLog_ERR(TAG, "Failure in VIDIOC_REQBUFS, errno %s [%d], count %u",
368 winpr_strerror(errno, buffer, sizeof(buffer)), errno, rbuffer.count);
369 return 0;
370 }
371
372 stream->nBuffers = rbuffer.count;
373
374 /* Map the buffers */
375 stream->buffers = (CamV4lBuffer*)calloc(rbuffer.count, sizeof(CamV4lBuffer));
376 if (!stream->buffers)
377 {
378 WLog_ERR(TAG, "Failure in calloc");
379 return 0;
380 }
381
382 for (unsigned int i = 0; i < rbuffer.count; i++)
383 {
384 struct v4l2_buffer vbuffer = WINPR_C_ARRAY_INIT;
385 vbuffer.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
386 vbuffer.memory = V4L2_MEMORY_MMAP;
387 vbuffer.index = i;
388
389 if (ioctl(stream->fd, VIDIOC_QUERYBUF, &vbuffer) < 0)
390 {
391 char buffer[64] = WINPR_C_ARRAY_INIT;
392 WLog_ERR(TAG, "Failure in VIDIOC_QUERYBUF, errno %s [%d]",
393 winpr_strerror(errno, buffer, sizeof(buffer)), errno);
394 cam_v4l_stream_free_buffers(stream);
395 return 0;
396 }
397
398 stream->buffers[i].start = mmap(nullptr, vbuffer.length, PROT_READ | PROT_WRITE, MAP_SHARED,
399 stream->fd, vbuffer.m.offset);
400
401 if (MAP_FAILED == stream->buffers[i].start)
402 {
403 char buffer[64] = WINPR_C_ARRAY_INIT;
404 WLog_ERR(TAG, "Failure in mmap, errno %s [%d]",
405 winpr_strerror(errno, buffer, sizeof(buffer)), errno);
406 cam_v4l_stream_free_buffers(stream);
407 return 0;
408 }
409
410 stream->buffers[i].length = vbuffer.length;
411
412 WLog_DBG(TAG, "Buffer %u mapped, size: %u", i, vbuffer.length);
413
414 if (ioctl(stream->fd, VIDIOC_QBUF, &vbuffer) < 0)
415 {
416 char buffer[64] = WINPR_C_ARRAY_INIT;
417 WLog_ERR(TAG, "Failure in VIDIOC_QBUF, errno %s [%d]",
418 winpr_strerror(errno, buffer, sizeof(buffer)), errno);
419 cam_v4l_stream_free_buffers(stream);
420 return 0;
421 }
422 }
423
424 return stream->buffers[0].length;
425}
426
432static DWORD WINAPI cam_v4l_stream_capture_thread(LPVOID param)
433{
434 CamV4lStream* stream = (CamV4lStream*)param;
435 WINPR_ASSERT(stream);
436
437 int fd = stream->fd;
438
439 do
440 {
441 int retVal = 0;
442 struct pollfd pfd = WINPR_C_ARRAY_INIT;
443
444 pfd.fd = fd;
445 pfd.events = POLLIN;
446
447 retVal = poll(&pfd, 1, CAM_V4L2_CAPTURE_THREAD_SLEEP_MS);
448
449 if (retVal == 0)
450 {
451 /* poll timed out */
452 continue;
453 }
454 else if (retVal < 0)
455 {
456 char buffer[64] = WINPR_C_ARRAY_INIT;
457 WLog_DBG(TAG, "Failure in poll, errno %s [%d]",
458 winpr_strerror(errno, buffer, sizeof(buffer)), errno);
459 Sleep(CAM_V4L2_CAPTURE_THREAD_SLEEP_MS); /* trying to recover */
460 continue;
461 }
462 else if (!(pfd.revents & POLLIN))
463 {
464 WLog_DBG(TAG, "poll reported non-read event %d", pfd.revents);
465 Sleep(CAM_V4L2_CAPTURE_THREAD_SLEEP_MS); /* also trying to recover */
466 continue;
467 }
468
469 EnterCriticalSection(&stream->lock);
470 if (stream->streaming)
471 {
472 struct v4l2_buffer buf = WINPR_C_ARRAY_INIT;
473 buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
474 buf.memory = V4L2_MEMORY_MMAP;
475
476 /* dequeue buffers until empty */
477 while (ioctl(fd, VIDIOC_DQBUF, &buf) != -1)
478 {
479 const UINT error =
480 stream->sampleCallback(stream->dev, stream->streamIndex,
481 stream->buffers[buf.index].start, buf.bytesused);
482 if (error != CHANNEL_RC_OK)
483 WLog_ERR(TAG, "Failure in sampleCallback: %" PRIu32, error);
484
485 /* enqueue buffer back */
486 if (ioctl(fd, VIDIOC_QBUF, &buf) == -1)
487 {
488 char buffer[64] = WINPR_C_ARRAY_INIT;
489 WLog_ERR(TAG, "Failure in VIDIOC_QBUF, errno %s [%d]",
490 winpr_strerror(errno, buffer, sizeof(buffer)), errno);
491 }
492 }
493 }
494 LeaveCriticalSection(&stream->lock);
495
496 } while (stream->streaming);
497
498 return CHANNEL_RC_OK;
499}
500
501void cam_v4l_stream_close_device(CamV4lStream* stream)
502{
503 if (stream->fd != -1)
504 {
505 close(stream->fd);
506 stream->fd = -1;
507 }
508}
509
515WINPR_ATTR_MALLOC(cam_v4l_stream_free, 1)
516CamV4lStream* cam_v4l_stream_create(const char* deviceId, size_t streamIndex)
517{
518 CamV4lStream* stream = calloc(1, sizeof(CamV4lStream));
519
520 if (!stream)
521 {
522 WLog_ERR(TAG, "Failure in calloc");
523 return nullptr;
524 }
525 stream->streamIndex = streamIndex;
526 stream->fd = -1;
527 stream->h264UnitId = get_uvc_h624_unit_id(deviceId);
528
529 if (!InitializeCriticalSectionEx(&stream->lock, 0, 0))
530 {
531 WLog_ERR(TAG, "Failure in calloc");
532 free(stream);
533 return nullptr;
534 }
535
536 return stream;
537}
538
544CAM_ERROR_CODE cam_v4l_stream_stop(CamV4lStream* stream)
545{
546 if (!stream || !stream->streaming)
547 return CAM_ERROR_CODE_None;
548
549 stream->streaming = FALSE; /* this will terminate capture thread */
550
551 if (stream->captureThread)
552 {
553 (void)WaitForSingleObject(stream->captureThread, INFINITE);
554 (void)CloseHandle(stream->captureThread);
555 stream->captureThread = nullptr;
556 }
557
558 EnterCriticalSection(&stream->lock);
559
560 /* stop streaming */
561 enum v4l2_buf_type type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
562 if (ioctl(stream->fd, VIDIOC_STREAMOFF, &type) < 0)
563 {
564 char buffer[64] = WINPR_C_ARRAY_INIT;
565 WLog_ERR(TAG, "Failure in VIDIOC_STREAMOFF, errno %s [%d]",
566 winpr_strerror(errno, buffer, sizeof(buffer)), errno);
567 }
568
569 cam_v4l_stream_free_buffers(stream);
570 cam_v4l_stream_close_device(stream);
571
572 LeaveCriticalSection(&stream->lock);
573
574 return CAM_ERROR_CODE_None;
575}
576
577static CAM_ERROR_CODE cam_v4l_stream_start(ICamHal* ihal, CameraDevice* dev, size_t streamIndex,
578 const CAM_MEDIA_TYPE_DESCRIPTION* mediaType,
579 ICamHalSampleCapturedCallback callback)
580{
581 CamV4lHal* hal = (CamV4lHal*)ihal;
582 WINPR_ASSERT(hal);
583
584 CamV4lStream* stream = (CamV4lStream*)HashTable_GetItemValue(hal->streams, dev->deviceId);
585
586 if (!stream)
587 {
588 WLog_ERR(TAG, "Unable to find stream, device %s, streamIndex %" PRIuz, dev->deviceId,
589 streamIndex);
590 return CAM_ERROR_CODE_UnexpectedError;
591 }
592
593 if (stream->streaming)
594 {
595 WLog_ERR(TAG, "Streaming already in progress, device %s, streamIndex %" PRIuz,
596 dev->deviceId, streamIndex);
597 return CAM_ERROR_CODE_UnexpectedError;
598 }
599
600 stream->dev = dev;
601 stream->sampleCallback = callback;
602
603 if ((stream->fd = cam_v4l_open_device(dev->deviceId, O_RDWR | O_NONBLOCK)) == -1)
604 {
605 WLog_ERR(TAG, "Unable to open device %s", dev->deviceId);
606 return CAM_ERROR_CODE_UnexpectedError;
607 }
608
609 struct v4l2_format video_fmt = WINPR_C_ARRAY_INIT;
610 video_fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
611
612 const UINT32 pixelFormat = ecamToV4L2PixFormat(mediaType->Format);
613 video_fmt.fmt.pix.width = mediaType->Width;
614 video_fmt.fmt.pix.height = mediaType->Height;
615
616 if (pixelFormat == 0)
617 {
618 cam_v4l_stream_close_device(stream);
619 return CAM_ERROR_CODE_InvalidMediaType;
620 }
621
622 video_fmt.fmt.pix.pixelformat = pixelFormat;
623
624 /* set format and frame size */
625 if (ioctl(stream->fd, VIDIOC_S_FMT, &video_fmt) < 0)
626 {
627 char buffer[64] = WINPR_C_ARRAY_INIT;
628 WLog_ERR(TAG, "Failure in VIDIOC_S_FMT, errno %s [%d]",
629 winpr_strerror(errno, buffer, sizeof(buffer)), errno);
630 cam_v4l_stream_close_device(stream);
631 return CAM_ERROR_CODE_InvalidMediaType;
632 }
633
634 /* trying to set frame rate, if driver supports it */
635 struct v4l2_streamparm sp1 = WINPR_C_ARRAY_INIT;
636 struct v4l2_streamparm sp2 = WINPR_C_ARRAY_INIT;
637 sp1.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
638 if (ioctl(stream->fd, VIDIOC_G_PARM, &sp1) < 0 ||
639 !(sp1.parm.capture.capability & V4L2_CAP_TIMEPERFRAME))
640 {
641 WLog_INFO(TAG, "Driver doesn't support setting framerate");
642 }
643 else
644 {
645 sp2.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
646
647 /* inverse of a fraction */
648 sp2.parm.capture.timeperframe.numerator = mediaType->FrameRateDenominator;
649 sp2.parm.capture.timeperframe.denominator = mediaType->FrameRateNumerator;
650
651 if (ioctl(stream->fd, VIDIOC_S_PARM, &sp2) < 0)
652 {
653 char buffer[64] = WINPR_C_ARRAY_INIT;
654 WLog_INFO(TAG, "Failed to set the framerate, errno %s [%d]",
655 winpr_strerror(errno, buffer, sizeof(buffer)), errno);
656 }
657 }
658
659 size_t maxSample = cam_v4l_stream_alloc_buffers(stream);
660 if (maxSample == 0)
661 {
662 WLog_ERR(TAG, "Failure to allocate video buffers");
663 cam_v4l_stream_close_device(stream);
664 return CAM_ERROR_CODE_OutOfMemory;
665 }
666
667 stream->streaming = TRUE;
668
669 /* start streaming */
670 enum v4l2_buf_type type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
671 if (ioctl(stream->fd, VIDIOC_STREAMON, &type) < 0)
672 {
673 char buffer[64] = WINPR_C_ARRAY_INIT;
674 WLog_ERR(TAG, "Failure in VIDIOC_STREAMON, errno %s [%d]",
675 winpr_strerror(errno, buffer, sizeof(buffer)), errno);
676 cam_v4l_stream_stop(stream);
677 return CAM_ERROR_CODE_UnexpectedError;
678 }
679
680 stream->captureThread =
681 CreateThread(nullptr, 0, cam_v4l_stream_capture_thread, stream, 0, nullptr);
682 if (!stream->captureThread)
683 {
684 WLog_ERR(TAG, "CreateThread failure");
685 cam_v4l_stream_stop(stream);
686 return CAM_ERROR_CODE_OutOfMemory;
687 }
688
689 char fourccstr[16] = WINPR_C_ARRAY_INIT;
690 cam_v4l_get_fourcc_str(pixelFormat, fourccstr, ARRAYSIZE(fourccstr));
691
692 WLog_INFO(TAG, "Camera format: %s, width: %u, height: %u, fps: %u/%u", fourccstr,
693 mediaType->Width, mediaType->Height, mediaType->FrameRateNumerator,
694 mediaType->FrameRateDenominator);
695
696 return CAM_ERROR_CODE_None;
697}
698
704static CAM_ERROR_CODE cam_v4l_stream_stop_by_device_id(ICamHal* ihal, const char* deviceId,
705 WINPR_ATTR_UNUSED size_t streamIndex)
706{
707 CamV4lHal* hal = (CamV4lHal*)ihal;
708
709 CamV4lStream* stream = (CamV4lStream*)HashTable_GetItemValue(hal->streams, deviceId);
710
711 if (!stream)
712 return CAM_ERROR_CODE_NotInitialized;
713
714 return cam_v4l_stream_stop(stream);
715}
716
723void cam_v4l_stream_free(void* obj)
724{
725 CamV4lStream* stream = (CamV4lStream*)obj;
726 if (!stream)
727 return;
728
729 cam_v4l_stream_stop(stream);
730
731 DeleteCriticalSection(&stream->lock);
732 free(stream);
733}
734
740static CAM_ERROR_CODE cam_v4l_free(ICamHal* ihal)
741{
742 CamV4lHal* hal = (CamV4lHal*)ihal;
743
744 if (hal == nullptr)
745 return CAM_ERROR_CODE_NotInitialized;
746
747 HashTable_Free(hal->streams);
748
749 free(hal);
750
751 return CAM_ERROR_CODE_None;
752}
753
759FREERDP_ENTRY_POINT(UINT VCAPITYPE v4l_freerdp_rdpecam_client_subsystem_entry(
761{
762 UINT ret = ERROR_INTERNAL_ERROR;
763 WINPR_ASSERT(pEntryPoints);
764
765 CamV4lHal* hal = (CamV4lHal*)calloc(1, sizeof(CamV4lHal));
766
767 if (hal == nullptr)
768 return CHANNEL_RC_NO_MEMORY;
769
770 hal->iHal.Enumerate = cam_v4l_enumerate;
771 hal->iHal.GetMediaTypeDescriptions = cam_v4l_get_media_type_descriptions;
772 hal->iHal.Activate = cam_v4l_activate;
773 hal->iHal.Deactivate = cam_v4l_deactivate;
774 hal->iHal.StartStream = cam_v4l_stream_start;
775 hal->iHal.StopStream = cam_v4l_stream_stop_by_device_id;
776 hal->iHal.Free = cam_v4l_free;
777
778 hal->streams = HashTable_New(FALSE);
779 if (!hal->streams)
780 goto error;
781
782 if (!HashTable_SetupForStringData(hal->streams, FALSE))
783 goto error;
784
785 wObject* obj = HashTable_ValueObject(hal->streams);
786 WINPR_ASSERT(obj);
787 obj->fnObjectFree = cam_v4l_stream_free;
788
789 ret = pEntryPoints->pRegisterCameraHal(pEntryPoints->plugin, &hal->iHal);
790 if (ret != CHANNEL_RC_OK)
791 {
792 WLog_ERR(TAG, "RegisterCameraHal failed with error %" PRIu32 "", ret);
793 goto error;
794 }
795
796 return ret;
797
798error:
799 cam_v4l_free(&hal->iHal);
800 return ret;
801}
Definition camera.h:221
This struct contains function pointer to initialize/free objects.
Definition collections.h:52
OBJECT_FREE_FN fnObjectFree
Definition collections.h:59