blob: 2bc194aeadb6b146176fe102c4b2305faf1eecc7 [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>
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -070022#include <ui/DetachedBufferHandle.h>
Alex Vakulenko4fe60582017-02-02 11:35:59 -080023
Alex Vakulenkoe4eec202017-01-27 14:41:04 -080024namespace android {
Alex Vakulenkoe4eec202017-01-27 14:41:04 -080025
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -080026using namespace dvr;
27
Jiwen 'Steve' Caid6cb17f2017-05-08 16:15:35 -070028/* static */
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -080029sp<BufferHubProducer> BufferHubProducer::Create(const std::shared_ptr<ProducerQueue>& queue) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080030 sp<BufferHubProducer> producer = new BufferHubProducer;
31 producer->queue_ = queue;
32 return producer;
Jiwen 'Steve' Caid6cb17f2017-05-08 16:15:35 -070033}
Alex Vakulenkoe4eec202017-01-27 14:41:04 -080034
Jiwen 'Steve' Cai5afb7402017-11-09 18:50:32 -080035/* static */
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -080036sp<BufferHubProducer> BufferHubProducer::Create(ProducerQueueParcelable parcelable) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080037 if (!parcelable.IsValid()) {
38 ALOGE("BufferHubProducer::Create: Invalid producer parcelable.");
39 return nullptr;
40 }
Jiwen 'Steve' Cai5afb7402017-11-09 18:50:32 -080041
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080042 sp<BufferHubProducer> producer = new BufferHubProducer;
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -080043 producer->queue_ = ProducerQueue::Import(parcelable.TakeChannelHandle());
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080044 return producer;
Jiwen 'Steve' Cai5afb7402017-11-09 18:50:32 -080045}
46
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -080047status_t BufferHubProducer::requestBuffer(int slot, sp<GraphicBuffer>* buf) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080048 ALOGV("requestBuffer: slot=%d", slot);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -080049
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080050 std::unique_lock<std::mutex> lock(mutex_);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -080051
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080052 if (connected_api_ == kNoConnectedApi) {
53 ALOGE("requestBuffer: BufferHubProducer has no connected producer");
54 return NO_INIT;
55 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -070056
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080057 if (slot < 0 || slot >= max_buffer_count_) {
58 ALOGE("requestBuffer: slot index %d out of range [0, %d)", slot, max_buffer_count_);
59 return BAD_VALUE;
60 } else if (!buffers_[slot].mBufferState.isDequeued()) {
61 ALOGE("requestBuffer: slot %d is not owned by the producer (state = %s)", slot,
62 buffers_[slot].mBufferState.string());
63 return BAD_VALUE;
64 } else if (buffers_[slot].mGraphicBuffer != nullptr) {
65 ALOGE("requestBuffer: slot %d is not empty.", slot);
66 return BAD_VALUE;
67 } else if (buffers_[slot].mBufferProducer == nullptr) {
68 ALOGE("requestBuffer: slot %d is not dequeued.", slot);
69 return BAD_VALUE;
70 }
Alex Vakulenkoe4eec202017-01-27 14:41:04 -080071
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080072 const auto& buffer_producer = buffers_[slot].mBufferProducer;
73 sp<GraphicBuffer> graphic_buffer = buffer_producer->buffer()->buffer();
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -070074
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080075 buffers_[slot].mGraphicBuffer = graphic_buffer;
76 buffers_[slot].mRequestBufferCalled = true;
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -070077
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080078 *buf = graphic_buffer;
79 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -080080}
81
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080082status_t BufferHubProducer::setMaxDequeuedBufferCount(int max_dequeued_buffers) {
83 ALOGV("setMaxDequeuedBufferCount: max_dequeued_buffers=%d", max_dequeued_buffers);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -080084
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080085 std::unique_lock<std::mutex> lock(mutex_);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -080086
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080087 if (max_dequeued_buffers <= 0 ||
88 max_dequeued_buffers >
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -080089 int(BufferHubQueue::kMaxQueueCapacity - kDefaultUndequeuedBuffers)) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080090 ALOGE("setMaxDequeuedBufferCount: %d out of range (0, %zu]", max_dequeued_buffers,
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -080091 BufferHubQueue::kMaxQueueCapacity);
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080092 return BAD_VALUE;
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -070093 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -070094
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -080095 // The new dequeued_buffers count should not be violated by the number
96 // of currently dequeued buffers.
97 int dequeued_count = 0;
98 for (const auto& buf : buffers_) {
99 if (buf.mBufferState.isDequeued()) {
100 dequeued_count++;
101 }
102 }
103 if (dequeued_count > max_dequeued_buffers) {
104 ALOGE("setMaxDequeuedBufferCount: the requested dequeued_buffers"
105 "count (%d) exceeds the current dequeued buffer count (%d)",
106 max_dequeued_buffers, dequeued_count);
107 return BAD_VALUE;
108 }
109
110 max_dequeued_buffer_count_ = max_dequeued_buffers;
111 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800112}
113
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800114status_t BufferHubProducer::setAsyncMode(bool async) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800115 if (async) {
116 // TODO(b/36724099) BufferHubQueue's consumer end always acquires the buffer
117 // automatically and behaves differently from IGraphicBufferConsumer. Thus,
118 // android::BufferQueue's async mode (a.k.a. allocating an additional buffer
119 // to prevent dequeueBuffer from being blocking) technically does not apply
120 // here.
121 //
122 // In Daydream, non-blocking producer side dequeue is guaranteed by careful
123 // buffer consumer implementations. In another word, BufferHubQueue based
124 // dequeueBuffer should never block whether setAsyncMode(true) is set or
125 // not.
126 //
127 // See: IGraphicBufferProducer::setAsyncMode and
128 // BufferQueueProducer::setAsyncMode for more about original implementation.
129 ALOGW("BufferHubProducer::setAsyncMode: BufferHubQueue should always be "
130 "asynchronous. This call makes no effact.");
131 return NO_ERROR;
132 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700133 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800134}
135
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800136status_t BufferHubProducer::dequeueBuffer(int* out_slot, sp<Fence>* out_fence, uint32_t width,
137 uint32_t height, PixelFormat format, uint64_t usage,
138 uint64_t* /*outBufferAge*/,
139 FrameEventHistoryDelta* /* out_timestamps */) {
Jiwen 'Steve' Caie7cf62a2018-04-30 22:56:08 -0700140 ALOGV("dequeueBuffer: w=%u, h=%u, format=%d, usage=%" PRIu64, width, height, format, usage);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800141
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800142 status_t ret;
143 std::unique_lock<std::mutex> lock(mutex_);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800144
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800145 if (connected_api_ == kNoConnectedApi) {
146 ALOGE("dequeueBuffer: BufferQueue has no connected producer");
147 return NO_INIT;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800148 }
149
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800150 const uint32_t kLayerCount = 1;
151 if (int32_t(queue_->capacity()) < max_dequeued_buffer_count_ + kDefaultUndequeuedBuffers) {
152 // Lazy allocation. When the capacity of |queue_| has not reached
153 // |max_dequeued_buffer_count_|, allocate new buffer.
154 // TODO(jwcai) To save memory, the really reasonable thing to do is to go
155 // over existing slots and find first existing one to dequeue.
156 ret = AllocateBuffer(width, height, kLayerCount, format, usage);
157 if (ret < 0) return ret;
158 }
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800159
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800160 size_t slot = 0;
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -0800161 std::shared_ptr<BufferProducer> buffer_producer;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800162
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -0800163 for (size_t retry = 0; retry < BufferHubQueue::kMaxQueueCapacity; retry++) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800164 LocalHandle fence;
165 auto buffer_status = queue_->Dequeue(dequeue_timeout_ms_, &slot, &fence);
166 if (!buffer_status) return NO_MEMORY;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800167
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800168 buffer_producer = buffer_status.take();
169 if (!buffer_producer) return NO_MEMORY;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800170
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800171 if (width == buffer_producer->width() && height == buffer_producer->height() &&
172 uint32_t(format) == buffer_producer->format()) {
173 // The producer queue returns a buffer producer matches the request.
174 break;
175 }
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800176
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800177 // Needs reallocation.
178 // TODO(jwcai) Consider use VLOG instead if we find this log is not useful.
179 ALOGI("dequeueBuffer: requested buffer (w=%u, h=%u, format=%u) is different "
180 "from the buffer returned at slot: %zu (w=%u, h=%u, format=%u). Need "
181 "re-allocattion.",
182 width, height, format, slot, buffer_producer->width(), buffer_producer->height(),
183 buffer_producer->format());
184 // Mark the slot as reallocating, so that later we can set
185 // BUFFER_NEEDS_REALLOCATION when the buffer actually get dequeued.
186 buffers_[slot].mIsReallocating = true;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800187
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800188 // Remove the old buffer once the allocation before allocating its
189 // replacement.
190 RemoveBuffer(slot);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800191
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800192 // Allocate a new producer buffer with new buffer configs. Note that if
193 // there are already multiple buffers in the queue, the next one returned
194 // from |queue_->Dequeue| may not be the new buffer we just reallocated.
195 // Retry up to BufferHubQueue::kMaxQueueCapacity times.
196 ret = AllocateBuffer(width, height, kLayerCount, format, usage);
197 if (ret < 0) return ret;
198 }
199
200 // With the BufferHub backed solution. Buffer slot returned from
201 // |queue_->Dequeue| is guaranteed to avaiable for producer's use.
202 // It's either in free state (if the buffer has never been used before) or
203 // in queued state (if the buffer has been dequeued and queued back to
204 // BufferHubQueue).
205 LOG_ALWAYS_FATAL_IF((!buffers_[slot].mBufferState.isFree() &&
206 !buffers_[slot].mBufferState.isQueued()),
207 "dequeueBuffer: slot %zu is not free or queued, actual state: %s.", slot,
208 buffers_[slot].mBufferState.string());
209
210 buffers_[slot].mBufferState.freeQueued();
211 buffers_[slot].mBufferState.dequeue();
Jiwen 'Steve' Caie7cf62a2018-04-30 22:56:08 -0700212 ALOGV("dequeueBuffer: slot=%zu", slot);
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800213
214 // TODO(jwcai) Handle fence properly. |BufferHub| has full fence support, we
215 // just need to exopose that through |BufferHubQueue| once we need fence.
216 *out_fence = Fence::NO_FENCE;
217 *out_slot = int(slot);
218 ret = NO_ERROR;
219
220 if (buffers_[slot].mIsReallocating) {
221 ret |= BUFFER_NEEDS_REALLOCATION;
222 buffers_[slot].mIsReallocating = false;
223 }
224
225 return ret;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800226}
227
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700228status_t BufferHubProducer::detachBuffer(int slot) {
229 ALOGV("detachBuffer: slot=%d", slot);
230 std::unique_lock<std::mutex> lock(mutex_);
231
232 return DetachBufferLocked(static_cast<size_t>(slot));
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800233}
234
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700235status_t BufferHubProducer::DetachBufferLocked(size_t slot) {
236 if (connected_api_ == kNoConnectedApi) {
237 ALOGE("detachBuffer: BufferHubProducer is not connected.");
238 return NO_INIT;
239 }
240
241 if (slot >= static_cast<size_t>(max_buffer_count_)) {
242 ALOGE("detachBuffer: slot index %zu out of range [0, %d)", slot, max_buffer_count_);
243 return BAD_VALUE;
244 } else if (!buffers_[slot].mBufferState.isDequeued()) {
245 ALOGE("detachBuffer: slot %zu is not owned by the producer (state = %s)", slot,
246 buffers_[slot].mBufferState.string());
247 return BAD_VALUE;
248 } else if (!buffers_[slot].mRequestBufferCalled) {
249 ALOGE("detachBuffer: buffer in slot %zu has not been requested", slot);
250 return BAD_VALUE;
251 }
252 std::shared_ptr<BufferProducer> buffer_producer = queue_->GetBuffer(slot);
253 if (buffer_producer == nullptr || buffer_producer->buffer() == nullptr) {
254 ALOGE("detachBuffer: Invalid BufferProducer at slot %zu.", slot);
255 return BAD_VALUE;
256 }
257 sp<GraphicBuffer> graphic_buffer = buffer_producer->buffer()->buffer();
258 if (graphic_buffer == nullptr) {
259 ALOGE("detachBuffer: Invalid GraphicBuffer at slot %zu.", slot);
260 return BAD_VALUE;
261 }
262
263 // Remove the BufferProducer from the ProducerQueue.
264 status_t error = RemoveBuffer(slot);
265 if (error != NO_ERROR) {
266 ALOGE("detachBuffer: Failed to remove buffer, slot=%zu, error=%d.", slot, error);
267 return error;
268 }
269
270 // Here we need to convert the existing ProducerBuffer into a DetachedBufferHandle and inject
271 // the handle into the GraphicBuffer object at the requested slot.
272 auto status_or_handle = buffer_producer->Detach();
273 if (!status_or_handle.ok()) {
274 ALOGE("detachBuffer: Failed to detach from a BufferProducer at slot %zu, error=%d.", slot,
275 status_or_handle.error());
276 return BAD_VALUE;
277 }
278 std::unique_ptr<DetachedBufferHandle> handle =
279 DetachedBufferHandle::Create(status_or_handle.take());
280 if (!handle->isValid()) {
281 ALOGE("detachBuffer: Failed to create a DetachedBufferHandle at slot %zu.", slot);
282 return BAD_VALUE;
283 }
284
285 return graphic_buffer->setDetachedBufferHandle(std::move(handle));
286}
287
288status_t BufferHubProducer::detachNextBuffer(sp<GraphicBuffer>* out_buffer, sp<Fence>* out_fence) {
289 ALOGV("detachNextBuffer.");
290
291 if (out_buffer == nullptr || out_fence == nullptr) {
292 ALOGE("detachNextBuffer: Invalid parameter: out_buffer=%p, out_fence=%p", out_buffer,
293 out_fence);
294 return BAD_VALUE;
295 }
296
297 std::unique_lock<std::mutex> lock(mutex_);
298
299 if (connected_api_ == kNoConnectedApi) {
300 ALOGE("detachNextBuffer: BufferHubProducer is not connected.");
301 return NO_INIT;
302 }
303
304 // detachNextBuffer is equivalent to calling dequeueBuffer, requestBuffer, and detachBuffer in
305 // sequence, except for two things:
306 //
307 // 1) It is unnecessary to know the dimensions, format, or usage of the next buffer, i.e. the
308 // function just returns whatever BufferProducer is available from the ProducerQueue and no
309 // buffer allocation or re-allocation will happen.
310 // 2) It will not block, since if it cannot find an appropriate buffer to return, it will return
311 // an error instead.
312 size_t slot = 0;
313 LocalHandle fence;
314
315 // First, dequeue a BufferProducer from the ProducerQueue with no timeout. Report error
316 // immediately if ProducerQueue::Dequeue() fails.
317 auto status_or_buffer = queue_->Dequeue(/*timeout=*/0, &slot, &fence);
318 if (!status_or_buffer.ok()) {
319 ALOGE("detachNextBuffer: Failed to dequeue buffer, error=%d.", status_or_buffer.error());
320 return NO_MEMORY;
321 }
322
323 std::shared_ptr<BufferProducer> buffer_producer = status_or_buffer.take();
324 if (buffer_producer == nullptr) {
325 ALOGE("detachNextBuffer: Dequeued buffer is null.");
326 return NO_MEMORY;
327 }
328
329 // With the BufferHub backed solution, slot returned from |queue_->Dequeue| is guaranteed to
330 // be available for producer's use. It's either in free state (if the buffer has never been used
331 // before) or in queued state (if the buffer has been dequeued and queued back to
332 // BufferHubQueue).
333 if (!buffers_[slot].mBufferState.isFree() && !buffers_[slot].mBufferState.isQueued()) {
334 ALOGE("detachNextBuffer: slot %zu is not free or queued, actual state: %s.", slot,
335 buffers_[slot].mBufferState.string());
336 return BAD_VALUE;
337 }
338 if (buffers_[slot].mBufferProducer == nullptr) {
339 ALOGE("detachNextBuffer: BufferProducer at slot %zu is null.", slot);
340 return BAD_VALUE;
341 }
342 if (buffers_[slot].mBufferProducer->id() != buffer_producer->id()) {
343 ALOGE("detachNextBuffer: BufferProducer at slot %zu has mismatched id, actual: "
344 "%d, expected: %d.",
345 slot, buffers_[slot].mBufferProducer->id(), buffer_producer->id());
346 return BAD_VALUE;
347 }
348
349 ALOGV("detachNextBuffer: slot=%zu", slot);
350 buffers_[slot].mBufferState.freeQueued();
351 buffers_[slot].mBufferState.dequeue();
352
353 // Second, request the buffer.
354 sp<GraphicBuffer> graphic_buffer = buffer_producer->buffer()->buffer();
355 buffers_[slot].mGraphicBuffer = buffer_producer->buffer()->buffer();
356
357 // Finally, detach the buffer and then return.
358 status_t error = DetachBufferLocked(slot);
359 if (error == NO_ERROR) {
360 *out_fence = new Fence(fence.Release());
361 *out_buffer = graphic_buffer;
362 }
363 return error;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800364}
365
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800366status_t BufferHubProducer::attachBuffer(int* /* out_slot */,
367 const sp<GraphicBuffer>& /* buffer */) {
368 // With this BufferHub backed implementation, we assume (for now) all buffers
369 // are allocated and owned by the BufferHub. Thus the attempt of transfering
370 // ownership of a buffer to the buffer queue is intentionally unsupported.
371 LOG_ALWAYS_FATAL("BufferHubProducer::attachBuffer not supported.");
372 return INVALID_OPERATION;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800373}
374
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800375status_t BufferHubProducer::queueBuffer(int slot, const QueueBufferInput& input,
376 QueueBufferOutput* output) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800377 ALOGV("queueBuffer: slot %d", slot);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800378
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800379 if (output == nullptr) {
380 return BAD_VALUE;
381 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700382
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800383 int64_t timestamp;
384 bool is_auto_timestamp;
385 android_dataspace dataspace;
386 Rect crop(Rect::EMPTY_RECT);
387 int scaling_mode;
388 uint32_t transform;
389 sp<Fence> fence;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800390
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800391 input.deflate(&timestamp, &is_auto_timestamp, &dataspace, &crop, &scaling_mode, &transform,
392 &fence);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800393
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800394 // Check input scaling mode is valid.
395 switch (scaling_mode) {
396 case NATIVE_WINDOW_SCALING_MODE_FREEZE:
397 case NATIVE_WINDOW_SCALING_MODE_SCALE_TO_WINDOW:
398 case NATIVE_WINDOW_SCALING_MODE_SCALE_CROP:
399 case NATIVE_WINDOW_SCALING_MODE_NO_SCALE_CROP:
400 break;
401 default:
402 ALOGE("queueBuffer: unknown scaling mode %d", scaling_mode);
403 return BAD_VALUE;
404 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700405
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800406 // Check input fence is valid.
407 if (fence == nullptr) {
408 ALOGE("queueBuffer: fence is NULL");
409 return BAD_VALUE;
410 }
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800411
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800412 std::unique_lock<std::mutex> lock(mutex_);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800413
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800414 if (connected_api_ == kNoConnectedApi) {
415 ALOGE("queueBuffer: BufferQueue has no connected producer");
416 return NO_INIT;
417 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700418
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800419 if (slot < 0 || slot >= max_buffer_count_) {
420 ALOGE("queueBuffer: slot index %d out of range [0, %d)", slot, max_buffer_count_);
421 return BAD_VALUE;
422 } else if (!buffers_[slot].mBufferState.isDequeued()) {
423 ALOGE("queueBuffer: slot %d is not owned by the producer (state = %s)", slot,
424 buffers_[slot].mBufferState.string());
425 return BAD_VALUE;
426 } else if ((!buffers_[slot].mRequestBufferCalled || buffers_[slot].mGraphicBuffer == nullptr)) {
427 ALOGE("queueBuffer: slot %d is not requested (mRequestBufferCalled=%d, "
428 "mGraphicBuffer=%p)",
429 slot, buffers_[slot].mRequestBufferCalled, buffers_[slot].mGraphicBuffer.get());
430 return BAD_VALUE;
431 }
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800432
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800433 // Post the buffer producer with timestamp in the metadata.
434 const auto& buffer_producer = buffers_[slot].mBufferProducer;
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700435
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800436 // Check input crop is not out of boundary of current buffer.
437 Rect buffer_rect(buffer_producer->width(), buffer_producer->height());
438 Rect cropped_rect(Rect::EMPTY_RECT);
439 crop.intersect(buffer_rect, &cropped_rect);
440 if (cropped_rect != crop) {
441 ALOGE("queueBuffer: slot %d has out-of-boundary crop.", slot);
442 return BAD_VALUE;
443 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700444
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800445 LocalHandle fence_fd(fence->isValid() ? fence->dup() : -1);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800446
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800447 DvrNativeBufferMetadata meta_data;
448 meta_data.timestamp = timestamp;
449 meta_data.is_auto_timestamp = int32_t(is_auto_timestamp);
450 meta_data.dataspace = int32_t(dataspace);
451 meta_data.crop_left = crop.left;
452 meta_data.crop_top = crop.top;
453 meta_data.crop_right = crop.right;
454 meta_data.crop_bottom = crop.bottom;
455 meta_data.scaling_mode = int32_t(scaling_mode);
456 meta_data.transform = int32_t(transform);
Jiwen 'Steve' Caicb4751c2017-04-14 17:56:55 -0700457
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800458 buffer_producer->PostAsync(&meta_data, fence_fd);
459 buffers_[slot].mBufferState.queue();
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800460
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800461 output->width = buffer_producer->width();
462 output->height = buffer_producer->height();
463 output->transformHint = 0; // default value, we don't use it yet.
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700464
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800465 // |numPendingBuffers| counts of the number of buffers that has been enqueued
466 // by the producer but not yet acquired by the consumer. Due to the nature
467 // of BufferHubQueue design, this is hard to trace from the producer's client
468 // side, but it's safe to assume it's zero.
469 output->numPendingBuffers = 0;
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700470
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800471 // Note that we are not setting nextFrameNumber here as it seems to be only
472 // used by surface flinger. See more at b/22802885, ag/791760.
473 output->nextFrameNumber = 0;
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700474
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800475 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800476}
477
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800478status_t BufferHubProducer::cancelBuffer(int slot, const sp<Fence>& fence) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800479 ALOGV(__FUNCTION__);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800480
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800481 std::unique_lock<std::mutex> lock(mutex_);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800482
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800483 if (connected_api_ == kNoConnectedApi) {
484 ALOGE("cancelBuffer: BufferQueue has no connected producer");
485 return NO_INIT;
486 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700487
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800488 if (slot < 0 || slot >= max_buffer_count_) {
489 ALOGE("cancelBuffer: slot index %d out of range [0, %d)", slot, max_buffer_count_);
490 return BAD_VALUE;
491 } else if (!buffers_[slot].mBufferState.isDequeued()) {
492 ALOGE("cancelBuffer: slot %d is not owned by the producer (state = %s)", slot,
493 buffers_[slot].mBufferState.string());
494 return BAD_VALUE;
495 } else if (fence == nullptr) {
496 ALOGE("cancelBuffer: fence is NULL");
497 return BAD_VALUE;
498 }
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800499
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800500 auto buffer_producer = buffers_[slot].mBufferProducer;
501 queue_->Enqueue(buffer_producer, size_t(slot), 0ULL);
502 buffers_[slot].mBufferState.cancel();
503 buffers_[slot].mFence = fence;
504 ALOGV("cancelBuffer: slot %d", slot);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800505
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800506 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800507}
508
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800509status_t BufferHubProducer::query(int what, int* out_value) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800510 ALOGV(__FUNCTION__);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800511
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800512 std::unique_lock<std::mutex> lock(mutex_);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800513
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800514 if (out_value == nullptr) {
515 ALOGE("query: out_value was NULL");
516 return BAD_VALUE;
517 }
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800518
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800519 int value = 0;
520 switch (what) {
521 case NATIVE_WINDOW_MIN_UNDEQUEUED_BUFFERS:
522 // TODO(b/36187402) This should be the maximum number of buffers that this
523 // producer queue's consumer can acquire. Set to be at least one. Need to
524 // find a way to set from the consumer side.
525 value = kDefaultUndequeuedBuffers;
526 break;
527 case NATIVE_WINDOW_BUFFER_AGE:
528 value = 0;
529 break;
530 case NATIVE_WINDOW_WIDTH:
531 value = int32_t(queue_->default_width());
532 break;
533 case NATIVE_WINDOW_HEIGHT:
534 value = int32_t(queue_->default_height());
535 break;
536 case NATIVE_WINDOW_FORMAT:
537 value = int32_t(queue_->default_format());
538 break;
539 case NATIVE_WINDOW_CONSUMER_RUNNING_BEHIND:
540 // BufferHubQueue is always operating in async mode, thus semantically
541 // consumer can never be running behind. See BufferQueueCore.cpp core
542 // for more information about the original meaning of this flag.
543 value = 0;
544 break;
545 case NATIVE_WINDOW_CONSUMER_USAGE_BITS:
546 // TODO(jwcai) This is currently not implement as we don't need
547 // IGraphicBufferConsumer parity.
548 value = 0;
549 break;
550 case NATIVE_WINDOW_DEFAULT_DATASPACE:
551 // TODO(jwcai) Return the default value android::BufferQueue is using as
552 // there is no way dvr::ConsumerQueue can set it.
553 value = 0; // HAL_DATASPACE_UNKNOWN
554 break;
555 case NATIVE_WINDOW_STICKY_TRANSFORM:
556 // TODO(jwcai) Return the default value android::BufferQueue is using as
557 // there is no way dvr::ConsumerQueue can set it.
558 value = 0;
559 break;
560 case NATIVE_WINDOW_CONSUMER_IS_PROTECTED:
561 // In Daydream's implementation, the consumer end (i.e. VR Compostior)
562 // knows how to handle protected buffers.
563 value = 1;
564 break;
565 default:
566 return BAD_VALUE;
567 }
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800568
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800569 ALOGV("query: key=%d, v=%d", what, value);
570 *out_value = value;
571 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800572}
573
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800574status_t BufferHubProducer::connect(const sp<IProducerListener>& /* listener */, int api,
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800575 bool /* producer_controlled_by_app */,
576 QueueBufferOutput* output) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800577 // Consumer interaction are actually handled by buffer hub, and we need
578 // to maintain consumer operations here. We only need to perform basic input
579 // parameter checks here.
580 ALOGV(__FUNCTION__);
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700581
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800582 if (output == nullptr) {
583 return BAD_VALUE;
584 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700585
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800586 std::unique_lock<std::mutex> lock(mutex_);
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700587
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800588 if (connected_api_ != kNoConnectedApi) {
589 return BAD_VALUE;
590 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700591
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800592 if (!queue_->is_connected()) {
593 ALOGE("BufferHubProducer::connect: This BufferHubProducer is not "
594 "connected to bufferhud. Has it been taken out as a parcelable?");
595 return BAD_VALUE;
596 }
Jiwen 'Steve' Cai5afb7402017-11-09 18:50:32 -0800597
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800598 switch (api) {
599 case NATIVE_WINDOW_API_EGL:
600 case NATIVE_WINDOW_API_CPU:
601 case NATIVE_WINDOW_API_MEDIA:
602 case NATIVE_WINDOW_API_CAMERA:
603 connected_api_ = api;
Jiwen 'Steve' Caicb4751c2017-04-14 17:56:55 -0700604
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800605 output->width = queue_->default_width();
606 output->height = queue_->default_height();
Jiwen 'Steve' Caicb4751c2017-04-14 17:56:55 -0700607
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800608 // default values, we don't use them yet.
609 output->transformHint = 0;
610 output->numPendingBuffers = 0;
611 output->nextFrameNumber = 0;
612 output->bufferReplaced = false;
Jiwen 'Steve' Caicb4751c2017-04-14 17:56:55 -0700613
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800614 break;
615 default:
616 ALOGE("BufferHubProducer::connect: unknow API %d", api);
617 return BAD_VALUE;
618 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700619
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800620 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800621}
622
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800623status_t BufferHubProducer::disconnect(int api, DisconnectMode /*mode*/) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800624 // Consumer interaction are actually handled by buffer hub, and we need
625 // to maintain consumer operations here. We only need to perform basic input
626 // parameter checks here.
627 ALOGV(__FUNCTION__);
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700628
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800629 std::unique_lock<std::mutex> lock(mutex_);
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700630
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800631 if (kNoConnectedApi == connected_api_) {
632 return NO_INIT;
633 } else if (api != connected_api_) {
634 return BAD_VALUE;
635 }
Jiwen 'Steve' Cai1601bcf2017-03-24 14:03:06 -0700636
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800637 FreeAllBuffers();
638 connected_api_ = kNoConnectedApi;
639 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800640}
641
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800642status_t BufferHubProducer::setSidebandStream(const sp<NativeHandle>& stream) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800643 if (stream != nullptr) {
644 // TODO(jwcai) Investigate how is is used, maybe use BufferHubBuffer's
645 // metadata.
646 ALOGE("SidebandStream is not currently supported.");
647 return INVALID_OPERATION;
648 }
649 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800650}
651
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800652void BufferHubProducer::allocateBuffers(uint32_t /* width */, uint32_t /* height */,
653 PixelFormat /* format */, uint64_t /* usage */) {
654 // TODO(jwcai) |allocateBuffers| aims to preallocate up to the maximum number
655 // of buffers permitted by the current BufferQueue configuration (aka
656 // |max_buffer_count_|).
657 ALOGE("BufferHubProducer::allocateBuffers not implemented.");
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800658}
659
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800660status_t BufferHubProducer::allowAllocation(bool /* allow */) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800661 ALOGE("BufferHubProducer::allowAllocation not implemented.");
662 return INVALID_OPERATION;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800663}
664
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800665status_t BufferHubProducer::setGenerationNumber(uint32_t generation_number) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800666 ALOGV(__FUNCTION__);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800667
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800668 std::unique_lock<std::mutex> lock(mutex_);
669 generation_number_ = generation_number;
670 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800671}
672
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800673String8 BufferHubProducer::getConsumerName() const {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800674 // BufferHub based implementation could have one to many producer/consumer
675 // relationship, thus |getConsumerName| from the producer side does not
676 // make any sense.
677 ALOGE("BufferHubProducer::getConsumerName not supported.");
678 return String8("BufferHubQueue::DummyConsumer");
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800679}
680
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800681status_t BufferHubProducer::setSharedBufferMode(bool shared_buffer_mode) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800682 if (shared_buffer_mode) {
683 ALOGE("BufferHubProducer::setSharedBufferMode(true) is not supported.");
684 // TODO(b/36373181) Front buffer mode for buffer hub queue as ANativeWindow.
685 return INVALID_OPERATION;
686 }
687 // Setting to default should just work as a no-op.
688 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800689}
690
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800691status_t BufferHubProducer::setAutoRefresh(bool auto_refresh) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800692 if (auto_refresh) {
693 ALOGE("BufferHubProducer::setAutoRefresh(true) is not supported.");
694 return INVALID_OPERATION;
695 }
696 // Setting to default should just work as a no-op.
697 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800698}
699
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800700status_t BufferHubProducer::setDequeueTimeout(nsecs_t timeout) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800701 ALOGV(__FUNCTION__);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800702
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800703 std::unique_lock<std::mutex> lock(mutex_);
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -0800704 dequeue_timeout_ms_ = static_cast<int>(timeout / (1000 * 1000));
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800705 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800706}
707
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800708status_t BufferHubProducer::getLastQueuedBuffer(sp<GraphicBuffer>* /* out_buffer */,
709 sp<Fence>* /* out_fence */,
710 float /*out_transform_matrix*/[16]) {
711 ALOGE("BufferHubProducer::getLastQueuedBuffer not implemented.");
712 return INVALID_OPERATION;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800713}
714
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800715void BufferHubProducer::getFrameTimestamps(FrameEventHistoryDelta* /*outDelta*/) {
716 ALOGE("BufferHubProducer::getFrameTimestamps not implemented.");
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800717}
718
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800719status_t BufferHubProducer::getUniqueId(uint64_t* out_id) const {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800720 ALOGV(__FUNCTION__);
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800721
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800722 *out_id = unique_id_;
723 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800724}
725
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800726status_t BufferHubProducer::getConsumerUsage(uint64_t* out_usage) const {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800727 ALOGV(__FUNCTION__);
Chia-I Wue2786ea2017-08-07 10:36:08 -0700728
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800729 // same value as returned by querying NATIVE_WINDOW_CONSUMER_USAGE_BITS
730 *out_usage = 0;
731 return NO_ERROR;
Chia-I Wue2786ea2017-08-07 10:36:08 -0700732}
733
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -0800734status_t BufferHubProducer::TakeAsParcelable(ProducerQueueParcelable* out_parcelable) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800735 if (!out_parcelable || out_parcelable->IsValid()) return BAD_VALUE;
Jiwen 'Steve' Cai5afb7402017-11-09 18:50:32 -0800736
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800737 if (connected_api_ != kNoConnectedApi) {
738 ALOGE("BufferHubProducer::TakeAsParcelable: BufferHubProducer has "
739 "connected client. Must disconnect first.");
740 return BAD_VALUE;
741 }
Jiwen 'Steve' Cai5afb7402017-11-09 18:50:32 -0800742
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800743 if (!queue_->is_connected()) {
744 ALOGE("BufferHubProducer::TakeAsParcelable: This BufferHubProducer "
745 "is not connected to bufferhud. Has it been taken out as a "
746 "parcelable?");
747 return BAD_VALUE;
748 }
Jiwen 'Steve' Cai5afb7402017-11-09 18:50:32 -0800749
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800750 auto status = queue_->TakeAsParcelable();
751 if (!status) {
752 ALOGE("BufferHubProducer::TakeAsParcelable: Failed to take out "
753 "ProducuerQueueParcelable from the producer queue, error: %s.",
754 status.GetErrorMessage().c_str());
755 return BAD_VALUE;
756 }
Jiwen 'Steve' Cai5afb7402017-11-09 18:50:32 -0800757
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800758 *out_parcelable = status.take();
759 return NO_ERROR;
Jiwen 'Steve' Cai5afb7402017-11-09 18:50:32 -0800760}
761
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800762status_t BufferHubProducer::AllocateBuffer(uint32_t width, uint32_t height, uint32_t layer_count,
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800763 PixelFormat format, uint64_t usage) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800764 auto status = queue_->AllocateBuffer(width, height, layer_count, uint32_t(format), usage);
765 if (!status) {
766 ALOGE("BufferHubProducer::AllocateBuffer: Failed to allocate buffer: %s",
767 status.GetErrorMessage().c_str());
768 return NO_MEMORY;
769 }
Jiwen 'Steve' Caid6cb17f2017-05-08 16:15:35 -0700770
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800771 size_t slot = status.get();
772 auto buffer_producer = queue_->GetBuffer(slot);
Jiwen 'Steve' Caid6cb17f2017-05-08 16:15:35 -0700773
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800774 LOG_ALWAYS_FATAL_IF(buffer_producer == nullptr, "Failed to get buffer producer at slot: %zu",
775 slot);
Jiwen 'Steve' Caid6cb17f2017-05-08 16:15:35 -0700776
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800777 buffers_[slot].mBufferProducer = buffer_producer;
Jiwen 'Steve' Caid6cb17f2017-05-08 16:15:35 -0700778
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800779 return NO_ERROR;
Jiwen 'Steve' Caid6cb17f2017-05-08 16:15:35 -0700780}
781
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800782status_t BufferHubProducer::RemoveBuffer(size_t slot) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800783 auto status = queue_->RemoveBuffer(slot);
784 if (!status) {
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700785 ALOGE("BufferHubProducer::RemoveBuffer: Failed to remove buffer at slot: %zu, error: %s.",
786 slot, status.GetErrorMessage().c_str());
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800787 return INVALID_OPERATION;
788 }
Jiwen 'Steve' Caid6cb17f2017-05-08 16:15:35 -0700789
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800790 // Reset in memory objects related the the buffer.
791 buffers_[slot].mBufferProducer = nullptr;
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800792 buffers_[slot].mBufferState.detachProducer();
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700793 buffers_[slot].mFence = Fence::NO_FENCE;
794 buffers_[slot].mGraphicBuffer = nullptr;
795 buffers_[slot].mRequestBufferCalled = false;
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800796 return NO_ERROR;
Alex Vakulenkoe4eec202017-01-27 14:41:04 -0800797}
798
Jiwen 'Steve' Cai0f950842018-01-16 17:05:54 -0800799status_t BufferHubProducer::FreeAllBuffers() {
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -0800800 for (size_t slot = 0; slot < BufferHubQueue::kMaxQueueCapacity; slot++) {
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800801 // Reset in memory objects related the the buffer.
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800802 buffers_[slot].mBufferProducer = nullptr;
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700803 buffers_[slot].mBufferState.reset();
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800804 buffers_[slot].mFence = Fence::NO_FENCE;
Jiwen 'Steve' Cai42875352018-04-30 17:55:11 -0700805 buffers_[slot].mGraphicBuffer = nullptr;
806 buffers_[slot].mRequestBufferCalled = false;
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800807 }
Jiwen 'Steve' Cai005f45d2017-08-04 17:34:37 -0700808
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800809 auto status = queue_->FreeAllBuffers();
810 if (!status) {
811 ALOGE("BufferHubProducer::FreeAllBuffers: Failed to free all buffers on "
812 "the queue: %s",
813 status.GetErrorMessage().c_str());
814 }
Jiwen 'Steve' Cai005f45d2017-08-04 17:34:37 -0700815
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800816 if (queue_->capacity() != 0 || queue_->count() != 0) {
817 LOG_ALWAYS_FATAL("BufferHubProducer::FreeAllBuffers: Not all buffers are freed.");
818 }
Jiwen 'Steve' Cai005f45d2017-08-04 17:34:37 -0700819
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800820 return NO_ERROR;
Jiwen 'Steve' Cai005f45d2017-08-04 17:34:37 -0700821}
822
Jiwen 'Steve' Caic90a77f2018-01-14 15:42:29 -0800823status_t BufferHubProducer::exportToParcel(Parcel* parcel) {
824 status_t res = TakeAsParcelable(&pending_producer_parcelable_);
825 if (res != NO_ERROR) return res;
826
827 if (!pending_producer_parcelable_.IsValid()) {
828 ALOGE("BufferHubProducer::exportToParcel: Invalid parcelable object.");
829 return BAD_VALUE;
830 }
831
832 res = parcel->writeUint32(USE_BUFFER_HUB);
833 if (res != NO_ERROR) {
834 ALOGE("BufferHubProducer::exportToParcel: Cannot write magic, res=%d.", res);
835 return res;
836 }
837
838 return pending_producer_parcelable_.writeToParcel(parcel);
839}
840
841IBinder* BufferHubProducer::onAsBinder() {
842 ALOGE("BufferHubProducer::onAsBinder: BufferHubProducer should never be used as an Binder "
843 "object.");
844 return nullptr;
845}
846
Jiwen 'Steve' Cai9a6ddf72018-01-18 14:37:24 -0800847} // namespace android