• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2016-2018 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 "Camera3-SharedOuStrm"
18 #define ATRACE_TAG ATRACE_TAG_CAMERA
19 //#define LOG_NDEBUG 0
20 
21 #include "Camera3SharedOutputStream.h"
22 
23 namespace android {
24 
25 namespace camera3 {
26 
27 const size_t Camera3SharedOutputStream::kMaxOutputs;
28 
Camera3SharedOutputStream(int id,const std::vector<sp<Surface>> & surfaces,uint32_t width,uint32_t height,int format,uint64_t consumerUsage,android_dataspace dataSpace,camera_stream_rotation_t rotation,nsecs_t timestampOffset,const String8 & physicalCameraId,const std::unordered_set<int32_t> & sensorPixelModesUsed,IPCTransport transport,int setId,bool useHalBufManager,int64_t dynamicProfile,int64_t streamUseCase,bool deviceTimeBaseIsRealtime,int timestampBase,int mirrorMode)29 Camera3SharedOutputStream::Camera3SharedOutputStream(int id,
30         const std::vector<sp<Surface>>& surfaces,
31         uint32_t width, uint32_t height, int format,
32         uint64_t consumerUsage, android_dataspace dataSpace,
33         camera_stream_rotation_t rotation,
34         nsecs_t timestampOffset, const String8& physicalCameraId,
35         const std::unordered_set<int32_t> &sensorPixelModesUsed, IPCTransport transport,
36         int setId, bool useHalBufManager, int64_t dynamicProfile,
37         int64_t streamUseCase, bool deviceTimeBaseIsRealtime, int timestampBase,
38         int mirrorMode) :
39         Camera3OutputStream(id, CAMERA_STREAM_OUTPUT, width, height,
40                             format, dataSpace, rotation, physicalCameraId, sensorPixelModesUsed,
41                             transport, consumerUsage, timestampOffset, setId,
42                             /*isMultiResolution*/false, dynamicProfile, streamUseCase,
43                             deviceTimeBaseIsRealtime, timestampBase, mirrorMode),
44         mUseHalBufManager(useHalBufManager) {
45     size_t consumerCount = std::min(surfaces.size(), kMaxOutputs);
46     if (surfaces.size() > consumerCount) {
47         ALOGE("%s: Trying to add more consumers than the maximum ", __func__);
48     }
49     for (size_t i = 0; i < consumerCount; i++) {
50         mSurfaceUniqueIds[i] = std::make_pair(surfaces[i], mNextUniqueSurfaceId++);
51     }
52 }
53 
~Camera3SharedOutputStream()54 Camera3SharedOutputStream::~Camera3SharedOutputStream() {
55     disconnectLocked();
56 }
57 
connectStreamSplitterLocked()58 status_t Camera3SharedOutputStream::connectStreamSplitterLocked() {
59     status_t res = OK;
60 
61     mStreamSplitter = new Camera3StreamSplitter(mUseHalBufManager);
62 
63     uint64_t usage = 0;
64     getEndpointUsage(&usage);
65 
66     std::unordered_map<size_t, sp<Surface>> initialSurfaces;
67     for (size_t i = 0; i < kMaxOutputs; i++) {
68         if (mSurfaceUniqueIds[i].first != nullptr) {
69             initialSurfaces.emplace(i, mSurfaceUniqueIds[i].first);
70         }
71     }
72 
73     res = mStreamSplitter->connect(initialSurfaces, usage, mUsage, camera_stream::max_buffers,
74             getWidth(), getHeight(), getFormat(), &mConsumer, camera_stream::dynamic_range_profile);
75     if (res != OK) {
76         ALOGE("%s: Failed to connect to stream splitter: %s(%d)",
77                 __FUNCTION__, strerror(-res), res);
78         return res;
79     }
80 
81     return res;
82 }
83 
attachBufferToSplitterLocked(ANativeWindowBuffer * anb,const std::vector<size_t> & surface_ids)84 status_t Camera3SharedOutputStream::attachBufferToSplitterLocked(
85         ANativeWindowBuffer* anb,
86         const std::vector<size_t>& surface_ids) {
87     status_t res = OK;
88 
89     // Attach the buffer to the splitter output queues. This could block if
90     // the output queue doesn't have any empty slot. So unlock during the course
91     // of attachBufferToOutputs.
92     sp<Camera3StreamSplitter> splitter = mStreamSplitter;
93     mLock.unlock();
94     res = splitter->attachBufferToOutputs(anb, surface_ids);
95     mLock.lock();
96     if (res != OK) {
97         ALOGE("%s: Stream %d: Cannot attach stream splitter buffer to outputs: %s (%d)",
98                 __FUNCTION__, mId, strerror(-res), res);
99         // Only transition to STATE_ABANDONED from STATE_CONFIGURED. (If it is STATE_PREPARING,
100         // let prepareNextBuffer handle the error.)
101         if (res == NO_INIT && mState == STATE_CONFIGURED) {
102             mState = STATE_ABANDONED;
103         }
104     }
105     return res;
106 }
107 
108 
notifyBufferReleased(ANativeWindowBuffer * anwBuffer)109 status_t Camera3SharedOutputStream::notifyBufferReleased(ANativeWindowBuffer *anwBuffer) {
110     Mutex::Autolock l(mLock);
111     status_t res = OK;
112     const sp<GraphicBuffer> buffer(static_cast<GraphicBuffer*>(anwBuffer));
113 
114     if (mStreamSplitter != nullptr) {
115         res = mStreamSplitter->notifyBufferReleased(buffer);
116     }
117 
118     return res;
119 }
120 
isConsumerConfigurationDeferred(size_t surface_id) const121 bool Camera3SharedOutputStream::isConsumerConfigurationDeferred(size_t surface_id) const {
122     Mutex::Autolock l(mLock);
123     if (surface_id >= kMaxOutputs) {
124         return true;
125     }
126 
127     return (mSurfaceUniqueIds[surface_id].first == nullptr);
128 }
129 
setConsumers(const std::vector<sp<Surface>> & surfaces)130 status_t Camera3SharedOutputStream::setConsumers(const std::vector<sp<Surface>>& surfaces) {
131     Mutex::Autolock l(mLock);
132     if (surfaces.size() == 0) {
133         ALOGE("%s: it's illegal to set zero consumer surfaces!", __FUNCTION__);
134         return INVALID_OPERATION;
135     }
136 
137     status_t ret = OK;
138     for (auto& surface : surfaces) {
139         if (surface == nullptr) {
140             ALOGE("%s: it's illegal to set a null consumer surface!", __FUNCTION__);
141             return INVALID_OPERATION;
142         }
143 
144         ssize_t id = getNextSurfaceIdLocked();
145         if (id < 0) {
146             ALOGE("%s: No surface ids available!", __func__);
147             return NO_MEMORY;
148         }
149 
150         mSurfaceUniqueIds[id] = std::make_pair(surface, mNextUniqueSurfaceId++);
151 
152         // Only call addOutput if the splitter has been connected.
153         if (mStreamSplitter != nullptr) {
154             ret = mStreamSplitter->addOutput(id, surface);
155             if (ret != OK) {
156                 ALOGE("%s: addOutput failed with error code %d", __FUNCTION__, ret);
157                 return ret;
158 
159             }
160         }
161     }
162     return ret;
163 }
164 
getBufferLocked(camera_stream_buffer * buffer,const std::vector<size_t> & surfaceIds)165 status_t Camera3SharedOutputStream::getBufferLocked(camera_stream_buffer *buffer,
166         const std::vector<size_t>& surfaceIds) {
167     ANativeWindowBuffer* anb;
168     int fenceFd = -1;
169 
170     status_t res;
171     res = getBufferLockedCommon(&anb, &fenceFd);
172     if (res != OK) {
173         return res;
174     }
175 
176     if (!mUseHalBufManager) {
177         res = attachBufferToSplitterLocked(anb, surfaceIds);
178         if (res != OK) {
179             return res;
180         }
181     }
182 
183     /**
184      * FenceFD now owned by HAL except in case of error,
185      * in which case we reassign it to acquire_fence
186      */
187     handoutBufferLocked(*buffer, &(anb->handle), /*acquireFence*/fenceFd,
188                         /*releaseFence*/-1, CAMERA_BUFFER_STATUS_OK, /*output*/true);
189 
190     return OK;
191 }
192 
queueBufferToConsumer(sp<ANativeWindow> & consumer,ANativeWindowBuffer * buffer,int anwReleaseFence,const std::vector<size_t> & uniqueSurfaceIds)193 status_t Camera3SharedOutputStream::queueBufferToConsumer(sp<ANativeWindow>& consumer,
194             ANativeWindowBuffer* buffer, int anwReleaseFence,
195             const std::vector<size_t>& uniqueSurfaceIds) {
196     status_t res = OK;
197     if (mUseHalBufManager) {
198         if (uniqueSurfaceIds.size() == 0) {
199             ALOGE("%s: uniqueSurfaceIds must not be empty!", __FUNCTION__);
200             return BAD_VALUE;
201         }
202         Mutex::Autolock l(mLock);
203         std::vector<size_t> surfaceIds;
204         for (const auto& uniqueId : uniqueSurfaceIds) {
205             bool uniqueIdFound = false;
206             for (size_t i = 0; i < kMaxOutputs; i++) {
207                 if (mSurfaceUniqueIds[i].second == uniqueId) {
208                     surfaceIds.push_back(i);
209                     uniqueIdFound = true;
210                     break;
211                 }
212             }
213             if (!uniqueIdFound) {
214                 ALOGV("%s: unknown unique surface ID %zu for stream %d: "
215                         "output might have been removed.",
216                         __FUNCTION__, uniqueId, mId);
217             }
218         }
219         res = attachBufferToSplitterLocked(buffer, surfaceIds);
220         if (res != OK) {
221             return res;
222         }
223     }
224 
225     res = consumer->queueBuffer(consumer.get(), buffer, anwReleaseFence);
226 
227     // After queuing buffer to the internal consumer queue, check whether the buffer is
228     // successfully queued to the output queues.
229     if (res == OK) {
230         res = mStreamSplitter->getOnFrameAvailableResult();
231         if (res != OK) {
232             ALOGE("%s: getOnFrameAvailable returns %d", __FUNCTION__, res);
233         }
234     } else {
235         ALOGE("%s: queueBufer failed %d", __FUNCTION__, res);
236     }
237 
238     return res;
239 }
240 
configureQueueLocked()241 status_t Camera3SharedOutputStream::configureQueueLocked() {
242     status_t res;
243 
244     if ((res = Camera3IOStreamBase::configureQueueLocked()) != OK) {
245         return res;
246     }
247 
248     res = connectStreamSplitterLocked();
249     if (res != OK) {
250         ALOGE("Cannot connect to stream splitter: %s(%d)", strerror(-res), res);
251         return res;
252     }
253 
254     res = configureConsumerQueueLocked(false/*allowPreviewRespace*/);
255     if (res != OK) {
256         ALOGE("Failed to configureConsumerQueueLocked: %s(%d)", strerror(-res), res);
257         return res;
258     }
259 
260     return OK;
261 }
262 
disconnectLocked()263 status_t Camera3SharedOutputStream::disconnectLocked() {
264     status_t res;
265     res = Camera3OutputStream::disconnectLocked();
266 
267     if (mStreamSplitter != nullptr) {
268         mStreamSplitter->disconnect();
269     }
270 
271     return res;
272 }
273 
getEndpointUsage(uint64_t * usage) const274 status_t Camera3SharedOutputStream::getEndpointUsage(uint64_t *usage) const {
275 
276     status_t res = OK;
277     uint64_t u = 0;
278 
279     if (mConsumer == nullptr) {
280         // Called before shared buffer queue is constructed.
281         *usage = getPresetConsumerUsage();
282 
283         for (size_t id = 0; id < kMaxOutputs; id++) {
284             if (mSurfaceUniqueIds[id].first != nullptr) {
285                 res = getEndpointUsageForSurface(&u, mSurfaceUniqueIds[id].first);
286                 *usage |= u;
287             }
288         }
289     } else {
290         // Called after shared buffer queue is constructed.
291         res = getEndpointUsageForSurface(&u, mConsumer);
292         *usage |= u;
293     }
294 
295     return res;
296 }
297 
getNextSurfaceIdLocked()298 ssize_t Camera3SharedOutputStream::getNextSurfaceIdLocked() {
299     ssize_t id = -1;
300     for (size_t i = 0; i < kMaxOutputs; i++) {
301         if (mSurfaceUniqueIds[i].first == nullptr) {
302             id = i;
303             break;
304         }
305     }
306 
307     return id;
308 }
309 
getSurfaceId(const sp<Surface> & surface)310 ssize_t Camera3SharedOutputStream::getSurfaceId(const sp<Surface> &surface) {
311     Mutex::Autolock l(mLock);
312     ssize_t id = -1;
313     for (size_t i = 0; i < kMaxOutputs; i++) {
314         if (mSurfaceUniqueIds[i].first == surface) {
315             id = i;
316             break;
317         }
318     }
319 
320     return id;
321 }
322 
getUniqueSurfaceIds(const std::vector<size_t> & surfaceIds,std::vector<size_t> * outUniqueIds)323 status_t Camera3SharedOutputStream::getUniqueSurfaceIds(
324         const std::vector<size_t>& surfaceIds,
325         /*out*/std::vector<size_t>* outUniqueIds) {
326     Mutex::Autolock l(mLock);
327     if (outUniqueIds == nullptr || surfaceIds.size() > kMaxOutputs) {
328         return BAD_VALUE;
329     }
330 
331     outUniqueIds->clear();
332     outUniqueIds->reserve(surfaceIds.size());
333 
334     for (const auto& surfaceId : surfaceIds) {
335         if (surfaceId >= kMaxOutputs) {
336             return BAD_VALUE;
337         }
338         outUniqueIds->push_back(mSurfaceUniqueIds[surfaceId].second);
339     }
340     return OK;
341 }
342 
revertPartialUpdateLocked(const KeyedVector<sp<Surface>,size_t> & removedSurfaces,const KeyedVector<sp<Surface>,size_t> & attachedSurfaces)343 status_t Camera3SharedOutputStream::revertPartialUpdateLocked(
344         const KeyedVector<sp<Surface>, size_t> &removedSurfaces,
345         const KeyedVector<sp<Surface>, size_t> &attachedSurfaces) {
346     status_t ret = OK;
347 
348     for (size_t i = 0; i < attachedSurfaces.size(); i++) {
349         size_t index = attachedSurfaces.valueAt(i);
350         if (mStreamSplitter != nullptr) {
351             ret = mStreamSplitter->removeOutput(index);
352             if (ret != OK) {
353                 return UNKNOWN_ERROR;
354             }
355         }
356         mSurfaceUniqueIds[index] = std::make_pair(nullptr, mNextUniqueSurfaceId++);
357     }
358 
359     for (size_t i = 0; i < removedSurfaces.size(); i++) {
360         size_t index = removedSurfaces.valueAt(i);
361         if (mStreamSplitter != nullptr) {
362             ret = mStreamSplitter->addOutput(index, removedSurfaces.keyAt(i));
363             if (ret != OK) {
364                 return UNKNOWN_ERROR;
365             }
366         }
367         mSurfaceUniqueIds[index] = std::make_pair(
368                 removedSurfaces.keyAt(i), mNextUniqueSurfaceId++);
369     }
370 
371     return ret;
372 }
373 
updateStream(const std::vector<sp<Surface>> & outputSurfaces,const std::vector<OutputStreamInfo> & outputInfo,const std::vector<size_t> & removedSurfaceIds,KeyedVector<sp<Surface>,size_t> * outputMap)374 status_t Camera3SharedOutputStream::updateStream(const std::vector<sp<Surface>> &outputSurfaces,
375         const std::vector<OutputStreamInfo> &outputInfo,
376         const std::vector<size_t> &removedSurfaceIds,
377         KeyedVector<sp<Surface>, size_t> *outputMap) {
378     status_t ret = OK;
379     Mutex::Autolock l(mLock);
380 
381     if ((outputMap == nullptr) || (outputInfo.size() != outputSurfaces.size()) ||
382             (outputSurfaces.size() > kMaxOutputs)) {
383         return BAD_VALUE;
384     }
385 
386     uint64_t usage;
387     getEndpointUsage(&usage);
388     KeyedVector<sp<Surface>, size_t> removedSurfaces;
389     //Check whether the new surfaces are compatible.
390     for (const auto &infoIt : outputInfo) {
391         bool imgReaderUsage = (infoIt.consumerUsage & GRALLOC_USAGE_SW_READ_OFTEN) ? true : false;
392         bool sizeMismatch = ((static_cast<uint32_t>(infoIt.width) != getWidth()) ||
393                                 (static_cast<uint32_t> (infoIt.height) != getHeight())) ?
394                                 true : false;
395         bool dynamicRangeMismatch = dynamic_range_profile != infoIt.dynamicRangeProfile;
396         if ((imgReaderUsage && sizeMismatch) || dynamicRangeMismatch ||
397                 (infoIt.format != getOriginalFormat() && infoIt.format != getFormat()) ||
398                 (infoIt.dataSpace != getDataSpace() &&
399                  infoIt.dataSpace != getOriginalDataSpace())) {
400             ALOGE("%s: Shared surface parameters format: 0x%x dataSpace: 0x%x dynamic range 0x%"
401                     PRIx64 " don't match source stream format: 0x%x  dataSpace: 0x%x dynamic"
402                     " range 0x%" PRIx64 , __FUNCTION__, infoIt.format, infoIt.dataSpace,
403                     infoIt.dynamicRangeProfile, getFormat(), getDataSpace(), dynamic_range_profile);
404             return BAD_VALUE;
405         }
406     }
407 
408     //First remove all absent outputs
409     for (const auto &it : removedSurfaceIds) {
410         if (mStreamSplitter != nullptr) {
411             ret = mStreamSplitter->removeOutput(it);
412             if (ret != OK) {
413                 ALOGE("%s: failed with error code %d", __FUNCTION__, ret);
414                 status_t res = revertPartialUpdateLocked(removedSurfaces, *outputMap);
415                 if (res != OK) {
416                     return res;
417                 }
418                 return ret;
419 
420             }
421         }
422         removedSurfaces.add(mSurfaceUniqueIds[it].first, it);
423         mSurfaceUniqueIds[it] = std::make_pair(nullptr, mNextUniqueSurfaceId++);
424     }
425 
426     //Next add the new outputs
427     for (const auto &it : outputSurfaces) {
428         ssize_t surfaceId = getNextSurfaceIdLocked();
429         if (surfaceId < 0) {
430             ALOGE("%s: No more available output slots!", __FUNCTION__);
431             status_t res = revertPartialUpdateLocked(removedSurfaces, *outputMap);
432             if (res != OK) {
433                 return res;
434             }
435             return NO_MEMORY;
436         }
437         if (mStreamSplitter != nullptr) {
438             ret = mStreamSplitter->addOutput(surfaceId, it);
439             if (ret != OK) {
440                 ALOGE("%s: failed with error code %d", __FUNCTION__, ret);
441                 status_t res = revertPartialUpdateLocked(removedSurfaces, *outputMap);
442                 if (res != OK) {
443                     return res;
444                 }
445                 return ret;
446             }
447         }
448         mSurfaceUniqueIds[surfaceId] = std::make_pair(it, mNextUniqueSurfaceId++);
449         outputMap->add(it, surfaceId);
450     }
451 
452     return ret;
453 }
454 
455 } // namespace camera3
456 
457 } // namespace android
458