• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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(&timestamp, &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