blob: 1f71e23b4dfa6c240fe9c1b52ca0e1c8a57ee4f8 [file] [log] [blame]
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -08001/*
2 * Copyright 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
Jiwen 'Steve' Caid6cb17f2017-05-08 16:15:35 -070017#include <dvr/dvr_api.h>
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -080018#include <gui/BufferHubProducer.h>
Alex Vakulenko4fe60582017-02-02 11:35:59 -080019#include <inttypes.h>
20#include <log/log.h>
Mathias Agopian6a3c05b2017-04-27 20:06:55 -070021#include <system/window.h>
Alex Vakulenko4fe60582017-02-02 11:35:59 -080022
Alex Vakulenkoe4eec202017-01-27 14:41:04 -080023namespace android {
Alex Vakulenkoe4eec202017-01-27 14:41:04 -080024
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -080025using namespace dvr;
26
Jiwen 'Steve' Caid6cb17f2017-05-08 16:15:35 -070027/* static */
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -080028sp<BufferHubProducer> BufferHubProducer::Create(const std::shared_ptr<ProducerQueue>& queue) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080029 sp<BufferHubProducer> producer = new BufferHubProducer;
30 producer->queue_ = queue;
31 return producer;
Jiwen 'Steve' Caid6cb17f2017-05-08 16:15:35 -070032}
Alex Vakulenkoe4eec202017-01-27 14:41:04 -080033
Jiwen 'Steve' Cai5afb7402017-11-09 18:50:32 -080034/* static */
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -080035sp<BufferHubProducer> BufferHubProducer::Create(ProducerQueueParcelable parcelable) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080036 if (!parcelable.IsValid()) {
37 ALOGE("BufferHubProducer::Create: Invalid producer parcelable.");
38 return nullptr;
39 }
Jiwen 'Steve' Cai5afb7402017-11-09 18:50:32 -080040
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080041 sp<BufferHubProducer> producer = new BufferHubProducer;
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -080042 producer->queue_ = ProducerQueue::Import(parcelable.TakeChannelHandle());
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080043 return producer;
Jiwen 'Steve' Cai5afb7402017-11-09 18:50:32 -080044}
45
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -080046status_t BufferHubProducer::requestBuffer(int slot, sp<GraphicBuffer>* buf) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080047 ALOGV("requestBuffer: slot=%d", slot);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -080048
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080049 std::unique_lock<std::mutex> lock(mutex_);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -080050
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080051 if (connected_api_ == kNoConnectedApi) {
52 ALOGE("requestBuffer: BufferHubProducer has no connected producer");
53 return NO_INIT;
54 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -070055
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080056 if (slot < 0 || slot >= max_buffer_count_) {
57 ALOGE("requestBuffer: slot index %d out of range [0, %d)", slot, max_buffer_count_);
58 return BAD_VALUE;
59 } else if (!buffers_[slot].mBufferState.isDequeued()) {
60 ALOGE("requestBuffer: slot %d is not owned by the producer (state = %s)", slot,
61 buffers_[slot].mBufferState.string());
62 return BAD_VALUE;
63 } else if (buffers_[slot].mGraphicBuffer != nullptr) {
64 ALOGE("requestBuffer: slot %d is not empty.", slot);
65 return BAD_VALUE;
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -080066 } else if (buffers_[slot].mProducerBuffer == nullptr) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080067 ALOGE("requestBuffer: slot %d is not dequeued.", slot);
68 return BAD_VALUE;
69 }
Alex Vakulenkoe4eec202017-01-27 14:41:04 -080070
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -080071 const auto& producer_buffer = buffers_[slot].mProducerBuffer;
72 sp<GraphicBuffer> graphic_buffer = producer_buffer->buffer()->buffer();
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -070073
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080074 buffers_[slot].mGraphicBuffer = graphic_buffer;
75 buffers_[slot].mRequestBufferCalled = true;
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -070076
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080077 *buf = graphic_buffer;
78 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -080079}
80
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080081status_t BufferHubProducer::setMaxDequeuedBufferCount(int max_dequeued_buffers) {
82 ALOGV("setMaxDequeuedBufferCount: max_dequeued_buffers=%d", max_dequeued_buffers);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -080083
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080084 std::unique_lock<std::mutex> lock(mutex_);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -080085
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080086 if (max_dequeued_buffers <= 0 ||
87 max_dequeued_buffers >
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -080088 int(BufferHubQueue::kMaxQueueCapacity - kDefaultUndequeuedBuffers)) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080089 ALOGE("setMaxDequeuedBufferCount: %d out of range (0, %zu]", max_dequeued_buffers,
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -080090 BufferHubQueue::kMaxQueueCapacity);
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080091 return BAD_VALUE;
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -070092 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -070093
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080094 // The new dequeued_buffers count should not be violated by the number
95 // of currently dequeued buffers.
96 int dequeued_count = 0;
97 for (const auto& buf : buffers_) {
98 if (buf.mBufferState.isDequeued()) {
99 dequeued_count++;
100 }
101 }
102 if (dequeued_count > max_dequeued_buffers) {
103 ALOGE("setMaxDequeuedBufferCount: the requested dequeued_buffers"
104 "count (%d) exceeds the current dequeued buffer count (%d)",
105 max_dequeued_buffers, dequeued_count);
106 return BAD_VALUE;
107 }
108
109 max_dequeued_buffer_count_ = max_dequeued_buffers;
110 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800111}
112
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800113status_t BufferHubProducer::setAsyncMode(bool async) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800114 if (async) {
115 // TODO(b/36724099) BufferHubQueue's consumer end always acquires the buffer
116 // automatically and behaves differently from IGraphicBufferConsumer. Thus,
117 // android::BufferQueue's async mode (a.k.a. allocating an additional buffer
118 // to prevent dequeueBuffer from being blocking) technically does not apply
119 // here.
120 //
121 // In Daydream, non-blocking producer side dequeue is guaranteed by careful
122 // buffer consumer implementations. In another word, BufferHubQueue based
123 // dequeueBuffer should never block whether setAsyncMode(true) is set or
124 // not.
125 //
126 // See: IGraphicBufferProducer::setAsyncMode and
127 // BufferQueueProducer::setAsyncMode for more about original implementation.
128 ALOGW("BufferHubProducer::setAsyncMode: BufferHubQueue should always be "
129 "asynchronous. This call makes no effact.");
130 return NO_ERROR;
131 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700132 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800133}
134
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800135status_t BufferHubProducer::dequeueBuffer(int* out_slot, sp<Fence>* out_fence, uint32_t width,
136 uint32_t height, PixelFormat format, uint64_t usage,
137 uint64_t* /*outBufferAge*/,
138 FrameEventHistoryDelta* /* out_timestamps */) {
Jiwen 'Steve' Caie7cf62a2018-04-30 22:56:08 -0700139 ALOGV("dequeueBuffer: w=%u, h=%u, format=%d, usage=%" PRIu64, width, height, format, usage);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800140
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800141 status_t ret;
142 std::unique_lock<std::mutex> lock(mutex_);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800143
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800144 if (connected_api_ == kNoConnectedApi) {
145 ALOGE("dequeueBuffer: BufferQueue has no connected producer");
146 return NO_INIT;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800147 }
148
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800149 const uint32_t kLayerCount = 1;
150 if (int32_t(queue_->capacity()) < max_dequeued_buffer_count_ + kDefaultUndequeuedBuffers) {
151 // Lazy allocation. When the capacity of |queue_| has not reached
152 // |max_dequeued_buffer_count_|, allocate new buffer.
153 // TODO(jwcai) To save memory, the really reasonable thing to do is to go
154 // over existing slots and find first existing one to dequeue.
155 ret = AllocateBuffer(width, height, kLayerCount, format, usage);
156 if (ret < 0) return ret;
157 }
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800158
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800159 size_t slot = 0;
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800160 std::shared_ptr<ProducerBuffer> producer_buffer;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800161
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -0800162 for (size_t retry = 0; retry < BufferHubQueue::kMaxQueueCapacity; retry++) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800163 LocalHandle fence;
164 auto buffer_status = queue_->Dequeue(dequeue_timeout_ms_, &slot, &fence);
165 if (!buffer_status) return NO_MEMORY;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800166
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800167 producer_buffer = buffer_status.take();
168 if (!producer_buffer) return NO_MEMORY;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800169
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800170 if (width == producer_buffer->width() && height == producer_buffer->height() &&
171 uint32_t(format) == producer_buffer->format()) {
172 // The producer queue returns a producer buffer matches the request.
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800173 break;
174 }
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800175
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800176 // Needs reallocation.
177 // TODO(jwcai) Consider use VLOG instead if we find this log is not useful.
178 ALOGI("dequeueBuffer: requested buffer (w=%u, h=%u, format=%u) is different "
179 "from the buffer returned at slot: %zu (w=%u, h=%u, format=%u). Need "
180 "re-allocattion.",
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800181 width, height, format, slot, producer_buffer->width(), producer_buffer->height(),
182 producer_buffer->format());
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800183 // Mark the slot as reallocating, so that later we can set
184 // BUFFER_NEEDS_REALLOCATION when the buffer actually get dequeued.
185 buffers_[slot].mIsReallocating = true;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800186
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800187 // Remove the old buffer once the allocation before allocating its
188 // replacement.
189 RemoveBuffer(slot);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800190
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800191 // Allocate a new producer buffer with new buffer configs. Note that if
192 // there are already multiple buffers in the queue, the next one returned
193 // from |queue_->Dequeue| may not be the new buffer we just reallocated.
194 // Retry up to BufferHubQueue::kMaxQueueCapacity times.
195 ret = AllocateBuffer(width, height, kLayerCount, format, usage);
196 if (ret < 0) return ret;
197 }
198
199 // With the BufferHub backed solution. Buffer slot returned from
200 // |queue_->Dequeue| is guaranteed to avaiable for producer's use.
201 // It's either in free state (if the buffer has never been used before) or
202 // in queued state (if the buffer has been dequeued and queued back to
203 // BufferHubQueue).
204 LOG_ALWAYS_FATAL_IF((!buffers_[slot].mBufferState.isFree() &&
205 !buffers_[slot].mBufferState.isQueued()),
206 "dequeueBuffer: slot %zu is not free or queued, actual state: %s.", slot,
207 buffers_[slot].mBufferState.string());
208
209 buffers_[slot].mBufferState.freeQueued();
210 buffers_[slot].mBufferState.dequeue();
Jiwen 'Steve' Caie7cf62a2018-04-30 22:56:08 -0700211 ALOGV("dequeueBuffer: slot=%zu", slot);
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800212
213 // TODO(jwcai) Handle fence properly. |BufferHub| has full fence support, we
214 // just need to exopose that through |BufferHubQueue| once we need fence.
215 *out_fence = Fence::NO_FENCE;
216 *out_slot = int(slot);
217 ret = NO_ERROR;
218
219 if (buffers_[slot].mIsReallocating) {
220 ret |= BUFFER_NEEDS_REALLOCATION;
221 buffers_[slot].mIsReallocating = false;
222 }
223
224 return ret;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800225}
226
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700227status_t BufferHubProducer::detachBuffer(int slot) {
228 ALOGV("detachBuffer: slot=%d", slot);
229 std::unique_lock<std::mutex> lock(mutex_);
230
231 return DetachBufferLocked(static_cast<size_t>(slot));
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800232}
233
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700234status_t BufferHubProducer::DetachBufferLocked(size_t slot) {
235 if (connected_api_ == kNoConnectedApi) {
236 ALOGE("detachBuffer: BufferHubProducer is not connected.");
237 return NO_INIT;
238 }
239
240 if (slot >= static_cast<size_t>(max_buffer_count_)) {
241 ALOGE("detachBuffer: slot index %zu out of range [0, %d)", slot, max_buffer_count_);
242 return BAD_VALUE;
243 } else if (!buffers_[slot].mBufferState.isDequeued()) {
244 ALOGE("detachBuffer: slot %zu is not owned by the producer (state = %s)", slot,
245 buffers_[slot].mBufferState.string());
246 return BAD_VALUE;
247 } else if (!buffers_[slot].mRequestBufferCalled) {
248 ALOGE("detachBuffer: buffer in slot %zu has not been requested", slot);
249 return BAD_VALUE;
250 }
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800251 std::shared_ptr<ProducerBuffer> producer_buffer = queue_->GetBuffer(slot);
252 if (producer_buffer == nullptr || producer_buffer->buffer() == nullptr) {
253 ALOGE("detachBuffer: Invalid ProducerBuffer at slot %zu.", slot);
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700254 return BAD_VALUE;
255 }
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800256 sp<GraphicBuffer> graphic_buffer = producer_buffer->buffer()->buffer();
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700257 if (graphic_buffer == nullptr) {
258 ALOGE("detachBuffer: Invalid GraphicBuffer at slot %zu.", slot);
259 return BAD_VALUE;
260 }
261
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800262 // Remove the ProducerBuffer from the ProducerQueue.
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700263 status_t error = RemoveBuffer(slot);
264 if (error != NO_ERROR) {
265 ALOGE("detachBuffer: Failed to remove buffer, slot=%zu, error=%d.", slot, error);
266 return error;
267 }
268
269 // Here we need to convert the existing ProducerBuffer into a DetachedBufferHandle and inject
270 // the handle into the GraphicBuffer object at the requested slot.
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800271 auto status_or_handle = producer_buffer->Detach();
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700272 if (!status_or_handle.ok()) {
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800273 ALOGE("detachBuffer: Failed to detach from a ProducerBuffer at slot %zu, error=%d.", slot,
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700274 status_or_handle.error());
275 return BAD_VALUE;
276 }
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700277
Jiwen 'Steve' Cai77565572018-10-15 11:31:29 -0700278 // TODO(b/70912269): Reimplement BufferHubProducer::DetachBufferLocked() once GraphicBuffer can
279 // be directly backed by BufferHub.
280 return INVALID_OPERATION;
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700281}
282
283status_t BufferHubProducer::detachNextBuffer(sp<GraphicBuffer>* out_buffer, sp<Fence>* out_fence) {
284 ALOGV("detachNextBuffer.");
285
286 if (out_buffer == nullptr || out_fence == nullptr) {
287 ALOGE("detachNextBuffer: Invalid parameter: out_buffer=%p, out_fence=%p", out_buffer,
288 out_fence);
289 return BAD_VALUE;
290 }
291
292 std::unique_lock<std::mutex> lock(mutex_);
293
294 if (connected_api_ == kNoConnectedApi) {
295 ALOGE("detachNextBuffer: BufferHubProducer is not connected.");
296 return NO_INIT;
297 }
298
299 // detachNextBuffer is equivalent to calling dequeueBuffer, requestBuffer, and detachBuffer in
300 // sequence, except for two things:
301 //
302 // 1) It is unnecessary to know the dimensions, format, or usage of the next buffer, i.e. the
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800303 // function just returns whatever ProducerBuffer is available from the ProducerQueue and no
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700304 // buffer allocation or re-allocation will happen.
305 // 2) It will not block, since if it cannot find an appropriate buffer to return, it will return
306 // an error instead.
307 size_t slot = 0;
308 LocalHandle fence;
309
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800310 // First, dequeue a ProducerBuffer from the ProducerQueue with no timeout. Report error
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700311 // immediately if ProducerQueue::Dequeue() fails.
312 auto status_or_buffer = queue_->Dequeue(/*timeout=*/0, &slot, &fence);
313 if (!status_or_buffer.ok()) {
314 ALOGE("detachNextBuffer: Failed to dequeue buffer, error=%d.", status_or_buffer.error());
315 return NO_MEMORY;
316 }
317
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800318 std::shared_ptr<ProducerBuffer> producer_buffer = status_or_buffer.take();
319 if (producer_buffer == nullptr) {
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700320 ALOGE("detachNextBuffer: Dequeued buffer is null.");
321 return NO_MEMORY;
322 }
323
324 // With the BufferHub backed solution, slot returned from |queue_->Dequeue| is guaranteed to
325 // be available for producer's use. It's either in free state (if the buffer has never been used
326 // before) or in queued state (if the buffer has been dequeued and queued back to
327 // BufferHubQueue).
328 if (!buffers_[slot].mBufferState.isFree() && !buffers_[slot].mBufferState.isQueued()) {
329 ALOGE("detachNextBuffer: slot %zu is not free or queued, actual state: %s.", slot,
330 buffers_[slot].mBufferState.string());
331 return BAD_VALUE;
332 }
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800333 if (buffers_[slot].mProducerBuffer == nullptr) {
334 ALOGE("detachNextBuffer: ProducerBuffer at slot %zu is null.", slot);
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700335 return BAD_VALUE;
336 }
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800337 if (buffers_[slot].mProducerBuffer->id() != producer_buffer->id()) {
338 ALOGE("detachNextBuffer: ProducerBuffer at slot %zu has mismatched id, actual: "
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700339 "%d, expected: %d.",
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800340 slot, buffers_[slot].mProducerBuffer->id(), producer_buffer->id());
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700341 return BAD_VALUE;
342 }
343
344 ALOGV("detachNextBuffer: slot=%zu", slot);
345 buffers_[slot].mBufferState.freeQueued();
346 buffers_[slot].mBufferState.dequeue();
347
348 // Second, request the buffer.
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800349 sp<GraphicBuffer> graphic_buffer = producer_buffer->buffer()->buffer();
350 buffers_[slot].mGraphicBuffer = producer_buffer->buffer()->buffer();
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700351
352 // Finally, detach the buffer and then return.
353 status_t error = DetachBufferLocked(slot);
354 if (error == NO_ERROR) {
355 *out_fence = new Fence(fence.Release());
356 *out_buffer = graphic_buffer;
357 }
358 return error;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800359}
360
Jiwen 'Steve' Cai57ae3ee2018-05-03 17:51:52 -0700361status_t BufferHubProducer::attachBuffer(int* out_slot, const sp<GraphicBuffer>& buffer) {
362 // In the BufferHub design, all buffers are allocated and owned by the BufferHub. Thus only
363 // GraphicBuffers that are originated from BufferHub can be attached to a BufferHubProducer.
364 ALOGV("queueBuffer: buffer=%p", buffer.get());
365
366 if (out_slot == nullptr) {
367 ALOGE("attachBuffer: out_slot cannot be NULL.");
368 return BAD_VALUE;
369 }
Jiwen 'Steve' Cai77565572018-10-15 11:31:29 -0700370 if (buffer == nullptr) {
Jiwen 'Steve' Cai57ae3ee2018-05-03 17:51:52 -0700371 ALOGE("attachBuffer: invalid GraphicBuffer.");
372 return BAD_VALUE;
373 }
374
375 std::unique_lock<std::mutex> lock(mutex_);
376
377 if (connected_api_ == kNoConnectedApi) {
378 ALOGE("attachBuffer: BufferQueue has no connected producer");
379 return NO_INIT;
380 }
381
382 // Before attaching the buffer, caller is supposed to call
383 // IGraphicBufferProducer::setGenerationNumber to inform the
384 // BufferHubProducer the next generation number.
385 if (buffer->getGenerationNumber() != generation_number_) {
386 ALOGE("attachBuffer: Mismatched generation number, buffer: %u, queue: %u.",
387 buffer->getGenerationNumber(), generation_number_);
388 return BAD_VALUE;
389 }
390
Jiwen 'Steve' Cai77565572018-10-15 11:31:29 -0700391 // TODO(b/70912269): Reimplement BufferHubProducer::DetachBufferLocked() once GraphicBuffer can
392 // be directly backed by BufferHub.
393 return INVALID_OPERATION;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800394}
395
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800396status_t BufferHubProducer::queueBuffer(int slot, const QueueBufferInput& input,
397 QueueBufferOutput* output) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800398 ALOGV("queueBuffer: slot %d", slot);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800399
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800400 if (output == nullptr) {
401 return BAD_VALUE;
402 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700403
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800404 int64_t timestamp;
405 bool is_auto_timestamp;
406 android_dataspace dataspace;
407 Rect crop(Rect::EMPTY_RECT);
408 int scaling_mode;
409 uint32_t transform;
410 sp<Fence> fence;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800411
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800412 input.deflate(&timestamp, &is_auto_timestamp, &dataspace, &crop, &scaling_mode, &transform,
413 &fence);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800414
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800415 // Check input scaling mode is valid.
416 switch (scaling_mode) {
417 case NATIVE_WINDOW_SCALING_MODE_FREEZE:
418 case NATIVE_WINDOW_SCALING_MODE_SCALE_TO_WINDOW:
419 case NATIVE_WINDOW_SCALING_MODE_SCALE_CROP:
420 case NATIVE_WINDOW_SCALING_MODE_NO_SCALE_CROP:
421 break;
422 default:
423 ALOGE("queueBuffer: unknown scaling mode %d", scaling_mode);
424 return BAD_VALUE;
425 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700426
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800427 // Check input fence is valid.
428 if (fence == nullptr) {
429 ALOGE("queueBuffer: fence is NULL");
430 return BAD_VALUE;
431 }
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800432
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800433 std::unique_lock<std::mutex> lock(mutex_);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800434
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800435 if (connected_api_ == kNoConnectedApi) {
436 ALOGE("queueBuffer: BufferQueue has no connected producer");
437 return NO_INIT;
438 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700439
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800440 if (slot < 0 || slot >= max_buffer_count_) {
441 ALOGE("queueBuffer: slot index %d out of range [0, %d)", slot, max_buffer_count_);
442 return BAD_VALUE;
443 } else if (!buffers_[slot].mBufferState.isDequeued()) {
444 ALOGE("queueBuffer: slot %d is not owned by the producer (state = %s)", slot,
445 buffers_[slot].mBufferState.string());
446 return BAD_VALUE;
447 } else if ((!buffers_[slot].mRequestBufferCalled || buffers_[slot].mGraphicBuffer == nullptr)) {
448 ALOGE("queueBuffer: slot %d is not requested (mRequestBufferCalled=%d, "
449 "mGraphicBuffer=%p)",
450 slot, buffers_[slot].mRequestBufferCalled, buffers_[slot].mGraphicBuffer.get());
451 return BAD_VALUE;
452 }
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800453
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800454 // Post the producer buffer with timestamp in the metadata.
455 const auto& producer_buffer = buffers_[slot].mProducerBuffer;
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700456
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800457 // Check input crop is not out of boundary of current buffer.
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800458 Rect buffer_rect(producer_buffer->width(), producer_buffer->height());
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800459 Rect cropped_rect(Rect::EMPTY_RECT);
460 crop.intersect(buffer_rect, &cropped_rect);
461 if (cropped_rect != crop) {
462 ALOGE("queueBuffer: slot %d has out-of-boundary crop.", slot);
463 return BAD_VALUE;
464 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700465
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800466 LocalHandle fence_fd(fence->isValid() ? fence->dup() : -1);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800467
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800468 DvrNativeBufferMetadata meta_data;
469 meta_data.timestamp = timestamp;
470 meta_data.is_auto_timestamp = int32_t(is_auto_timestamp);
471 meta_data.dataspace = int32_t(dataspace);
472 meta_data.crop_left = crop.left;
473 meta_data.crop_top = crop.top;
474 meta_data.crop_right = crop.right;
475 meta_data.crop_bottom = crop.bottom;
476 meta_data.scaling_mode = int32_t(scaling_mode);
477 meta_data.transform = int32_t(transform);
Jiwen 'Steve' Caicb4751c2017-04-14 17:56:55 -0700478
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800479 producer_buffer->PostAsync(&meta_data, fence_fd);
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800480 buffers_[slot].mBufferState.queue();
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800481
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800482 output->width = producer_buffer->width();
483 output->height = producer_buffer->height();
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800484 output->transformHint = 0; // default value, we don't use it yet.
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700485
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800486 // |numPendingBuffers| counts of the number of buffers that has been enqueued
487 // by the producer but not yet acquired by the consumer. Due to the nature
488 // of BufferHubQueue design, this is hard to trace from the producer's client
489 // side, but it's safe to assume it's zero.
490 output->numPendingBuffers = 0;
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700491
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800492 // Note that we are not setting nextFrameNumber here as it seems to be only
493 // used by surface flinger. See more at b/22802885, ag/791760.
494 output->nextFrameNumber = 0;
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700495
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800496 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800497}
498
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800499status_t BufferHubProducer::cancelBuffer(int slot, const sp<Fence>& fence) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800500 ALOGV(__FUNCTION__);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800501
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800502 std::unique_lock<std::mutex> lock(mutex_);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800503
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800504 if (connected_api_ == kNoConnectedApi) {
505 ALOGE("cancelBuffer: BufferQueue has no connected producer");
506 return NO_INIT;
507 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700508
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800509 if (slot < 0 || slot >= max_buffer_count_) {
510 ALOGE("cancelBuffer: slot index %d out of range [0, %d)", slot, max_buffer_count_);
511 return BAD_VALUE;
512 } else if (!buffers_[slot].mBufferState.isDequeued()) {
513 ALOGE("cancelBuffer: slot %d is not owned by the producer (state = %s)", slot,
514 buffers_[slot].mBufferState.string());
515 return BAD_VALUE;
516 } else if (fence == nullptr) {
517 ALOGE("cancelBuffer: fence is NULL");
518 return BAD_VALUE;
519 }
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800520
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800521 auto producer_buffer = buffers_[slot].mProducerBuffer;
522 queue_->Enqueue(producer_buffer, size_t(slot), 0U);
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800523 buffers_[slot].mBufferState.cancel();
524 buffers_[slot].mFence = fence;
525 ALOGV("cancelBuffer: slot %d", slot);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800526
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800527 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800528}
529
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800530status_t BufferHubProducer::query(int what, int* out_value) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800531 ALOGV(__FUNCTION__);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800532
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800533 std::unique_lock<std::mutex> lock(mutex_);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800534
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800535 if (out_value == nullptr) {
536 ALOGE("query: out_value was NULL");
537 return BAD_VALUE;
538 }
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800539
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800540 int value = 0;
541 switch (what) {
542 case NATIVE_WINDOW_MIN_UNDEQUEUED_BUFFERS:
543 // TODO(b/36187402) This should be the maximum number of buffers that this
544 // producer queue's consumer can acquire. Set to be at least one. Need to
545 // find a way to set from the consumer side.
546 value = kDefaultUndequeuedBuffers;
547 break;
548 case NATIVE_WINDOW_BUFFER_AGE:
549 value = 0;
550 break;
551 case NATIVE_WINDOW_WIDTH:
552 value = int32_t(queue_->default_width());
553 break;
554 case NATIVE_WINDOW_HEIGHT:
555 value = int32_t(queue_->default_height());
556 break;
557 case NATIVE_WINDOW_FORMAT:
558 value = int32_t(queue_->default_format());
559 break;
560 case NATIVE_WINDOW_CONSUMER_RUNNING_BEHIND:
561 // BufferHubQueue is always operating in async mode, thus semantically
562 // consumer can never be running behind. See BufferQueueCore.cpp core
563 // for more information about the original meaning of this flag.
564 value = 0;
565 break;
566 case NATIVE_WINDOW_CONSUMER_USAGE_BITS:
567 // TODO(jwcai) This is currently not implement as we don't need
568 // IGraphicBufferConsumer parity.
569 value = 0;
570 break;
571 case NATIVE_WINDOW_DEFAULT_DATASPACE:
572 // TODO(jwcai) Return the default value android::BufferQueue is using as
573 // there is no way dvr::ConsumerQueue can set it.
574 value = 0; // HAL_DATASPACE_UNKNOWN
575 break;
576 case NATIVE_WINDOW_STICKY_TRANSFORM:
577 // TODO(jwcai) Return the default value android::BufferQueue is using as
578 // there is no way dvr::ConsumerQueue can set it.
579 value = 0;
580 break;
581 case NATIVE_WINDOW_CONSUMER_IS_PROTECTED:
582 // In Daydream's implementation, the consumer end (i.e. VR Compostior)
583 // knows how to handle protected buffers.
584 value = 1;
585 break;
586 default:
587 return BAD_VALUE;
588 }
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800589
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800590 ALOGV("query: key=%d, v=%d", what, value);
591 *out_value = value;
592 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800593}
594
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800595status_t BufferHubProducer::connect(const sp<IProducerListener>& /* listener */, int api,
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800596 bool /* producer_controlled_by_app */,
597 QueueBufferOutput* output) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800598 // Consumer interaction are actually handled by buffer hub, and we need
599 // to maintain consumer operations here. We only need to perform basic input
600 // parameter checks here.
601 ALOGV(__FUNCTION__);
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700602
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800603 if (output == nullptr) {
604 return BAD_VALUE;
605 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700606
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800607 std::unique_lock<std::mutex> lock(mutex_);
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700608
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800609 if (connected_api_ != kNoConnectedApi) {
610 return BAD_VALUE;
611 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700612
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800613 if (!queue_->is_connected()) {
614 ALOGE("BufferHubProducer::connect: This BufferHubProducer is not "
615 "connected to bufferhud. Has it been taken out as a parcelable?");
616 return BAD_VALUE;
617 }
Jiwen 'Steve' Cai5afb7402017-11-09 18:50:32 -0800618
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800619 switch (api) {
620 case NATIVE_WINDOW_API_EGL:
621 case NATIVE_WINDOW_API_CPU:
622 case NATIVE_WINDOW_API_MEDIA:
623 case NATIVE_WINDOW_API_CAMERA:
624 connected_api_ = api;
Jiwen 'Steve' Caicb4751c2017-04-14 17:56:55 -0700625
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800626 output->width = queue_->default_width();
627 output->height = queue_->default_height();
Jiwen 'Steve' Caicb4751c2017-04-14 17:56:55 -0700628
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800629 // default values, we don't use them yet.
630 output->transformHint = 0;
631 output->numPendingBuffers = 0;
632 output->nextFrameNumber = 0;
633 output->bufferReplaced = false;
Jiwen 'Steve' Caicb4751c2017-04-14 17:56:55 -0700634
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800635 break;
636 default:
637 ALOGE("BufferHubProducer::connect: unknow API %d", api);
638 return BAD_VALUE;
639 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700640
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800641 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800642}
643
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800644status_t BufferHubProducer::disconnect(int api, DisconnectMode /*mode*/) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800645 // Consumer interaction are actually handled by buffer hub, and we need
646 // to maintain consumer operations here. We only need to perform basic input
647 // parameter checks here.
648 ALOGV(__FUNCTION__);
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700649
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800650 std::unique_lock<std::mutex> lock(mutex_);
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700651
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800652 if (kNoConnectedApi == connected_api_) {
653 return NO_INIT;
654 } else if (api != connected_api_) {
655 return BAD_VALUE;
656 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700657
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800658 FreeAllBuffers();
659 connected_api_ = kNoConnectedApi;
660 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800661}
662
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800663status_t BufferHubProducer::setSidebandStream(const sp<NativeHandle>& stream) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800664 if (stream != nullptr) {
665 // TODO(jwcai) Investigate how is is used, maybe use BufferHubBuffer's
666 // metadata.
667 ALOGE("SidebandStream is not currently supported.");
668 return INVALID_OPERATION;
669 }
670 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800671}
672
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800673void BufferHubProducer::allocateBuffers(uint32_t /* width */, uint32_t /* height */,
674 PixelFormat /* format */, uint64_t /* usage */) {
675 // TODO(jwcai) |allocateBuffers| aims to preallocate up to the maximum number
676 // of buffers permitted by the current BufferQueue configuration (aka
677 // |max_buffer_count_|).
678 ALOGE("BufferHubProducer::allocateBuffers not implemented.");
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800679}
680
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800681status_t BufferHubProducer::allowAllocation(bool /* allow */) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800682 ALOGE("BufferHubProducer::allowAllocation not implemented.");
683 return INVALID_OPERATION;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800684}
685
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800686status_t BufferHubProducer::setGenerationNumber(uint32_t generation_number) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800687 ALOGV(__FUNCTION__);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800688
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800689 std::unique_lock<std::mutex> lock(mutex_);
690 generation_number_ = generation_number;
691 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800692}
693
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800694String8 BufferHubProducer::getConsumerName() const {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800695 // BufferHub based implementation could have one to many producer/consumer
696 // relationship, thus |getConsumerName| from the producer side does not
697 // make any sense.
698 ALOGE("BufferHubProducer::getConsumerName not supported.");
Peiyong Lind8460c82020-07-28 16:04:22 -0700699 return String8("BufferHubQueue::StubConsumer");
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800700}
701
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800702status_t BufferHubProducer::setSharedBufferMode(bool shared_buffer_mode) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800703 if (shared_buffer_mode) {
704 ALOGE("BufferHubProducer::setSharedBufferMode(true) is not supported.");
705 // TODO(b/36373181) Front buffer mode for buffer hub queue as ANativeWindow.
706 return INVALID_OPERATION;
707 }
708 // Setting to default should just work as a no-op.
709 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800710}
711
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800712status_t BufferHubProducer::setAutoRefresh(bool auto_refresh) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800713 if (auto_refresh) {
714 ALOGE("BufferHubProducer::setAutoRefresh(true) is not supported.");
715 return INVALID_OPERATION;
716 }
717 // Setting to default should just work as a no-op.
718 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800719}
720
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800721status_t BufferHubProducer::setDequeueTimeout(nsecs_t timeout) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800722 ALOGV(__FUNCTION__);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800723
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800724 std::unique_lock<std::mutex> lock(mutex_);
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -0800725 dequeue_timeout_ms_ = static_cast<int>(timeout / (1000 * 1000));
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800726 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800727}
728
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800729status_t BufferHubProducer::getLastQueuedBuffer(sp<GraphicBuffer>* /* out_buffer */,
730 sp<Fence>* /* out_fence */,
731 float /*out_transform_matrix*/[16]) {
732 ALOGE("BufferHubProducer::getLastQueuedBuffer not implemented.");
733 return INVALID_OPERATION;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800734}
735
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800736void BufferHubProducer::getFrameTimestamps(FrameEventHistoryDelta* /*outDelta*/) {
737 ALOGE("BufferHubProducer::getFrameTimestamps not implemented.");
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800738}
739
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800740status_t BufferHubProducer::getUniqueId(uint64_t* out_id) const {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800741 ALOGV(__FUNCTION__);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800742
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800743 *out_id = unique_id_;
744 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800745}
746
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800747status_t BufferHubProducer::getConsumerUsage(uint64_t* out_usage) const {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800748 ALOGV(__FUNCTION__);
Chia-I Wue2786ea2017-08-07 10:36:08 -0700749
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800750 // same value as returned by querying NATIVE_WINDOW_CONSUMER_USAGE_BITS
751 *out_usage = 0;
752 return NO_ERROR;
Chia-I Wue2786ea2017-08-07 10:36:08 -0700753}
754
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -0800755status_t BufferHubProducer::TakeAsParcelable(ProducerQueueParcelable* out_parcelable) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800756 if (!out_parcelable || out_parcelable->IsValid()) return BAD_VALUE;
Jiwen 'Steve' Cai5afb7402017-11-09 18:50:32 -0800757
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800758 if (connected_api_ != kNoConnectedApi) {
759 ALOGE("BufferHubProducer::TakeAsParcelable: BufferHubProducer has "
760 "connected client. Must disconnect first.");
761 return BAD_VALUE;
762 }
Jiwen 'Steve' Cai5afb7402017-11-09 18:50:32 -0800763
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800764 if (!queue_->is_connected()) {
765 ALOGE("BufferHubProducer::TakeAsParcelable: This BufferHubProducer "
766 "is not connected to bufferhud. Has it been taken out as a "
767 "parcelable?");
768 return BAD_VALUE;
769 }
Jiwen 'Steve' Cai5afb7402017-11-09 18:50:32 -0800770
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800771 auto status = queue_->TakeAsParcelable();
772 if (!status) {
773 ALOGE("BufferHubProducer::TakeAsParcelable: Failed to take out "
774 "ProducuerQueueParcelable from the producer queue, error: %s.",
775 status.GetErrorMessage().c_str());
776 return BAD_VALUE;
777 }
Jiwen 'Steve' Cai5afb7402017-11-09 18:50:32 -0800778
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800779 *out_parcelable = status.take();
780 return NO_ERROR;
Jiwen 'Steve' Cai5afb7402017-11-09 18:50:32 -0800781}
782
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800783status_t BufferHubProducer::AllocateBuffer(uint32_t width, uint32_t height, uint32_t layer_count,
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800784 PixelFormat format, uint64_t usage) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800785 auto status = queue_->AllocateBuffer(width, height, layer_count, uint32_t(format), usage);
786 if (!status) {
787 ALOGE("BufferHubProducer::AllocateBuffer: Failed to allocate buffer: %s",
788 status.GetErrorMessage().c_str());
789 return NO_MEMORY;
790 }
Jiwen 'Steve' Caid6cb17f2017-05-08 16:15:35 -0700791
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800792 size_t slot = status.get();
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800793 auto producer_buffer = queue_->GetBuffer(slot);
Jiwen 'Steve' Caid6cb17f2017-05-08 16:15:35 -0700794
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800795 LOG_ALWAYS_FATAL_IF(producer_buffer == nullptr,
796 "Failed to get the producer buffer at slot: %zu", slot);
Jiwen 'Steve' Caid6cb17f2017-05-08 16:15:35 -0700797
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800798 buffers_[slot].mProducerBuffer = producer_buffer;
Jiwen 'Steve' Caid6cb17f2017-05-08 16:15:35 -0700799
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800800 return NO_ERROR;
Jiwen 'Steve' Caid6cb17f2017-05-08 16:15:35 -0700801}
802
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800803status_t BufferHubProducer::RemoveBuffer(size_t slot) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800804 auto status = queue_->RemoveBuffer(slot);
805 if (!status) {
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700806 ALOGE("BufferHubProducer::RemoveBuffer: Failed to remove buffer at slot: %zu, error: %s.",
807 slot, status.GetErrorMessage().c_str());
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800808 return INVALID_OPERATION;
809 }
Jiwen 'Steve' Caid6cb17f2017-05-08 16:15:35 -0700810
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800811 // Reset in memory objects related the the buffer.
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800812 buffers_[slot].mProducerBuffer = nullptr;
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800813 buffers_[slot].mBufferState.detachProducer();
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700814 buffers_[slot].mFence = Fence::NO_FENCE;
815 buffers_[slot].mGraphicBuffer = nullptr;
816 buffers_[slot].mRequestBufferCalled = false;
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800817 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800818}
819
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800820status_t BufferHubProducer::FreeAllBuffers() {
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -0800821 for (size_t slot = 0; slot < BufferHubQueue::kMaxQueueCapacity; slot++) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800822 // Reset in memory objects related the the buffer.
Jiwen 'Steve' Cai1c730242018-12-31 18:40:02 -0800823 buffers_[slot].mProducerBuffer = nullptr;
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700824 buffers_[slot].mBufferState.reset();
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800825 buffers_[slot].mFence = Fence::NO_FENCE;
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700826 buffers_[slot].mGraphicBuffer = nullptr;
827 buffers_[slot].mRequestBufferCalled = false;
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800828 }
Jiwen 'Steve' Cai005f45d2017-08-04 17:34:37 -0700829
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800830 auto status = queue_->FreeAllBuffers();
831 if (!status) {
832 ALOGE("BufferHubProducer::FreeAllBuffers: Failed to free all buffers on "
833 "the queue: %s",
834 status.GetErrorMessage().c_str());
835 }
Jiwen 'Steve' Cai005f45d2017-08-04 17:34:37 -0700836
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800837 if (queue_->capacity() != 0 || queue_->count() != 0) {
838 LOG_ALWAYS_FATAL("BufferHubProducer::FreeAllBuffers: Not all buffers are freed.");
839 }
Jiwen 'Steve' Cai005f45d2017-08-04 17:34:37 -0700840
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800841 return NO_ERROR;
Jiwen 'Steve' Cai005f45d2017-08-04 17:34:37 -0700842}
843
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -0800844status_t BufferHubProducer::exportToParcel(Parcel* parcel) {
845 status_t res = TakeAsParcelable(&pending_producer_parcelable_);
846 if (res != NO_ERROR) return res;
847
848 if (!pending_producer_parcelable_.IsValid()) {
849 ALOGE("BufferHubProducer::exportToParcel: Invalid parcelable object.");
850 return BAD_VALUE;
851 }
852
853 res = parcel->writeUint32(USE_BUFFER_HUB);
854 if (res != NO_ERROR) {
855 ALOGE("BufferHubProducer::exportToParcel: Cannot write magic, res=%d.", res);
856 return res;
857 }
858
859 return pending_producer_parcelable_.writeToParcel(parcel);
860}
861
862IBinder* BufferHubProducer::onAsBinder() {
863 ALOGE("BufferHubProducer::onAsBinder: BufferHubProducer should never be used as an Binder "
864 "object.");
865 return nullptr;
866}
867
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800868} // namespace android