1 /*
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
17 #include <dvr/dvr_api.h>
18 #include <gui/BufferHubProducer.h>
19 #include <inttypes.h>
20 #include <log/log.h>
21 #include <system/window.h>
22
23 namespace android {
24
25 using namespace dvr;
26
27 /* static */
Create(const std::shared_ptr<ProducerQueue> & queue)28 sp<BufferHubProducer> BufferHubProducer::Create(const std::shared_ptr<ProducerQueue>& queue) {
29 sp<BufferHubProducer> producer = new BufferHubProducer;
30 producer->queue_ = queue;
31 return producer;
32 }
33
34 /* static */
Create(ProducerQueueParcelable parcelable)35 sp<BufferHubProducer> BufferHubProducer::Create(ProducerQueueParcelable parcelable) {
36 if (!parcelable.IsValid()) {
37 ALOGE("BufferHubProducer::Create: Invalid producer parcelable.");
38 return nullptr;
39 }
40
41 sp<BufferHubProducer> producer = new BufferHubProducer;
42 producer->queue_ = ProducerQueue::Import(parcelable.TakeChannelHandle());
43 return producer;
44 }
45
requestBuffer(int slot,sp<GraphicBuffer> * buf)46 status_t BufferHubProducer::requestBuffer(int slot, sp<GraphicBuffer>* buf) {
47 ALOGV("requestBuffer: slot=%d", slot);
48
49 std::unique_lock<std::mutex> lock(mutex_);
50
51 if (connected_api_ == kNoConnectedApi) {
52 ALOGE("requestBuffer: BufferHubProducer has no connected producer");
53 return NO_INIT;
54 }
55
56 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;
66 } else if (buffers_[slot].mBufferProducer == nullptr) {
67 ALOGE("requestBuffer: slot %d is not dequeued.", slot);
68 return BAD_VALUE;
69 }
70
71 const auto& buffer_producer = buffers_[slot].mBufferProducer;
72 sp<GraphicBuffer> graphic_buffer = buffer_producer->buffer()->buffer();
73
74 buffers_[slot].mGraphicBuffer = graphic_buffer;
75 buffers_[slot].mRequestBufferCalled = true;
76
77 *buf = graphic_buffer;
78 return NO_ERROR;
79 }
80
setMaxDequeuedBufferCount(int max_dequeued_buffers)81 status_t BufferHubProducer::setMaxDequeuedBufferCount(int max_dequeued_buffers) {
82 ALOGV("setMaxDequeuedBufferCount: max_dequeued_buffers=%d", max_dequeued_buffers);
83
84 std::unique_lock<std::mutex> lock(mutex_);
85
86 if (max_dequeued_buffers <= 0 ||
87 max_dequeued_buffers >
88 int(BufferHubQueue::kMaxQueueCapacity - kDefaultUndequeuedBuffers)) {
89 ALOGE("setMaxDequeuedBufferCount: %d out of range (0, %zu]", max_dequeued_buffers,
90 BufferHubQueue::kMaxQueueCapacity);
91 return BAD_VALUE;
92 }
93
94 // 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;
111 }
112
setAsyncMode(bool async)113 status_t BufferHubProducer::setAsyncMode(bool async) {
114 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 }
132 return NO_ERROR;
133 }
134
dequeueBuffer(int * out_slot,sp<Fence> * out_fence,uint32_t width,uint32_t height,PixelFormat format,uint64_t usage,uint64_t *,FrameEventHistoryDelta *)135 status_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 */) {
139 ALOGV("dequeueBuffer: w=%u, h=%u, format=%d, usage=%" PRIu64, width, height, format, usage);
140
141 status_t ret;
142 std::unique_lock<std::mutex> lock(mutex_);
143
144 if (connected_api_ == kNoConnectedApi) {
145 ALOGE("dequeueBuffer: BufferQueue has no connected producer");
146 return NO_INIT;
147 }
148
149 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 }
158
159 size_t slot = 0;
160 std::shared_ptr<BufferProducer> buffer_producer;
161
162 for (size_t retry = 0; retry < BufferHubQueue::kMaxQueueCapacity; retry++) {
163 LocalHandle fence;
164 auto buffer_status = queue_->Dequeue(dequeue_timeout_ms_, &slot, &fence);
165 if (!buffer_status) return NO_MEMORY;
166
167 buffer_producer = buffer_status.take();
168 if (!buffer_producer) return NO_MEMORY;
169
170 if (width == buffer_producer->width() && height == buffer_producer->height() &&
171 uint32_t(format) == buffer_producer->format()) {
172 // The producer queue returns a buffer producer matches the request.
173 break;
174 }
175
176 // 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.",
181 width, height, format, slot, buffer_producer->width(), buffer_producer->height(),
182 buffer_producer->format());
183 // 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;
186
187 // Remove the old buffer once the allocation before allocating its
188 // replacement.
189 RemoveBuffer(slot);
190
191 // 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();
211 ALOGV("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 = 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;
225 }
226
detachBuffer(int)227 status_t BufferHubProducer::detachBuffer(int /* slot */) {
228 ALOGE("BufferHubProducer::detachBuffer not implemented.");
229 return INVALID_OPERATION;
230 }
231
detachNextBuffer(sp<GraphicBuffer> *,sp<Fence> *)232 status_t BufferHubProducer::detachNextBuffer(sp<GraphicBuffer>* /* out_buffer */,
233 sp<Fence>* /* out_fence */) {
234 ALOGE("BufferHubProducer::detachNextBuffer not implemented.");
235 return INVALID_OPERATION;
236 }
237
attachBuffer(int *,const sp<GraphicBuffer> &)238 status_t BufferHubProducer::attachBuffer(int* /* out_slot */,
239 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("BufferHubProducer::attachBuffer not supported.");
244 return INVALID_OPERATION;
245 }
246
queueBuffer(int slot,const QueueBufferInput & input,QueueBufferOutput * output)247 status_t BufferHubProducer::queueBuffer(int slot, const QueueBufferInput& input,
248 QueueBufferOutput* output) {
249 ALOGV("queueBuffer: slot %d", slot);
250
251 if (output == nullptr) {
252 return BAD_VALUE;
253 }
254
255 int64_t timestamp;
256 bool is_auto_timestamp;
257 android_dataspace dataspace;
258 Rect crop(Rect::EMPTY_RECT);
259 int scaling_mode;
260 uint32_t transform;
261 sp<Fence> fence;
262
263 input.deflate(×tamp, &is_auto_timestamp, &dataspace, &crop, &scaling_mode, &transform,
264 &fence);
265
266 // Check input scaling mode is valid.
267 switch (scaling_mode) {
268 case NATIVE_WINDOW_SCALING_MODE_FREEZE:
269 case NATIVE_WINDOW_SCALING_MODE_SCALE_TO_WINDOW:
270 case NATIVE_WINDOW_SCALING_MODE_SCALE_CROP:
271 case NATIVE_WINDOW_SCALING_MODE_NO_SCALE_CROP:
272 break;
273 default:
274 ALOGE("queueBuffer: unknown scaling mode %d", scaling_mode);
275 return BAD_VALUE;
276 }
277
278 // Check input fence is valid.
279 if (fence == nullptr) {
280 ALOGE("queueBuffer: fence is NULL");
281 return BAD_VALUE;
282 }
283
284 std::unique_lock<std::mutex> lock(mutex_);
285
286 if (connected_api_ == kNoConnectedApi) {
287 ALOGE("queueBuffer: BufferQueue has no connected producer");
288 return NO_INIT;
289 }
290
291 if (slot < 0 || slot >= max_buffer_count_) {
292 ALOGE("queueBuffer: slot index %d out of range [0, %d)", slot, max_buffer_count_);
293 return BAD_VALUE;
294 } else if (!buffers_[slot].mBufferState.isDequeued()) {
295 ALOGE("queueBuffer: slot %d is not owned by the producer (state = %s)", slot,
296 buffers_[slot].mBufferState.string());
297 return BAD_VALUE;
298 } else if ((!buffers_[slot].mRequestBufferCalled || buffers_[slot].mGraphicBuffer == nullptr)) {
299 ALOGE("queueBuffer: slot %d is not requested (mRequestBufferCalled=%d, "
300 "mGraphicBuffer=%p)",
301 slot, buffers_[slot].mRequestBufferCalled, buffers_[slot].mGraphicBuffer.get());
302 return BAD_VALUE;
303 }
304
305 // Post the buffer producer with timestamp in the metadata.
306 const auto& buffer_producer = buffers_[slot].mBufferProducer;
307
308 // Check input crop is not out of boundary of current buffer.
309 Rect buffer_rect(buffer_producer->width(), buffer_producer->height());
310 Rect cropped_rect(Rect::EMPTY_RECT);
311 crop.intersect(buffer_rect, &cropped_rect);
312 if (cropped_rect != crop) {
313 ALOGE("queueBuffer: slot %d has out-of-boundary crop.", slot);
314 return BAD_VALUE;
315 }
316
317 LocalHandle fence_fd(fence->isValid() ? fence->dup() : -1);
318
319 DvrNativeBufferMetadata meta_data;
320 meta_data.timestamp = timestamp;
321 meta_data.is_auto_timestamp = int32_t(is_auto_timestamp);
322 meta_data.dataspace = int32_t(dataspace);
323 meta_data.crop_left = crop.left;
324 meta_data.crop_top = crop.top;
325 meta_data.crop_right = crop.right;
326 meta_data.crop_bottom = crop.bottom;
327 meta_data.scaling_mode = int32_t(scaling_mode);
328 meta_data.transform = int32_t(transform);
329
330 buffer_producer->PostAsync(&meta_data, fence_fd);
331 buffers_[slot].mBufferState.queue();
332
333 output->width = buffer_producer->width();
334 output->height = buffer_producer->height();
335 output->transformHint = 0; // default value, we don't use it yet.
336
337 // |numPendingBuffers| counts of the number of buffers that has been enqueued
338 // by the producer but not yet acquired by the consumer. Due to the nature
339 // of BufferHubQueue design, this is hard to trace from the producer's client
340 // side, but it's safe to assume it's zero.
341 output->numPendingBuffers = 0;
342
343 // Note that we are not setting nextFrameNumber here as it seems to be only
344 // used by surface flinger. See more at b/22802885, ag/791760.
345 output->nextFrameNumber = 0;
346
347 return NO_ERROR;
348 }
349
cancelBuffer(int slot,const sp<Fence> & fence)350 status_t BufferHubProducer::cancelBuffer(int slot, const sp<Fence>& fence) {
351 ALOGV(__FUNCTION__);
352
353 std::unique_lock<std::mutex> lock(mutex_);
354
355 if (connected_api_ == kNoConnectedApi) {
356 ALOGE("cancelBuffer: BufferQueue has no connected producer");
357 return NO_INIT;
358 }
359
360 if (slot < 0 || slot >= max_buffer_count_) {
361 ALOGE("cancelBuffer: slot index %d out of range [0, %d)", slot, max_buffer_count_);
362 return BAD_VALUE;
363 } else if (!buffers_[slot].mBufferState.isDequeued()) {
364 ALOGE("cancelBuffer: slot %d is not owned by the producer (state = %s)", slot,
365 buffers_[slot].mBufferState.string());
366 return BAD_VALUE;
367 } else if (fence == nullptr) {
368 ALOGE("cancelBuffer: fence is NULL");
369 return BAD_VALUE;
370 }
371
372 auto buffer_producer = buffers_[slot].mBufferProducer;
373 queue_->Enqueue(buffer_producer, size_t(slot), 0ULL);
374 buffers_[slot].mBufferState.cancel();
375 buffers_[slot].mFence = fence;
376 ALOGV("cancelBuffer: slot %d", slot);
377
378 return NO_ERROR;
379 }
380
query(int what,int * out_value)381 status_t BufferHubProducer::query(int what, int* out_value) {
382 ALOGV(__FUNCTION__);
383
384 std::unique_lock<std::mutex> lock(mutex_);
385
386 if (out_value == nullptr) {
387 ALOGE("query: out_value was NULL");
388 return BAD_VALUE;
389 }
390
391 int value = 0;
392 switch (what) {
393 case NATIVE_WINDOW_MIN_UNDEQUEUED_BUFFERS:
394 // TODO(b/36187402) This should be the maximum number of buffers that this
395 // producer queue's consumer can acquire. Set to be at least one. Need to
396 // find a way to set from the consumer side.
397 value = kDefaultUndequeuedBuffers;
398 break;
399 case NATIVE_WINDOW_BUFFER_AGE:
400 value = 0;
401 break;
402 case NATIVE_WINDOW_WIDTH:
403 value = int32_t(queue_->default_width());
404 break;
405 case NATIVE_WINDOW_HEIGHT:
406 value = int32_t(queue_->default_height());
407 break;
408 case NATIVE_WINDOW_FORMAT:
409 value = int32_t(queue_->default_format());
410 break;
411 case NATIVE_WINDOW_CONSUMER_RUNNING_BEHIND:
412 // BufferHubQueue is always operating in async mode, thus semantically
413 // consumer can never be running behind. See BufferQueueCore.cpp core
414 // for more information about the original meaning of this flag.
415 value = 0;
416 break;
417 case NATIVE_WINDOW_CONSUMER_USAGE_BITS:
418 // TODO(jwcai) This is currently not implement as we don't need
419 // IGraphicBufferConsumer parity.
420 value = 0;
421 break;
422 case NATIVE_WINDOW_DEFAULT_DATASPACE:
423 // TODO(jwcai) Return the default value android::BufferQueue is using as
424 // there is no way dvr::ConsumerQueue can set it.
425 value = 0; // HAL_DATASPACE_UNKNOWN
426 break;
427 case NATIVE_WINDOW_STICKY_TRANSFORM:
428 // TODO(jwcai) Return the default value android::BufferQueue is using as
429 // there is no way dvr::ConsumerQueue can set it.
430 value = 0;
431 break;
432 case NATIVE_WINDOW_CONSUMER_IS_PROTECTED:
433 // In Daydream's implementation, the consumer end (i.e. VR Compostior)
434 // knows how to handle protected buffers.
435 value = 1;
436 break;
437 default:
438 return BAD_VALUE;
439 }
440
441 ALOGV("query: key=%d, v=%d", what, value);
442 *out_value = value;
443 return NO_ERROR;
444 }
445
connect(const sp<IProducerListener> &,int api,bool,QueueBufferOutput * output)446 status_t BufferHubProducer::connect(const sp<IProducerListener>& /* listener */, int api,
447 bool /* producer_controlled_by_app */,
448 QueueBufferOutput* output) {
449 // Consumer interaction are actually handled by buffer hub, and we need
450 // to maintain consumer operations here. We only need to perform basic input
451 // parameter checks here.
452 ALOGV(__FUNCTION__);
453
454 if (output == nullptr) {
455 return BAD_VALUE;
456 }
457
458 std::unique_lock<std::mutex> lock(mutex_);
459
460 if (connected_api_ != kNoConnectedApi) {
461 return BAD_VALUE;
462 }
463
464 if (!queue_->is_connected()) {
465 ALOGE("BufferHubProducer::connect: This BufferHubProducer is not "
466 "connected to bufferhud. Has it been taken out as a parcelable?");
467 return BAD_VALUE;
468 }
469
470 switch (api) {
471 case NATIVE_WINDOW_API_EGL:
472 case NATIVE_WINDOW_API_CPU:
473 case NATIVE_WINDOW_API_MEDIA:
474 case NATIVE_WINDOW_API_CAMERA:
475 connected_api_ = api;
476
477 output->width = queue_->default_width();
478 output->height = queue_->default_height();
479
480 // default values, we don't use them yet.
481 output->transformHint = 0;
482 output->numPendingBuffers = 0;
483 output->nextFrameNumber = 0;
484 output->bufferReplaced = false;
485
486 break;
487 default:
488 ALOGE("BufferHubProducer::connect: unknow API %d", api);
489 return BAD_VALUE;
490 }
491
492 return NO_ERROR;
493 }
494
disconnect(int api,DisconnectMode)495 status_t BufferHubProducer::disconnect(int api, DisconnectMode /*mode*/) {
496 // Consumer interaction are actually handled by buffer hub, and we need
497 // to maintain consumer operations here. We only need to perform basic input
498 // parameter checks here.
499 ALOGV(__FUNCTION__);
500
501 std::unique_lock<std::mutex> lock(mutex_);
502
503 if (kNoConnectedApi == connected_api_) {
504 return NO_INIT;
505 } else if (api != connected_api_) {
506 return BAD_VALUE;
507 }
508
509 FreeAllBuffers();
510 connected_api_ = kNoConnectedApi;
511 return NO_ERROR;
512 }
513
setSidebandStream(const sp<NativeHandle> & stream)514 status_t BufferHubProducer::setSidebandStream(const sp<NativeHandle>& stream) {
515 if (stream != nullptr) {
516 // TODO(jwcai) Investigate how is is used, maybe use BufferHubBuffer's
517 // metadata.
518 ALOGE("SidebandStream is not currently supported.");
519 return INVALID_OPERATION;
520 }
521 return NO_ERROR;
522 }
523
allocateBuffers(uint32_t,uint32_t,PixelFormat,uint64_t)524 void BufferHubProducer::allocateBuffers(uint32_t /* width */, uint32_t /* height */,
525 PixelFormat /* format */, uint64_t /* usage */) {
526 // TODO(jwcai) |allocateBuffers| aims to preallocate up to the maximum number
527 // of buffers permitted by the current BufferQueue configuration (aka
528 // |max_buffer_count_|).
529 ALOGE("BufferHubProducer::allocateBuffers not implemented.");
530 }
531
allowAllocation(bool)532 status_t BufferHubProducer::allowAllocation(bool /* allow */) {
533 ALOGE("BufferHubProducer::allowAllocation not implemented.");
534 return INVALID_OPERATION;
535 }
536
setGenerationNumber(uint32_t generation_number)537 status_t BufferHubProducer::setGenerationNumber(uint32_t generation_number) {
538 ALOGV(__FUNCTION__);
539
540 std::unique_lock<std::mutex> lock(mutex_);
541 generation_number_ = generation_number;
542 return NO_ERROR;
543 }
544
getConsumerName() const545 String8 BufferHubProducer::getConsumerName() const {
546 // BufferHub based implementation could have one to many producer/consumer
547 // relationship, thus |getConsumerName| from the producer side does not
548 // make any sense.
549 ALOGE("BufferHubProducer::getConsumerName not supported.");
550 return String8("BufferHubQueue::DummyConsumer");
551 }
552
setSharedBufferMode(bool shared_buffer_mode)553 status_t BufferHubProducer::setSharedBufferMode(bool shared_buffer_mode) {
554 if (shared_buffer_mode) {
555 ALOGE("BufferHubProducer::setSharedBufferMode(true) is not supported.");
556 // TODO(b/36373181) Front buffer mode for buffer hub queue as ANativeWindow.
557 return INVALID_OPERATION;
558 }
559 // Setting to default should just work as a no-op.
560 return NO_ERROR;
561 }
562
setAutoRefresh(bool auto_refresh)563 status_t BufferHubProducer::setAutoRefresh(bool auto_refresh) {
564 if (auto_refresh) {
565 ALOGE("BufferHubProducer::setAutoRefresh(true) is not supported.");
566 return INVALID_OPERATION;
567 }
568 // Setting to default should just work as a no-op.
569 return NO_ERROR;
570 }
571
setDequeueTimeout(nsecs_t timeout)572 status_t BufferHubProducer::setDequeueTimeout(nsecs_t timeout) {
573 ALOGV(__FUNCTION__);
574
575 std::unique_lock<std::mutex> lock(mutex_);
576 dequeue_timeout_ms_ = static_cast<int>(timeout / (1000 * 1000));
577 return NO_ERROR;
578 }
579
getLastQueuedBuffer(sp<GraphicBuffer> *,sp<Fence> *,float[16])580 status_t BufferHubProducer::getLastQueuedBuffer(sp<GraphicBuffer>* /* out_buffer */,
581 sp<Fence>* /* out_fence */,
582 float /*out_transform_matrix*/[16]) {
583 ALOGE("BufferHubProducer::getLastQueuedBuffer not implemented.");
584 return INVALID_OPERATION;
585 }
586
getFrameTimestamps(FrameEventHistoryDelta *)587 void BufferHubProducer::getFrameTimestamps(FrameEventHistoryDelta* /*outDelta*/) {
588 ALOGE("BufferHubProducer::getFrameTimestamps not implemented.");
589 }
590
getUniqueId(uint64_t * out_id) const591 status_t BufferHubProducer::getUniqueId(uint64_t* out_id) const {
592 ALOGV(__FUNCTION__);
593
594 *out_id = unique_id_;
595 return NO_ERROR;
596 }
597
getConsumerUsage(uint64_t * out_usage) const598 status_t BufferHubProducer::getConsumerUsage(uint64_t* out_usage) const {
599 ALOGV(__FUNCTION__);
600
601 // same value as returned by querying NATIVE_WINDOW_CONSUMER_USAGE_BITS
602 *out_usage = 0;
603 return NO_ERROR;
604 }
605
TakeAsParcelable(ProducerQueueParcelable * out_parcelable)606 status_t BufferHubProducer::TakeAsParcelable(ProducerQueueParcelable* out_parcelable) {
607 if (!out_parcelable || out_parcelable->IsValid()) return BAD_VALUE;
608
609 if (connected_api_ != kNoConnectedApi) {
610 ALOGE("BufferHubProducer::TakeAsParcelable: BufferHubProducer has "
611 "connected client. Must disconnect first.");
612 return BAD_VALUE;
613 }
614
615 if (!queue_->is_connected()) {
616 ALOGE("BufferHubProducer::TakeAsParcelable: This BufferHubProducer "
617 "is not connected to bufferhud. Has it been taken out as a "
618 "parcelable?");
619 return BAD_VALUE;
620 }
621
622 auto status = queue_->TakeAsParcelable();
623 if (!status) {
624 ALOGE("BufferHubProducer::TakeAsParcelable: Failed to take out "
625 "ProducuerQueueParcelable from the producer queue, error: %s.",
626 status.GetErrorMessage().c_str());
627 return BAD_VALUE;
628 }
629
630 *out_parcelable = status.take();
631 return NO_ERROR;
632 }
633
AllocateBuffer(uint32_t width,uint32_t height,uint32_t layer_count,PixelFormat format,uint64_t usage)634 status_t BufferHubProducer::AllocateBuffer(uint32_t width, uint32_t height, uint32_t layer_count,
635 PixelFormat format, uint64_t usage) {
636 auto status = queue_->AllocateBuffer(width, height, layer_count, uint32_t(format), usage);
637 if (!status) {
638 ALOGE("BufferHubProducer::AllocateBuffer: Failed to allocate buffer: %s",
639 status.GetErrorMessage().c_str());
640 return NO_MEMORY;
641 }
642
643 size_t slot = status.get();
644 auto buffer_producer = queue_->GetBuffer(slot);
645
646 LOG_ALWAYS_FATAL_IF(buffer_producer == nullptr, "Failed to get buffer producer at slot: %zu",
647 slot);
648
649 buffers_[slot].mBufferProducer = buffer_producer;
650
651 return NO_ERROR;
652 }
653
RemoveBuffer(size_t slot)654 status_t BufferHubProducer::RemoveBuffer(size_t slot) {
655 auto status = queue_->RemoveBuffer(slot);
656 if (!status) {
657 ALOGE("BufferHubProducer::RemoveBuffer: Failed to remove buffer: %s",
658 status.GetErrorMessage().c_str());
659 return INVALID_OPERATION;
660 }
661
662 // Reset in memory objects related the the buffer.
663 buffers_[slot].mBufferProducer = nullptr;
664 buffers_[slot].mGraphicBuffer = nullptr;
665 buffers_[slot].mBufferState.detachProducer();
666 return NO_ERROR;
667 }
668
FreeAllBuffers()669 status_t BufferHubProducer::FreeAllBuffers() {
670 for (size_t slot = 0; slot < BufferHubQueue::kMaxQueueCapacity; slot++) {
671 // Reset in memory objects related the the buffer.
672 buffers_[slot].mGraphicBuffer = nullptr;
673 buffers_[slot].mBufferState.reset();
674 buffers_[slot].mRequestBufferCalled = false;
675 buffers_[slot].mBufferProducer = nullptr;
676 buffers_[slot].mFence = Fence::NO_FENCE;
677 }
678
679 auto status = queue_->FreeAllBuffers();
680 if (!status) {
681 ALOGE("BufferHubProducer::FreeAllBuffers: Failed to free all buffers on "
682 "the queue: %s",
683 status.GetErrorMessage().c_str());
684 }
685
686 if (queue_->capacity() != 0 || queue_->count() != 0) {
687 LOG_ALWAYS_FATAL("BufferHubProducer::FreeAllBuffers: Not all buffers are freed.");
688 }
689
690 return NO_ERROR;
691 }
692
exportToParcel(Parcel * parcel)693 status_t BufferHubProducer::exportToParcel(Parcel* parcel) {
694 status_t res = TakeAsParcelable(&pending_producer_parcelable_);
695 if (res != NO_ERROR) return res;
696
697 if (!pending_producer_parcelable_.IsValid()) {
698 ALOGE("BufferHubProducer::exportToParcel: Invalid parcelable object.");
699 return BAD_VALUE;
700 }
701
702 res = parcel->writeUint32(USE_BUFFER_HUB);
703 if (res != NO_ERROR) {
704 ALOGE("BufferHubProducer::exportToParcel: Cannot write magic, res=%d.", res);
705 return res;
706 }
707
708 return pending_producer_parcelable_.writeToParcel(parcel);
709 }
710
onAsBinder()711 IBinder* BufferHubProducer::onAsBinder() {
712 ALOGE("BufferHubProducer::onAsBinder: BufferHubProducer should never be used as an Binder "
713 "object.");
714 return nullptr;
715 }
716
717 } // namespace android
718