1 #include "include/private/dvr/buffer_hub_queue_producer.h"
2
3 #include <dvr/dvr_api.h>
4 #include <inttypes.h>
5 #include <log/log.h>
6
7 namespace android {
8 namespace dvr {
9
10 /* static */
Create()11 sp<BufferHubQueueProducer> BufferHubQueueProducer::Create() {
12 sp<BufferHubQueueProducer> producer = new BufferHubQueueProducer;
13 producer->queue_ = ProducerQueue::Create<DvrNativeBufferMetadata>();
14 return producer;
15 }
16
17 /* static */
Create(const std::shared_ptr<ProducerQueue> & queue)18 sp<BufferHubQueueProducer> BufferHubQueueProducer::Create(
19 const std::shared_ptr<ProducerQueue>& queue) {
20 if (queue->metadata_size() != sizeof(DvrNativeBufferMetadata)) {
21 ALOGE(
22 "BufferHubQueueProducer::Create producer's metadata size is different "
23 "than the size of DvrNativeBufferMetadata");
24 return nullptr;
25 }
26
27 sp<BufferHubQueueProducer> producer = new BufferHubQueueProducer;
28 producer->queue_ = queue;
29 return producer;
30 }
31
requestBuffer(int slot,sp<GraphicBuffer> * buf)32 status_t BufferHubQueueProducer::requestBuffer(int slot,
33 sp<GraphicBuffer>* buf) {
34 ALOGD_IF(TRACE, "requestBuffer: slot=%d", slot);
35
36 std::unique_lock<std::mutex> lock(mutex_);
37
38 if (connected_api_ == kNoConnectedApi) {
39 ALOGE("requestBuffer: BufferHubQueueProducer has no connected producer");
40 return NO_INIT;
41 }
42
43 if (slot < 0 || slot >= max_buffer_count_) {
44 ALOGE("requestBuffer: slot index %d out of range [0, %d)", slot,
45 max_buffer_count_);
46 return BAD_VALUE;
47 } else if (!buffers_[slot].mBufferState.isDequeued()) {
48 ALOGE("requestBuffer: slot %d is not owned by the producer (state = %s)",
49 slot, buffers_[slot].mBufferState.string());
50 return BAD_VALUE;
51 } else if (buffers_[slot].mGraphicBuffer != nullptr) {
52 ALOGE("requestBuffer: slot %d is not empty.", slot);
53 return BAD_VALUE;
54 } else if (buffers_[slot].mBufferProducer == nullptr) {
55 ALOGE("requestBuffer: slot %d is not dequeued.", slot);
56 return BAD_VALUE;
57 }
58
59 const auto& buffer_producer = buffers_[slot].mBufferProducer;
60 sp<GraphicBuffer> graphic_buffer = buffer_producer->buffer()->buffer();
61
62 buffers_[slot].mGraphicBuffer = graphic_buffer;
63 buffers_[slot].mRequestBufferCalled = true;
64
65 *buf = graphic_buffer;
66 return NO_ERROR;
67 }
68
setMaxDequeuedBufferCount(int max_dequeued_buffers)69 status_t BufferHubQueueProducer::setMaxDequeuedBufferCount(
70 int max_dequeued_buffers) {
71 ALOGD_IF(TRACE, "setMaxDequeuedBufferCount: max_dequeued_buffers=%d",
72 max_dequeued_buffers);
73
74 std::unique_lock<std::mutex> lock(mutex_);
75
76 if (max_dequeued_buffers <= 0 ||
77 max_dequeued_buffers >
78 static_cast<int>(BufferHubQueue::kMaxQueueCapacity -
79 kDefaultUndequeuedBuffers)) {
80 ALOGE("setMaxDequeuedBufferCount: %d out of range (0, %zu]",
81 max_dequeued_buffers, BufferHubQueue::kMaxQueueCapacity);
82 return BAD_VALUE;
83 }
84
85 // The new dequeued_buffers count should not be violated by the number
86 // of currently dequeued buffers.
87 int dequeued_count = 0;
88 for (const auto& buf : buffers_) {
89 if (buf.mBufferState.isDequeued()) {
90 dequeued_count++;
91 }
92 }
93 if (dequeued_count > max_dequeued_buffers) {
94 ALOGE(
95 "setMaxDequeuedBufferCount: the requested dequeued_buffers"
96 "count (%d) exceeds the current dequeued buffer count (%d)",
97 max_dequeued_buffers, dequeued_count);
98 return BAD_VALUE;
99 }
100
101 max_dequeued_buffer_count_ = max_dequeued_buffers;
102 return NO_ERROR;
103 }
104
setAsyncMode(bool async)105 status_t BufferHubQueueProducer::setAsyncMode(bool async) {
106 if (async) {
107 // TODO(b/36724099) BufferHubQueue's consumer end always acquires the buffer
108 // automatically and behaves differently from IGraphicBufferConsumer. Thus,
109 // android::BufferQueue's async mode (a.k.a. allocating an additional buffer
110 // to prevent dequeueBuffer from being blocking) technically does not apply
111 // here.
112 //
113 // In Daydream, non-blocking producer side dequeue is guaranteed by careful
114 // buffer consumer implementations. In another word, BufferHubQueue based
115 // dequeueBuffer should never block whether setAsyncMode(true) is set or
116 // not.
117 //
118 // See: IGraphicBufferProducer::setAsyncMode and
119 // BufferQueueProducer::setAsyncMode for more about original implementation.
120 ALOGW(
121 "BufferHubQueueProducer::setAsyncMode: BufferHubQueue should always be "
122 "asynchronous. This call makes no effact.");
123 return NO_ERROR;
124 }
125 return NO_ERROR;
126 }
127
dequeueBuffer(int * out_slot,sp<Fence> * out_fence,uint32_t width,uint32_t height,PixelFormat format,uint32_t usage,FrameEventHistoryDelta *)128 status_t BufferHubQueueProducer::dequeueBuffer(
129 int* out_slot, sp<Fence>* out_fence, uint32_t width, uint32_t height,
130 PixelFormat format, uint32_t usage,
131 FrameEventHistoryDelta* /* out_timestamps */) {
132 ALOGD_IF(TRACE, "dequeueBuffer: w=%u, h=%u, format=%d, usage=%u", width,
133 height, format, usage);
134
135 status_t ret;
136 std::unique_lock<std::mutex> lock(mutex_);
137
138 if (connected_api_ == kNoConnectedApi) {
139 ALOGE("dequeueBuffer: BufferQueue has no connected producer");
140 return NO_INIT;
141 }
142
143 const uint32_t kLayerCount = 1;
144 if (static_cast<int32_t>(queue_->capacity()) <
145 max_dequeued_buffer_count_ + kDefaultUndequeuedBuffers) {
146 // Lazy allocation. When the capacity of |queue_| has not reached
147 // |max_dequeued_buffer_count_|, allocate new buffer.
148 // TODO(jwcai) To save memory, the really reasonable thing to do is to go
149 // over existing slots and find first existing one to dequeue.
150 ret = AllocateBuffer(width, height, kLayerCount, format, usage);
151 if (ret < 0)
152 return ret;
153 }
154
155 size_t slot;
156 std::shared_ptr<BufferProducer> buffer_producer;
157
158 for (size_t retry = 0; retry < BufferHubQueue::kMaxQueueCapacity; retry++) {
159 LocalHandle fence;
160 auto buffer_status = queue_->Dequeue(dequeue_timeout_ms_, &slot, &fence);
161
162 buffer_producer = buffer_status.take();
163 if (!buffer_producer)
164 return NO_MEMORY;
165
166 if (width == buffer_producer->width() &&
167 height == buffer_producer->height() &&
168 static_cast<uint32_t>(format) == buffer_producer->format()) {
169 // The producer queue returns a buffer producer matches the request.
170 break;
171 }
172
173 // Needs reallocation.
174 // TODO(jwcai) Consider use VLOG instead if we find this log is not useful.
175 ALOGI(
176 "dequeueBuffer: requested buffer (w=%u, h=%u, format=%u) is different "
177 "from the buffer returned at slot: %zu (w=%u, h=%u, format=%u). Need "
178 "re-allocattion.",
179 width, height, format, slot, buffer_producer->width(),
180 buffer_producer->height(), buffer_producer->format());
181 // Mark the slot as reallocating, so that later we can set
182 // BUFFER_NEEDS_REALLOCATION when the buffer actually get dequeued.
183 buffers_[slot].mIsReallocating = true;
184
185 // Remove the old buffer once the allocation before allocating its
186 // replacement.
187 RemoveBuffer(slot);
188
189 // Allocate a new producer buffer with new buffer configs. Note that if
190 // there are already multiple buffers in the queue, the next one returned
191 // from |queue_->Dequeue| may not be the new buffer we just reallocated.
192 // Retry up to BufferHubQueue::kMaxQueueCapacity times.
193 ret = AllocateBuffer(width, height, kLayerCount, format, usage);
194 if (ret < 0)
195 return ret;
196 }
197
198 // With the BufferHub backed solution. Buffer slot returned from
199 // |queue_->Dequeue| is guaranteed to avaiable for producer's use.
200 // It's either in free state (if the buffer has never been used before) or
201 // in queued state (if the buffer has been dequeued and queued back to
202 // BufferHubQueue).
203 // TODO(jwcai) Clean this up, make mBufferState compatible with BufferHub's
204 // model.
205 LOG_ALWAYS_FATAL_IF((!buffers_[slot].mBufferState.isFree() &&
206 !buffers_[slot].mBufferState.isQueued()),
207 "dequeueBuffer: slot %zu is not free or queued.", slot);
208
209 buffers_[slot].mBufferState.freeQueued();
210 buffers_[slot].mBufferState.dequeue();
211 ALOGD_IF(TRACE, "dequeueBuffer: slot=%zu", slot);
212
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 = 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;
225 }
226
detachBuffer(int)227 status_t BufferHubQueueProducer::detachBuffer(int /* slot */) {
228 ALOGE("BufferHubQueueProducer::detachBuffer not implemented.");
229 return INVALID_OPERATION;
230 }
231
detachNextBuffer(sp<GraphicBuffer> *,sp<Fence> *)232 status_t BufferHubQueueProducer::detachNextBuffer(
233 sp<GraphicBuffer>* /* out_buffer */, sp<Fence>* /* out_fence */) {
234 ALOGE("BufferHubQueueProducer::detachNextBuffer not implemented.");
235 return INVALID_OPERATION;
236 }
237
attachBuffer(int *,const sp<GraphicBuffer> &)238 status_t BufferHubQueueProducer::attachBuffer(
239 int* /* out_slot */, const sp<GraphicBuffer>& /* buffer */) {
240 // With this BufferHub backed implementation, we assume (for now) all buffers
241 // are allocated and owned by the BufferHub. Thus the attempt of transfering
242 // ownership of a buffer to the buffer queue is intentionally unsupported.
243 LOG_ALWAYS_FATAL("BufferHubQueueProducer::attachBuffer not supported.");
244 return INVALID_OPERATION;
245 }
246
queueBuffer(int slot,const QueueBufferInput & input,QueueBufferOutput * output)247 status_t BufferHubQueueProducer::queueBuffer(int slot,
248 const QueueBufferInput& input,
249 QueueBufferOutput* output) {
250 ALOGD_IF(TRACE, "queueBuffer: slot %d", slot);
251
252 if (output == nullptr) {
253 return BAD_VALUE;
254 }
255
256 int64_t timestamp;
257 bool is_auto_timestamp;
258 android_dataspace dataspace;
259 Rect crop(Rect::EMPTY_RECT);
260 int scaling_mode;
261 uint32_t transform;
262 sp<Fence> fence;
263
264 input.deflate(×tamp, &is_auto_timestamp, &dataspace, &crop,
265 &scaling_mode, &transform, &fence);
266
267 // Check input scaling mode is valid.
268 switch (scaling_mode) {
269 case NATIVE_WINDOW_SCALING_MODE_FREEZE:
270 case NATIVE_WINDOW_SCALING_MODE_SCALE_TO_WINDOW:
271 case NATIVE_WINDOW_SCALING_MODE_SCALE_CROP:
272 case NATIVE_WINDOW_SCALING_MODE_NO_SCALE_CROP:
273 break;
274 default:
275 ALOGE("queueBuffer: unknown scaling mode %d", scaling_mode);
276 return BAD_VALUE;
277 }
278
279 // Check input fence is valid.
280 if (fence == nullptr) {
281 ALOGE("queueBuffer: fence is NULL");
282 return BAD_VALUE;
283 }
284
285 status_t ret;
286 std::unique_lock<std::mutex> lock(mutex_);
287
288 if (connected_api_ == kNoConnectedApi) {
289 ALOGE("queueBuffer: BufferQueue has no connected producer");
290 return NO_INIT;
291 }
292
293 if (slot < 0 || slot >= max_buffer_count_) {
294 ALOGE("queueBuffer: slot index %d out of range [0, %d)", slot,
295 max_buffer_count_);
296 return BAD_VALUE;
297 } else if (!buffers_[slot].mBufferState.isDequeued()) {
298 ALOGE("queueBuffer: slot %d is not owned by the producer (state = %s)",
299 slot, buffers_[slot].mBufferState.string());
300 return BAD_VALUE;
301 } else if ((!buffers_[slot].mRequestBufferCalled ||
302 buffers_[slot].mGraphicBuffer == nullptr)) {
303 ALOGE(
304 "queueBuffer: slot %d is not requested (mRequestBufferCalled=%d, "
305 "mGraphicBuffer=%p)",
306 slot, buffers_[slot].mRequestBufferCalled,
307 buffers_[slot].mGraphicBuffer.get());
308 return BAD_VALUE;
309 }
310
311 // Post the buffer producer with timestamp in the metadata.
312 const auto& buffer_producer = buffers_[slot].mBufferProducer;
313
314 // Check input crop is not out of boundary of current buffer.
315 Rect buffer_rect(buffer_producer->width(), buffer_producer->height());
316 Rect cropped_rect(Rect::EMPTY_RECT);
317 crop.intersect(buffer_rect, &cropped_rect);
318 if (cropped_rect != crop) {
319 ALOGE("queueBuffer: slot %d has out-of-boundary crop.", slot);
320 return BAD_VALUE;
321 }
322
323 LocalHandle fence_fd(fence->isValid() ? fence->dup() : -1);
324
325 DvrNativeBufferMetadata meta_data = {};
326 meta_data.timestamp = timestamp;
327 meta_data.is_auto_timestamp = static_cast<int32_t>(is_auto_timestamp);
328 meta_data.dataspace = static_cast<int32_t>(dataspace);
329 meta_data.crop_left = crop.left;
330 meta_data.crop_top = crop.top;
331 meta_data.crop_right = crop.right;
332 meta_data.crop_bottom = crop.bottom;
333 meta_data.scaling_mode = static_cast<int32_t>(scaling_mode);
334 meta_data.transform = static_cast<int32_t>(transform);
335
336 buffer_producer->Post(fence_fd, &meta_data, sizeof(meta_data));
337 buffers_[slot].mBufferState.queue();
338
339 output->width = buffer_producer->width();
340 output->height = buffer_producer->height();
341 output->transformHint = 0; // default value, we don't use it yet.
342
343 // |numPendingBuffers| counts of the number of buffers that has been enqueued
344 // by the producer but not yet acquired by the consumer. Due to the nature
345 // of BufferHubQueue design, this is hard to trace from the producer's client
346 // side, but it's safe to assume it's zero.
347 output->numPendingBuffers = 0;
348
349 // Note that we are not setting nextFrameNumber here as it seems to be only
350 // used by surface flinger. See more at b/22802885, ag/791760.
351 output->nextFrameNumber = 0;
352
353 return NO_ERROR;
354 }
355
cancelBuffer(int slot,const sp<Fence> & fence)356 status_t BufferHubQueueProducer::cancelBuffer(int slot,
357 const sp<Fence>& fence) {
358 ALOGD_IF(TRACE, __FUNCTION__);
359
360 std::unique_lock<std::mutex> lock(mutex_);
361
362 if (connected_api_ == kNoConnectedApi) {
363 ALOGE("cancelBuffer: BufferQueue has no connected producer");
364 return NO_INIT;
365 }
366
367 if (slot < 0 || slot >= max_buffer_count_) {
368 ALOGE("cancelBuffer: slot index %d out of range [0, %d)", slot,
369 max_buffer_count_);
370 return BAD_VALUE;
371 } else if (!buffers_[slot].mBufferState.isDequeued()) {
372 ALOGE("cancelBuffer: slot %d is not owned by the producer (state = %s)",
373 slot, buffers_[slot].mBufferState.string());
374 return BAD_VALUE;
375 } else if (fence == nullptr) {
376 ALOGE("cancelBuffer: fence is NULL");
377 return BAD_VALUE;
378 }
379
380 auto buffer_producer = buffers_[slot].mBufferProducer;
381 queue_->Enqueue(buffer_producer, slot);
382 buffers_[slot].mBufferState.cancel();
383 buffers_[slot].mFence = fence;
384 ALOGD_IF(TRACE, "cancelBuffer: slot %d", slot);
385
386 return NO_ERROR;
387 }
388
query(int what,int * out_value)389 status_t BufferHubQueueProducer::query(int what, int* out_value) {
390 ALOGD_IF(TRACE, __FUNCTION__);
391
392 std::unique_lock<std::mutex> lock(mutex_);
393
394 if (out_value == nullptr) {
395 ALOGE("query: out_value was NULL");
396 return BAD_VALUE;
397 }
398
399 int value = 0;
400 switch (what) {
401 case NATIVE_WINDOW_MIN_UNDEQUEUED_BUFFERS:
402 // TODO(b/36187402) This should be the maximum number of buffers that this
403 // producer queue's consumer can acquire. Set to be at least one. Need to
404 // find a way to set from the consumer side.
405 value = kDefaultUndequeuedBuffers;
406 break;
407 case NATIVE_WINDOW_BUFFER_AGE:
408 value = 0;
409 break;
410 case NATIVE_WINDOW_WIDTH:
411 value = queue_->default_width();
412 break;
413 case NATIVE_WINDOW_HEIGHT:
414 value = queue_->default_height();
415 break;
416 case NATIVE_WINDOW_FORMAT:
417 value = queue_->default_format();
418 break;
419 case NATIVE_WINDOW_CONSUMER_RUNNING_BEHIND:
420 // BufferHubQueue is always operating in async mode, thus semantically
421 // consumer can never be running behind. See BufferQueueCore.cpp core
422 // for more information about the original meaning of this flag.
423 value = 0;
424 break;
425 case NATIVE_WINDOW_CONSUMER_USAGE_BITS:
426 // TODO(jwcai) This is currently not implement as we don't need
427 // IGraphicBufferConsumer parity.
428 value = 0;
429 break;
430 case NATIVE_WINDOW_DEFAULT_DATASPACE:
431 // TODO(jwcai) Return the default value android::BufferQueue is using as
432 // there is no way dvr::ConsumerQueue can set it.
433 value = 0; // HAL_DATASPACE_UNKNOWN
434 break;
435 case NATIVE_WINDOW_STICKY_TRANSFORM:
436 // TODO(jwcai) Return the default value android::BufferQueue is using as
437 // there is no way dvr::ConsumerQueue can set it.
438 value = 0;
439 break;
440 case NATIVE_WINDOW_CONSUMER_IS_PROTECTED:
441 // In Daydream's implementation, the consumer end (i.e. VR Compostior)
442 // knows how to handle protected buffers.
443 value = 1;
444 break;
445 default:
446 return BAD_VALUE;
447 }
448
449 ALOGD_IF(TRACE, "query: key=%d, v=%d", what, value);
450 *out_value = value;
451 return NO_ERROR;
452 }
453
connect(const sp<IProducerListener> &,int api,bool,QueueBufferOutput * output)454 status_t BufferHubQueueProducer::connect(
455 const sp<IProducerListener>& /* listener */, int api,
456 bool /* producer_controlled_by_app */, QueueBufferOutput* output) {
457 // Consumer interaction are actually handled by buffer hub, and we need
458 // to maintain consumer operations here. We only need to perform basic input
459 // parameter checks here.
460 ALOGD_IF(TRACE, __FUNCTION__);
461
462 if (output == nullptr) {
463 return BAD_VALUE;
464 }
465
466 std::unique_lock<std::mutex> lock(mutex_);
467
468 if (connected_api_ != kNoConnectedApi) {
469 return BAD_VALUE;
470 }
471
472 switch (api) {
473 case NATIVE_WINDOW_API_EGL:
474 case NATIVE_WINDOW_API_CPU:
475 case NATIVE_WINDOW_API_MEDIA:
476 case NATIVE_WINDOW_API_CAMERA:
477 connected_api_ = api;
478
479 output->width = queue_->default_width();
480 output->height = queue_->default_height();
481
482 // default values, we don't use them yet.
483 output->transformHint = 0;
484 output->numPendingBuffers = 0;
485 output->nextFrameNumber = 0;
486 output->bufferReplaced = false;
487
488 break;
489 default:
490 ALOGE("BufferHubQueueProducer::connect: unknow API %d", api);
491 return BAD_VALUE;
492 }
493
494 return NO_ERROR;
495 }
496
disconnect(int api,DisconnectMode)497 status_t BufferHubQueueProducer::disconnect(int api, DisconnectMode /*mode*/) {
498 // Consumer interaction are actually handled by buffer hub, and we need
499 // to maintain consumer operations here. We only need to perform basic input
500 // parameter checks here.
501 ALOGD_IF(TRACE, __FUNCTION__);
502
503 std::unique_lock<std::mutex> lock(mutex_);
504
505 if (kNoConnectedApi == connected_api_) {
506 return NO_INIT;
507 } else if (api != connected_api_) {
508 return BAD_VALUE;
509 }
510
511 connected_api_ = kNoConnectedApi;
512 return NO_ERROR;
513 }
514
setSidebandStream(const sp<NativeHandle> & stream)515 status_t BufferHubQueueProducer::setSidebandStream(
516 const sp<NativeHandle>& stream) {
517 if (stream != nullptr) {
518 // TODO(jwcai) Investigate how is is used, maybe use BufferHubBuffer's
519 // metadata.
520 ALOGE("SidebandStream is not currently supported.");
521 return INVALID_OPERATION;
522 }
523 return NO_ERROR;
524 }
525
allocateBuffers(uint32_t,uint32_t,PixelFormat,uint32_t)526 void BufferHubQueueProducer::allocateBuffers(uint32_t /* width */,
527 uint32_t /* height */,
528 PixelFormat /* format */,
529 uint32_t /* usage */) {
530 // TODO(jwcai) |allocateBuffers| aims to preallocate up to the maximum number
531 // of buffers permitted by the current BufferQueue configuration (aka
532 // |max_buffer_count_|).
533 ALOGE("BufferHubQueueProducer::allocateBuffers not implemented.");
534 }
535
allowAllocation(bool)536 status_t BufferHubQueueProducer::allowAllocation(bool /* allow */) {
537 ALOGE("BufferHubQueueProducer::allowAllocation not implemented.");
538 return INVALID_OPERATION;
539 }
540
setGenerationNumber(uint32_t generation_number)541 status_t BufferHubQueueProducer::setGenerationNumber(
542 uint32_t generation_number) {
543 ALOGD_IF(TRACE, __FUNCTION__);
544
545 std::unique_lock<std::mutex> lock(mutex_);
546 generation_number_ = generation_number;
547 return NO_ERROR;
548 }
549
getConsumerName() const550 String8 BufferHubQueueProducer::getConsumerName() const {
551 // BufferHub based implementation could have one to many producer/consumer
552 // relationship, thus |getConsumerName| from the producer side does not
553 // make any sense.
554 ALOGE("BufferHubQueueProducer::getConsumerName not supported.");
555 return String8("BufferHubQueue::DummyConsumer");
556 }
557
setSharedBufferMode(bool shared_buffer_mode)558 status_t BufferHubQueueProducer::setSharedBufferMode(bool shared_buffer_mode) {
559 if (shared_buffer_mode) {
560 ALOGE(
561 "BufferHubQueueProducer::setSharedBufferMode(true) is not supported.");
562 // TODO(b/36373181) Front buffer mode for buffer hub queue as ANativeWindow.
563 return INVALID_OPERATION;
564 }
565 // Setting to default should just work as a no-op.
566 return NO_ERROR;
567 }
568
setAutoRefresh(bool auto_refresh)569 status_t BufferHubQueueProducer::setAutoRefresh(bool auto_refresh) {
570 if (auto_refresh) {
571 ALOGE("BufferHubQueueProducer::setAutoRefresh(true) is not supported.");
572 return INVALID_OPERATION;
573 }
574 // Setting to default should just work as a no-op.
575 return NO_ERROR;
576 }
577
setDequeueTimeout(nsecs_t timeout)578 status_t BufferHubQueueProducer::setDequeueTimeout(nsecs_t timeout) {
579 ALOGD_IF(TRACE, __FUNCTION__);
580
581 std::unique_lock<std::mutex> lock(mutex_);
582 dequeue_timeout_ms_ = static_cast<int>(timeout / (1000 * 1000));
583 return NO_ERROR;
584 }
585
getLastQueuedBuffer(sp<GraphicBuffer> *,sp<Fence> *,float[16])586 status_t BufferHubQueueProducer::getLastQueuedBuffer(
587 sp<GraphicBuffer>* /* out_buffer */, sp<Fence>* /* out_fence */,
588 float /*out_transform_matrix*/[16]) {
589 ALOGE("BufferHubQueueProducer::getLastQueuedBuffer not implemented.");
590 return INVALID_OPERATION;
591 }
592
getFrameTimestamps(FrameEventHistoryDelta *)593 void BufferHubQueueProducer::getFrameTimestamps(
594 FrameEventHistoryDelta* /*outDelta*/) {
595 ALOGE("BufferHubQueueProducer::getFrameTimestamps not implemented.");
596 }
597
getUniqueId(uint64_t * out_id) const598 status_t BufferHubQueueProducer::getUniqueId(uint64_t* out_id) const {
599 ALOGD_IF(TRACE, __FUNCTION__);
600
601 *out_id = unique_id_;
602 return NO_ERROR;
603 }
604
AllocateBuffer(uint32_t width,uint32_t height,uint32_t layer_count,PixelFormat format,uint64_t usage)605 status_t BufferHubQueueProducer::AllocateBuffer(uint32_t width, uint32_t height,
606 uint32_t layer_count,
607 PixelFormat format,
608 uint64_t usage) {
609 size_t slot;
610
611 if (queue_->AllocateBuffer(width, height, layer_count, format, usage, &slot) <
612 0) {
613 ALOGE("Failed to allocate new buffer in BufferHub.");
614 return NO_MEMORY;
615 }
616
617 auto buffer_producer = queue_->GetBuffer(slot);
618
619 LOG_ALWAYS_FATAL_IF(buffer_producer == nullptr,
620 "Failed to get buffer producer at slot: %zu", slot);
621
622 buffers_[slot].mBufferProducer = buffer_producer;
623
624 return NO_ERROR;
625 }
626
RemoveBuffer(size_t slot)627 status_t BufferHubQueueProducer::RemoveBuffer(size_t slot) {
628 int ret = queue_->DetachBuffer(slot);
629 if (ret < 0) {
630 ALOGE("BufferHubQueueProducer::RemoveBuffer failed through RPC, ret=%s",
631 strerror(-ret));
632 return ret;
633 }
634
635 // Reset in memory objects related the the buffer.
636 buffers_[slot].mBufferProducer = nullptr;
637 buffers_[slot].mGraphicBuffer = nullptr;
638 buffers_[slot].mBufferState.detachProducer();
639 return NO_ERROR;
640 }
641
642 } // namespace dvr
643 } // namespace android
644