• 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 implementation of classes that encapsulate connection to camera
19  * services in the emulator via qemu pipe.
20  */
21 
22 #define LOG_NDEBUG 1
23 #define LOG_TAG "EmulatedCamera_QemuClient"
24 #include <cutils/log.h>
25 #include "EmulatedCamera.h"
26 #include "QemuClient.h"
27 
28 #define LOG_QUERIES 0
29 #if LOG_QUERIES
30 #define LOGQ(...)   ALOGD(__VA_ARGS__)
31 #else
32 #define LOGQ(...)   (void(0))
33 
34 #endif  // LOG_QUERIES
35 namespace android {
36 
37 /****************************************************************************
38  * Qemu query
39  ***************************************************************************/
40 
QemuQuery()41 QemuQuery::QemuQuery()
42     : mQuery(mQueryPrealloc),
43       mQueryDeliveryStatus(NO_ERROR),
44       mReplyBuffer(NULL),
45       mReplyData(NULL),
46       mReplySize(0),
47       mReplyDataSize(0),
48       mReplyStatus(0)
49 {
50     *mQuery = '\0';
51 }
52 
QemuQuery(const char * query_string)53 QemuQuery::QemuQuery(const char* query_string)
54     : mQuery(mQueryPrealloc),
55       mQueryDeliveryStatus(NO_ERROR),
56       mReplyBuffer(NULL),
57       mReplyData(NULL),
58       mReplySize(0),
59       mReplyDataSize(0),
60       mReplyStatus(0)
61 {
62     mQueryDeliveryStatus = QemuQuery::createQuery(query_string, NULL);
63 }
64 
QemuQuery(const char * query_name,const char * query_param)65 QemuQuery::QemuQuery(const char* query_name, const char* query_param)
66     : mQuery(mQueryPrealloc),
67       mQueryDeliveryStatus(NO_ERROR),
68       mReplyBuffer(NULL),
69       mReplyData(NULL),
70       mReplySize(0),
71       mReplyDataSize(0),
72       mReplyStatus(0)
73 {
74     mQueryDeliveryStatus = QemuQuery::createQuery(query_name, query_param);
75 }
76 
~QemuQuery()77 QemuQuery::~QemuQuery()
78 {
79     QemuQuery::resetQuery();
80 }
81 
createQuery(const char * name,const char * param)82 status_t QemuQuery::createQuery(const char* name, const char* param)
83 {
84     /* Reset from the previous use. */
85     resetQuery();
86 
87     /* Query name cannot be NULL or an empty string. */
88     if (name == NULL || *name == '\0') {
89         ALOGE("%s: NULL or an empty string is passed as query name.",
90              __FUNCTION__);
91         mQueryDeliveryStatus = EINVAL;
92         return EINVAL;
93     }
94 
95     const size_t name_len = strlen(name);
96     const size_t param_len = (param != NULL) ? strlen(param) : 0;
97     const size_t required = strlen(name) + (param_len ? (param_len + 2) : 1);
98 
99     if (required > sizeof(mQueryPrealloc)) {
100         /* Preallocated buffer was too small. Allocate a bigger query buffer. */
101         mQuery = new char[required];
102         if (mQuery == NULL) {
103             ALOGE("%s: Unable to allocate %d bytes for query buffer",
104                  __FUNCTION__, required);
105             mQueryDeliveryStatus = ENOMEM;
106             return ENOMEM;
107         }
108     }
109 
110     /* At this point mQuery buffer is big enough for the query. */
111     if (param_len) {
112         sprintf(mQuery, "%s %s", name, param);
113     } else {
114         memcpy(mQuery, name, name_len + 1);
115     }
116 
117     return NO_ERROR;
118 }
119 
completeQuery(status_t status)120 status_t QemuQuery::completeQuery(status_t status)
121 {
122     /* Save query completion status. */
123     mQueryDeliveryStatus = status;
124     if (mQueryDeliveryStatus != NO_ERROR) {
125         return mQueryDeliveryStatus;
126     }
127 
128     /* Make sure reply buffer contains at least 'ok', or 'ko'.
129      * Note that 'ok', or 'ko' prefixes are always 3 characters long: in case
130      * there are more data in the reply, that data will be separated from 'ok'/'ko'
131      * with a ':'. If there is no more data in the reply, the prefix will be
132      * zero-terminated, and the terminator will be inculded in the reply. */
133     if (mReplyBuffer == NULL || mReplySize < 3) {
134         ALOGE("%s: Invalid reply to the query", __FUNCTION__);
135         mQueryDeliveryStatus = EINVAL;
136         return EINVAL;
137     }
138 
139     /* Lets see the reply status. */
140     if (!memcmp(mReplyBuffer, "ok", 2)) {
141         mReplyStatus = 1;
142     } else if (!memcmp(mReplyBuffer, "ko", 2)) {
143         mReplyStatus = 0;
144     } else {
145         ALOGE("%s: Invalid query reply: '%s'", __FUNCTION__, mReplyBuffer);
146         mQueryDeliveryStatus = EINVAL;
147         return EINVAL;
148     }
149 
150     /* Lets see if there are reply data that follow. */
151     if (mReplySize > 3) {
152         /* There are extra data. Make sure they are separated from the status
153          * with a ':' */
154         if (mReplyBuffer[2] != ':') {
155             ALOGE("%s: Invalid query reply: '%s'", __FUNCTION__, mReplyBuffer);
156             mQueryDeliveryStatus = EINVAL;
157             return EINVAL;
158         }
159         mReplyData = mReplyBuffer + 3;
160         mReplyDataSize = mReplySize - 3;
161     } else {
162         /* Make sure reply buffer containing just 'ok'/'ko' ends with
163          * zero-terminator. */
164         if (mReplyBuffer[2] != '\0') {
165             ALOGE("%s: Invalid query reply: '%s'", __FUNCTION__, mReplyBuffer);
166             mQueryDeliveryStatus = EINVAL;
167             return EINVAL;
168         }
169     }
170 
171     return NO_ERROR;
172 }
173 
resetQuery()174 void QemuQuery::resetQuery()
175 {
176     if (mQuery != NULL && mQuery != mQueryPrealloc) {
177         delete[] mQuery;
178     }
179     mQuery = mQueryPrealloc;
180     mQueryDeliveryStatus = NO_ERROR;
181     if (mReplyBuffer != NULL) {
182         free(mReplyBuffer);
183         mReplyBuffer = NULL;
184     }
185     mReplyData = NULL;
186     mReplySize = mReplyDataSize = 0;
187     mReplyStatus = 0;
188 }
189 
190 /****************************************************************************
191  * Qemu client base
192  ***************************************************************************/
193 
194 /* Camera service name. */
195 const char QemuClient::mCameraServiceName[]   = "camera";
196 
QemuClient()197 QemuClient::QemuClient()
198     : mPipeFD(-1)
199 {
200 }
201 
~QemuClient()202 QemuClient::~QemuClient()
203 {
204     if (mPipeFD >= 0) {
205         close(mPipeFD);
206     }
207 }
208 
209 /****************************************************************************
210  * Qemu client API
211  ***************************************************************************/
212 
connectClient(const char * param)213 status_t QemuClient::connectClient(const char* param)
214 {
215     ALOGV("%s: '%s'", __FUNCTION__, param ? param : "");
216 
217     /* Make sure that client is not connected already. */
218     if (mPipeFD >= 0) {
219         ALOGE("%s: Qemu client is already connected", __FUNCTION__);
220         return EINVAL;
221     }
222 
223     /* Select one of the two: 'factory', or 'emulated camera' service */
224     if (param == NULL || *param == '\0') {
225         /* No parameters: connect to the factory service. */
226         char pipe_name[512];
227         snprintf(pipe_name, sizeof(pipe_name), "qemud:%s", mCameraServiceName);
228         mPipeFD = qemu_pipe_open(pipe_name);
229     } else {
230         /* One extra char ':' that separates service name and parameters + six
231          * characters for 'qemud:'. This is required by qemu pipe protocol. */
232         char* connection_str = new char[strlen(mCameraServiceName) +
233                                         strlen(param) + 8];
234         sprintf(connection_str, "qemud:%s:%s", mCameraServiceName, param);
235 
236         mPipeFD = qemu_pipe_open(connection_str);
237         delete[] connection_str;
238     }
239     if (mPipeFD < 0) {
240         ALOGE("%s: Unable to connect to the camera service '%s': %s",
241              __FUNCTION__, param ? param : "Factory", strerror(errno));
242         return errno ? errno : EINVAL;
243     }
244 
245     return NO_ERROR;
246 }
247 
disconnectClient()248 void QemuClient::disconnectClient()
249 {
250     ALOGV("%s", __FUNCTION__);
251 
252     if (mPipeFD >= 0) {
253         close(mPipeFD);
254         mPipeFD = -1;
255     }
256 }
257 
sendMessage(const void * data,size_t data_size)258 status_t QemuClient::sendMessage(const void* data, size_t data_size)
259 {
260     if (mPipeFD < 0) {
261         ALOGE("%s: Qemu client is not connected", __FUNCTION__);
262         return EINVAL;
263     }
264 
265     /* Note that we don't use here qemud_client_send, since with qemu pipes we
266      * don't need to provide payload size prior to payload when we're writing to
267      * the pipe. So, we can use simple write, and qemu pipe will take care of the
268      * rest, calling the receiving end with the number of bytes transferred. */
269     const size_t written = qemud_fd_write(mPipeFD, data, data_size);
270     if (written == data_size) {
271         return NO_ERROR;
272     } else {
273         ALOGE("%s: Error sending data via qemu pipe: '%s'",
274              __FUNCTION__, strerror(errno));
275         return errno ? errno : EIO;
276     }
277 }
278 
receiveMessage(void ** data,size_t * data_size)279 status_t QemuClient::receiveMessage(void** data, size_t* data_size)
280 {
281     *data = NULL;
282     *data_size = 0;
283 
284     if (mPipeFD < 0) {
285         ALOGE("%s: Qemu client is not connected", __FUNCTION__);
286         return EINVAL;
287     }
288 
289     /* The way the service replies to a query, it sends payload size first, and
290      * then it sends the payload itself. Note that payload size is sent as a
291      * string, containing 8 characters representing a hexadecimal payload size
292      * value. Note also, that the string doesn't contain zero-terminator. */
293     size_t payload_size;
294     char payload_size_str[9];
295     int rd_res = qemud_fd_read(mPipeFD, payload_size_str, 8);
296     if (rd_res != 8) {
297         ALOGE("%s: Unable to obtain payload size: %s",
298              __FUNCTION__, strerror(errno));
299         return errno ? errno : EIO;
300     }
301 
302     /* Convert payload size. */
303     errno = 0;
304     payload_size_str[8] = '\0';
305     payload_size = strtol(payload_size_str, NULL, 16);
306     if (errno) {
307         ALOGE("%s: Invalid payload size '%s'", __FUNCTION__, payload_size_str);
308         return EIO;
309     }
310 
311     /* Allocate payload data buffer, and read the payload there. */
312     *data = malloc(payload_size);
313     if (*data == NULL) {
314         ALOGE("%s: Unable to allocate %d bytes payload buffer",
315              __FUNCTION__, payload_size);
316         return ENOMEM;
317     }
318     rd_res = qemud_fd_read(mPipeFD, *data, payload_size);
319     if (static_cast<size_t>(rd_res) == payload_size) {
320         *data_size = payload_size;
321         return NO_ERROR;
322     } else {
323         ALOGE("%s: Read size %d doesnt match expected payload size %d: %s",
324              __FUNCTION__, rd_res, payload_size, strerror(errno));
325         free(*data);
326         *data = NULL;
327         return errno ? errno : EIO;
328     }
329 }
330 
doQuery(QemuQuery * query)331 status_t QemuClient::doQuery(QemuQuery* query)
332 {
333     /* Make sure that query has been successfuly constructed. */
334     if (query->mQueryDeliveryStatus != NO_ERROR) {
335         ALOGE("%s: Query is invalid", __FUNCTION__);
336         return query->mQueryDeliveryStatus;
337     }
338 
339     LOGQ("Send query '%s'", query->mQuery);
340 
341     /* Send the query. */
342     status_t res = sendMessage(query->mQuery, strlen(query->mQuery) + 1);
343     if (res == NO_ERROR) {
344         /* Read the response. */
345         res = receiveMessage(reinterpret_cast<void**>(&query->mReplyBuffer),
346                       &query->mReplySize);
347         if (res == NO_ERROR) {
348             LOGQ("Response to query '%s': Status = '%.2s', %d bytes in response",
349                  query->mQuery, query->mReplyBuffer, query->mReplySize);
350         } else {
351             ALOGE("%s Response to query '%s' has failed: %s",
352                  __FUNCTION__, query->mQuery, strerror(res));
353         }
354     } else {
355         ALOGE("%s: Send query '%s' failed: %s",
356              __FUNCTION__, query->mQuery, strerror(res));
357     }
358 
359     /* Complete the query, and return its completion handling status. */
360     const status_t res1 = query->completeQuery(res);
361     ALOGE_IF(res1 != NO_ERROR && res1 != res,
362             "%s: Error %d in query '%s' completion",
363             __FUNCTION__, res1, query->mQuery);
364     return res1;
365 }
366 
367 /****************************************************************************
368  * Qemu client for the 'factory' service.
369  ***************************************************************************/
370 
371 /*
372  * Factory service queries.
373  */
374 
375 /* Queries list of cameras connected to the host. */
376 const char FactoryQemuClient::mQueryList[] = "list";
377 
FactoryQemuClient()378 FactoryQemuClient::FactoryQemuClient()
379     : QemuClient()
380 {
381 }
382 
~FactoryQemuClient()383 FactoryQemuClient::~FactoryQemuClient()
384 {
385 }
386 
listCameras(char ** list)387 status_t FactoryQemuClient::listCameras(char** list)
388 {
389     ALOGV("%s", __FUNCTION__);
390 
391     QemuQuery query(mQueryList);
392     if (doQuery(&query) || !query.isQuerySucceeded()) {
393         ALOGE("%s: List cameras query failed: %s", __FUNCTION__,
394              query.mReplyData ? query.mReplyData : "No error message");
395         return query.getCompletionStatus();
396     }
397 
398     /* Make sure there is a list returned. */
399     if (query.mReplyDataSize == 0) {
400         ALOGE("%s: No camera list is returned.", __FUNCTION__);
401         return EINVAL;
402     }
403 
404     /* Copy the list over. */
405     *list = (char*)malloc(query.mReplyDataSize);
406     if (*list != NULL) {
407         memcpy(*list, query.mReplyData, query.mReplyDataSize);
408         ALOGD("Emulated camera list: %s", *list);
409         return NO_ERROR;
410     } else {
411         ALOGE("%s: Unable to allocate %d bytes",
412              __FUNCTION__, query.mReplyDataSize);
413         return ENOMEM;
414     }
415 }
416 
417 /****************************************************************************
418  * Qemu client for an 'emulated camera' service.
419  ***************************************************************************/
420 
421 /*
422  * Emulated camera queries
423  */
424 
425 /* Connect to the camera device. */
426 const char CameraQemuClient::mQueryConnect[]    = "connect";
427 /* Disconect from the camera device. */
428 const char CameraQemuClient::mQueryDisconnect[] = "disconnect";
429 /* Start capturing video from the camera device. */
430 const char CameraQemuClient::mQueryStart[]      = "start";
431 /* Stop capturing video from the camera device. */
432 const char CameraQemuClient::mQueryStop[]       = "stop";
433 /* Get next video frame from the camera device. */
434 const char CameraQemuClient::mQueryFrame[]      = "frame";
435 
CameraQemuClient()436 CameraQemuClient::CameraQemuClient()
437     : QemuClient()
438 {
439 }
440 
~CameraQemuClient()441 CameraQemuClient::~CameraQemuClient()
442 {
443 
444 }
445 
queryConnect()446 status_t CameraQemuClient::queryConnect()
447 {
448     ALOGV("%s", __FUNCTION__);
449 
450     QemuQuery query(mQueryConnect);
451     doQuery(&query);
452     const status_t res = query.getCompletionStatus();
453     ALOGE_IF(res != NO_ERROR, "%s: Query failed: %s",
454             __FUNCTION__, query.mReplyData ? query.mReplyData :
455                                              "No error message");
456     return res;
457 }
458 
queryDisconnect()459 status_t CameraQemuClient::queryDisconnect()
460 {
461     ALOGV("%s", __FUNCTION__);
462 
463     QemuQuery query(mQueryDisconnect);
464     doQuery(&query);
465     const status_t res = query.getCompletionStatus();
466     ALOGE_IF(res != NO_ERROR, "%s: Query failed: %s",
467             __FUNCTION__, query.mReplyData ? query.mReplyData :
468                                              "No error message");
469     return res;
470 }
471 
queryStart(uint32_t pixel_format,int width,int height)472 status_t CameraQemuClient::queryStart(uint32_t pixel_format,
473                                       int width,
474                                       int height)
475 {
476     ALOGV("%s", __FUNCTION__);
477 
478     char query_str[256];
479     snprintf(query_str, sizeof(query_str), "%s dim=%dx%d pix=%d",
480              mQueryStart, width, height, pixel_format);
481     QemuQuery query(query_str);
482     doQuery(&query);
483     const status_t res = query.getCompletionStatus();
484     ALOGE_IF(res != NO_ERROR, "%s: Query failed: %s",
485             __FUNCTION__, query.mReplyData ? query.mReplyData :
486                                              "No error message");
487     return res;
488 }
489 
queryStop()490 status_t CameraQemuClient::queryStop()
491 {
492     ALOGV("%s", __FUNCTION__);
493 
494     QemuQuery query(mQueryStop);
495     doQuery(&query);
496     const status_t res = query.getCompletionStatus();
497     ALOGE_IF(res != NO_ERROR, "%s: Query failed: %s",
498             __FUNCTION__, query.mReplyData ? query.mReplyData :
499                                              "No error message");
500     return res;
501 }
502 
queryFrame(void * vframe,void * pframe,size_t vframe_size,size_t pframe_size,float r_scale,float g_scale,float b_scale,float exposure_comp)503 status_t CameraQemuClient::queryFrame(void* vframe,
504                                       void* pframe,
505                                       size_t vframe_size,
506                                       size_t pframe_size,
507                                       float r_scale,
508                                       float g_scale,
509                                       float b_scale,
510                                       float exposure_comp)
511 {
512     ALOGV("%s", __FUNCTION__);
513 
514     char query_str[256];
515     snprintf(query_str, sizeof(query_str), "%s video=%d preview=%d whiteb=%g,%g,%g expcomp=%g",
516              mQueryFrame, (vframe && vframe_size) ? vframe_size : 0,
517              (pframe && pframe_size) ? pframe_size : 0, r_scale, g_scale, b_scale,
518              exposure_comp);
519     QemuQuery query(query_str);
520     doQuery(&query);
521     const status_t res = query.getCompletionStatus();
522     if( res != NO_ERROR) {
523         ALOGE("%s: Query failed: %s",
524              __FUNCTION__, query.mReplyData ? query.mReplyData :
525                                               "No error message");
526         return res;
527     }
528 
529     /* Copy requested frames. */
530     size_t cur_offset = 0;
531     const uint8_t* frame = reinterpret_cast<const uint8_t*>(query.mReplyData);
532     /* Video frame is always first. */
533     if (vframe != NULL && vframe_size != 0) {
534         /* Make sure that video frame is in. */
535         if ((query.mReplyDataSize - cur_offset) >= vframe_size) {
536             memcpy(vframe, frame, vframe_size);
537             cur_offset += vframe_size;
538         } else {
539             ALOGE("%s: Reply %d bytes is to small to contain %d bytes video frame",
540                  __FUNCTION__, query.mReplyDataSize - cur_offset, vframe_size);
541             return EINVAL;
542         }
543     }
544     if (pframe != NULL && pframe_size != 0) {
545         /* Make sure that preview frame is in. */
546         if ((query.mReplyDataSize - cur_offset) >= pframe_size) {
547             memcpy(pframe, frame + cur_offset, pframe_size);
548             cur_offset += pframe_size;
549         } else {
550             ALOGE("%s: Reply %d bytes is to small to contain %d bytes preview frame",
551                  __FUNCTION__, query.mReplyDataSize - cur_offset, pframe_size);
552             return EINVAL;
553         }
554     }
555 
556     return NO_ERROR;
557 }
558 
559 }; /* namespace android */
560