blob: 1dfc1e9e88b08cd23710f9cb0cb77456d6948afa [file] [log] [blame]
Jiwen 'Steve' Cai8f51ec62018-08-07 21:50:51 -07001/*
2 * Copyright (C) 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
Tianyu Jiang83e514e2019-05-15 15:05:30 -070017#define LOG_TAG "BufferHubBuffer"
Jiwen 'Steve' Cai0728fa92018-04-24 19:03:14 -070018#include <poll.h>
19
Fan Xu069e8382018-11-16 16:28:08 -080020#include <android-base/unique_fd.h>
Fan Xu021776e2018-12-05 13:34:48 -080021#include <android/frameworks/bufferhub/1.0/IBufferHub.h>
22#include <log/log.h>
Fan Xu069e8382018-11-16 16:28:08 -080023#include <ui/BufferHubBuffer.h>
Fan Xucfbe0742018-11-21 15:03:32 -080024#include <ui/BufferHubDefs.h>
Fan Xu021776e2018-12-05 13:34:48 -080025#include <utils/Trace.h>
Fan Xu069e8382018-11-16 16:28:08 -080026
Fan Xu021776e2018-12-05 13:34:48 -080027using ::android::base::unique_fd;
Tianyu Jiang727ede42019-02-01 11:44:51 -080028using ::android::BufferHubDefs::isAnyClientAcquired;
29using ::android::BufferHubDefs::isAnyClientGained;
30using ::android::BufferHubDefs::isClientAcquired;
31using ::android::BufferHubDefs::isClientGained;
32using ::android::BufferHubDefs::isClientPosted;
33using ::android::BufferHubDefs::isClientReleased;
Fan Xu021776e2018-12-05 13:34:48 -080034using ::android::frameworks::bufferhub::V1_0::BufferHubStatus;
35using ::android::frameworks::bufferhub::V1_0::BufferTraits;
36using ::android::frameworks::bufferhub::V1_0::IBufferClient;
37using ::android::frameworks::bufferhub::V1_0::IBufferHub;
38using ::android::hardware::hidl_handle;
39using ::android::hardware::graphics::common::V1_2::HardwareBufferDescription;
Jiwen 'Steve' Caia8049a22018-03-28 15:14:02 -070040
41namespace android {
Jiwen 'Steve' Caia8049a22018-03-28 15:14:02 -070042
Tianyu Jiang727ede42019-02-01 11:44:51 -080043std::unique_ptr<BufferHubBuffer> BufferHubBuffer::create(uint32_t width, uint32_t height,
Fan Xu021776e2018-12-05 13:34:48 -080044 uint32_t layerCount, uint32_t format,
45 uint64_t usage, size_t userMetadataSize) {
46 auto buffer = std::unique_ptr<BufferHubBuffer>(
47 new BufferHubBuffer(width, height, layerCount, format, usage, userMetadataSize));
Tianyu Jiang727ede42019-02-01 11:44:51 -080048 return buffer->isValid() ? std::move(buffer) : nullptr;
Jiwen 'Steve' Cai088b3b62018-10-03 18:49:07 -070049}
50
Tianyu Jiang8a7e7652019-02-11 13:41:20 -080051std::unique_ptr<BufferHubBuffer> BufferHubBuffer::import(const sp<NativeHandle>& token) {
52 if (token == nullptr || token.get() == nullptr) {
Fan Xu021776e2018-12-05 13:34:48 -080053 ALOGE("%s: token cannot be nullptr!", __FUNCTION__);
54 return nullptr;
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -070055 }
Fan Xu021776e2018-12-05 13:34:48 -080056
57 auto buffer = std::unique_ptr<BufferHubBuffer>(new BufferHubBuffer(token));
Tianyu Jiang727ede42019-02-01 11:44:51 -080058 return buffer->isValid() ? std::move(buffer) : nullptr;
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -070059}
60
61BufferHubBuffer::BufferHubBuffer(uint32_t width, uint32_t height, uint32_t layerCount,
Fan Xu021776e2018-12-05 13:34:48 -080062 uint32_t format, uint64_t usage, size_t userMetadataSize) {
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -070063 ATRACE_CALL();
Fan Xu021776e2018-12-05 13:34:48 -080064 ALOGD("%s: width=%u height=%u layerCount=%u, format=%u "
65 "usage=%" PRIx64 " mUserMetadataSize=%zu",
66 __FUNCTION__, width, height, layerCount, format, usage, userMetadataSize);
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -070067
Fan Xu021776e2018-12-05 13:34:48 -080068 sp<IBufferHub> bufferhub = IBufferHub::getService();
69 if (bufferhub.get() == nullptr) {
70 ALOGE("%s: BufferHub service not found!", __FUNCTION__);
71 return;
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -070072 }
73
Fan Xu021776e2018-12-05 13:34:48 -080074 AHardwareBuffer_Desc aDesc = {width, height, layerCount, format,
75 usage, /*stride=*/0UL, /*rfu0=*/0UL, /*rfu1=*/0ULL};
76 HardwareBufferDescription desc;
77 memcpy(&desc, &aDesc, sizeof(HardwareBufferDescription));
78
79 BufferHubStatus ret;
80 sp<IBufferClient> client;
81 BufferTraits bufferTraits;
Tianyu Jiangf377a762019-02-13 13:46:42 -080082 IBufferHub::allocateBuffer_cb allocCb = [&](const auto& status, const auto& outClient,
83 const auto& outTraits) {
Fan Xu021776e2018-12-05 13:34:48 -080084 ret = status;
85 client = std::move(outClient);
Jiwen 'Steve' Caidfe0f4c2019-01-15 21:49:07 -080086 bufferTraits = std::move(outTraits);
Fan Xu021776e2018-12-05 13:34:48 -080087 };
88
Tianyu Jiangf377a762019-02-13 13:46:42 -080089 if (!bufferhub->allocateBuffer(desc, static_cast<uint32_t>(userMetadataSize), allocCb).isOk()) {
Fan Xu021776e2018-12-05 13:34:48 -080090 ALOGE("%s: allocateBuffer transaction failed!", __FUNCTION__);
91 return;
92 } else if (ret != BufferHubStatus::NO_ERROR) {
93 ALOGE("%s: allocateBuffer failed with error %u.", __FUNCTION__, ret);
94 return;
95 } else if (client == nullptr) {
96 ALOGE("%s: allocateBuffer got null BufferClient.", __FUNCTION__);
97 return;
98 }
99
100 const int importRet = initWithBufferTraits(bufferTraits);
101 if (importRet < 0) {
102 ALOGE("%s: Failed to import buffer: %s", __FUNCTION__, strerror(-importRet));
103 client->close();
104 }
105 mBufferClient = std::move(client);
106}
107
Tianyu Jiang8a7e7652019-02-11 13:41:20 -0800108BufferHubBuffer::BufferHubBuffer(const sp<NativeHandle>& token) {
Fan Xu021776e2018-12-05 13:34:48 -0800109 sp<IBufferHub> bufferhub = IBufferHub::getService();
110 if (bufferhub.get() == nullptr) {
111 ALOGE("%s: BufferHub service not found!", __FUNCTION__);
112 return;
113 }
114
115 BufferHubStatus ret;
116 sp<IBufferClient> client;
117 BufferTraits bufferTraits;
Tianyu Jiangf377a762019-02-13 13:46:42 -0800118 IBufferHub::importBuffer_cb importCb = [&](const auto& status, const auto& outClient,
119 const auto& outTraits) {
Fan Xu021776e2018-12-05 13:34:48 -0800120 ret = status;
121 client = std::move(outClient);
Jiwen 'Steve' Caidfe0f4c2019-01-15 21:49:07 -0800122 bufferTraits = std::move(outTraits);
Fan Xu021776e2018-12-05 13:34:48 -0800123 };
124
125 // hidl_handle(native_handle_t*) simply creates a raw pointer reference withouth ownership
126 // transfer.
Tianyu Jiangf377a762019-02-13 13:46:42 -0800127 if (!bufferhub->importBuffer(hidl_handle(token.get()->handle()), importCb).isOk()) {
Fan Xu021776e2018-12-05 13:34:48 -0800128 ALOGE("%s: importBuffer transaction failed!", __FUNCTION__);
129 return;
130 } else if (ret != BufferHubStatus::NO_ERROR) {
131 ALOGE("%s: importBuffer failed with error %u.", __FUNCTION__, ret);
132 return;
133 } else if (client == nullptr) {
134 ALOGE("%s: importBuffer got null BufferClient.", __FUNCTION__);
135 return;
136 }
137
138 const int importRet = initWithBufferTraits(bufferTraits);
139 if (importRet < 0) {
140 ALOGE("%s: Failed to import buffer: %s", __FUNCTION__, strerror(-importRet));
141 client->close();
142 }
143 mBufferClient = std::move(client);
144}
145
146BufferHubBuffer::~BufferHubBuffer() {
147 // Close buffer client to avoid possible race condition: user could first duplicate and hold
148 // token with the original buffer gone, and then try to import the token. The close function
149 // will explicitly invalidate the token to avoid this.
150 if (mBufferClient != nullptr) {
151 if (!mBufferClient->close().isOk()) {
152 ALOGE("%s: close BufferClient transaction failed!", __FUNCTION__);
153 }
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -0700154 }
155}
156
Fan Xu021776e2018-12-05 13:34:48 -0800157int BufferHubBuffer::initWithBufferTraits(const BufferTraits& bufferTraits) {
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -0700158 ATRACE_CALL();
159
Fan Xu021776e2018-12-05 13:34:48 -0800160 if (bufferTraits.bufferInfo.getNativeHandle() == nullptr) {
161 ALOGE("%s: missing buffer info handle.", __FUNCTION__);
162 return -EINVAL;
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -0700163 }
164
Fan Xu021776e2018-12-05 13:34:48 -0800165 if (bufferTraits.bufferHandle.getNativeHandle() == nullptr) {
166 ALOGE("%s: missing gralloc handle.", __FUNCTION__);
167 return -EINVAL;
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -0700168 }
169
Tianyu Jiang6f54dca2019-01-29 12:46:35 -0800170 // Import fds. Dup fds because hidl_handle owns the fds.
171 unique_fd ashmemFd(fcntl(bufferTraits.bufferInfo->data[0], F_DUPFD_CLOEXEC, 0));
Tianyu Jiang727ede42019-02-01 11:44:51 -0800172 mMetadata = BufferHubMetadata::import(std::move(ashmemFd));
173 if (!mMetadata.isValid()) {
Tianyu Jiang6f54dca2019-01-29 12:46:35 -0800174 ALOGE("%s: Received an invalid metadata.", __FUNCTION__);
175 return -EINVAL;
176 }
177
178 mEventFd = BufferHubEventFd(fcntl(bufferTraits.bufferInfo->data[1], F_DUPFD_CLOEXEC, 0));
179 if (!mEventFd.isValid()) {
180 ALOGE("%s: Received ad invalid event fd.", __FUNCTION__);
181 return -EINVAL;
182 }
183
Fan Xu5cf47bc2019-01-15 15:02:15 -0800184 int bufferId = bufferTraits.bufferInfo->data[2];
Fan Xu021776e2018-12-05 13:34:48 -0800185 if (bufferId < 0) {
Tianyu Jiang6f54dca2019-01-29 12:46:35 -0800186 ALOGE("%s: Received an invalid (negative) id.", __FUNCTION__);
Fan Xu021776e2018-12-05 13:34:48 -0800187 return -EINVAL;
188 }
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -0700189
Fan Xu021776e2018-12-05 13:34:48 -0800190 uint32_t clientBitMask;
Fan Xu5cf47bc2019-01-15 15:02:15 -0800191 memcpy(&clientBitMask, &bufferTraits.bufferInfo->data[3], sizeof(clientBitMask));
Fan Xu021776e2018-12-05 13:34:48 -0800192 if (clientBitMask == 0U) {
Tianyu Jiang6f54dca2019-01-29 12:46:35 -0800193 ALOGE("%s: Received an invalid client state mask.", __FUNCTION__);
Fan Xu021776e2018-12-05 13:34:48 -0800194 return -EINVAL;
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -0700195 }
196
Fan Xu021776e2018-12-05 13:34:48 -0800197 uint32_t userMetadataSize;
Fan Xu5cf47bc2019-01-15 15:02:15 -0800198 memcpy(&userMetadataSize, &bufferTraits.bufferInfo->data[4], sizeof(userMetadataSize));
Tianyu Jiang727ede42019-02-01 11:44:51 -0800199 if (mMetadata.userMetadataSize() != userMetadataSize) {
Fan Xu021776e2018-12-05 13:34:48 -0800200 ALOGE("%s: user metadata size not match: expected %u, actual %zu.", __FUNCTION__,
Tianyu Jiang727ede42019-02-01 11:44:51 -0800201 userMetadataSize, mMetadata.userMetadataSize());
Fan Xu021776e2018-12-05 13:34:48 -0800202 return -EINVAL;
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -0700203 }
204
Tianyu Jiang727ede42019-02-01 11:44:51 -0800205 size_t metadataSize = static_cast<size_t>(mMetadata.metadataSize());
Fan Xucfbe0742018-11-21 15:03:32 -0800206 if (metadataSize < BufferHubDefs::kMetadataHeaderSize) {
Tianyu Jiangdf9d91d2018-12-17 10:48:54 -0800207 ALOGE("%s: metadata too small: %zu", __FUNCTION__, metadataSize);
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -0700208 return -EINVAL;
209 }
210
Tianyu Jiangb08b7222018-11-16 17:55:26 -0800211 // Populate shortcuts to the atomics in metadata.
Tianyu Jiangc9d8d582019-02-10 22:05:34 -0800212 auto metadataHeader = mMetadata.metadataHeader();
Tianyu Jiangf377a762019-02-13 13:46:42 -0800213 mBufferState = &metadataHeader->bufferState;
214 mFenceState = &metadataHeader->fenceState;
215 mActiveClientsBitMask = &metadataHeader->activeClientsBitMask;
Tianyu Jiang2ceb3202018-12-17 12:58:34 -0800216 // The C++ standard recommends (but does not require) that lock-free atomic operations are
217 // also address-free, that is, suitable for communication between processes using shared
218 // memory.
Tianyu Jiang1543d882019-02-04 10:53:53 -0800219 LOG_ALWAYS_FATAL_IF(!std::atomic_is_lock_free(mBufferState) ||
220 !std::atomic_is_lock_free(mFenceState) ||
221 !std::atomic_is_lock_free(mActiveClientsBitMask),
Tianyu Jiang2ceb3202018-12-17 12:58:34 -0800222 "Atomic variables in ashmen are not lock free.");
Tianyu Jiangb08b7222018-11-16 17:55:26 -0800223
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -0700224 // Import the buffer: We only need to hold on the native_handle_t here so that
225 // GraphicBuffer instance can be created in future.
Fan Xu021776e2018-12-05 13:34:48 -0800226 mBufferHandle = std::move(bufferTraits.bufferHandle);
227 memcpy(&mBufferDesc, &bufferTraits.bufferDesc, sizeof(AHardwareBuffer_Desc));
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -0700228
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -0700229 mId = bufferId;
Fan Xu021776e2018-12-05 13:34:48 -0800230 mClientStateMask = clientBitMask;
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -0700231
232 // TODO(b/112012161) Set up shared fences.
Tianyu Jiangf377a762019-02-13 13:46:42 -0800233 ALOGD("%s: id=%d, mBufferState=%" PRIx32 ".", __FUNCTION__, mId,
Tianyu Jiang1543d882019-02-04 10:53:53 -0800234 mBufferState->load(std::memory_order_acquire));
Tianyu Jiangb08b7222018-11-16 17:55:26 -0800235 return 0;
236}
237
Tianyu Jiang727ede42019-02-01 11:44:51 -0800238int BufferHubBuffer::gain() {
Tianyu Jiang1543d882019-02-04 10:53:53 -0800239 uint32_t currentBufferState = mBufferState->load(std::memory_order_acquire);
Tianyu Jiang727ede42019-02-01 11:44:51 -0800240 if (isClientGained(currentBufferState, mClientStateMask)) {
Tianyu Jianga99f9112018-12-13 18:23:07 -0800241 ALOGV("%s: Buffer is already gained by this client %" PRIx32 ".", __FUNCTION__,
Tianyu Jiangb08b7222018-11-16 17:55:26 -0800242 mClientStateMask);
243 return 0;
244 }
245 do {
Tianyu Jiang727ede42019-02-01 11:44:51 -0800246 if (isAnyClientGained(currentBufferState & (~mClientStateMask)) ||
247 isAnyClientAcquired(currentBufferState)) {
Tianyu Jianga99f9112018-12-13 18:23:07 -0800248 ALOGE("%s: Buffer is in use, id=%d mClientStateMask=%" PRIx32 " state=%" PRIx32 ".",
Tianyu Jiang1543d882019-02-04 10:53:53 -0800249 __FUNCTION__, mId, mClientStateMask, currentBufferState);
Tianyu Jiangb08b7222018-11-16 17:55:26 -0800250 return -EBUSY;
251 }
252 // Change the buffer state to gained state, whose value happens to be the same as
253 // mClientStateMask.
Tianyu Jiang1543d882019-02-04 10:53:53 -0800254 } while (!mBufferState->compare_exchange_weak(currentBufferState, mClientStateMask,
255 std::memory_order_acq_rel,
256 std::memory_order_acquire));
Tianyu Jiangb08b7222018-11-16 17:55:26 -0800257 // TODO(b/119837586): Update fence state and return GPU fence.
258 return 0;
259}
260
Tianyu Jiang727ede42019-02-01 11:44:51 -0800261int BufferHubBuffer::post() {
Tianyu Jiang1543d882019-02-04 10:53:53 -0800262 uint32_t currentBufferState = mBufferState->load(std::memory_order_acquire);
263 uint32_t updatedBufferState = (~mClientStateMask) & BufferHubDefs::kHighBitsMask;
Tianyu Jiangb08b7222018-11-16 17:55:26 -0800264 do {
Tianyu Jiang727ede42019-02-01 11:44:51 -0800265 if (!isClientGained(currentBufferState, mClientStateMask)) {
Tianyu Jiangb08b7222018-11-16 17:55:26 -0800266 ALOGE("%s: Cannot post a buffer that is not gained by this client. buffer_id=%d "
Tianyu Jianga99f9112018-12-13 18:23:07 -0800267 "mClientStateMask=%" PRIx32 " state=%" PRIx32 ".",
Tianyu Jiang1543d882019-02-04 10:53:53 -0800268 __FUNCTION__, mId, mClientStateMask, currentBufferState);
Tianyu Jiangb08b7222018-11-16 17:55:26 -0800269 return -EBUSY;
270 }
271 // Set the producer client buffer state to released, other clients' buffer state to posted.
Tianyu Jiangec97b762019-01-07 17:14:02 -0800272 // Post to all existing and non-existing clients.
Tianyu Jiang1543d882019-02-04 10:53:53 -0800273 } while (!mBufferState->compare_exchange_weak(currentBufferState, updatedBufferState,
274 std::memory_order_acq_rel,
275 std::memory_order_acquire));
Tianyu Jiangb08b7222018-11-16 17:55:26 -0800276 // TODO(b/119837586): Update fence state and return GPU fence if needed.
277 return 0;
278}
279
Tianyu Jiang727ede42019-02-01 11:44:51 -0800280int BufferHubBuffer::acquire() {
Tianyu Jiang1543d882019-02-04 10:53:53 -0800281 uint32_t currentBufferState = mBufferState->load(std::memory_order_acquire);
Tianyu Jiang727ede42019-02-01 11:44:51 -0800282 if (isClientAcquired(currentBufferState, mClientStateMask)) {
Tianyu Jianga99f9112018-12-13 18:23:07 -0800283 ALOGV("%s: Buffer is already acquired by this client %" PRIx32 ".", __FUNCTION__,
Tianyu Jiangb08b7222018-11-16 17:55:26 -0800284 mClientStateMask);
285 return 0;
286 }
Tianyu Jiang1543d882019-02-04 10:53:53 -0800287 uint32_t updatedBufferState = 0U;
Tianyu Jiangb08b7222018-11-16 17:55:26 -0800288 do {
Tianyu Jiang727ede42019-02-01 11:44:51 -0800289 if (!isClientPosted(currentBufferState, mClientStateMask)) {
Tianyu Jiangb08b7222018-11-16 17:55:26 -0800290 ALOGE("%s: Cannot acquire a buffer that is not in posted state. buffer_id=%d "
Tianyu Jianga99f9112018-12-13 18:23:07 -0800291 "mClientStateMask=%" PRIx32 " state=%" PRIx32 ".",
Tianyu Jiang1543d882019-02-04 10:53:53 -0800292 __FUNCTION__, mId, mClientStateMask, currentBufferState);
Tianyu Jiangb08b7222018-11-16 17:55:26 -0800293 return -EBUSY;
294 }
295 // Change the buffer state for this consumer from posted to acquired.
Tianyu Jiang1543d882019-02-04 10:53:53 -0800296 updatedBufferState = currentBufferState ^ mClientStateMask;
297 } while (!mBufferState->compare_exchange_weak(currentBufferState, updatedBufferState,
298 std::memory_order_acq_rel,
299 std::memory_order_acquire));
Tianyu Jiangb08b7222018-11-16 17:55:26 -0800300 // TODO(b/119837586): Update fence state and return GPU fence.
301 return 0;
302}
303
Tianyu Jiang727ede42019-02-01 11:44:51 -0800304int BufferHubBuffer::release() {
Tianyu Jiang1543d882019-02-04 10:53:53 -0800305 uint32_t currentBufferState = mBufferState->load(std::memory_order_acquire);
Tianyu Jiang727ede42019-02-01 11:44:51 -0800306 if (isClientReleased(currentBufferState, mClientStateMask)) {
Tianyu Jianga99f9112018-12-13 18:23:07 -0800307 ALOGV("%s: Buffer is already released by this client %" PRIx32 ".", __FUNCTION__,
Tianyu Jiangb08b7222018-11-16 17:55:26 -0800308 mClientStateMask);
309 return 0;
310 }
Tianyu Jiang1543d882019-02-04 10:53:53 -0800311 uint32_t updatedBufferState = 0U;
Tianyu Jiangb08b7222018-11-16 17:55:26 -0800312 do {
Tianyu Jiang1543d882019-02-04 10:53:53 -0800313 updatedBufferState = currentBufferState & (~mClientStateMask);
314 } while (!mBufferState->compare_exchange_weak(currentBufferState, updatedBufferState,
315 std::memory_order_acq_rel,
316 std::memory_order_acquire));
Tianyu Jiangb08b7222018-11-16 17:55:26 -0800317 // TODO(b/119837586): Update fence state and return GPU fence if needed.
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -0700318 return 0;
319}
320
Tianyu Jiang727ede42019-02-01 11:44:51 -0800321bool BufferHubBuffer::isReleased() const {
Tianyu Jiang1543d882019-02-04 10:53:53 -0800322 return (mBufferState->load(std::memory_order_acquire) &
323 mActiveClientsBitMask->load(std::memory_order_acquire)) == 0;
Tianyu Jianga8df5f32019-01-14 18:42:12 -0800324}
325
Tianyu Jiang727ede42019-02-01 11:44:51 -0800326bool BufferHubBuffer::isValid() const {
Fan Xu021776e2018-12-05 13:34:48 -0800327 return mBufferHandle.getNativeHandle() != nullptr && mId >= 0 && mClientStateMask != 0U &&
Tianyu Jiang727ede42019-02-01 11:44:51 -0800328 mEventFd.get() >= 0 && mMetadata.isValid() && mBufferClient != nullptr;
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -0700329}
330
Tianyu Jiang8a7e7652019-02-11 13:41:20 -0800331sp<NativeHandle> BufferHubBuffer::duplicate() {
Fan Xu021776e2018-12-05 13:34:48 -0800332 if (mBufferClient == nullptr) {
333 ALOGE("%s: missing BufferClient!", __FUNCTION__);
334 return nullptr;
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -0700335 }
Fan Xu021776e2018-12-05 13:34:48 -0800336
337 hidl_handle token;
338 BufferHubStatus ret;
Tianyu Jiangf377a762019-02-13 13:46:42 -0800339 IBufferClient::duplicate_cb dupCb = [&](const auto& outToken, const auto& status) {
Fan Xu021776e2018-12-05 13:34:48 -0800340 token = std::move(outToken);
341 ret = status;
342 };
343
Tianyu Jiangf377a762019-02-13 13:46:42 -0800344 if (!mBufferClient->duplicate(dupCb).isOk()) {
Fan Xu021776e2018-12-05 13:34:48 -0800345 ALOGE("%s: duplicate transaction failed!", __FUNCTION__);
346 return nullptr;
347 } else if (ret != BufferHubStatus::NO_ERROR) {
348 ALOGE("%s: duplicate failed with error %u.", __FUNCTION__, ret);
349 return nullptr;
350 } else if (token.getNativeHandle() == nullptr) {
351 ALOGE("%s: duplicate got null token.", __FUNCTION__);
352 return nullptr;
353 }
354
Tianyu Jiang8a7e7652019-02-11 13:41:20 -0800355 return NativeHandle::create(native_handle_clone(token.getNativeHandle()), /*ownsHandle=*/true);
Jiwen 'Steve' Caia8049a22018-03-28 15:14:02 -0700356}
357
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -0700358} // namespace android