/* * Copyright (C) 2016-2018 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #define LOG_TAG "Camera3-SharedOuStrm" #define ATRACE_TAG ATRACE_TAG_CAMERA //#define LOG_NDEBUG 0 #include "Camera3SharedOutputStream.h" namespace android { namespace camera3 { const size_t Camera3SharedOutputStream::kMaxOutputs; Camera3SharedOutputStream::Camera3SharedOutputStream(int id, const std::vector>& surfaces, uint32_t width, uint32_t height, int format, uint64_t consumerUsage, android_dataspace dataSpace, camera_stream_rotation_t rotation, nsecs_t timestampOffset, const std::string& physicalCameraId, const std::unordered_set &sensorPixelModesUsed, IPCTransport transport, int setId, bool useHalBufManager, int64_t dynamicProfile, int64_t streamUseCase, bool deviceTimeBaseIsRealtime, int timestampBase, int mirrorMode, int32_t colorSpace, bool useReadoutTimestamp) : Camera3OutputStream(id, CAMERA_STREAM_OUTPUT, width, height, format, dataSpace, rotation, physicalCameraId, sensorPixelModesUsed, transport, consumerUsage, timestampOffset, setId, /*isMultiResolution*/false, dynamicProfile, streamUseCase, deviceTimeBaseIsRealtime, timestampBase, mirrorMode, colorSpace, useReadoutTimestamp), mUseHalBufManager(useHalBufManager) { size_t consumerCount = std::min(surfaces.size(), kMaxOutputs); if (surfaces.size() > consumerCount) { ALOGE("%s: Trying to add more consumers than the maximum ", __func__); } for (size_t i = 0; i < consumerCount; i++) { mSurfaceUniqueIds[i] = std::make_pair(surfaces[i], mNextUniqueSurfaceId++); } } Camera3SharedOutputStream::~Camera3SharedOutputStream() { disconnectLocked(); } status_t Camera3SharedOutputStream::connectStreamSplitterLocked() { status_t res = OK; mStreamSplitter = new Camera3StreamSplitter(mUseHalBufManager); uint64_t usage = 0; getEndpointUsage(&usage); std::unordered_map> initialSurfaces; for (size_t i = 0; i < kMaxOutputs; i++) { if (mSurfaceUniqueIds[i].first != nullptr) { initialSurfaces.emplace(i, mSurfaceUniqueIds[i].first); } } res = mStreamSplitter->connect(initialSurfaces, usage, mUsage, camera_stream::max_buffers, getWidth(), getHeight(), getFormat(), &mConsumer, camera_stream::dynamic_range_profile); if (res != OK) { ALOGE("%s: Failed to connect to stream splitter: %s(%d)", __FUNCTION__, strerror(-res), res); return res; } return res; } status_t Camera3SharedOutputStream::attachBufferToSplitterLocked( ANativeWindowBuffer* anb, const std::vector& surface_ids) { status_t res = OK; // Attach the buffer to the splitter output queues. This could block if // the output queue doesn't have any empty slot. So unlock during the course // of attachBufferToOutputs. sp splitter = mStreamSplitter; mLock.unlock(); res = splitter->attachBufferToOutputs(anb, surface_ids); mLock.lock(); if (res != OK) { ALOGE("%s: Stream %d: Cannot attach stream splitter buffer to outputs: %s (%d)", __FUNCTION__, mId, strerror(-res), res); // Only transition to STATE_ABANDONED from STATE_CONFIGURED. (If it is STATE_PREPARING, // let prepareNextBuffer handle the error.) if (res == NO_INIT && mState == STATE_CONFIGURED) { mState = STATE_ABANDONED; } } return res; } void Camera3SharedOutputStream::setHalBufferManager(bool enabled) { Mutex::Autolock l(mLock); mUseHalBufManager = enabled; if (mStreamSplitter != nullptr) { mStreamSplitter->setHalBufferManager(enabled); } } status_t Camera3SharedOutputStream::notifyBufferReleased(ANativeWindowBuffer *anwBuffer) { Mutex::Autolock l(mLock); status_t res = OK; const sp buffer(static_cast(anwBuffer)); if (mStreamSplitter != nullptr) { res = mStreamSplitter->notifyBufferReleased(buffer); } return res; } bool Camera3SharedOutputStream::isConsumerConfigurationDeferred(size_t surface_id) const { Mutex::Autolock l(mLock); if (surface_id >= kMaxOutputs) { return true; } return (mSurfaceUniqueIds[surface_id].first == nullptr); } status_t Camera3SharedOutputStream::setConsumers(const std::vector>& surfaces) { Mutex::Autolock l(mLock); if (surfaces.size() == 0) { ALOGE("%s: it's illegal to set zero consumer surfaces!", __FUNCTION__); return INVALID_OPERATION; } status_t ret = OK; for (auto& surface : surfaces) { if (surface == nullptr) { ALOGE("%s: it's illegal to set a null consumer surface!", __FUNCTION__); return INVALID_OPERATION; } ssize_t id = getNextSurfaceIdLocked(); if (id < 0) { ALOGE("%s: No surface ids available!", __func__); return NO_MEMORY; } mSurfaceUniqueIds[id] = std::make_pair(surface, mNextUniqueSurfaceId++); // Only call addOutput if the splitter has been connected. if (mStreamSplitter != nullptr) { ret = mStreamSplitter->addOutput(id, surface); if (ret != OK) { ALOGE("%s: addOutput failed with error code %d", __FUNCTION__, ret); return ret; } } } return ret; } status_t Camera3SharedOutputStream::getBufferLocked(camera_stream_buffer *buffer, const std::vector& surfaceIds) { ANativeWindowBuffer* anb; int fenceFd = -1; status_t res; res = getBufferLockedCommon(&anb, &fenceFd); if (res != OK) { return res; } if (!mUseHalBufManager) { res = attachBufferToSplitterLocked(anb, surfaceIds); if (res != OK) { return res; } } /** * FenceFD now owned by HAL except in case of error, * in which case we reassign it to acquire_fence */ handoutBufferLocked(*buffer, &(anb->handle), /*acquireFence*/fenceFd, /*releaseFence*/-1, CAMERA_BUFFER_STATUS_OK, /*output*/true); return OK; } status_t Camera3SharedOutputStream::queueBufferToConsumer(sp& consumer, ANativeWindowBuffer* buffer, int anwReleaseFence, const std::vector& uniqueSurfaceIds) { status_t res = OK; if (mUseHalBufManager) { if (uniqueSurfaceIds.size() == 0) { ALOGE("%s: uniqueSurfaceIds must not be empty!", __FUNCTION__); return BAD_VALUE; } Mutex::Autolock l(mLock); std::vector surfaceIds; for (const auto& uniqueId : uniqueSurfaceIds) { bool uniqueIdFound = false; for (size_t i = 0; i < kMaxOutputs; i++) { if (mSurfaceUniqueIds[i].second == uniqueId) { surfaceIds.push_back(i); uniqueIdFound = true; break; } } if (!uniqueIdFound) { ALOGV("%s: unknown unique surface ID %zu for stream %d: " "output might have been removed.", __FUNCTION__, uniqueId, mId); } } res = attachBufferToSplitterLocked(buffer, surfaceIds); if (res != OK) { return res; } } res = consumer->queueBuffer(consumer.get(), buffer, anwReleaseFence); // After queuing buffer to the internal consumer queue, check whether the buffer is // successfully queued to the output queues. if (res == OK) { res = mStreamSplitter->getOnFrameAvailableResult(); if (res != OK) { ALOGE("%s: getOnFrameAvailable returns %d", __FUNCTION__, res); } } else { ALOGE("%s: queueBufer failed %d", __FUNCTION__, res); } return res; } status_t Camera3SharedOutputStream::configureQueueLocked() { status_t res; if ((res = Camera3IOStreamBase::configureQueueLocked()) != OK) { return res; } res = connectStreamSplitterLocked(); if (res != OK) { ALOGE("Cannot connect to stream splitter: %s(%d)", strerror(-res), res); return res; } res = configureConsumerQueueLocked(false/*allowPreviewRespace*/); if (res != OK) { ALOGE("Failed to configureConsumerQueueLocked: %s(%d)", strerror(-res), res); return res; } return OK; } status_t Camera3SharedOutputStream::disconnectLocked() { status_t res; res = Camera3OutputStream::disconnectLocked(); if (mStreamSplitter != nullptr) { mStreamSplitter->disconnect(); } return res; } status_t Camera3SharedOutputStream::getEndpointUsage(uint64_t *usage) { status_t res = OK; uint64_t u = 0; if (mConsumer == nullptr) { // Called before shared buffer queue is constructed. *usage = getPresetConsumerUsage(); for (size_t id = 0; id < kMaxOutputs; id++) { if (mSurfaceUniqueIds[id].first != nullptr) { res = getEndpointUsageForSurface(&u, mSurfaceUniqueIds[id].first); *usage |= u; } } } else { // Called after shared buffer queue is constructed. res = getEndpointUsageForSurface(&u, mConsumer); *usage |= u; } return res; } ssize_t Camera3SharedOutputStream::getNextSurfaceIdLocked() { ssize_t id = -1; for (size_t i = 0; i < kMaxOutputs; i++) { if (mSurfaceUniqueIds[i].first == nullptr) { id = i; break; } } return id; } ssize_t Camera3SharedOutputStream::getSurfaceId(const sp &surface) { Mutex::Autolock l(mLock); ssize_t id = -1; for (size_t i = 0; i < kMaxOutputs; i++) { if (mSurfaceUniqueIds[i].first == surface) { id = i; break; } } return id; } status_t Camera3SharedOutputStream::getUniqueSurfaceIds( const std::vector& surfaceIds, /*out*/std::vector* outUniqueIds) { Mutex::Autolock l(mLock); if (outUniqueIds == nullptr || surfaceIds.size() > kMaxOutputs) { return BAD_VALUE; } outUniqueIds->clear(); outUniqueIds->reserve(surfaceIds.size()); for (const auto& surfaceId : surfaceIds) { if (surfaceId >= kMaxOutputs) { return BAD_VALUE; } outUniqueIds->push_back(mSurfaceUniqueIds[surfaceId].second); } return OK; } status_t Camera3SharedOutputStream::revertPartialUpdateLocked( const KeyedVector, size_t> &removedSurfaces, const KeyedVector, size_t> &attachedSurfaces) { status_t ret = OK; for (size_t i = 0; i < attachedSurfaces.size(); i++) { size_t index = attachedSurfaces.valueAt(i); if (mStreamSplitter != nullptr) { ret = mStreamSplitter->removeOutput(index); if (ret != OK) { return UNKNOWN_ERROR; } } mSurfaceUniqueIds[index] = std::make_pair(nullptr, mNextUniqueSurfaceId++); } for (size_t i = 0; i < removedSurfaces.size(); i++) { size_t index = removedSurfaces.valueAt(i); if (mStreamSplitter != nullptr) { ret = mStreamSplitter->addOutput(index, removedSurfaces.keyAt(i)); if (ret != OK) { return UNKNOWN_ERROR; } } mSurfaceUniqueIds[index] = std::make_pair( removedSurfaces.keyAt(i), mNextUniqueSurfaceId++); } return ret; } status_t Camera3SharedOutputStream::updateStream(const std::vector> &outputSurfaces, const std::vector &outputInfo, const std::vector &removedSurfaceIds, KeyedVector, size_t> *outputMap) { status_t ret = OK; Mutex::Autolock l(mLock); if ((outputMap == nullptr) || (outputInfo.size() != outputSurfaces.size()) || (outputSurfaces.size() > kMaxOutputs)) { return BAD_VALUE; } uint64_t usage; getEndpointUsage(&usage); KeyedVector, size_t> removedSurfaces; //Check whether the new surfaces are compatible. for (const auto &infoIt : outputInfo) { bool imgReaderUsage = (infoIt.consumerUsage & GRALLOC_USAGE_SW_READ_OFTEN) ? true : false; bool sizeMismatch = ((static_cast(infoIt.width) != getWidth()) || (static_cast (infoIt.height) != getHeight())) ? true : false; bool dynamicRangeMismatch = dynamic_range_profile != infoIt.dynamicRangeProfile; if ((imgReaderUsage && sizeMismatch) || dynamicRangeMismatch || (infoIt.format != getOriginalFormat() && infoIt.format != getFormat()) || (infoIt.dataSpace != getDataSpace() && infoIt.dataSpace != getOriginalDataSpace())) { ALOGE("%s: Shared surface parameters format: 0x%x dataSpace: 0x%x dynamic range 0x%" PRIx64 " don't match source stream format: 0x%x dataSpace: 0x%x dynamic" " range 0x%" PRIx64 , __FUNCTION__, infoIt.format, infoIt.dataSpace, infoIt.dynamicRangeProfile, getFormat(), getDataSpace(), dynamic_range_profile); return BAD_VALUE; } } //First remove all absent outputs for (const auto &it : removedSurfaceIds) { if (mStreamSplitter != nullptr) { ret = mStreamSplitter->removeOutput(it); if (ret != OK) { ALOGE("%s: failed with error code %d", __FUNCTION__, ret); status_t res = revertPartialUpdateLocked(removedSurfaces, *outputMap); if (res != OK) { return res; } return ret; } } removedSurfaces.add(mSurfaceUniqueIds[it].first, it); mSurfaceUniqueIds[it] = std::make_pair(nullptr, mNextUniqueSurfaceId++); } //Next add the new outputs for (const auto &it : outputSurfaces) { ssize_t surfaceId = getNextSurfaceIdLocked(); if (surfaceId < 0) { ALOGE("%s: No more available output slots!", __FUNCTION__); status_t res = revertPartialUpdateLocked(removedSurfaces, *outputMap); if (res != OK) { return res; } return NO_MEMORY; } if (mStreamSplitter != nullptr) { ret = mStreamSplitter->addOutput(surfaceId, it); if (ret != OK) { ALOGE("%s: failed with error code %d", __FUNCTION__, ret); status_t res = revertPartialUpdateLocked(removedSurfaces, *outputMap); if (res != OK) { return res; } return ret; } } mSurfaceUniqueIds[surfaceId] = std::make_pair(it, mNextUniqueSurfaceId++); outputMap->add(it, surfaceId); } return ret; } } // namespace camera3 } // namespace android