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 implementation of a class EmulatedCameraFactory that manages cameras
19  * available for emulation.
20  */
21 
22 //#define LOG_NDEBUG 0
23 #define LOG_TAG "EmulatedCamera_Factory"
24 #include <cutils/log.h>
25 #include <cutils/properties.h>
26 #include "EmulatedQemuCamera.h"
27 #include "EmulatedFakeCamera.h"
28 #include "EmulatedFakeCamera2.h"
29 #include "EmulatedFakeCamera3.h"
30 #include "EmulatedCameraHotplugThread.h"
31 #include "EmulatedCameraFactory.h"
32 
33 extern camera_module_t HAL_MODULE_INFO_SYM;
34 
35 /* A global instance of EmulatedCameraFactory is statically instantiated and
36  * initialized when camera emulation HAL is loaded.
37  */
38 android::EmulatedCameraFactory  gEmulatedCameraFactory;
39 
40 namespace android {
41 
EmulatedCameraFactory()42 EmulatedCameraFactory::EmulatedCameraFactory()
43         : mQemuClient(),
44           mEmulatedCameras(NULL),
45           mEmulatedCameraNum(0),
46           mFakeCameraNum(0),
47           mConstructedOK(false),
48           mCallbacks(NULL)
49 {
50     status_t res;
51     /* Connect to the factory service in the emulator, and create Qemu cameras. */
52     if (mQemuClient.connectClient(NULL) == NO_ERROR) {
53         /* Connection has succeeded. Create emulated cameras for each camera
54          * device, reported by the service. */
55         createQemuCameras();
56     }
57 
58     if (isBackFakeCameraEmulationOn()) {
59         /* Camera ID. */
60         const int camera_id = mEmulatedCameraNum;
61         /* Use fake camera to emulate back-facing camera. */
62         mEmulatedCameraNum++;
63 
64         /* Make sure that array is allocated (in case there were no 'qemu'
65          * cameras created. Note that we preallocate the array so it may contain
66          * two fake cameras: one facing back, and another facing front. */
67         if (mEmulatedCameras == NULL) {
68             mEmulatedCameras = new EmulatedBaseCamera*[mEmulatedCameraNum + 1];
69             if (mEmulatedCameras == NULL) {
70                 ALOGE("%s: Unable to allocate emulated camera array for %d entries",
71                      __FUNCTION__, mEmulatedCameraNum);
72                 return;
73             }
74             memset(mEmulatedCameras, 0,
75                     (mEmulatedCameraNum + 1) * sizeof(EmulatedBaseCamera*));
76         }
77 
78         /* Create, and initialize the fake camera */
79         switch (getBackCameraHalVersion()) {
80             case 1:
81                 mEmulatedCameras[camera_id] =
82                         new EmulatedFakeCamera(camera_id, true,
83                                 &HAL_MODULE_INFO_SYM.common);
84                 break;
85             case 2:
86                 mEmulatedCameras[camera_id] =
87                         new EmulatedFakeCamera2(camera_id, true,
88                                 &HAL_MODULE_INFO_SYM.common);
89                 break;
90             case 3:
91                 mEmulatedCameras[camera_id] =
92                         new EmulatedFakeCamera3(camera_id, true,
93                                 &HAL_MODULE_INFO_SYM.common);
94                 break;
95             default:
96                 ALOGE("%s: Unknown back camera hal version requested: %d", __FUNCTION__,
97                         getBackCameraHalVersion());
98         }
99         if (mEmulatedCameras[camera_id] != NULL) {
100             ALOGV("%s: Back camera device version is %d", __FUNCTION__,
101                     getBackCameraHalVersion());
102             res = mEmulatedCameras[camera_id]->Initialize();
103             if (res != NO_ERROR) {
104                 ALOGE("%s: Unable to intialize back camera %d: %s (%d)",
105                         __FUNCTION__, camera_id, strerror(-res), res);
106                 delete mEmulatedCameras[camera_id];
107                 mEmulatedCameraNum--;
108             }
109         } else {
110             mEmulatedCameraNum--;
111             ALOGE("%s: Unable to instantiate fake camera class", __FUNCTION__);
112         }
113     }
114 
115     if (isFrontFakeCameraEmulationOn()) {
116         /* Camera ID. */
117         const int camera_id = mEmulatedCameraNum;
118         /* Use fake camera to emulate front-facing camera. */
119         mEmulatedCameraNum++;
120 
121         /* Make sure that array is allocated (in case there were no 'qemu'
122          * cameras created. */
123         if (mEmulatedCameras == NULL) {
124             mEmulatedCameras = new EmulatedBaseCamera*[mEmulatedCameraNum];
125             if (mEmulatedCameras == NULL) {
126                 ALOGE("%s: Unable to allocate emulated camera array for %d entries",
127                      __FUNCTION__, mEmulatedCameraNum);
128                 return;
129             }
130             memset(mEmulatedCameras, 0,
131                     mEmulatedCameraNum * sizeof(EmulatedBaseCamera*));
132         }
133 
134         /* Create, and initialize the fake camera */
135         switch (getFrontCameraHalVersion()) {
136             case 1:
137                 mEmulatedCameras[camera_id] =
138                         new EmulatedFakeCamera(camera_id, false,
139                                 &HAL_MODULE_INFO_SYM.common);
140                 break;
141             case 2:
142                 mEmulatedCameras[camera_id] =
143                         new EmulatedFakeCamera2(camera_id, false,
144                                 &HAL_MODULE_INFO_SYM.common);
145                 break;
146             case 3:
147                 mEmulatedCameras[camera_id] =
148                         new EmulatedFakeCamera3(camera_id, false,
149                                 &HAL_MODULE_INFO_SYM.common);
150                 break;
151             default:
152                 ALOGE("%s: Unknown front camera hal version requested: %d",
153                         __FUNCTION__,
154                         getFrontCameraHalVersion());
155         }
156         if (mEmulatedCameras[camera_id] != NULL) {
157             ALOGV("%s: Front camera device version is %d", __FUNCTION__,
158                     getFrontCameraHalVersion());
159             res = mEmulatedCameras[camera_id]->Initialize();
160             if (res != NO_ERROR) {
161                 ALOGE("%s: Unable to intialize front camera %d: %s (%d)",
162                         __FUNCTION__, camera_id, strerror(-res), res);
163                 delete mEmulatedCameras[camera_id];
164                 mEmulatedCameraNum--;
165             }
166         } else {
167             mEmulatedCameraNum--;
168             ALOGE("%s: Unable to instantiate fake camera class", __FUNCTION__);
169         }
170     }
171 
172     ALOGV("%d cameras are being emulated. %d of them are fake cameras.",
173           mEmulatedCameraNum, mFakeCameraNum);
174 
175     /* Create hotplug thread */
176     {
177         Vector<int> cameraIdVector;
178         for (int i = 0; i < mEmulatedCameraNum; ++i) {
179             cameraIdVector.push_back(i);
180         }
181         mHotplugThread = new EmulatedCameraHotplugThread(&cameraIdVector[0],
182                                                          mEmulatedCameraNum);
183         mHotplugThread->run();
184     }
185 
186     mConstructedOK = true;
187 }
188 
~EmulatedCameraFactory()189 EmulatedCameraFactory::~EmulatedCameraFactory()
190 {
191     if (mEmulatedCameras != NULL) {
192         for (int n = 0; n < mEmulatedCameraNum; n++) {
193             if (mEmulatedCameras[n] != NULL) {
194                 delete mEmulatedCameras[n];
195             }
196         }
197         delete[] mEmulatedCameras;
198     }
199 
200     if (mHotplugThread != NULL) {
201         mHotplugThread->requestExit();
202         mHotplugThread->join();
203     }
204 }
205 
206 /****************************************************************************
207  * Camera HAL API handlers.
208  *
209  * Each handler simply verifies existence of an appropriate EmulatedBaseCamera
210  * instance, and dispatches the call to that instance.
211  *
212  ***************************************************************************/
213 
cameraDeviceOpen(int camera_id,hw_device_t ** device)214 int EmulatedCameraFactory::cameraDeviceOpen(int camera_id, hw_device_t** device)
215 {
216     ALOGV("%s: id = %d", __FUNCTION__, camera_id);
217 
218     *device = NULL;
219 
220     if (!isConstructedOK()) {
221         ALOGE("%s: EmulatedCameraFactory has failed to initialize", __FUNCTION__);
222         return -EINVAL;
223     }
224 
225     if (camera_id < 0 || camera_id >= getEmulatedCameraNum()) {
226         ALOGE("%s: Camera id %d is out of bounds (%d)",
227              __FUNCTION__, camera_id, getEmulatedCameraNum());
228         return -ENODEV;
229     }
230 
231     return mEmulatedCameras[camera_id]->connectCamera(device);
232 }
233 
getCameraInfo(int camera_id,struct camera_info * info)234 int EmulatedCameraFactory::getCameraInfo(int camera_id, struct camera_info* info)
235 {
236     ALOGV("%s: id = %d", __FUNCTION__, camera_id);
237 
238     if (!isConstructedOK()) {
239         ALOGE("%s: EmulatedCameraFactory has failed to initialize", __FUNCTION__);
240         return -EINVAL;
241     }
242 
243     if (camera_id < 0 || camera_id >= getEmulatedCameraNum()) {
244         ALOGE("%s: Camera id %d is out of bounds (%d)",
245              __FUNCTION__, camera_id, getEmulatedCameraNum());
246         return -ENODEV;
247     }
248 
249     return mEmulatedCameras[camera_id]->getCameraInfo(info);
250 }
251 
setCallbacks(const camera_module_callbacks_t * callbacks)252 int EmulatedCameraFactory::setCallbacks(
253         const camera_module_callbacks_t *callbacks)
254 {
255     ALOGV("%s: callbacks = %p", __FUNCTION__, callbacks);
256 
257     mCallbacks = callbacks;
258 
259     return OK;
260 }
261 
getVendorTagOps(vendor_tag_ops_t * ops)262 void EmulatedCameraFactory::getVendorTagOps(vendor_tag_ops_t* ops) {
263     ALOGV("%s: ops = %p", __FUNCTION__, ops);
264 
265     // No vendor tags defined for emulator yet, so not touching ops
266 }
267 
268 /****************************************************************************
269  * Camera HAL API callbacks.
270  ***************************************************************************/
271 
device_open(const hw_module_t * module,const char * name,hw_device_t ** device)272 int EmulatedCameraFactory::device_open(const hw_module_t* module,
273                                        const char* name,
274                                        hw_device_t** device)
275 {
276     /*
277      * Simply verify the parameters, and dispatch the call inside the
278      * EmulatedCameraFactory instance.
279      */
280 
281     if (module != &HAL_MODULE_INFO_SYM.common) {
282         ALOGE("%s: Invalid module %p expected %p",
283              __FUNCTION__, module, &HAL_MODULE_INFO_SYM.common);
284         return -EINVAL;
285     }
286     if (name == NULL) {
287         ALOGE("%s: NULL name is not expected here", __FUNCTION__);
288         return -EINVAL;
289     }
290 
291     return gEmulatedCameraFactory.cameraDeviceOpen(atoi(name), device);
292 }
293 
get_number_of_cameras(void)294 int EmulatedCameraFactory::get_number_of_cameras(void)
295 {
296     return gEmulatedCameraFactory.getEmulatedCameraNum();
297 }
298 
get_camera_info(int camera_id,struct camera_info * info)299 int EmulatedCameraFactory::get_camera_info(int camera_id,
300                                            struct camera_info* info)
301 {
302     return gEmulatedCameraFactory.getCameraInfo(camera_id, info);
303 }
304 
set_callbacks(const camera_module_callbacks_t * callbacks)305 int EmulatedCameraFactory::set_callbacks(
306         const camera_module_callbacks_t *callbacks)
307 {
308     return gEmulatedCameraFactory.setCallbacks(callbacks);
309 }
310 
get_vendor_tag_ops(vendor_tag_ops_t * ops)311 void EmulatedCameraFactory::get_vendor_tag_ops(vendor_tag_ops_t* ops)
312 {
313     gEmulatedCameraFactory.getVendorTagOps(ops);
314 }
315 
open_legacy(const struct hw_module_t * module,const char * id,uint32_t halVersion,struct hw_device_t ** device)316 int EmulatedCameraFactory::open_legacy(const struct hw_module_t* module,
317         const char* id, uint32_t halVersion, struct hw_device_t** device) {
318     // Not supporting legacy open
319     return -ENOSYS;
320 }
321 
322 /********************************************************************************
323  * Internal API
324  *******************************************************************************/
325 
326 /*
327  * Camera information tokens passed in response to the "list" factory query.
328  */
329 
330 /* Device name token. */
331 static const char lListNameToken[]    = "name=";
332 /* Frame dimensions token. */
333 static const char lListDimsToken[]    = "framedims=";
334 /* Facing direction token. */
335 static const char lListDirToken[]     = "dir=";
336 
createQemuCameras()337 void EmulatedCameraFactory::createQemuCameras()
338 {
339     /* Obtain camera list. */
340     char* camera_list = NULL;
341     status_t res = mQemuClient.listCameras(&camera_list);
342     /* Empty list, or list containing just an EOL means that there were no
343      * connected cameras found. */
344     if (res != NO_ERROR || camera_list == NULL || *camera_list == '\0' ||
345         *camera_list == '\n') {
346         if (camera_list != NULL) {
347             free(camera_list);
348         }
349         return;
350     }
351 
352     /*
353      * Calculate number of connected cameras. Number of EOLs in the camera list
354      * is the number of the connected cameras.
355      */
356 
357     int num = 0;
358     const char* eol = strchr(camera_list, '\n');
359     while (eol != NULL) {
360         num++;
361         eol = strchr(eol + 1, '\n');
362     }
363 
364     /* Allocate the array for emulated camera instances. Note that we allocate
365      * two more entries for back and front fake camera emulation. */
366     mEmulatedCameras = new EmulatedBaseCamera*[num + 2];
367     if (mEmulatedCameras == NULL) {
368         ALOGE("%s: Unable to allocate emulated camera array for %d entries",
369              __FUNCTION__, num + 1);
370         free(camera_list);
371         return;
372     }
373     memset(mEmulatedCameras, 0, sizeof(EmulatedBaseCamera*) * (num + 1));
374 
375     /*
376      * Iterate the list, creating, and initializin emulated qemu cameras for each
377      * entry (line) in the list.
378      */
379 
380     int index = 0;
381     char* cur_entry = camera_list;
382     while (cur_entry != NULL && *cur_entry != '\0' && index < num) {
383         /* Find the end of the current camera entry, and terminate it with zero
384          * for simpler string manipulation. */
385         char* next_entry = strchr(cur_entry, '\n');
386         if (next_entry != NULL) {
387             *next_entry = '\0';
388             next_entry++;   // Start of the next entry.
389         }
390 
391         /* Find 'name', 'framedims', and 'dir' tokens that are required here. */
392         char* name_start = strstr(cur_entry, lListNameToken);
393         char* dim_start = strstr(cur_entry, lListDimsToken);
394         char* dir_start = strstr(cur_entry, lListDirToken);
395         if (name_start != NULL && dim_start != NULL && dir_start != NULL) {
396             /* Advance to the token values. */
397             name_start += strlen(lListNameToken);
398             dim_start += strlen(lListDimsToken);
399             dir_start += strlen(lListDirToken);
400 
401             /* Terminate token values with zero. */
402             char* s = strchr(name_start, ' ');
403             if (s != NULL) {
404                 *s = '\0';
405             }
406             s = strchr(dim_start, ' ');
407             if (s != NULL) {
408                 *s = '\0';
409             }
410             s = strchr(dir_start, ' ');
411             if (s != NULL) {
412                 *s = '\0';
413             }
414 
415             /* Create and initialize qemu camera. */
416             EmulatedQemuCamera* qemu_cam =
417                 new EmulatedQemuCamera(index, &HAL_MODULE_INFO_SYM.common);
418             if (NULL != qemu_cam) {
419                 res = qemu_cam->Initialize(name_start, dim_start, dir_start);
420                 if (res == NO_ERROR) {
421                     mEmulatedCameras[index] = qemu_cam;
422                     index++;
423                 } else {
424                     delete qemu_cam;
425                 }
426             } else {
427                 ALOGE("%s: Unable to instantiate EmulatedQemuCamera",
428                      __FUNCTION__);
429             }
430         } else {
431             ALOGW("%s: Bad camera information: %s", __FUNCTION__, cur_entry);
432         }
433 
434         cur_entry = next_entry;
435     }
436 
437     mEmulatedCameraNum = index;
438 }
439 
isBackFakeCameraEmulationOn()440 bool EmulatedCameraFactory::isBackFakeCameraEmulationOn()
441 {
442     /* Defined by 'qemu.sf.fake_camera' boot property: if property exist, and
443      * is set to 'both', or 'back', then fake camera is used to emulate back
444      * camera. */
445     char prop[PROPERTY_VALUE_MAX];
446     if ((property_get("qemu.sf.fake_camera", prop, NULL) > 0) &&
447         (!strcmp(prop, "both") || !strcmp(prop, "back"))) {
448         return true;
449     } else {
450         return false;
451     }
452 }
453 
getBackCameraHalVersion()454 int EmulatedCameraFactory::getBackCameraHalVersion()
455 {
456     /* Defined by 'qemu.sf.back_camera_hal_version' boot property: if the
457      * property doesn't exist, it is assumed to be 1. */
458     char prop[PROPERTY_VALUE_MAX];
459     if (property_get("qemu.sf.back_camera_hal", prop, NULL) > 0) {
460         char *prop_end = prop;
461         int val = strtol(prop, &prop_end, 10);
462         if (*prop_end == '\0') {
463             return val;
464         }
465         // Badly formatted property, should just be a number
466         ALOGE("qemu.sf.back_camera_hal is not a number: %s", prop);
467     }
468     return 1;
469 }
470 
isFrontFakeCameraEmulationOn()471 bool EmulatedCameraFactory::isFrontFakeCameraEmulationOn()
472 {
473     /* Defined by 'qemu.sf.fake_camera' boot property: if property exist, and
474      * is set to 'both', or 'front', then fake camera is used to emulate front
475      * camera. */
476     char prop[PROPERTY_VALUE_MAX];
477     if ((property_get("qemu.sf.fake_camera", prop, NULL) > 0) &&
478         (!strcmp(prop, "both") || !strcmp(prop, "front"))) {
479         return true;
480     } else {
481         return false;
482     }
483 }
484 
getFrontCameraHalVersion()485 int EmulatedCameraFactory::getFrontCameraHalVersion()
486 {
487     /* Defined by 'qemu.sf.front_camera_hal_version' boot property: if the
488      * property doesn't exist, it is assumed to be 1. */
489     char prop[PROPERTY_VALUE_MAX];
490     if (property_get("qemu.sf.front_camera_hal", prop, NULL) > 0) {
491         char *prop_end = prop;
492         int val = strtol(prop, &prop_end, 10);
493         if (*prop_end == '\0') {
494             return val;
495         }
496         // Badly formatted property, should just be a number
497         ALOGE("qemu.sf.front_camera_hal is not a number: %s", prop);
498     }
499     return 1;
500 }
501 
onStatusChanged(int cameraId,int newStatus)502 void EmulatedCameraFactory::onStatusChanged(int cameraId, int newStatus) {
503 
504     EmulatedBaseCamera *cam = mEmulatedCameras[cameraId];
505     if (!cam) {
506         ALOGE("%s: Invalid camera ID %d", __FUNCTION__, cameraId);
507         return;
508     }
509 
510     /**
511      * (Order is important)
512      * Send the callback first to framework, THEN close the camera.
513      */
514 
515     if (newStatus == cam->getHotplugStatus()) {
516         ALOGW("%s: Ignoring transition to the same status", __FUNCTION__);
517         return;
518     }
519 
520     const camera_module_callbacks_t* cb = mCallbacks;
521     if (cb != NULL && cb->camera_device_status_change != NULL) {
522         cb->camera_device_status_change(cb, cameraId, newStatus);
523     }
524 
525     if (newStatus == CAMERA_DEVICE_STATUS_NOT_PRESENT) {
526         cam->unplugCamera();
527     } else if (newStatus == CAMERA_DEVICE_STATUS_PRESENT) {
528         cam->plugCamera();
529     }
530 
531 }
532 
533 /********************************************************************************
534  * Initializer for the static member structure.
535  *******************************************************************************/
536 
537 /* Entry point for camera HAL API. */
538 struct hw_module_methods_t EmulatedCameraFactory::mCameraModuleMethods = {
539     open: EmulatedCameraFactory::device_open
540 };
541 
542 }; /* namespace android */
543