• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2012 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 #define LOG_TAG "Camera2-CallbackProcessor"
18 #define ATRACE_TAG ATRACE_TAG_CAMERA
19 //#define LOG_NDEBUG 0
20 
21 #include <utils/Log.h>
22 #include <utils/Trace.h>
23 #include <gui/Surface.h>
24 
25 #include "common/CameraDeviceBase.h"
26 #include "api1/Camera2Client.h"
27 #include "api1/client2/CallbackProcessor.h"
28 
29 #define ALIGN(x, mask) ( ((x) + (mask) - 1) & ~((mask) - 1) )
30 
31 namespace android {
32 namespace camera2 {
33 
CallbackProcessor(sp<Camera2Client> client)34 CallbackProcessor::CallbackProcessor(sp<Camera2Client> client):
35         Thread(false),
36         mClient(client),
37         mDevice(client->getCameraDevice()),
38         mId(client->getCameraId()),
39         mCallbackAvailable(false),
40         mCallbackToApp(false),
41         mCallbackStreamId(NO_STREAM) {
42 }
43 
~CallbackProcessor()44 CallbackProcessor::~CallbackProcessor() {
45     ALOGV("%s: Exit", __FUNCTION__);
46     deleteStream();
47 }
48 
onFrameAvailable(const BufferItem &)49 void CallbackProcessor::onFrameAvailable(const BufferItem& /*item*/) {
50     Mutex::Autolock l(mInputMutex);
51     if (!mCallbackAvailable) {
52         mCallbackAvailable = true;
53         mCallbackAvailableSignal.signal();
54     }
55 }
56 
setCallbackWindow(sp<Surface> callbackWindow)57 status_t CallbackProcessor::setCallbackWindow(
58         sp<Surface> callbackWindow) {
59     ATRACE_CALL();
60     status_t res;
61 
62     Mutex::Autolock l(mInputMutex);
63 
64     sp<Camera2Client> client = mClient.promote();
65     if (client == 0) return OK;
66     sp<CameraDeviceBase> device = client->getCameraDevice();
67 
68     // If the window is changing, clear out stream if it already exists
69     if (mCallbackWindow != callbackWindow && mCallbackStreamId != NO_STREAM) {
70         res = device->deleteStream(mCallbackStreamId);
71         if (res != OK) {
72             ALOGE("%s: Camera %d: Unable to delete old stream "
73                     "for callbacks: %s (%d)", __FUNCTION__,
74                     client->getCameraId(), strerror(-res), res);
75             return res;
76         }
77         mCallbackStreamId = NO_STREAM;
78         mCallbackConsumer.clear();
79     }
80     mCallbackWindow = callbackWindow;
81     mCallbackToApp = (mCallbackWindow != NULL);
82 
83     return OK;
84 }
85 
updateStream(const Parameters & params)86 status_t CallbackProcessor::updateStream(const Parameters &params) {
87     ATRACE_CALL();
88     status_t res;
89 
90     Mutex::Autolock l(mInputMutex);
91 
92     sp<CameraDeviceBase> device = mDevice.promote();
93     if (device == 0) {
94         ALOGE("%s: Camera %d: Device does not exist", __FUNCTION__, mId);
95         return INVALID_OPERATION;
96     }
97 
98     // If possible, use the flexible YUV format
99     int32_t callbackFormat = params.previewFormat;
100     if (mCallbackToApp) {
101         // TODO: etalvala: This should use the flexible YUV format as well, but
102         // need to reconcile HAL2/HAL3 requirements.
103         callbackFormat = HAL_PIXEL_FORMAT_YV12;
104     } else if(params.fastInfo.useFlexibleYuv &&
105             (params.previewFormat == HAL_PIXEL_FORMAT_YCrCb_420_SP ||
106              params.previewFormat == HAL_PIXEL_FORMAT_YV12) ) {
107         callbackFormat = HAL_PIXEL_FORMAT_YCbCr_420_888;
108     }
109 
110     if (!mCallbackToApp && mCallbackConsumer == 0) {
111         // Create CPU buffer queue endpoint, since app hasn't given us one
112         // Make it async to avoid disconnect deadlocks
113         sp<IGraphicBufferProducer> producer;
114         sp<IGraphicBufferConsumer> consumer;
115         BufferQueue::createBufferQueue(&producer, &consumer);
116         mCallbackConsumer = new CpuConsumer(consumer, kCallbackHeapCount);
117         mCallbackConsumer->setFrameAvailableListener(this);
118         mCallbackConsumer->setName(String8("Camera2-CallbackConsumer"));
119         mCallbackWindow = new Surface(producer);
120     }
121 
122     if (mCallbackStreamId != NO_STREAM) {
123         // Check if stream parameters have to change
124         uint32_t currentWidth, currentHeight, currentFormat;
125         res = device->getStreamInfo(mCallbackStreamId,
126                 &currentWidth, &currentHeight, &currentFormat, 0);
127         if (res != OK) {
128             ALOGE("%s: Camera %d: Error querying callback output stream info: "
129                     "%s (%d)", __FUNCTION__, mId,
130                     strerror(-res), res);
131             return res;
132         }
133         if (currentWidth != (uint32_t)params.previewWidth ||
134                 currentHeight != (uint32_t)params.previewHeight ||
135                 currentFormat != (uint32_t)callbackFormat) {
136             // Since size should only change while preview is not running,
137             // assuming that all existing use of old callback stream is
138             // completed.
139             ALOGV("%s: Camera %d: Deleting stream %d since the buffer "
140                     "parameters changed", __FUNCTION__, mId, mCallbackStreamId);
141             res = device->deleteStream(mCallbackStreamId);
142             if (res != OK) {
143                 ALOGE("%s: Camera %d: Unable to delete old output stream "
144                         "for callbacks: %s (%d)", __FUNCTION__,
145                         mId, strerror(-res), res);
146                 return res;
147             }
148             mCallbackStreamId = NO_STREAM;
149         }
150     }
151 
152     if (mCallbackStreamId == NO_STREAM) {
153         ALOGV("Creating callback stream: %d x %d, format 0x%x, API format 0x%x",
154                 params.previewWidth, params.previewHeight,
155                 callbackFormat, params.previewFormat);
156         res = device->createStream(mCallbackWindow,
157                 params.previewWidth, params.previewHeight, callbackFormat,
158                 HAL_DATASPACE_V0_JFIF, CAMERA3_STREAM_ROTATION_0, &mCallbackStreamId);
159         if (res != OK) {
160             ALOGE("%s: Camera %d: Can't create output stream for callbacks: "
161                     "%s (%d)", __FUNCTION__, mId,
162                     strerror(-res), res);
163             return res;
164         }
165     }
166 
167     return OK;
168 }
169 
deleteStream()170 status_t CallbackProcessor::deleteStream() {
171     ATRACE_CALL();
172     sp<CameraDeviceBase> device;
173     status_t res;
174     {
175         Mutex::Autolock l(mInputMutex);
176 
177         if (mCallbackStreamId == NO_STREAM) {
178             return OK;
179         }
180         device = mDevice.promote();
181         if (device == 0) {
182             ALOGE("%s: Camera %d: Device does not exist", __FUNCTION__, mId);
183             return INVALID_OPERATION;
184         }
185     }
186     res = device->waitUntilDrained();
187     if (res != OK) {
188         ALOGE("%s: Error waiting for HAL to drain: %s (%d)",
189                 __FUNCTION__, strerror(-res), res);
190         return res;
191     }
192 
193     res = device->deleteStream(mCallbackStreamId);
194     if (res != OK) {
195         ALOGE("%s: Unable to delete callback stream: %s (%d)",
196                 __FUNCTION__, strerror(-res), res);
197         return res;
198     }
199 
200     {
201         Mutex::Autolock l(mInputMutex);
202 
203         mCallbackHeap.clear();
204         mCallbackWindow.clear();
205         mCallbackConsumer.clear();
206 
207         mCallbackStreamId = NO_STREAM;
208     }
209     return OK;
210 }
211 
getStreamId() const212 int CallbackProcessor::getStreamId() const {
213     Mutex::Autolock l(mInputMutex);
214     return mCallbackStreamId;
215 }
216 
dump(int,const Vector<String16> &) const217 void CallbackProcessor::dump(int /*fd*/, const Vector<String16>& /*args*/) const {
218 }
219 
threadLoop()220 bool CallbackProcessor::threadLoop() {
221     status_t res;
222 
223     {
224         Mutex::Autolock l(mInputMutex);
225         while (!mCallbackAvailable) {
226             res = mCallbackAvailableSignal.waitRelative(mInputMutex,
227                     kWaitDuration);
228             if (res == TIMED_OUT) return true;
229         }
230         mCallbackAvailable = false;
231     }
232 
233     do {
234         sp<Camera2Client> client = mClient.promote();
235         if (client == 0) {
236             res = discardNewCallback();
237         } else {
238             res = processNewCallback(client);
239         }
240     } while (res == OK);
241 
242     return true;
243 }
244 
discardNewCallback()245 status_t CallbackProcessor::discardNewCallback() {
246     ATRACE_CALL();
247     status_t res;
248     CpuConsumer::LockedBuffer imgBuffer;
249     res = mCallbackConsumer->lockNextBuffer(&imgBuffer);
250     if (res != OK) {
251         if (res != BAD_VALUE) {
252             ALOGE("%s: Camera %d: Error receiving next callback buffer: "
253                     "%s (%d)", __FUNCTION__, mId, strerror(-res), res);
254         }
255         return res;
256     }
257     mCallbackConsumer->unlockBuffer(imgBuffer);
258     return OK;
259 }
260 
processNewCallback(sp<Camera2Client> & client)261 status_t CallbackProcessor::processNewCallback(sp<Camera2Client> &client) {
262     ATRACE_CALL();
263     status_t res;
264 
265     sp<Camera2Heap> callbackHeap;
266     bool useFlexibleYuv = false;
267     int32_t previewFormat = 0;
268     size_t heapIdx;
269 
270     {
271         /* acquire SharedParameters before mMutex so we don't dead lock
272             with Camera2Client code calling into StreamingProcessor */
273         SharedParameters::Lock l(client->getParameters());
274         Mutex::Autolock m(mInputMutex);
275         CpuConsumer::LockedBuffer imgBuffer;
276         if (mCallbackStreamId == NO_STREAM) {
277             ALOGV("%s: Camera %d:No stream is available"
278                     , __FUNCTION__, mId);
279             return INVALID_OPERATION;
280         }
281 
282         ALOGV("%s: Getting buffer", __FUNCTION__);
283         res = mCallbackConsumer->lockNextBuffer(&imgBuffer);
284         if (res != OK) {
285             if (res != BAD_VALUE) {
286                 ALOGE("%s: Camera %d: Error receiving next callback buffer: "
287                         "%s (%d)", __FUNCTION__, mId, strerror(-res), res);
288             }
289             return res;
290         }
291         ALOGV("%s: Camera %d: Preview callback available", __FUNCTION__,
292                 mId);
293 
294         if ( l.mParameters.state != Parameters::PREVIEW
295                 && l.mParameters.state != Parameters::RECORD
296                 && l.mParameters.state != Parameters::VIDEO_SNAPSHOT) {
297             ALOGV("%s: Camera %d: No longer streaming",
298                     __FUNCTION__, mId);
299             mCallbackConsumer->unlockBuffer(imgBuffer);
300             return OK;
301         }
302 
303         if (! (l.mParameters.previewCallbackFlags &
304                 CAMERA_FRAME_CALLBACK_FLAG_ENABLE_MASK) ) {
305             ALOGV("%s: No longer enabled, dropping", __FUNCTION__);
306             mCallbackConsumer->unlockBuffer(imgBuffer);
307             return OK;
308         }
309         if ((l.mParameters.previewCallbackFlags &
310                         CAMERA_FRAME_CALLBACK_FLAG_ONE_SHOT_MASK) &&
311                 !l.mParameters.previewCallbackOneShot) {
312             ALOGV("%s: One shot mode, already sent, dropping", __FUNCTION__);
313             mCallbackConsumer->unlockBuffer(imgBuffer);
314             return OK;
315         }
316 
317         if (imgBuffer.width != static_cast<uint32_t>(l.mParameters.previewWidth) ||
318                 imgBuffer.height != static_cast<uint32_t>(l.mParameters.previewHeight)) {
319             ALOGW("%s: The preview size has changed to %d x %d from %d x %d, this buffer is"
320                     " no longer valid, dropping",__FUNCTION__,
321                     l.mParameters.previewWidth, l.mParameters.previewHeight,
322                     imgBuffer.width, imgBuffer.height);
323             mCallbackConsumer->unlockBuffer(imgBuffer);
324             return OK;
325         }
326 
327         previewFormat = l.mParameters.previewFormat;
328         useFlexibleYuv = l.mParameters.fastInfo.useFlexibleYuv &&
329                 (previewFormat == HAL_PIXEL_FORMAT_YCrCb_420_SP ||
330                  previewFormat == HAL_PIXEL_FORMAT_YV12);
331 
332         int32_t expectedFormat = useFlexibleYuv ?
333                 HAL_PIXEL_FORMAT_YCbCr_420_888 : previewFormat;
334 
335         if (imgBuffer.format != expectedFormat) {
336             ALOGE("%s: Camera %d: Unexpected format for callback: "
337                     "0x%x, expected 0x%x", __FUNCTION__, mId,
338                     imgBuffer.format, expectedFormat);
339             mCallbackConsumer->unlockBuffer(imgBuffer);
340             return INVALID_OPERATION;
341         }
342 
343         // In one-shot mode, stop sending callbacks after the first one
344         if (l.mParameters.previewCallbackFlags &
345                 CAMERA_FRAME_CALLBACK_FLAG_ONE_SHOT_MASK) {
346             ALOGV("%s: clearing oneshot", __FUNCTION__);
347             l.mParameters.previewCallbackOneShot = false;
348         }
349 
350         uint32_t destYStride = 0;
351         uint32_t destCStride = 0;
352         if (useFlexibleYuv) {
353             if (previewFormat == HAL_PIXEL_FORMAT_YV12) {
354                 // Strides must align to 16 for YV12
355                 destYStride = ALIGN(imgBuffer.width, 16);
356                 destCStride = ALIGN(destYStride / 2, 16);
357             } else {
358                 // No padding for NV21
359                 ALOG_ASSERT(previewFormat == HAL_PIXEL_FORMAT_YCrCb_420_SP,
360                         "Unexpected preview format 0x%x", previewFormat);
361                 destYStride = imgBuffer.width;
362                 destCStride = destYStride / 2;
363             }
364         } else {
365             destYStride = imgBuffer.stride;
366             // don't care about cStride
367         }
368 
369         size_t bufferSize = Camera2Client::calculateBufferSize(
370                 imgBuffer.width, imgBuffer.height,
371                 previewFormat, destYStride);
372         size_t currentBufferSize = (mCallbackHeap == 0) ?
373                 0 : (mCallbackHeap->mHeap->getSize() / kCallbackHeapCount);
374         if (bufferSize != currentBufferSize) {
375             mCallbackHeap.clear();
376             mCallbackHeap = new Camera2Heap(bufferSize, kCallbackHeapCount,
377                     "Camera2Client::CallbackHeap");
378             if (mCallbackHeap->mHeap->getSize() == 0) {
379                 ALOGE("%s: Camera %d: Unable to allocate memory for callbacks",
380                         __FUNCTION__, mId);
381                 mCallbackConsumer->unlockBuffer(imgBuffer);
382                 return INVALID_OPERATION;
383             }
384 
385             mCallbackHeapHead = 0;
386             mCallbackHeapFree = kCallbackHeapCount;
387         }
388 
389         if (mCallbackHeapFree == 0) {
390             ALOGE("%s: Camera %d: No free callback buffers, dropping frame",
391                     __FUNCTION__, mId);
392             mCallbackConsumer->unlockBuffer(imgBuffer);
393             return OK;
394         }
395 
396         heapIdx = mCallbackHeapHead;
397 
398         mCallbackHeapHead = (mCallbackHeapHead + 1) % kCallbackHeapCount;
399         mCallbackHeapFree--;
400 
401         // TODO: Get rid of this copy by passing the gralloc queue all the way
402         // to app
403 
404         ssize_t offset;
405         size_t size;
406         sp<IMemoryHeap> heap =
407                 mCallbackHeap->mBuffers[heapIdx]->getMemory(&offset,
408                         &size);
409         uint8_t *data = (uint8_t*)heap->getBase() + offset;
410 
411         if (!useFlexibleYuv) {
412             // Can just memcpy when HAL format matches API format
413             memcpy(data, imgBuffer.data, bufferSize);
414         } else {
415             res = convertFromFlexibleYuv(previewFormat, data, imgBuffer,
416                     destYStride, destCStride);
417             if (res != OK) {
418                 ALOGE("%s: Camera %d: Can't convert between 0x%x and 0x%x formats!",
419                         __FUNCTION__, mId, imgBuffer.format, previewFormat);
420                 mCallbackConsumer->unlockBuffer(imgBuffer);
421                 return BAD_VALUE;
422             }
423         }
424 
425         ALOGV("%s: Freeing buffer", __FUNCTION__);
426         mCallbackConsumer->unlockBuffer(imgBuffer);
427 
428         // mCallbackHeap may get freed up once input mutex is released
429         callbackHeap = mCallbackHeap;
430     }
431 
432     // Call outside parameter lock to allow re-entrancy from notification
433     {
434         Camera2Client::SharedCameraCallbacks::Lock
435             l(client->mSharedCameraCallbacks);
436         if (l.mRemoteCallback != 0) {
437             ALOGV("%s: Camera %d: Invoking client data callback",
438                     __FUNCTION__, mId);
439             l.mRemoteCallback->dataCallback(CAMERA_MSG_PREVIEW_FRAME,
440                     callbackHeap->mBuffers[heapIdx], NULL);
441         }
442     }
443 
444     // Only increment free if we're still using the same heap
445     mCallbackHeapFree++;
446 
447     ALOGV("%s: exit", __FUNCTION__);
448 
449     return OK;
450 }
451 
convertFromFlexibleYuv(int32_t previewFormat,uint8_t * dst,const CpuConsumer::LockedBuffer & src,uint32_t dstYStride,uint32_t dstCStride) const452 status_t CallbackProcessor::convertFromFlexibleYuv(int32_t previewFormat,
453         uint8_t *dst,
454         const CpuConsumer::LockedBuffer &src,
455         uint32_t dstYStride,
456         uint32_t dstCStride) const {
457 
458     if (previewFormat != HAL_PIXEL_FORMAT_YCrCb_420_SP &&
459             previewFormat != HAL_PIXEL_FORMAT_YV12) {
460         ALOGE("%s: Camera %d: Unexpected preview format when using "
461                 "flexible YUV: 0x%x", __FUNCTION__, mId, previewFormat);
462         return INVALID_OPERATION;
463     }
464 
465     // Copy Y plane, adjusting for stride
466     const uint8_t *ySrc = src.data;
467     uint8_t *yDst = dst;
468     for (size_t row = 0; row < src.height; row++) {
469         memcpy(yDst, ySrc, src.width);
470         ySrc += src.stride;
471         yDst += dstYStride;
472     }
473 
474     // Copy/swizzle chroma planes, 4:2:0 subsampling
475     const uint8_t *cbSrc = src.dataCb;
476     const uint8_t *crSrc = src.dataCr;
477     size_t chromaHeight = src.height / 2;
478     size_t chromaWidth = src.width / 2;
479     ssize_t chromaGap = src.chromaStride -
480             (chromaWidth * src.chromaStep);
481     size_t dstChromaGap = dstCStride - chromaWidth;
482 
483     if (previewFormat == HAL_PIXEL_FORMAT_YCrCb_420_SP) {
484         // Flexible YUV chroma to NV21 chroma
485         uint8_t *crcbDst = yDst;
486         // Check for shortcuts
487         if (cbSrc == crSrc + 1 && src.chromaStep == 2) {
488             ALOGV("%s: Fast NV21->NV21", __FUNCTION__);
489             // Source has semiplanar CrCb chroma layout, can copy by rows
490             for (size_t row = 0; row < chromaHeight; row++) {
491                 memcpy(crcbDst, crSrc, src.width);
492                 crcbDst += src.width;
493                 crSrc += src.chromaStride;
494             }
495         } else {
496             ALOGV("%s: Generic->NV21", __FUNCTION__);
497             // Generic copy, always works but not very efficient
498             for (size_t row = 0; row < chromaHeight; row++) {
499                 for (size_t col = 0; col < chromaWidth; col++) {
500                     *(crcbDst++) = *crSrc;
501                     *(crcbDst++) = *cbSrc;
502                     crSrc += src.chromaStep;
503                     cbSrc += src.chromaStep;
504                 }
505                 crSrc += chromaGap;
506                 cbSrc += chromaGap;
507             }
508         }
509     } else {
510         // flexible YUV chroma to YV12 chroma
511         ALOG_ASSERT(previewFormat == HAL_PIXEL_FORMAT_YV12,
512                 "Unexpected preview format 0x%x", previewFormat);
513         uint8_t *crDst = yDst;
514         uint8_t *cbDst = yDst + chromaHeight * dstCStride;
515         if (src.chromaStep == 1) {
516             ALOGV("%s: Fast YV12->YV12", __FUNCTION__);
517             // Source has planar chroma layout, can copy by row
518             for (size_t row = 0; row < chromaHeight; row++) {
519                 memcpy(crDst, crSrc, chromaWidth);
520                 crDst += dstCStride;
521                 crSrc += src.chromaStride;
522             }
523             for (size_t row = 0; row < chromaHeight; row++) {
524                 memcpy(cbDst, cbSrc, chromaWidth);
525                 cbDst += dstCStride;
526                 cbSrc += src.chromaStride;
527             }
528         } else {
529             ALOGV("%s: Generic->YV12", __FUNCTION__);
530             // Generic copy, always works but not very efficient
531             for (size_t row = 0; row < chromaHeight; row++) {
532                 for (size_t col = 0; col < chromaWidth; col++) {
533                     *(crDst++) = *crSrc;
534                     *(cbDst++) = *cbSrc;
535                     crSrc += src.chromaStep;
536                     cbSrc += src.chromaStep;
537                 }
538                 crSrc += chromaGap;
539                 cbSrc += chromaGap;
540                 crDst += dstChromaGap;
541                 cbDst += dstChromaGap;
542             }
543         }
544     }
545 
546     return OK;
547 }
548 
549 }; // namespace camera2
550 }; // namespace android
551