| /* |
| * Copyright (C) 2013 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. |
| */ |
| |
| #include <inttypes.h> |
| |
| #define LOG_TAG "GraphicBufferSource" |
| //#define LOG_NDEBUG 0 |
| #include <utils/Log.h> |
| |
| #define STRINGIFY_ENUMS // for asString in HardwareAPI.h/VideoAPI.h |
| |
| #include <media/stagefright/bqhelper/GraphicBufferSource.h> |
| #include <media/stagefright/bqhelper/FrameDropper.h> |
| #include <media/stagefright/foundation/ADebug.h> |
| #include <media/stagefright/foundation/AMessage.h> |
| #include <media/stagefright/foundation/ColorUtils.h> |
| #include <media/stagefright/foundation/FileDescriptor.h> |
| |
| #include <android-base/properties.h> |
| #include <media/hardware/MetadataBufferType.h> |
| #include <ui/GraphicBuffer.h> |
| #include <gui/BufferItem.h> |
| #include <gui/BufferQueue.h> |
| #include <gui/bufferqueue/1.0/WGraphicBufferProducer.h> |
| #include <gui/bufferqueue/2.0/B2HGraphicBufferProducer.h> |
| #include <gui/IGraphicBufferProducer.h> |
| #include <gui/IGraphicBufferConsumer.h> |
| #include <media/hardware/HardwareAPI.h> |
| |
| #include <inttypes.h> |
| |
| #include <functional> |
| #include <memory> |
| #include <cmath> |
| |
| namespace android { |
| |
| namespace { |
| // kTimestampFluctuation is an upper bound of timestamp fluctuation from the |
| // source that GraphicBufferSource allows. The unit of kTimestampFluctuation is |
| // frames. More specifically, GraphicBufferSource will drop a frame if |
| // |
| // expectedNewFrametimestamp - actualNewFrameTimestamp < |
| // (0.5 - kTimestampFluctuation) * expectedtimePeriodBetweenFrames |
| // |
| // where |
| // - expectedNewFrameTimestamp is the calculated ideal timestamp of the new |
| // incoming frame |
| // - actualNewFrameTimestamp is the timestamp received from the source |
| // - expectedTimePeriodBetweenFrames is the ideal difference of the timestamps |
| // of two adjacent frames |
| // |
| // See GraphicBufferSource::calculateCodecTimestamp_l() for more detail about |
| // how kTimestampFluctuation is used. |
| // |
| // kTimestampFluctuation should be non-negative. A higher value causes a smaller |
| // chance of dropping frames, but at the same time a higher bound on the |
| // difference between the source timestamp and the interpreted (snapped) |
| // timestamp. |
| // |
| // The value of 0.05 means that GraphicBufferSource expects the input timestamps |
| // to fluctuate no more than 5% from the regular time period. |
| // |
| // TODO: Justify the choice of this value, or make it configurable. |
| constexpr double kTimestampFluctuation = 0.05; |
| } |
| |
| /** |
| * A copiable object managing a buffer in the buffer cache managed by the producer. This object |
| * holds a reference to the buffer, and maintains which buffer slot it belongs to (if any), and |
| * whether it is still in a buffer slot. It also maintains whether there are any outstanging acquire |
| * references to it (by buffers acquired from the slot) mainly so that we can keep a debug |
| * count of how many buffers we need to still release back to the producer. |
| */ |
| struct GraphicBufferSource::CachedBuffer { |
| /** |
| * Token that is used to track acquire counts (as opposed to all references to this object). |
| */ |
| struct Acquirable { }; |
| |
| /** |
| * Create using a buffer cached in a slot. |
| */ |
| CachedBuffer(slot_id slot, const sp<GraphicBuffer> &graphicBuffer) |
| : mIsCached(true), |
| mSlot(slot), |
| mGraphicBuffer(graphicBuffer), |
| mAcquirable(std::make_shared<Acquirable>()) { |
| } |
| |
| /** |
| * Returns the cache slot that this buffer is cached in, or -1 if it is no longer cached. |
| * |
| * This assumes that -1 slot id is invalid; though, it is just a benign collision used for |
| * debugging. This object explicitly manages whether it is still cached. |
| */ |
| slot_id getSlot() const { |
| return mIsCached ? mSlot : -1; |
| } |
| |
| /** |
| * Returns the cached buffer. |
| */ |
| sp<GraphicBuffer> getGraphicBuffer() const { |
| return mGraphicBuffer; |
| } |
| |
| /** |
| * Checks whether this buffer is still in the buffer cache. |
| */ |
| bool isCached() const { |
| return mIsCached; |
| } |
| |
| /** |
| * Checks whether this buffer has an acquired reference. |
| */ |
| bool isAcquired() const { |
| return mAcquirable.use_count() > 1; |
| } |
| |
| /** |
| * Gets and returns a shared acquired reference. |
| */ |
| std::shared_ptr<Acquirable> getAcquirable() { |
| return mAcquirable; |
| } |
| |
| private: |
| friend void GraphicBufferSource::discardBufferAtSlotIndex_l(ssize_t); |
| |
| /** |
| * This method to be called when the buffer is no longer in the buffer cache. |
| * Called from discardBufferAtSlotIndex_l. |
| */ |
| void onDroppedFromCache() { |
| CHECK_DBG(mIsCached); |
| mIsCached = false; |
| } |
| |
| bool mIsCached; |
| slot_id mSlot; |
| sp<GraphicBuffer> mGraphicBuffer; |
| std::shared_ptr<Acquirable> mAcquirable; |
| }; |
| |
| /** |
| * A copiable object managing a buffer acquired from the producer. This must always be a cached |
| * buffer. This objects also manages its acquire fence and any release fences that may be returned |
| * by the encoder for this buffer (this buffer may be queued to the encoder multiple times). |
| * If no release fences are added by the encoder, the acquire fence is returned as the release |
| * fence for this - as it is assumed that noone waited for the acquire fence. Otherwise, it is |
| * assumed that the encoder has waited for the acquire fence (or returned it as the release |
| * fence). |
| */ |
| struct GraphicBufferSource::AcquiredBuffer { |
| AcquiredBuffer( |
| const std::shared_ptr<CachedBuffer> &buffer, |
| std::function<void(AcquiredBuffer *)> onReleased, |
| const sp<Fence> &acquireFence) |
| : mBuffer(buffer), |
| mAcquirable(buffer->getAcquirable()), |
| mAcquireFence(acquireFence), |
| mGotReleaseFences(false), |
| mOnReleased(onReleased) { |
| } |
| |
| /** |
| * Adds a release fence returned by the encoder to this object. If this is called with an |
| * valid file descriptor, it is added to the list of release fences. These are returned to the |
| * producer on release() as a merged fence. Regardless of the validity of the file descriptor, |
| * we take note that a release fence was attempted to be added and the acquire fence can now be |
| * assumed as acquired. |
| */ |
| void addReleaseFenceFd(int fenceFd) { |
| // save all release fences - these will be propagated to the producer if this buffer is |
| // ever released to it |
| if (fenceFd >= 0) { |
| mReleaseFenceFds.push_back(fenceFd); |
| } |
| mGotReleaseFences = true; |
| } |
| |
| /** |
| * Returns the acquire fence file descriptor associated with this object. |
| */ |
| int getAcquireFenceFd() { |
| if (mAcquireFence == nullptr || !mAcquireFence->isValid()) { |
| return -1; |
| } |
| return mAcquireFence->dup(); |
| } |
| |
| /** |
| * Returns whether the buffer is still in the buffer cache. |
| */ |
| bool isCached() const { |
| return mBuffer->isCached(); |
| } |
| |
| /** |
| * Returns the acquired buffer. |
| */ |
| sp<GraphicBuffer> getGraphicBuffer() const { |
| return mBuffer->getGraphicBuffer(); |
| } |
| |
| /** |
| * Returns the slot that this buffer is cached at, or -1 otherwise. |
| * |
| * This assumes that -1 slot id is invalid; though, it is just a benign collision used for |
| * debugging. This object explicitly manages whether it is still cached. |
| */ |
| slot_id getSlot() const { |
| return mBuffer->getSlot(); |
| } |
| |
| /** |
| * Creates and returns a release fence object from the acquire fence and/or any release fences |
| * added. If no release fences were added (even if invalid), returns the acquire fence. |
| * Otherwise, it returns a merged fence from all the valid release fences added. |
| */ |
| sp<Fence> getReleaseFence() { |
| // If did not receive release fences, we assume this buffer was not consumed (it was |
| // discarded or dropped). In this case release the acquire fence as the release fence. |
| // We do this here to avoid a dup, close and recreation of the Fence object. |
| if (!mGotReleaseFences) { |
| return mAcquireFence; |
| } |
| sp<Fence> ret = getReleaseFence(0, mReleaseFenceFds.size()); |
| // clear fds as fence took ownership of them |
| mReleaseFenceFds.clear(); |
| return ret; |
| } |
| |
| // this video buffer is no longer referenced by the codec (or kept for later encoding) |
| // it is now safe to release to the producer |
| ~AcquiredBuffer() { |
| //mAcquirable.clear(); |
| mOnReleased(this); |
| // mOnRelease method should call getReleaseFence() that releases all fds but just in case |
| ALOGW_IF(!mReleaseFenceFds.empty(), "release fences were not obtained, closing fds"); |
| for (int fildes : mReleaseFenceFds) { |
| ::close(fildes); |
| TRESPASS_DBG(); |
| } |
| } |
| |
| private: |
| std::shared_ptr<GraphicBufferSource::CachedBuffer> mBuffer; |
| std::shared_ptr<GraphicBufferSource::CachedBuffer::Acquirable> mAcquirable; |
| sp<Fence> mAcquireFence; |
| Vector<int> mReleaseFenceFds; |
| bool mGotReleaseFences; |
| std::function<void(AcquiredBuffer *)> mOnReleased; |
| |
| /** |
| * Creates and returns a release fence from 0 or more release fence file descriptors in from |
| * the specified range in the array. |
| * |
| * @param start start index |
| * @param num number of release fds to merge |
| */ |
| sp<Fence> getReleaseFence(size_t start, size_t num) const { |
| if (num == 0) { |
| return Fence::NO_FENCE; |
| } else if (num == 1) { |
| return new Fence(mReleaseFenceFds[start]); |
| } else { |
| return Fence::merge("GBS::AB", |
| getReleaseFence(start, num >> 1), |
| getReleaseFence(start + (num >> 1), num - (num >> 1))); |
| } |
| } |
| }; |
| |
| struct GraphicBufferSource::ConsumerProxy : public BufferQueue::ConsumerListener { |
| ConsumerProxy(const wp<GraphicBufferSource> &gbs) : mGbs(gbs) {} |
| |
| ~ConsumerProxy() = default; |
| |
| void onFrameAvailable(const BufferItem& item) override { |
| sp<GraphicBufferSource> gbs = mGbs.promote(); |
| if (gbs != nullptr) { |
| gbs->onFrameAvailable(item); |
| } |
| } |
| |
| void onBuffersReleased() override { |
| sp<GraphicBufferSource> gbs = mGbs.promote(); |
| if (gbs != nullptr) { |
| gbs->onBuffersReleased(); |
| } |
| } |
| |
| void onSidebandStreamChanged() override { |
| sp<GraphicBufferSource> gbs = mGbs.promote(); |
| if (gbs != nullptr) { |
| gbs->onSidebandStreamChanged(); |
| } |
| } |
| |
| private: |
| // Note that GraphicBufferSource is holding an sp to us, we can't hold |
| // an sp back to GraphicBufferSource as the circular dependency will |
| // make both immortal. |
| wp<GraphicBufferSource> mGbs; |
| }; |
| |
| GraphicBufferSource::GraphicBufferSource() : |
| mInitCheck(UNKNOWN_ERROR), |
| mNumAvailableUnacquiredBuffers(0), |
| mNumOutstandingAcquires(0), |
| mEndOfStream(false), |
| mEndOfStreamSent(false), |
| mLastDataspace(HAL_DATASPACE_UNKNOWN), |
| mExecuting(false), |
| mSuspended(false), |
| mLastFrameTimestampUs(-1), |
| mStopTimeUs(-1), |
| mLastActionTimeUs(-1LL), |
| mSkipFramesBeforeNs(-1LL), |
| mFrameRepeatIntervalUs(-1LL), |
| mRepeatLastFrameGeneration(0), |
| mOutstandingFrameRepeatCount(0), |
| mFrameRepeatBlockedOnCodecBuffer(false), |
| mFps(-1.0), |
| mCaptureFps(-1.0), |
| mBaseCaptureUs(-1LL), |
| mBaseFrameUs(-1LL), |
| mFrameCount(0), |
| mPrevCaptureUs(-1LL), |
| mPrevFrameUs(-1LL), |
| mInputBufferTimeOffsetUs(0LL) { |
| ALOGV("GraphicBufferSource"); |
| |
| String8 name("GraphicBufferSource"); |
| |
| BufferQueue::createBufferQueue(&mProducer, &mConsumer); |
| mConsumer->setConsumerName(name); |
| |
| // create the consumer listener interface, and hold sp so that this |
| // interface lives as long as the GraphicBufferSource. |
| mConsumerProxy = new ConsumerProxy(this); |
| |
| sp<IConsumerListener> proxy = |
| new BufferQueue::ProxyConsumerListener(mConsumerProxy); |
| |
| mInitCheck = mConsumer->consumerConnect(proxy, false); |
| if (mInitCheck != NO_ERROR) { |
| ALOGE("Error connecting to BufferQueue: %s (%d)", |
| strerror(-mInitCheck), mInitCheck); |
| return; |
| } |
| |
| memset(&mDefaultColorAspectsPacked, 0, sizeof(mDefaultColorAspectsPacked)); |
| |
| CHECK(mInitCheck == NO_ERROR); |
| } |
| |
| GraphicBufferSource::~GraphicBufferSource() { |
| ALOGV("~GraphicBufferSource"); |
| { |
| // all acquired buffers must be freed with the mutex locked otherwise our debug assertion |
| // may trigger |
| Mutex::Autolock autoLock(mMutex); |
| mAvailableBuffers.clear(); |
| mSubmittedCodecBuffers.clear(); |
| mLatestBuffer.mBuffer.reset(); |
| } |
| |
| if (mNumOutstandingAcquires != 0) { |
| ALOGW("potential buffer leak: acquired=%d", mNumOutstandingAcquires); |
| TRESPASS_DBG(); |
| } |
| if (mConsumer != NULL) { |
| status_t err = mConsumer->consumerDisconnect(); |
| if (err != NO_ERROR) { |
| ALOGW("consumerDisconnect failed: %d", err); |
| } |
| } |
| } |
| |
| sp<IGraphicBufferProducer> GraphicBufferSource::getIGraphicBufferProducer() const { |
| return mProducer; |
| } |
| |
| sp<::android::hardware::graphics::bufferqueue::V1_0::IGraphicBufferProducer> |
| GraphicBufferSource::getHGraphicBufferProducer_V1_0() const { |
| using TWGraphicBufferProducer = ::android::TWGraphicBufferProducer< |
| ::android::hardware::graphics::bufferqueue::V1_0::IGraphicBufferProducer>; |
| |
| return new TWGraphicBufferProducer(getIGraphicBufferProducer()); |
| } |
| |
| sp<::android::hardware::graphics::bufferqueue::V2_0::IGraphicBufferProducer> |
| GraphicBufferSource::getHGraphicBufferProducer() const { |
| return new ::android::hardware::graphics::bufferqueue::V2_0::utils:: |
| B2HGraphicBufferProducer(getIGraphicBufferProducer()); |
| } |
| |
| status_t GraphicBufferSource::start() { |
| Mutex::Autolock autoLock(mMutex); |
| ALOGV("--> start; available=%zu, submittable=%zd", |
| mAvailableBuffers.size(), mFreeCodecBuffers.size()); |
| CHECK(!mExecuting); |
| mExecuting = true; |
| mLastDataspace = HAL_DATASPACE_UNKNOWN; |
| ALOGV("clearing last dataSpace"); |
| |
| // Start by loading up as many buffers as possible. We want to do this, |
| // rather than just submit the first buffer, to avoid a degenerate case: |
| // if all BQ buffers arrive before we start executing, and we only submit |
| // one here, the other BQ buffers will just sit until we get notified |
| // that the codec buffer has been released. We'd then acquire and |
| // submit a single additional buffer, repeatedly, never using more than |
| // one codec buffer simultaneously. (We could instead try to submit |
| // all BQ buffers whenever any codec buffer is freed, but if we get the |
| // initial conditions right that will never be useful.) |
| while (haveAvailableBuffers_l()) { |
| if (!fillCodecBuffer_l()) { |
| ALOGV("stop load with available=%zu+%d", |
| mAvailableBuffers.size(), mNumAvailableUnacquiredBuffers); |
| break; |
| } |
| } |
| |
| ALOGV("done loading initial frames, available=%zu+%d", |
| mAvailableBuffers.size(), mNumAvailableUnacquiredBuffers); |
| |
| // If EOS has already been signaled, and there are no more frames to |
| // submit, try to send EOS now as well. |
| if (mStopTimeUs == -1 && mEndOfStream && !haveAvailableBuffers_l()) { |
| submitEndOfInputStream_l(); |
| } |
| |
| if (mFrameRepeatIntervalUs > 0LL && mLooper == NULL) { |
| mReflector = new AHandlerReflector<GraphicBufferSource>(this); |
| |
| mLooper = new ALooper; |
| mLooper->registerHandler(mReflector); |
| mLooper->start(); |
| |
| if (mLatestBuffer.mBuffer != nullptr) { |
| queueFrameRepeat_l(); |
| } |
| } |
| |
| return OK; |
| } |
| |
| status_t GraphicBufferSource::stop() { |
| ALOGV("stop"); |
| |
| Mutex::Autolock autoLock(mMutex); |
| |
| if (mExecuting) { |
| // We are only interested in the transition from executing->idle, |
| // not loaded->idle. |
| mExecuting = false; |
| } |
| return OK; |
| } |
| |
| status_t GraphicBufferSource::release(){ |
| sp<ALooper> looper; |
| { |
| Mutex::Autolock autoLock(mMutex); |
| looper = mLooper; |
| if (mLooper != NULL) { |
| mLooper->unregisterHandler(mReflector->id()); |
| mReflector.clear(); |
| |
| mLooper.clear(); |
| } |
| |
| ALOGV("--> release; available=%zu+%d eos=%d eosSent=%d acquired=%d", |
| mAvailableBuffers.size(), mNumAvailableUnacquiredBuffers, |
| mEndOfStream, mEndOfStreamSent, mNumOutstandingAcquires); |
| |
| // Codec is no longer executing. Releasing all buffers to bq. |
| mFreeCodecBuffers.clear(); |
| mSubmittedCodecBuffers.clear(); |
| mLatestBuffer.mBuffer.reset(); |
| mComponent.clear(); |
| mExecuting = false; |
| } |
| if (looper != NULL) { |
| looper->stop(); |
| } |
| return OK; |
| } |
| |
| status_t GraphicBufferSource::onInputBufferAdded(codec_buffer_id bufferId) { |
| Mutex::Autolock autoLock(mMutex); |
| |
| if (mExecuting) { |
| // This should never happen -- buffers can only be allocated when |
| // transitioning from "loaded" to "idle". |
| ALOGE("addCodecBuffer: buffer added while executing"); |
| return INVALID_OPERATION; |
| } |
| |
| ALOGV("addCodecBuffer: bufferId=%u", bufferId); |
| |
| mFreeCodecBuffers.push_back(bufferId); |
| return OK; |
| } |
| |
| status_t GraphicBufferSource::onInputBufferEmptied(codec_buffer_id bufferId, int fenceFd) { |
| Mutex::Autolock autoLock(mMutex); |
| FileDescriptor::Autoclose fence(fenceFd); |
| |
| ssize_t cbi = mSubmittedCodecBuffers.indexOfKey(bufferId); |
| if (cbi < 0) { |
| // This should never happen. |
| ALOGE("onInputBufferEmptied: buffer not recognized (bufferId=%u)", bufferId); |
| return BAD_VALUE; |
| } |
| |
| std::shared_ptr<AcquiredBuffer> buffer = mSubmittedCodecBuffers.valueAt(cbi); |
| |
| // Move buffer to available buffers |
| mSubmittedCodecBuffers.removeItemsAt(cbi); |
| mFreeCodecBuffers.push_back(bufferId); |
| |
| // header->nFilledLen may not be the original value, so we can't compare |
| // that to zero to see of this was the EOS buffer. Instead we just |
| // see if there is a null AcquiredBuffer, which should only ever happen for EOS. |
| if (buffer == nullptr) { |
| if (!(mEndOfStream && mEndOfStreamSent)) { |
| // This can happen when broken code sends us the same buffer twice in a row. |
| ALOGE("onInputBufferEmptied: non-EOS null buffer (bufferId=%u)", bufferId); |
| } else { |
| ALOGV("onInputBufferEmptied: EOS null buffer (bufferId=%u@%zd)", bufferId, cbi); |
| } |
| // No GraphicBuffer to deal with, no additional input or output is expected, so just return. |
| return BAD_VALUE; |
| } |
| |
| if (!mExecuting) { |
| // this is fine since this could happen when going from Idle to Loaded |
| ALOGV("onInputBufferEmptied: no longer executing (bufferId=%u@%zd)", bufferId, cbi); |
| return OK; |
| } |
| |
| ALOGV("onInputBufferEmptied: bufferId=%d@%zd [slot=%d, useCount=%ld, handle=%p] acquired=%d", |
| bufferId, cbi, buffer->getSlot(), buffer.use_count(), buffer->getGraphicBuffer()->handle, |
| mNumOutstandingAcquires); |
| |
| buffer->addReleaseFenceFd(fence.release()); |
| // release codec reference for video buffer just in case remove does not it |
| buffer.reset(); |
| |
| if (haveAvailableBuffers_l()) { |
| // Fill this codec buffer. |
| CHECK(!mEndOfStreamSent); |
| ALOGV("onInputBufferEmptied: buffer freed, feeding codec (available=%zu+%d, eos=%d)", |
| mAvailableBuffers.size(), mNumAvailableUnacquiredBuffers, mEndOfStream); |
| fillCodecBuffer_l(); |
| } else if (mEndOfStream && mStopTimeUs == -1) { |
| // No frames available, but EOS is pending and no stop time, so use this buffer to |
| // send that. |
| ALOGV("onInputBufferEmptied: buffer freed, submitting EOS"); |
| submitEndOfInputStream_l(); |
| } else if (mFrameRepeatBlockedOnCodecBuffer) { |
| bool success = repeatLatestBuffer_l(); |
| ALOGV("onInputBufferEmptied: completing deferred repeatLatestBuffer_l %s", |
| success ? "SUCCESS" : "FAILURE"); |
| mFrameRepeatBlockedOnCodecBuffer = false; |
| } |
| |
| // releaseReleasableBuffers_l(); |
| return OK; |
| } |
| |
| void GraphicBufferSource::onDataspaceChanged_l( |
| android_dataspace dataspace, android_pixel_format pixelFormat) { |
| ALOGD("got buffer with new dataSpace #%x", dataspace); |
| mLastDataspace = dataspace; |
| |
| if (ColorUtils::convertDataSpaceToV0(dataspace)) { |
| mComponent->dispatchDataSpaceChanged( |
| mLastDataspace, mDefaultColorAspectsPacked, pixelFormat); |
| } |
| } |
| |
| bool GraphicBufferSource::fillCodecBuffer_l() { |
| CHECK(mExecuting && haveAvailableBuffers_l()); |
| |
| if (mFreeCodecBuffers.empty()) { |
| // No buffers available, bail. |
| ALOGV("fillCodecBuffer_l: no codec buffers, available=%zu+%d", |
| mAvailableBuffers.size(), mNumAvailableUnacquiredBuffers); |
| return false; |
| } |
| |
| VideoBuffer item; |
| if (mAvailableBuffers.empty()) { |
| ALOGV("fillCodecBuffer_l: acquiring available buffer, available=%zu+%d", |
| mAvailableBuffers.size(), mNumAvailableUnacquiredBuffers); |
| if (acquireBuffer_l(&item) != OK) { |
| ALOGE("fillCodecBuffer_l: failed to acquire available buffer"); |
| return false; |
| } |
| } else { |
| ALOGV("fillCodecBuffer_l: getting available buffer, available=%zu+%d", |
| mAvailableBuffers.size(), mNumAvailableUnacquiredBuffers); |
| item = *mAvailableBuffers.begin(); |
| mAvailableBuffers.erase(mAvailableBuffers.begin()); |
| } |
| |
| int64_t itemTimeUs = item.mTimestampNs / 1000; |
| |
| // Process ActionItem in the Queue if there is any. If a buffer's timestamp |
| // is smaller than the first action's timestamp, no action need to be performed. |
| // If buffer's timestamp is larger or equal than the last action's timestamp, |
| // only the last action needs to be performed as all the acitions before the |
| // the action are overridden by the last action. For the other cases, traverse |
| // the Queue to find the newest action that with timestamp smaller or equal to |
| // the buffer's timestamp. For example, an action queue like |
| // [pause 1us], [resume 2us], [pause 3us], [resume 4us], [pause 5us].... Upon |
| // receiving a buffer with timestamp 3.5us, only the action [pause, 3us] needs |
| // to be handled and [pause, 1us], [resume 2us] will be discarded. |
| bool done = false; |
| bool seeStopAction = false; |
| if (!mActionQueue.empty()) { |
| // First scan to check if bufferTimestamp is smaller than first action's timestamp. |
| ActionItem nextAction = *(mActionQueue.begin()); |
| if (itemTimeUs < nextAction.mActionTimeUs) { |
| ALOGV("No action. buffer timestamp %lld us < action timestamp: %lld us", |
| (long long)itemTimeUs, (long long)nextAction.mActionTimeUs); |
| // All the actions are ahead. No action need to perform now. |
| // Release the buffer if is in suspended state, or process the buffer |
| // if not in suspended state. |
| done = true; |
| } |
| |
| if (!done) { |
| // Find the newest action that with timestamp smaller than itemTimeUs. Then |
| // remove all the actions before and include the newest action. |
| List<ActionItem>::iterator it = mActionQueue.begin(); |
| while (it != mActionQueue.end() && it->mActionTimeUs <= itemTimeUs |
| && nextAction.mAction != ActionItem::STOP) { |
| nextAction = *it; |
| ++it; |
| } |
| mActionQueue.erase(mActionQueue.begin(), it); |
| |
| CHECK(itemTimeUs >= nextAction.mActionTimeUs); |
| switch (nextAction.mAction) { |
| case ActionItem::PAUSE: |
| { |
| mSuspended = true; |
| ALOGV("RUNNING/PAUSE -> PAUSE at buffer %lld us PAUSE Time: %lld us", |
| (long long)itemTimeUs, (long long)nextAction.mActionTimeUs); |
| break; |
| } |
| case ActionItem::RESUME: |
| { |
| mSuspended = false; |
| ALOGV("PAUSE/RUNNING -> RUNNING at buffer %lld us RESUME Time: %lld us", |
| (long long)itemTimeUs, (long long)nextAction.mActionTimeUs); |
| break; |
| } |
| case ActionItem::STOP: |
| { |
| ALOGV("RUNNING/PAUSE -> STOP at buffer %lld us STOP Time: %lld us", |
| (long long)itemTimeUs, (long long)nextAction.mActionTimeUs); |
| // Clear the whole ActionQueue as recording is done |
| mActionQueue.clear(); |
| seeStopAction = true; |
| break; |
| } |
| default: |
| TRESPASS_DBG("Unknown action type"); |
| // return true here because we did consume an available buffer, so the |
| // loop in start will eventually terminate even if we hit this. |
| return false; |
| } |
| } |
| } |
| |
| if (seeStopAction) { |
| // Clear all the buffers before setting mEndOfStream and signal EndOfInputStream. |
| releaseAllAvailableBuffers_l(); |
| mEndOfStream = true; |
| submitEndOfInputStream_l(); |
| return true; |
| } |
| |
| if (mSuspended) { |
| return true; |
| } |
| |
| int err = UNKNOWN_ERROR; |
| |
| // only submit sample if start time is unspecified, or sample |
| // is queued after the specified start time |
| if (mSkipFramesBeforeNs < 0LL || item.mTimestampNs >= mSkipFramesBeforeNs) { |
| // if start time is set, offset time stamp by start time |
| if (mSkipFramesBeforeNs > 0) { |
| item.mTimestampNs -= mSkipFramesBeforeNs; |
| } |
| |
| int64_t timeUs = item.mTimestampNs / 1000; |
| if (mFrameDropper != NULL && mFrameDropper->shouldDrop(timeUs)) { |
| ALOGV("skipping frame (%lld) to meet max framerate", static_cast<long long>(timeUs)); |
| // set err to OK so that the skipped frame can still be saved as the lastest frame |
| err = OK; |
| } else { |
| err = submitBuffer_l(item); // this takes shared ownership of the acquired buffer on succeess |
| } |
| } |
| |
| if (err != OK) { |
| ALOGV("submitBuffer_l failed, will release bq slot %d", item.mBuffer->getSlot()); |
| return true; |
| } else { |
| // Don't set the last buffer id if we're not repeating, |
| // we'll be holding on to the last buffer for nothing. |
| if (mFrameRepeatIntervalUs > 0LL) { |
| setLatestBuffer_l(item); |
| } |
| ALOGV("buffer submitted [slot=%d, useCount=%ld] acquired=%d", |
| item.mBuffer->getSlot(), item.mBuffer.use_count(), mNumOutstandingAcquires); |
| mLastFrameTimestampUs = itemTimeUs; |
| } |
| |
| return true; |
| } |
| |
| bool GraphicBufferSource::repeatLatestBuffer_l() { |
| CHECK(mExecuting && !haveAvailableBuffers_l()); |
| |
| if (mLatestBuffer.mBuffer == nullptr || mSuspended) { |
| return false; |
| } |
| |
| if (mFreeCodecBuffers.empty()) { |
| // No buffers available, bail. |
| ALOGV("repeatLatestBuffer_l: no codec buffers."); |
| return false; |
| } |
| |
| if (!mLatestBuffer.mBuffer->isCached()) { |
| ALOGV("repeatLatestBuffer_l: slot was discarded, but repeating our own reference"); |
| } |
| |
| // it is ok to update the timestamp of latest buffer as it is only used for submission |
| status_t err = submitBuffer_l(mLatestBuffer); |
| if (err != OK) { |
| return false; |
| } |
| |
| /* repeat last frame up to kRepeatLastFrameCount times. |
| * in case of static scene, a single repeat might not get rid of encoder |
| * ghosting completely, refresh a couple more times to get better quality |
| */ |
| if (--mOutstandingFrameRepeatCount > 0) { |
| // set up timestamp for repeat frame |
| mLatestBuffer.mTimestampNs += mFrameRepeatIntervalUs * 1000; |
| queueFrameRepeat_l(); |
| } |
| |
| return true; |
| } |
| |
| void GraphicBufferSource::setLatestBuffer_l(const VideoBuffer &item) { |
| mLatestBuffer = item; |
| |
| ALOGV("setLatestBuffer_l: [slot=%d, useCount=%ld]", |
| mLatestBuffer.mBuffer->getSlot(), mLatestBuffer.mBuffer.use_count()); |
| |
| mOutstandingFrameRepeatCount = kRepeatLastFrameCount; |
| // set up timestamp for repeat frame |
| mLatestBuffer.mTimestampNs += mFrameRepeatIntervalUs * 1000; |
| queueFrameRepeat_l(); |
| } |
| |
| void GraphicBufferSource::queueFrameRepeat_l() { |
| mFrameRepeatBlockedOnCodecBuffer = false; |
| |
| if (mReflector != NULL) { |
| sp<AMessage> msg = new AMessage(kWhatRepeatLastFrame, mReflector); |
| msg->setInt32("generation", ++mRepeatLastFrameGeneration); |
| msg->post(mFrameRepeatIntervalUs); |
| } |
| } |
| |
| #ifdef __clang__ |
| __attribute__((no_sanitize("integer"))) |
| #endif |
| bool GraphicBufferSource::calculateCodecTimestamp_l( |
| nsecs_t bufferTimeNs, int64_t *codecTimeUs) { |
| int64_t timeUs = bufferTimeNs / 1000; |
| timeUs += mInputBufferTimeOffsetUs; |
| |
| if (mCaptureFps > 0. |
| && (mFps > 2 * mCaptureFps |
| || mCaptureFps > 2 * mFps)) { |
| // Time lapse or slow motion mode |
| if (mPrevCaptureUs < 0LL) { |
| // first capture |
| mPrevCaptureUs = mBaseCaptureUs = timeUs; |
| // adjust the first sample timestamp. |
| mPrevFrameUs = mBaseFrameUs = |
| std::llround((timeUs * mCaptureFps) / mFps); |
| mFrameCount = 0; |
| } else if (mSnapTimestamps) { |
| double nFrames = (timeUs - mPrevCaptureUs) * mCaptureFps / 1000000; |
| if (nFrames < 0.5 - kTimestampFluctuation) { |
| // skip this frame as it's too close to previous capture |
| ALOGD("skipping frame, timeUs %lld", |
| static_cast<long long>(timeUs)); |
| return false; |
| } |
| // snap to nearest capture point |
| if (nFrames <= 1.0) { |
| nFrames = 1.0; |
| } |
| mFrameCount += std::llround(nFrames); |
| mPrevCaptureUs = mBaseCaptureUs + std::llround( |
| mFrameCount * 1000000 / mCaptureFps); |
| mPrevFrameUs = mBaseFrameUs + std::llround( |
| mFrameCount * 1000000 / mFps); |
| } else { |
| if (timeUs <= mPrevCaptureUs) { |
| if (mFrameDropper != NULL && mFrameDropper->disabled()) { |
| // Warn only, client has disabled frame drop logic possibly for image |
| // encoding cases where camera's ZSL mode could send out of order frames. |
| ALOGW("Received frame that's going backward in time"); |
| } else { |
| // Drop the frame if it's going backward in time. Bad timestamp |
| // could disrupt encoder's rate control completely. |
| ALOGW("Dropping frame that's going backward in time"); |
| return false; |
| } |
| } |
| mPrevCaptureUs = timeUs; |
| mPrevFrameUs = mBaseFrameUs + std::llround( |
| (timeUs - mBaseCaptureUs) * (mCaptureFps / mFps)); |
| } |
| |
| ALOGV("timeUs %lld, captureUs %lld, frameUs %lld", |
| static_cast<long long>(timeUs), |
| static_cast<long long>(mPrevCaptureUs), |
| static_cast<long long>(mPrevFrameUs)); |
| } else { |
| if (timeUs <= mPrevFrameUs) { |
| if (mFrameDropper != NULL && mFrameDropper->disabled()) { |
| // Warn only, client has disabled frame drop logic possibly for image |
| // encoding cases where camera's ZSL mode could send out of order frames. |
| ALOGW("Received frame that's going backward in time"); |
| } else { |
| // Drop the frame if it's going backward in time. Bad timestamp |
| // could disrupt encoder's rate control completely. |
| ALOGW("Dropping frame that's going backward in time"); |
| return false; |
| } |
| } |
| |
| mPrevFrameUs = timeUs; |
| } |
| |
| *codecTimeUs = mPrevFrameUs; |
| return true; |
| } |
| |
| status_t GraphicBufferSource::submitBuffer_l(const VideoBuffer &item) { |
| CHECK(!mFreeCodecBuffers.empty()); |
| uint32_t codecBufferId = *mFreeCodecBuffers.begin(); |
| |
| ALOGV("submitBuffer_l [slot=%d, bufferId=%d]", item.mBuffer->getSlot(), codecBufferId); |
| |
| int64_t codecTimeUs; |
| if (!calculateCodecTimestamp_l(item.mTimestampNs, &codecTimeUs)) { |
| return UNKNOWN_ERROR; |
| } |
| |
| if ((android_dataspace)item.mDataspace != mLastDataspace) { |
| onDataspaceChanged_l( |
| item.mDataspace, |
| (android_pixel_format)item.mBuffer->getGraphicBuffer()->format); |
| } |
| |
| std::shared_ptr<AcquiredBuffer> buffer = item.mBuffer; |
| // use a GraphicBuffer for now as component is using GraphicBuffers to hold references |
| // and it requires this graphic buffer to be able to hold its reference |
| // and thus we would need to create a new GraphicBuffer from an ANWBuffer separate from the |
| // acquired GraphicBuffer. |
| // TODO: this can be reworked globally to use ANWBuffer references |
| sp<GraphicBuffer> graphicBuffer = buffer->getGraphicBuffer(); |
| status_t err = mComponent->submitBuffer( |
| codecBufferId, graphicBuffer, codecTimeUs, buffer->getAcquireFenceFd()); |
| |
| if (err != OK) { |
| ALOGW("WARNING: emptyGraphicBuffer failed: 0x%x", err); |
| return err; |
| } |
| |
| mFreeCodecBuffers.erase(mFreeCodecBuffers.begin()); |
| |
| ssize_t cbix = mSubmittedCodecBuffers.add(codecBufferId, buffer); |
| ALOGV("emptyGraphicBuffer succeeded, bufferId=%u@%zd bufhandle=%p", |
| codecBufferId, cbix, graphicBuffer->handle); |
| return OK; |
| } |
| |
| void GraphicBufferSource::submitEndOfInputStream_l() { |
| CHECK(mEndOfStream); |
| if (mEndOfStreamSent) { |
| ALOGV("EOS already sent"); |
| return; |
| } |
| |
| if (mFreeCodecBuffers.empty()) { |
| ALOGV("submitEndOfInputStream_l: no codec buffers available"); |
| return; |
| } |
| uint32_t codecBufferId = *mFreeCodecBuffers.begin(); |
| |
| // We reject any additional incoming graphic buffers. There is no acquired buffer used for EOS |
| status_t err = mComponent->submitEos(codecBufferId); |
| if (err != OK) { |
| ALOGW("emptyDirectBuffer EOS failed: 0x%x", err); |
| } else { |
| mFreeCodecBuffers.erase(mFreeCodecBuffers.begin()); |
| ssize_t cbix = mSubmittedCodecBuffers.add(codecBufferId, nullptr); |
| ALOGV("submitEndOfInputStream_l: buffer submitted, bufferId=%u@%zd", codecBufferId, cbix); |
| mEndOfStreamSent = true; |
| |
| // no need to hold onto any buffers for frame repeating |
| ++mRepeatLastFrameGeneration; |
| mLatestBuffer.mBuffer.reset(); |
| } |
| } |
| |
| status_t GraphicBufferSource::acquireBuffer_l(VideoBuffer *ab) { |
| BufferItem bi; |
| status_t err = mConsumer->acquireBuffer(&bi, 0); |
| if (err == BufferQueue::NO_BUFFER_AVAILABLE) { |
| // shouldn't happen |
| ALOGW("acquireBuffer_l: frame was not available"); |
| return err; |
| } else if (err != OK) { |
| ALOGW("acquireBuffer_l: failed with err=%d", err); |
| return err; |
| } |
| --mNumAvailableUnacquiredBuffers; |
| |
| // Manage our buffer cache. |
| std::shared_ptr<CachedBuffer> buffer; |
| ssize_t bsi = mBufferSlots.indexOfKey(bi.mSlot); |
| if (bi.mGraphicBuffer != NULL) { |
| // replace/initialize slot with new buffer |
| ALOGV("acquireBuffer_l: %s buffer slot %d", bsi < 0 ? "setting" : "UPDATING", bi.mSlot); |
| if (bsi >= 0) { |
| discardBufferAtSlotIndex_l(bsi); |
| } else { |
| bsi = mBufferSlots.add(bi.mSlot, nullptr); |
| } |
| buffer = std::make_shared<CachedBuffer>(bi.mSlot, bi.mGraphicBuffer); |
| mBufferSlots.replaceValueAt(bsi, buffer); |
| } else { |
| buffer = mBufferSlots.valueAt(bsi); |
| } |
| int64_t frameNum = bi.mFrameNumber; |
| |
| std::shared_ptr<AcquiredBuffer> acquiredBuffer = |
| std::make_shared<AcquiredBuffer>( |
| buffer, |
| [frameNum, this](AcquiredBuffer *buffer){ |
| // AcquiredBuffer's destructor should always be called when mMutex is locked. |
| // If we had a reentrant mutex, we could just lock it again to ensure this. |
| if (mMutex.tryLock() == 0) { |
| TRESPASS_DBG(); |
| mMutex.unlock(); |
| } |
| |
| // we can release buffers immediately if not using adapters |
| // alternately, we could add them to mSlotsToRelease, but we would |
| // somehow need to propagate frame number to that queue |
| if (buffer->isCached()) { |
| --mNumOutstandingAcquires; |
| mConsumer->releaseBuffer( |
| buffer->getSlot(), frameNum, EGL_NO_DISPLAY, EGL_NO_SYNC_KHR, |
| buffer->getReleaseFence()); |
| } |
| }, |
| bi.mFence); |
| VideoBuffer videoBuffer{acquiredBuffer, bi.mTimestamp, bi.mDataSpace}; |
| *ab = videoBuffer; |
| ++mNumOutstandingAcquires; |
| return OK; |
| } |
| |
| // BufferQueue::ConsumerListener callback |
| void GraphicBufferSource::onFrameAvailable(const BufferItem& item __unused) { |
| Mutex::Autolock autoLock(mMutex); |
| |
| ALOGV("onFrameAvailable: executing=%d available=%zu+%d", |
| mExecuting, mAvailableBuffers.size(), mNumAvailableUnacquiredBuffers); |
| ++mNumAvailableUnacquiredBuffers; |
| |
| // For BufferQueue we cannot acquire a buffer if we cannot immediately feed it to the codec |
| // UNLESS we are discarding this buffer (acquiring and immediately releasing it), which makes |
| // this an ugly logic. |
| // NOTE: We could also rely on our debug counter but that is meant only as a debug counter. |
| if (!areWeDiscardingAvailableBuffers_l() && mFreeCodecBuffers.empty()) { |
| // we may not be allowed to acquire a possibly encodable buffer, so just note that |
| // it is available |
| ALOGV("onFrameAvailable: cannot acquire buffer right now, do it later"); |
| |
| ++mRepeatLastFrameGeneration; // cancel any pending frame repeat |
| return; |
| } |
| |
| VideoBuffer buffer; |
| status_t err = acquireBuffer_l(&buffer); |
| if (err != OK) { |
| ALOGE("onFrameAvailable: acquireBuffer returned err=%d", err); |
| } else { |
| onBufferAcquired_l(buffer); |
| } |
| } |
| |
| bool GraphicBufferSource::areWeDiscardingAvailableBuffers_l() { |
| return mEndOfStreamSent // already sent EOS to codec |
| || mComponent == nullptr // there is no codec connected |
| || (mSuspended && mActionQueue.empty()) // we are suspended and not waiting for |
| // any further action |
| || !mExecuting; |
| } |
| |
| void GraphicBufferSource::onBufferAcquired_l(const VideoBuffer &buffer) { |
| if (mEndOfStreamSent) { |
| // This should only be possible if a new buffer was queued after |
| // EOS was signaled, i.e. the app is misbehaving. |
| ALOGW("onFrameAvailable: EOS is sent, ignoring frame"); |
| } else if (mComponent == NULL || (mSuspended && mActionQueue.empty())) { |
| // FIXME: if we are suspended but have a resume queued we will stop repeating the last |
| // frame. Is that the desired behavior? |
| ALOGV("onFrameAvailable: suspended, ignoring frame"); |
| } else { |
| ++mRepeatLastFrameGeneration; // cancel any pending frame repeat |
| mAvailableBuffers.push_back(buffer); |
| if (mExecuting) { |
| fillCodecBuffer_l(); |
| } |
| } |
| } |
| |
| // BufferQueue::ConsumerListener callback |
| void GraphicBufferSource::onBuffersReleased() { |
| Mutex::Autolock lock(mMutex); |
| |
| uint64_t slotMask; |
| uint64_t releaseMask; |
| if (mConsumer->getReleasedBuffers(&releaseMask) != NO_ERROR) { |
| slotMask = 0xffffffffffffffffULL; |
| ALOGW("onBuffersReleased: unable to get released buffer set"); |
| } else { |
| slotMask = releaseMask; |
| ALOGV("onBuffersReleased: 0x%016" PRIx64, slotMask); |
| } |
| |
| AString unpopulated; |
| for (int i = 0; i < BufferQueue::NUM_BUFFER_SLOTS; i++) { |
| if ((slotMask & 0x01) != 0) { |
| if (!discardBufferInSlot_l(i)) { |
| if (!unpopulated.empty()) { |
| unpopulated.append(", "); |
| } |
| unpopulated.append(i); |
| } |
| } |
| slotMask >>= 1; |
| } |
| if (!unpopulated.empty()) { |
| ALOGW("released unpopulated slots: [%s]", unpopulated.c_str()); |
| } |
| } |
| |
| bool GraphicBufferSource::discardBufferInSlot_l(GraphicBufferSource::slot_id i) { |
| ssize_t bsi = mBufferSlots.indexOfKey(i); |
| if (bsi < 0) { |
| return false; |
| } else { |
| discardBufferAtSlotIndex_l(bsi); |
| mBufferSlots.removeItemsAt(bsi); |
| return true; |
| } |
| } |
| |
| void GraphicBufferSource::discardBufferAtSlotIndex_l(ssize_t bsi) { |
| const std::shared_ptr<CachedBuffer>& buffer = mBufferSlots.valueAt(bsi); |
| // use -2 if there is no latest buffer, and -1 if it is no longer cached |
| slot_id latestBufferSlot = |
| mLatestBuffer.mBuffer == nullptr ? -2 : mLatestBuffer.mBuffer->getSlot(); |
| ALOGV("releasing acquired buffer: [slot=%d, useCount=%ld], latest: [slot=%d]", |
| mBufferSlots.keyAt(bsi), buffer.use_count(), latestBufferSlot); |
| mBufferSlots.valueAt(bsi)->onDroppedFromCache(); |
| |
| // If the slot of an acquired buffer is discarded, that buffer will not have to be |
| // released to the producer, so account it here. However, it is possible that the |
| // acquired buffer has already been discarded so check if it still is. |
| if (buffer->isAcquired()) { |
| --mNumOutstandingAcquires; |
| } |
| |
| // clear the buffer reference (not technically needed as caller either replaces or deletes |
| // it; done here for safety). |
| mBufferSlots.editValueAt(bsi).reset(); |
| CHECK_DBG(buffer == nullptr); |
| } |
| |
| void GraphicBufferSource::releaseAllAvailableBuffers_l() { |
| mAvailableBuffers.clear(); |
| while (mNumAvailableUnacquiredBuffers > 0) { |
| VideoBuffer item; |
| if (acquireBuffer_l(&item) != OK) { |
| ALOGW("releaseAllAvailableBuffers: failed to acquire available unacquired buffer"); |
| break; |
| } |
| } |
| } |
| |
| // BufferQueue::ConsumerListener callback |
| void GraphicBufferSource::onSidebandStreamChanged() { |
| ALOG_ASSERT(false, "GraphicBufferSource can't consume sideband streams"); |
| } |
| |
| status_t GraphicBufferSource::configure( |
| const sp<ComponentWrapper>& component, |
| int32_t dataSpace, |
| int32_t bufferCount, |
| uint32_t frameWidth, |
| uint32_t frameHeight, |
| uint32_t consumerUsage) { |
| if (component == NULL) { |
| return BAD_VALUE; |
| } |
| |
| |
| // Call setMaxAcquiredBufferCount without lock. |
| // setMaxAcquiredBufferCount could call back to onBuffersReleased |
| // if the buffer count change results in releasing of existing buffers, |
| // which would lead to deadlock. |
| status_t err = mConsumer->setMaxAcquiredBufferCount(bufferCount); |
| if (err != NO_ERROR) { |
| ALOGE("Unable to set BQ max acquired buffer count to %u: %d", |
| bufferCount, err); |
| return err; |
| } |
| |
| { |
| Mutex::Autolock autoLock(mMutex); |
| mComponent = component; |
| |
| err = mConsumer->setDefaultBufferSize(frameWidth, frameHeight); |
| if (err != NO_ERROR) { |
| ALOGE("Unable to set BQ default buffer size to %ux%u: %d", |
| frameWidth, frameHeight, err); |
| return err; |
| } |
| |
| consumerUsage |= GRALLOC_USAGE_HW_VIDEO_ENCODER; |
| mConsumer->setConsumerUsageBits(consumerUsage); |
| |
| // Set impl. defined format as default. Depending on the usage flags |
| // the device-specific implementation will derive the exact format. |
| err = mConsumer->setDefaultBufferFormat(HAL_PIXEL_FORMAT_IMPLEMENTATION_DEFINED); |
| if (err != NO_ERROR) { |
| ALOGE("Failed to configure surface default format ret: %d", err); |
| return err; |
| } |
| |
| // Sets the default buffer data space |
| ALOGD("setting dataspace: %#x, acquired=%d", dataSpace, mNumOutstandingAcquires); |
| mConsumer->setDefaultBufferDataSpace((android_dataspace)dataSpace); |
| mLastDataspace = (android_dataspace)dataSpace; |
| |
| mExecuting = false; |
| mSuspended = false; |
| mEndOfStream = false; |
| mEndOfStreamSent = false; |
| mSkipFramesBeforeNs = -1LL; |
| mFrameDropper.clear(); |
| mFrameRepeatIntervalUs = -1LL; |
| mRepeatLastFrameGeneration = 0; |
| mOutstandingFrameRepeatCount = 0; |
| mLatestBuffer.mBuffer.reset(); |
| mFrameRepeatBlockedOnCodecBuffer = false; |
| mFps = -1.0; |
| mCaptureFps = -1.0; |
| mBaseCaptureUs = -1LL; |
| mBaseFrameUs = -1LL; |
| mPrevCaptureUs = -1LL; |
| mPrevFrameUs = -1LL; |
| mFrameCount = 0; |
| mInputBufferTimeOffsetUs = 0; |
| mStopTimeUs = -1; |
| mActionQueue.clear(); |
| } |
| |
| return OK; |
| } |
| |
| status_t GraphicBufferSource::setSuspend(bool suspend, int64_t suspendStartTimeUs) { |
| ALOGV("setSuspend=%d at time %lld us", suspend, (long long)suspendStartTimeUs); |
| |
| Mutex::Autolock autoLock(mMutex); |
| |
| if (mStopTimeUs != -1) { |
| ALOGE("setSuspend failed as STOP action is pending"); |
| return INVALID_OPERATION; |
| } |
| |
| // Push the action to the queue. |
| if (suspendStartTimeUs != -1) { |
| // suspendStartTimeUs must be smaller or equal to current systemTime. |
| int64_t currentSystemTimeUs = systemTime() / 1000; |
| if (suspendStartTimeUs > currentSystemTimeUs) { |
| ALOGE("setSuspend failed. %lld is larger than current system time %lld us", |
| (long long)suspendStartTimeUs, (long long)currentSystemTimeUs); |
| return INVALID_OPERATION; |
| } |
| if (mLastActionTimeUs != -1 && suspendStartTimeUs < mLastActionTimeUs) { |
| ALOGE("setSuspend failed. %lld is smaller than last action time %lld us", |
| (long long)suspendStartTimeUs, (long long)mLastActionTimeUs); |
| return INVALID_OPERATION; |
| } |
| mLastActionTimeUs = suspendStartTimeUs; |
| ActionItem action; |
| action.mAction = suspend ? ActionItem::PAUSE : ActionItem::RESUME; |
| action.mActionTimeUs = suspendStartTimeUs; |
| ALOGV("Push %s action into actionQueue", suspend ? "PAUSE" : "RESUME"); |
| mActionQueue.push_back(action); |
| } else { |
| if (suspend) { |
| mSuspended = true; |
| releaseAllAvailableBuffers_l(); |
| return OK; |
| } else { |
| mSuspended = false; |
| if (mExecuting && !haveAvailableBuffers_l() |
| && mFrameRepeatBlockedOnCodecBuffer) { |
| if (repeatLatestBuffer_l()) { |
| ALOGV("suspend/deferred repeatLatestBuffer_l SUCCESS"); |
| mFrameRepeatBlockedOnCodecBuffer = false; |
| } else { |
| ALOGV("suspend/deferred repeatLatestBuffer_l FAILURE"); |
| } |
| } |
| } |
| } |
| return OK; |
| } |
| |
| status_t GraphicBufferSource::setRepeatPreviousFrameDelayUs(int64_t repeatAfterUs) { |
| ALOGV("setRepeatPreviousFrameDelayUs: delayUs=%lld", (long long)repeatAfterUs); |
| |
| Mutex::Autolock autoLock(mMutex); |
| |
| if (mExecuting || repeatAfterUs <= 0LL) { |
| return INVALID_OPERATION; |
| } |
| |
| mFrameRepeatIntervalUs = repeatAfterUs; |
| return OK; |
| } |
| |
| status_t GraphicBufferSource::setTimeOffsetUs(int64_t timeOffsetUs) { |
| Mutex::Autolock autoLock(mMutex); |
| |
| // timeOffsetUs must be negative for adjustment. |
| if (timeOffsetUs >= 0LL) { |
| return INVALID_OPERATION; |
| } |
| |
| mInputBufferTimeOffsetUs = timeOffsetUs; |
| return OK; |
| } |
| |
| status_t GraphicBufferSource::setMaxFps(float maxFps) { |
| ALOGV("setMaxFps: maxFps=%lld", (long long)maxFps); |
| |
| Mutex::Autolock autoLock(mMutex); |
| |
| if (mExecuting) { |
| return INVALID_OPERATION; |
| } |
| |
| mFrameDropper = new FrameDropper(); |
| status_t err = mFrameDropper->setMaxFrameRate(maxFps); |
| if (err != OK) { |
| mFrameDropper.clear(); |
| return err; |
| } |
| |
| return OK; |
| } |
| |
| status_t GraphicBufferSource::setStartTimeUs(int64_t skipFramesBeforeUs) { |
| ALOGV("setStartTimeUs: skipFramesBeforeUs=%lld", (long long)skipFramesBeforeUs); |
| |
| Mutex::Autolock autoLock(mMutex); |
| |
| mSkipFramesBeforeNs = |
| (skipFramesBeforeUs > 0 && skipFramesBeforeUs <= INT64_MAX / 1000) ? |
| (skipFramesBeforeUs * 1000) : -1LL; |
| |
| return OK; |
| } |
| |
| status_t GraphicBufferSource::setStopTimeUs(int64_t stopTimeUs) { |
| ALOGV("setStopTimeUs: %lld us", (long long)stopTimeUs); |
| Mutex::Autolock autoLock(mMutex); |
| |
| if (mStopTimeUs != -1) { |
| // Ignore if stop time has already been set |
| return OK; |
| } |
| |
| // stopTimeUs must be smaller or equal to current systemTime. |
| int64_t currentSystemTimeUs = systemTime() / 1000; |
| if (stopTimeUs > currentSystemTimeUs) { |
| ALOGE("setStopTimeUs failed. %lld is larger than current system time %lld us", |
| (long long)stopTimeUs, (long long)currentSystemTimeUs); |
| return INVALID_OPERATION; |
| } |
| if (mLastActionTimeUs != -1 && stopTimeUs < mLastActionTimeUs) { |
| ALOGE("setSuspend failed. %lld is smaller than last action time %lld us", |
| (long long)stopTimeUs, (long long)mLastActionTimeUs); |
| return INVALID_OPERATION; |
| } |
| mLastActionTimeUs = stopTimeUs; |
| ActionItem action; |
| action.mAction = ActionItem::STOP; |
| action.mActionTimeUs = stopTimeUs; |
| mActionQueue.push_back(action); |
| mStopTimeUs = stopTimeUs; |
| return OK; |
| } |
| |
| status_t GraphicBufferSource::getStopTimeOffsetUs(int64_t *stopTimeOffsetUs) { |
| ALOGV("getStopTimeOffsetUs"); |
| Mutex::Autolock autoLock(mMutex); |
| if (mStopTimeUs == -1) { |
| ALOGW("Fail to return stopTimeOffsetUs as stop time is not set"); |
| return INVALID_OPERATION; |
| } |
| *stopTimeOffsetUs = |
| mLastFrameTimestampUs == -1 ? 0 : mStopTimeUs - mLastFrameTimestampUs; |
| return OK; |
| } |
| |
| status_t GraphicBufferSource::setTimeLapseConfig(double fps, double captureFps) { |
| ALOGV("setTimeLapseConfig: fps=%lg, captureFps=%lg", |
| fps, captureFps); |
| Mutex::Autolock autoLock(mMutex); |
| |
| if (mExecuting || !(fps > 0) || !(captureFps > 0)) { |
| return INVALID_OPERATION; |
| } |
| |
| mFps = fps; |
| mCaptureFps = captureFps; |
| if (captureFps > fps) { |
| mSnapTimestamps = 1 == base::GetIntProperty( |
| "debug.stagefright.snap_timestamps", int64_t(0)); |
| } else { |
| mSnapTimestamps = false; |
| } |
| |
| return OK; |
| } |
| |
| status_t GraphicBufferSource::setColorAspects(int32_t aspectsPacked) { |
| Mutex::Autolock autoLock(mMutex); |
| mDefaultColorAspectsPacked = aspectsPacked; |
| ColorAspects colorAspects = ColorUtils::unpackToColorAspects(aspectsPacked); |
| ALOGD("requesting color aspects (R:%d(%s), P:%d(%s), M:%d(%s), T:%d(%s))", |
| colorAspects.mRange, asString(colorAspects.mRange), |
| colorAspects.mPrimaries, asString(colorAspects.mPrimaries), |
| colorAspects.mMatrixCoeffs, asString(colorAspects.mMatrixCoeffs), |
| colorAspects.mTransfer, asString(colorAspects.mTransfer)); |
| |
| return OK; |
| } |
| |
| status_t GraphicBufferSource::signalEndOfInputStream() { |
| Mutex::Autolock autoLock(mMutex); |
| ALOGV("signalEndOfInputStream: executing=%d available=%zu+%d eos=%d", |
| mExecuting, mAvailableBuffers.size(), mNumAvailableUnacquiredBuffers, mEndOfStream); |
| |
| if (mEndOfStream) { |
| ALOGE("EOS was already signaled"); |
| return INVALID_OPERATION; |
| } |
| |
| // Set the end-of-stream flag. If no frames are pending from the |
| // BufferQueue, and a codec buffer is available, and we're executing, |
| // and there is no stop timestamp, we initiate the EOS from here. |
| // Otherwise, we'll let codecBufferEmptied() (or start) do it. |
| // |
| // Note: if there are no pending frames and all codec buffers are |
| // available, we *must* submit the EOS from here or we'll just |
| // stall since no future events are expected. |
| mEndOfStream = true; |
| |
| if (mStopTimeUs == -1 && mExecuting && !haveAvailableBuffers_l()) { |
| submitEndOfInputStream_l(); |
| } |
| |
| return OK; |
| } |
| |
| void GraphicBufferSource::onMessageReceived(const sp<AMessage> &msg) { |
| switch (msg->what()) { |
| case kWhatRepeatLastFrame: |
| { |
| Mutex::Autolock autoLock(mMutex); |
| |
| int32_t generation; |
| CHECK(msg->findInt32("generation", &generation)); |
| |
| if (generation != mRepeatLastFrameGeneration) { |
| // stale |
| break; |
| } |
| |
| if (!mExecuting || haveAvailableBuffers_l()) { |
| break; |
| } |
| |
| bool success = repeatLatestBuffer_l(); |
| if (success) { |
| ALOGV("repeatLatestBuffer_l SUCCESS"); |
| } else { |
| ALOGV("repeatLatestBuffer_l FAILURE"); |
| mFrameRepeatBlockedOnCodecBuffer = true; |
| } |
| break; |
| } |
| |
| default: |
| TRESPASS(); |
| } |
| } |
| |
| } // namespace android |