• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1/*
2 * Copyright (C) 2011 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 *      http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17/*
18 * Contains code that is used to capture video frames from a camera device
19 * on Mac. This code uses QTKit API to work with camera devices, and requires
20 * Mac OS at least 10.5
21 */
22
23#import <Cocoa/Cocoa.h>
24#import <QTKit/QTKit.h>
25#import <CoreAudio/CoreAudio.h>
26#include "android/camera/camera-capture.h"
27#include "android/camera/camera-format-converters.h"
28
29#define  E(...)    derror(__VA_ARGS__)
30#define  W(...)    dwarning(__VA_ARGS__)
31#define  D(...)    VERBOSE_PRINT(camera,__VA_ARGS__)
32
33/*******************************************************************************
34 *                     Helper routines
35 ******************************************************************************/
36
37/* Converts internal QT pixel format to a FOURCC value. */
38static uint32_t
39_QTtoFOURCC(uint32_t qt_pix_format)
40{
41  switch (qt_pix_format) {
42    case kCVPixelFormatType_24RGB:
43      return V4L2_PIX_FMT_RGB24;
44
45    case kCVPixelFormatType_24BGR:
46      return V4L2_PIX_FMT_BGR32;
47
48    case kCVPixelFormatType_32ARGB:
49    case kCVPixelFormatType_32RGBA:
50      return V4L2_PIX_FMT_RGB32;
51
52    case kCVPixelFormatType_32BGRA:
53    case kCVPixelFormatType_32ABGR:
54      return V4L2_PIX_FMT_BGR32;
55
56    case kCVPixelFormatType_422YpCbCr8:
57      return V4L2_PIX_FMT_UYVY;
58
59    case kCVPixelFormatType_420YpCbCr8Planar:
60      return V4L2_PIX_FMT_YVU420;
61
62    case 'yuvs':  // kCVPixelFormatType_422YpCbCr8_yuvs - undeclared?
63      return V4L2_PIX_FMT_YUYV;
64
65    default:
66      E("Unrecognized pixel format '%.4s'", (const char*)&qt_pix_format);
67      return 0;
68  }
69}
70
71/*******************************************************************************
72 *                     MacCamera implementation
73 ******************************************************************************/
74
75/* Encapsulates a camera device on MacOS */
76@interface MacCamera : NSObject {
77    /* Capture session. */
78    QTCaptureSession*             capture_session;
79    /* Camera capture device. */
80    QTCaptureDevice*              capture_device;
81    /* Input device registered with the capture session. */
82    QTCaptureDeviceInput*         input_device;
83    /* Output device registered with the capture session. */
84    QTCaptureVideoPreviewOutput*  output_device;
85    /* Current framebuffer. */
86    CVImageBufferRef              current_frame;
87    /* Desired frame width */
88    int                           desired_width;
89    /* Desired frame height */
90    int                           desired_height;
91}
92
93/* Initializes MacCamera instance.
94 * Return:
95 *  Pointer to initialized instance on success, or nil on failure.
96 */
97- (MacCamera*)init;
98
99/* Undoes 'init' */
100- (void)free;
101
102/* Starts capturing video frames.
103 * Param:
104 *  width, height - Requested dimensions for the captured video frames.
105 * Return:
106 *  0 on success, or !=0 on failure.
107 */
108- (int)start_capturing:(int)width:(int)height;
109
110/* Captures a frame from the camera device.
111 * Param:
112 *  framebuffers - Array of framebuffers where to read the frame. Size of this
113 *      array is defined by the 'fbs_num' parameter. Note that the caller must
114 *      make sure that buffers are large enough to contain entire frame captured
115 *      from the device.
116 *  fbs_num - Number of entries in the 'framebuffers' array.
117 * Return:
118 *  0 on success, or non-zero value on failure. There is a special vaule 1
119 *  returned from this routine which indicates that frames are not yet available
120 *  in the device. The client should respond to this value by repeating the
121 *  read, rather than reporting an error.
122 */
123- (int)read_frame:(ClientFrameBuffer*)framebuffers:(int)fbs_num:(float)r_scale:(float)g_scale:(float)b_scale:(float)exp_comp;
124
125@end
126
127@implementation MacCamera
128
129- (MacCamera*)init
130{
131    NSError *error;
132    BOOL success;
133
134    /* Obtain the capture device, make sure it's not used by another
135     * application, and open it. */
136    capture_device =
137        [QTCaptureDevice defaultInputDeviceWithMediaType:QTMediaTypeVideo];
138    if (capture_device == nil) {
139        E("There are no available video devices found.");
140        [self release];
141        return nil;
142    }
143    if ([capture_device isInUseByAnotherApplication]) {
144        E("Default camera device is in use by another application.");
145        [capture_device release];
146        capture_device = nil;
147        [self release];
148        return nil;
149    }
150    success = [capture_device open:&error];
151    if (!success) {
152        E("Unable to open camera device: '%s'",
153          [[error localizedDescription] UTF8String]);
154        [self free];
155        [self release];
156        return nil;
157    }
158
159    /* Create capture session. */
160    capture_session = [[QTCaptureSession alloc] init];
161    if (capture_session == nil) {
162        E("Unable to create capure session.");
163        [self free];
164        [self release];
165        return nil;
166    }
167
168    /* Create an input device and register it with the capture session. */
169    input_device = [[QTCaptureDeviceInput alloc] initWithDevice:capture_device];
170    success = [capture_session addInput:input_device error:&error];
171    if (!success) {
172        E("Unable to initialize input device: '%s'",
173          [[error localizedDescription] UTF8String]);
174        [input_device release];
175        input_device = nil;
176        [self free];
177        [self release];
178        return nil;
179    }
180
181    /* Create an output device and register it with the capture session. */
182    output_device = [[QTCaptureVideoPreviewOutput alloc] init];
183    success = [capture_session addOutput:output_device error:&error];
184    if (!success) {
185        E("Unable to initialize output device: '%s'",
186          [[error localizedDescription] UTF8String]);
187        [output_device release];
188        output_device = nil;
189        [self free];
190        [self release];
191        return nil;
192    }
193    [output_device setDelegate:self];
194
195    return self;
196}
197
198- (void)free
199{
200    /* Uninitialize capture session. */
201    if (capture_session != nil) {
202        /* Make sure that capturing is stopped. */
203        if ([capture_session isRunning]) {
204            [capture_session stopRunning];
205        }
206        /* Detach input and output devices from the session. */
207        if (input_device != nil) {
208            [capture_session removeInput:input_device];
209            [input_device release];
210            input_device = nil;
211        }
212        if (output_device != nil) {
213            [capture_session removeOutput:output_device];
214            [output_device release];
215            output_device = nil;
216        }
217        /* Destroy capture session. */
218        [capture_session release];
219        capture_session = nil;
220    }
221
222    /* Uninitialize capture device. */
223    if (capture_device != nil) {
224        /* Make sure device is not opened. */
225        if ([capture_device isOpen]) {
226            [capture_device close];
227        }
228        [capture_device release];
229        capture_device = nil;
230    }
231
232    /* Release current framebuffer. */
233    if (current_frame != nil) {
234       CVBufferRelease(current_frame);
235       current_frame = nil;
236    }
237}
238
239- (int)start_capturing:(int)width:(int)height
240{
241  if (![capture_session isRunning]) {
242        /* Set desired frame dimensions. */
243        desired_width = width;
244        desired_height = height;
245        [output_device setPixelBufferAttributes:
246          [NSDictionary dictionaryWithObjectsAndKeys:
247              [NSNumber numberWithInt: width], kCVPixelBufferWidthKey,
248              [NSNumber numberWithInt: height], kCVPixelBufferHeightKey,
249              nil]];
250        [capture_session startRunning];
251        return 0;
252  } else if (width == desired_width && height == desired_height) {
253      W("%s: Already capturing %dx%d frames",
254        __FUNCTION__, desired_width, desired_height);
255      return -1;
256  } else {
257      E("%s: Already capturing %dx%d frames. Requested frame dimensions are %dx%d",
258        __FUNCTION__, desired_width, desired_height, width, height);
259      return -1;
260  }
261}
262
263- (int)read_frame:(ClientFrameBuffer*)framebuffers:(int)fbs_num:(float)r_scale:(float)g_scale:(float)b_scale:(float)exp_comp
264{
265    int res = -1;
266
267    /* Frames are pushed by QT in another thread.
268     * So we need a protection here. */
269    @synchronized (self)
270    {
271        if (current_frame != nil) {
272            /* Collect frame info. */
273            const uint32_t pixel_format =
274                _QTtoFOURCC(CVPixelBufferGetPixelFormatType(current_frame));
275            const int frame_width = CVPixelBufferGetWidth(current_frame);
276            const int frame_height = CVPixelBufferGetHeight(current_frame);
277            const size_t frame_size =
278                CVPixelBufferGetBytesPerRow(current_frame) * frame_height;
279
280            /* Get framebuffer pointer. */
281            CVPixelBufferLockBaseAddress(current_frame, 0);
282            const void* pixels = CVPixelBufferGetBaseAddress(current_frame);
283            if (pixels != nil) {
284                /* Convert framebuffer. */
285                res = convert_frame(pixels, pixel_format, frame_size,
286                                    frame_width, frame_height,
287                                    framebuffers, fbs_num,
288                                    r_scale, g_scale, b_scale, exp_comp);
289            } else {
290                E("%s: Unable to obtain framebuffer", __FUNCTION__);
291                res = -1;
292            }
293            CVPixelBufferUnlockBaseAddress(current_frame, 0);
294        } else {
295            /* First frame didn't come in just yet. Let the caller repeat. */
296            res = 1;
297        }
298    }
299
300    return res;
301}
302
303- (void)captureOutput:(QTCaptureOutput*) captureOutput
304                      didOutputVideoFrame:(CVImageBufferRef)videoFrame
305                      withSampleBuffer:(QTSampleBuffer*) sampleBuffer
306                      fromConnection:(QTCaptureConnection*) connection
307{
308    CVImageBufferRef to_release;
309    CVBufferRetain(videoFrame);
310
311    /* Frames are pulled by the client in another thread.
312     * So we need a protection here. */
313    @synchronized (self)
314    {
315        to_release = current_frame;
316        current_frame = videoFrame;
317    }
318    CVBufferRelease(to_release);
319}
320
321@end
322
323/*******************************************************************************
324 *                     CameraDevice routines
325 ******************************************************************************/
326
327typedef struct MacCameraDevice MacCameraDevice;
328/* MacOS-specific camera device descriptor. */
329struct MacCameraDevice {
330    /* Common camera device descriptor. */
331    CameraDevice  header;
332    /* Actual camera device object. */
333    MacCamera*    device;
334};
335
336/* Allocates an instance of MacCameraDevice structure.
337 * Return:
338 *  Allocated instance of MacCameraDevice structure. Note that this routine
339 *  also sets 'opaque' field in the 'header' structure to point back to the
340 *  containing MacCameraDevice instance.
341 */
342static MacCameraDevice*
343_camera_device_alloc(void)
344{
345    MacCameraDevice* cd = (MacCameraDevice*)malloc(sizeof(MacCameraDevice));
346    if (cd != NULL) {
347        memset(cd, 0, sizeof(MacCameraDevice));
348        cd->header.opaque = cd;
349    } else {
350        E("%s: Unable to allocate MacCameraDevice instance", __FUNCTION__);
351    }
352    return cd;
353}
354
355/* Uninitializes and frees MacCameraDevice descriptor.
356 * Note that upon return from this routine memory allocated for the descriptor
357 * will be freed.
358 */
359static void
360_camera_device_free(MacCameraDevice* cd)
361{
362    if (cd != NULL) {
363        if (cd->device != NULL) {
364            [cd->device free];
365            [cd->device release];
366            cd->device = nil;
367        }
368        AFREE(cd);
369    } else {
370        W("%s: No descriptor", __FUNCTION__);
371    }
372}
373
374/* Resets camera device after capturing.
375 * Since new capture request may require different frame dimensions we must
376 * reset frame info cached in the capture window. The only way to do that would
377 * be closing, and reopening it again. */
378static void
379_camera_device_reset(MacCameraDevice* cd)
380{
381    if (cd != NULL && cd->device) {
382        [cd->device free];
383        cd->device = [cd->device init];
384    }
385}
386
387/*******************************************************************************
388 *                     CameraDevice API
389 ******************************************************************************/
390
391CameraDevice*
392camera_device_open(const char* name, int inp_channel)
393{
394    MacCameraDevice* mcd;
395
396    mcd = _camera_device_alloc();
397    if (mcd == NULL) {
398        E("%s: Unable to allocate MacCameraDevice instance", __FUNCTION__);
399        return NULL;
400    }
401    mcd->device = [[MacCamera alloc] init];
402    if (mcd->device == nil) {
403        E("%s: Unable to initialize camera device.", __FUNCTION__);
404        return NULL;
405    }
406    return &mcd->header;
407}
408
409int
410camera_device_start_capturing(CameraDevice* cd,
411                              uint32_t pixel_format,
412                              int frame_width,
413                              int frame_height)
414{
415    MacCameraDevice* mcd;
416
417    /* Sanity checks. */
418    if (cd == NULL || cd->opaque == NULL) {
419        E("%s: Invalid camera device descriptor", __FUNCTION__);
420        return -1;
421    }
422    mcd = (MacCameraDevice*)cd->opaque;
423    if (mcd->device == nil) {
424        E("%s: Camera device is not opened", __FUNCTION__);
425        return -1;
426    }
427
428    return [mcd->device start_capturing:frame_width:frame_height];
429}
430
431int
432camera_device_stop_capturing(CameraDevice* cd)
433{
434    MacCameraDevice* mcd;
435
436    /* Sanity checks. */
437    if (cd == NULL || cd->opaque == NULL) {
438        E("%s: Invalid camera device descriptor", __FUNCTION__);
439        return -1;
440    }
441    mcd = (MacCameraDevice*)cd->opaque;
442    if (mcd->device == nil) {
443        E("%s: Camera device is not opened", __FUNCTION__);
444        return -1;
445    }
446
447    /* Reset capture settings, so next call to capture can set its own. */
448    _camera_device_reset(mcd);
449
450    return 0;
451}
452
453int
454camera_device_read_frame(CameraDevice* cd,
455                         ClientFrameBuffer* framebuffers,
456                         int fbs_num,
457                         float r_scale,
458                         float g_scale,
459                         float b_scale,
460                         float exp_comp)
461{
462    MacCameraDevice* mcd;
463
464    /* Sanity checks. */
465    if (cd == NULL || cd->opaque == NULL) {
466        E("%s: Invalid camera device descriptor", __FUNCTION__);
467        return -1;
468    }
469    mcd = (MacCameraDevice*)cd->opaque;
470    if (mcd->device == nil) {
471        E("%s: Camera device is not opened", __FUNCTION__);
472        return -1;
473    }
474
475    return [mcd->device read_frame:framebuffers:fbs_num:r_scale:g_scale:b_scale:exp_comp];
476}
477
478void
479camera_device_close(CameraDevice* cd)
480{
481    /* Sanity checks. */
482    if (cd == NULL || cd->opaque == NULL) {
483        E("%s: Invalid camera device descriptor", __FUNCTION__);
484    } else {
485        _camera_device_free((MacCameraDevice*)cd->opaque);
486    }
487}
488
489int
490enumerate_camera_devices(CameraInfo* cis, int max)
491{
492/* Array containing emulated webcam frame dimensions.
493 * QT API provides device independent frame dimensions, by scaling frames
494 * received from the device to whatever dimensions were requested for the
495 * output device. So, we can just use a small set of frame dimensions to
496 * emulate.
497 */
498static const CameraFrameDim _emulate_dims[] =
499{
500  /* Emulates 640x480 frame. */
501  {640, 480},
502  /* Emulates 352x288 frame (required by camera framework). */
503  {352, 288},
504  /* Emulates 320x240 frame (required by camera framework). */
505  {320, 240},
506  /* Emulates 176x144 frame (required by camera framework). */
507  {176, 144}
508};
509
510    /* Obtain default video device. QT API doesn't really provide a reliable
511     * way to identify camera devices. There is a QTCaptureDevice::uniqueId
512     * method that supposedly does that, but in some cases it just doesn't
513     * work. Until we figure out a reliable device identification, we will
514     * stick to using only one (default) camera for emulation. */
515    QTCaptureDevice* video_dev =
516        [QTCaptureDevice defaultInputDeviceWithMediaType:QTMediaTypeVideo];
517    if (video_dev == nil) {
518        D("No web cameras are connected to the host.");
519        return 0;
520    }
521
522    /* Obtain pixel format for the device. */
523    NSArray* pix_formats = [video_dev formatDescriptions];
524    if (pix_formats == nil || [pix_formats count] == 0) {
525        E("Unable to obtain pixel format for the default camera device.");
526        [video_dev release];
527        return 0;
528    }
529    const uint32_t qt_pix_format = [[pix_formats objectAtIndex:0] formatType];
530    [pix_formats release];
531
532    /* Obtain FOURCC pixel format for the device. */
533    cis[0].pixel_format = _QTtoFOURCC(qt_pix_format);
534    if (cis[0].pixel_format == 0) {
535        /* Unsupported pixel format. */
536        E("Pixel format '%.4s' reported by the camera device is unsupported",
537          (const char*)&qt_pix_format);
538        [video_dev release];
539        return 0;
540    }
541
542    /* Initialize camera info structure. */
543    cis[0].frame_sizes = (CameraFrameDim*)malloc(sizeof(_emulate_dims));
544    if (cis[0].frame_sizes != NULL) {
545        cis[0].frame_sizes_num = sizeof(_emulate_dims) / sizeof(*_emulate_dims);
546        memcpy(cis[0].frame_sizes, _emulate_dims, sizeof(_emulate_dims));
547        cis[0].device_name = ASTRDUP("webcam0");
548        cis[0].inp_channel = 0;
549        cis[0].display_name = ASTRDUP("webcam0");
550        cis[0].in_use = 0;
551        [video_dev release];
552        return 1;
553    } else {
554        E("Unable to allocate memory for camera information.");
555        [video_dev release];
556        return 0;
557    }
558}
559