| #include "include/private/dvr/buffer_hub_queue_producer.h" |
| |
| #include <inttypes.h> |
| #include <log/log.h> |
| |
| namespace android { |
| namespace dvr { |
| |
| BufferHubQueueProducer::BufferHubQueueProducer( |
| const std::shared_ptr<BufferHubQueueCore>& core) |
| : core_(core), req_buffer_count_(kInvalidBufferCount) {} |
| |
| status_t BufferHubQueueProducer::requestBuffer(int slot, |
| sp<GraphicBuffer>* buf) { |
| ALOGD("requestBuffer: slot=%d", slot); |
| |
| std::unique_lock<std::mutex> lock(core_->mutex_); |
| |
| if (slot < 0 || slot >= req_buffer_count_) { |
| ALOGE("requestBuffer: slot index %d out of range [0, %d)", slot, |
| req_buffer_count_); |
| return BAD_VALUE; |
| } else if (!core_->buffers_[slot].mBufferState.isDequeued()) { |
| ALOGE("requestBuffer: slot %d is not owned by the producer (state = %s)", |
| slot, core_->buffers_[slot].mBufferState.string()); |
| return BAD_VALUE; |
| } |
| |
| core_->buffers_[slot].mRequestBufferCalled = true; |
| *buf = core_->buffers_[slot].mGraphicBuffer; |
| return NO_ERROR; |
| } |
| |
| status_t BufferHubQueueProducer::setMaxDequeuedBufferCount( |
| int max_dequeued_buffers) { |
| ALOGD("setMaxDequeuedBufferCount: max_dequeued_buffers=%d", |
| max_dequeued_buffers); |
| |
| std::unique_lock<std::mutex> lock(core_->mutex_); |
| |
| if (max_dequeued_buffers <= 0 || |
| max_dequeued_buffers > |
| static_cast<int>(BufferHubQueue::kMaxQueueCapacity)) { |
| ALOGE("setMaxDequeuedBufferCount: %d out of range (0, %zu]", |
| max_dequeued_buffers, BufferHubQueue::kMaxQueueCapacity); |
| return BAD_VALUE; |
| } |
| |
| req_buffer_count_ = max_dequeued_buffers; |
| return NO_ERROR; |
| } |
| |
| status_t BufferHubQueueProducer::setAsyncMode(bool /* async */) { |
| ALOGE("BufferHubQueueProducer::setAsyncMode not implemented."); |
| return INVALID_OPERATION; |
| } |
| |
| status_t BufferHubQueueProducer::dequeueBuffer(int* out_slot, |
| sp<Fence>* out_fence, |
| uint32_t width, uint32_t height, |
| PixelFormat format, |
| uint32_t usage, |
| FrameEventHistoryDelta* /* outTimestamps */) { |
| ALOGD("dequeueBuffer: w=%u, h=%u, format=%d, usage=%u", width, height, format, |
| usage); |
| |
| status_t ret; |
| std::unique_lock<std::mutex> lock(core_->mutex_); |
| |
| if (static_cast<int32_t>(core_->producer_->capacity()) < req_buffer_count_) { |
| // Lazy allocation. When the capacity of |core_->producer_| has not reach |
| // |req_buffer_count_|, allocate new buffer. |
| // TODO(jwcai) To save memory, the really reasonable thing to do is to go |
| // over existing slots and find first existing one to dequeue. |
| ret = core_->AllocateBuffer(width, height, format, usage, 1); |
| if (ret < 0) |
| return ret; |
| } |
| |
| size_t slot; |
| std::shared_ptr<BufferProducer> buffer_producer; |
| |
| for (size_t retry = 0; retry < BufferHubQueue::kMaxQueueCapacity; retry++) { |
| buffer_producer = |
| core_->producer_->Dequeue(core_->dequeue_timeout_ms_, &slot); |
| if (!buffer_producer) |
| return NO_MEMORY; |
| |
| if (static_cast<int>(width) == buffer_producer->width() && |
| static_cast<int>(height) == buffer_producer->height() && |
| static_cast<int>(format) == buffer_producer->format()) { |
| // The producer queue returns a buffer producer matches the request. |
| break; |
| } |
| |
| // Needs reallocation. |
| // TODO(jwcai) Consider use VLOG instead if we find this log is not useful. |
| ALOGI( |
| "dequeueBuffer: requested buffer (w=%u, h=%u, format=%d) is different " |
| "from the buffer returned at slot: %zu (w=%d, h=%d, format=%d). Need " |
| "re-allocattion.", |
| width, height, format, slot, buffer_producer->width(), |
| buffer_producer->height(), buffer_producer->format()); |
| // Mark the slot as reallocating, so that later we can set |
| // BUFFER_NEEDS_REALLOCATION when the buffer actually get dequeued. |
| core_->buffers_[slot].mIsReallocating = true; |
| |
| // Detach the old buffer once the allocation before allocating its |
| // replacement. |
| core_->DetachBuffer(slot); |
| |
| // Allocate a new producer buffer with new buffer configs. Note that if |
| // there are already multiple buffers in the queue, the next one returned |
| // from |core_->producer_->Dequeue| may not be the new buffer we just |
| // reallocated. Retry up to BufferHubQueue::kMaxQueueCapacity times. |
| ret = core_->AllocateBuffer(width, height, format, usage, 1); |
| if (ret < 0) |
| return ret; |
| } |
| |
| // With the BufferHub backed solution. Buffer slot returned from |
| // |core_->producer_->Dequeue| is guaranteed to avaiable for producer's use. |
| // It's either in free state (if the buffer has never been used before) or |
| // in queued state (if the buffer has been dequeued and queued back to |
| // BufferHubQueue). |
| // TODO(jwcai) Clean this up, make mBufferState compatible with BufferHub's |
| // model. |
| LOG_ALWAYS_FATAL_IF(!core_->buffers_[slot].mBufferState.isFree() && |
| !core_->buffers_[slot].mBufferState.isQueued(), |
| "dequeueBuffer: slot %zu is not free or queued.", slot); |
| |
| core_->buffers_[slot].mBufferState.freeQueued(); |
| core_->buffers_[slot].mBufferState.dequeue(); |
| ALOGD("dequeueBuffer: slot=%zu", slot); |
| |
| // TODO(jwcai) Handle fence properly. |BufferHub| has full fence support, we |
| // just need to exopose that through |BufferHubQueue| once we need fence. |
| *out_fence = Fence::NO_FENCE; |
| *out_slot = slot; |
| ret = NO_ERROR; |
| |
| if (core_->buffers_[slot].mIsReallocating) { |
| ret |= BUFFER_NEEDS_REALLOCATION; |
| core_->buffers_[slot].mIsReallocating = false; |
| } |
| |
| return ret; |
| } |
| |
| status_t BufferHubQueueProducer::detachBuffer(int /* slot */) { |
| ALOGE("BufferHubQueueProducer::detachBuffer not implemented."); |
| return INVALID_OPERATION; |
| } |
| |
| status_t BufferHubQueueProducer::detachNextBuffer( |
| sp<GraphicBuffer>* /* out_buffer */, sp<Fence>* /* out_fence */) { |
| ALOGE("BufferHubQueueProducer::detachNextBuffer not implemented."); |
| return INVALID_OPERATION; |
| } |
| |
| status_t BufferHubQueueProducer::attachBuffer( |
| int* /* out_slot */, const sp<GraphicBuffer>& /* buffer */) { |
| // With this BufferHub backed implementation, we assume (for now) all buffers |
| // are allocated and owned by the BufferHub. Thus the attempt of transfering |
| // ownership of a buffer to the buffer queue is intentionally unsupported. |
| LOG_ALWAYS_FATAL("BufferHubQueueProducer::attachBuffer not supported."); |
| return INVALID_OPERATION; |
| } |
| |
| status_t BufferHubQueueProducer::queueBuffer(int slot, |
| const QueueBufferInput& input, |
| QueueBufferOutput* /* output */) { |
| ALOGD("queueBuffer: slot %d", slot); |
| |
| int64_t timestamp; |
| sp<Fence> fence; |
| |
| // TODO(jwcai) The following attributes are ignored. |
| bool is_auto_timestamp; |
| android_dataspace data_space; |
| Rect crop(Rect::EMPTY_RECT); |
| int scaling_mode; |
| uint32_t transform; |
| |
| input.deflate(×tamp, &is_auto_timestamp, &data_space, &crop, |
| &scaling_mode, &transform, &fence); |
| |
| if (fence == nullptr) { |
| ALOGE("queueBuffer: fence is NULL"); |
| return BAD_VALUE; |
| } |
| |
| status_t ret; |
| std::unique_lock<std::mutex> lock(core_->mutex_); |
| |
| if (slot < 0 || slot >= req_buffer_count_) { |
| ALOGE("queueBuffer: slot index %d out of range [0, %d)", slot, |
| req_buffer_count_); |
| return BAD_VALUE; |
| } else if (!core_->buffers_[slot].mBufferState.isDequeued()) { |
| ALOGE("queueBuffer: slot %d is not owned by the producer (state = %s)", |
| slot, core_->buffers_[slot].mBufferState.string()); |
| return BAD_VALUE; |
| } |
| |
| // Post the buffer producer with timestamp in the metadata. |
| auto buffer_producer = core_->buffers_[slot].mBufferProducer; |
| LocalHandle fence_fd(fence->isValid() ? fence->dup() : -1); |
| |
| BufferHubQueueCore::BufferMetadata meta_data = {.timestamp = timestamp}; |
| buffer_producer->Post(fence_fd, &meta_data, sizeof(meta_data)); |
| core_->buffers_[slot].mBufferState.queue(); |
| |
| // TODO(jwcai) check how to fill in output properly. |
| return NO_ERROR; |
| } |
| |
| status_t BufferHubQueueProducer::cancelBuffer(int slot, |
| const sp<Fence>& fence) { |
| ALOGD(__FUNCTION__); |
| |
| std::unique_lock<std::mutex> lock(core_->mutex_); |
| |
| if (slot < 0 || slot >= req_buffer_count_) { |
| ALOGE("cancelBuffer: slot index %d out of range [0, %d)", slot, |
| req_buffer_count_); |
| return BAD_VALUE; |
| } else if (!core_->buffers_[slot].mBufferState.isDequeued()) { |
| ALOGE("cancelBuffer: slot %d is not owned by the producer (state = %s)", |
| slot, core_->buffers_[slot].mBufferState.string()); |
| return BAD_VALUE; |
| } else if (fence == NULL) { |
| ALOGE("cancelBuffer: fence is NULL"); |
| return BAD_VALUE; |
| } |
| |
| auto buffer_producer = core_->buffers_[slot].mBufferProducer; |
| core_->producer_->Enqueue(buffer_producer, slot); |
| core_->buffers_[slot].mBufferState.cancel(); |
| core_->buffers_[slot].mFence = fence; |
| ALOGD("cancelBuffer: slot %d", slot); |
| |
| return NO_ERROR; |
| } |
| |
| status_t BufferHubQueueProducer::query(int what, int* out_value) { |
| ALOGD(__FUNCTION__); |
| |
| std::unique_lock<std::mutex> lock(core_->mutex_); |
| |
| if (out_value == NULL) { |
| ALOGE("query: out_value was NULL"); |
| return BAD_VALUE; |
| } |
| |
| int value = 0; |
| switch (what) { |
| case NATIVE_WINDOW_MIN_UNDEQUEUED_BUFFERS: |
| value = 0; |
| break; |
| case NATIVE_WINDOW_BUFFER_AGE: |
| value = 0; |
| break; |
| // The following queries are currently considered as unsupported. |
| // TODO(jwcai) Need to carefully check the whether they should be |
| // supported after all. |
| case NATIVE_WINDOW_WIDTH: |
| case NATIVE_WINDOW_HEIGHT: |
| case NATIVE_WINDOW_FORMAT: |
| case NATIVE_WINDOW_STICKY_TRANSFORM: |
| case NATIVE_WINDOW_CONSUMER_RUNNING_BEHIND: |
| case NATIVE_WINDOW_CONSUMER_USAGE_BITS: |
| case NATIVE_WINDOW_DEFAULT_DATASPACE: |
| default: |
| return BAD_VALUE; |
| } |
| |
| ALOGD("query: key=%d, v=%d", what, value); |
| *out_value = value; |
| return NO_ERROR; |
| } |
| |
| status_t BufferHubQueueProducer::connect( |
| const sp<IProducerListener>& /* listener */, int /* api */, |
| bool /* producer_controlled_by_app */, QueueBufferOutput* /* output */) { |
| // Consumer interaction are actually handled by buffer hub, and we need |
| // to maintain consumer operations here. Hence |connect| is a NO-OP. |
| ALOGD(__FUNCTION__); |
| return NO_ERROR; |
| } |
| |
| status_t BufferHubQueueProducer::disconnect(int /* api */, DisconnectMode /* mode */) { |
| // Consumer interaction are actually handled by buffer hub, and we need |
| // to maintain consumer operations here. Hence |disconnect| is a NO-OP. |
| ALOGD(__FUNCTION__); |
| return NO_ERROR; |
| } |
| |
| status_t BufferHubQueueProducer::setSidebandStream( |
| const sp<NativeHandle>& stream) { |
| if (stream != NULL) { |
| // TODO(jwcai) Investigate how is is used, maybe use BufferHubBuffer's |
| // metadata. |
| ALOGE("SidebandStream is not currently supported."); |
| return INVALID_OPERATION; |
| } |
| return NO_ERROR; |
| } |
| |
| void BufferHubQueueProducer::allocateBuffers(uint32_t /* width */, |
| uint32_t /* height */, |
| PixelFormat /* format */, |
| uint32_t /* usage */) { |
| // TODO(jwcai) |allocateBuffers| aims to preallocate up to the maximum number |
| // of buffers permitted by the current BufferQueue configuration (aka |
| // |req_buffer_count_|). |
| ALOGE("BufferHubQueueProducer::allocateBuffers not implemented."); |
| } |
| |
| status_t BufferHubQueueProducer::allowAllocation(bool /* allow */) { |
| ALOGE("BufferHubQueueProducer::allowAllocation not implemented."); |
| return INVALID_OPERATION; |
| } |
| |
| status_t BufferHubQueueProducer::setGenerationNumber( |
| uint32_t generation_number) { |
| ALOGD(__FUNCTION__); |
| |
| std::unique_lock<std::mutex> lock(core_->mutex_); |
| core_->generation_number_ = generation_number; |
| return NO_ERROR; |
| } |
| |
| String8 BufferHubQueueProducer::getConsumerName() const { |
| // BufferHub based implementation could have one to many producer/consumer |
| // relationship, thus |getConsumerName| from the producer side does not |
| // make any sense. |
| ALOGE("BufferHubQueueProducer::getConsumerName not supported."); |
| return String8("BufferHubQueue::DummyConsumer"); |
| } |
| |
| status_t BufferHubQueueProducer::setSharedBufferMode( |
| bool /* shared_buffer_mode */) { |
| ALOGE("BufferHubQueueProducer::setSharedBufferMode not implemented."); |
| return INVALID_OPERATION; |
| } |
| |
| status_t BufferHubQueueProducer::setAutoRefresh(bool /* auto_refresh */) { |
| ALOGE("BufferHubQueueProducer::setAutoRefresh not implemented."); |
| return INVALID_OPERATION; |
| } |
| |
| status_t BufferHubQueueProducer::setDequeueTimeout(nsecs_t timeout) { |
| ALOGD(__FUNCTION__); |
| |
| std::unique_lock<std::mutex> lock(core_->mutex_); |
| core_->dequeue_timeout_ms_ = static_cast<int>(timeout / (1000 * 1000)); |
| return NO_ERROR; |
| } |
| |
| status_t BufferHubQueueProducer::getLastQueuedBuffer( |
| sp<GraphicBuffer>* /* out_buffer */, sp<Fence>* /* out_fence */, |
| float /*out_transform_matrix*/[16]) { |
| ALOGE("BufferHubQueueProducer::getLastQueuedBuffer not implemented."); |
| return INVALID_OPERATION; |
| } |
| |
| void BufferHubQueueProducer::getFrameTimestamps( |
| FrameEventHistoryDelta* /*outDelta*/) { |
| ALOGE("BufferHubQueueProducer::getFrameTimestamps not implemented."); |
| } |
| |
| status_t BufferHubQueueProducer::getUniqueId(uint64_t* out_id) const { |
| ALOGD(__FUNCTION__); |
| |
| *out_id = core_->unique_id_; |
| return NO_ERROR; |
| } |
| |
| IBinder* BufferHubQueueProducer::onAsBinder() { |
| // BufferHubQueueProducer is a non-binder implementation of |
| // IGraphicBufferProducer. |
| ALOGW("BufferHubQueueProducer::onAsBinder is not supported."); |
| return nullptr; |
| } |
| |
| } // namespace dvr |
| } // namespace android |