Home | History | Annotate | Download | only in ui
      1 /*
      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 
     17 #include <poll.h>
     18 
     19 #include <android-base/unique_fd.h>
     20 #include <android/frameworks/bufferhub/1.0/IBufferHub.h>
     21 #include <log/log.h>
     22 #include <ui/BufferHubBuffer.h>
     23 #include <ui/BufferHubDefs.h>
     24 #include <utils/Trace.h>
     25 
     26 using ::android::base::unique_fd;
     27 using ::android::BufferHubDefs::isAnyClientAcquired;
     28 using ::android::BufferHubDefs::isAnyClientGained;
     29 using ::android::BufferHubDefs::isClientAcquired;
     30 using ::android::BufferHubDefs::isClientGained;
     31 using ::android::BufferHubDefs::isClientPosted;
     32 using ::android::BufferHubDefs::isClientReleased;
     33 using ::android::frameworks::bufferhub::V1_0::BufferHubStatus;
     34 using ::android::frameworks::bufferhub::V1_0::BufferTraits;
     35 using ::android::frameworks::bufferhub::V1_0::IBufferClient;
     36 using ::android::frameworks::bufferhub::V1_0::IBufferHub;
     37 using ::android::hardware::hidl_handle;
     38 using ::android::hardware::graphics::common::V1_2::HardwareBufferDescription;
     39 
     40 namespace android {
     41 
     42 std::unique_ptr<BufferHubBuffer> BufferHubBuffer::create(uint32_t width, uint32_t height,
     43                                                          uint32_t layerCount, uint32_t format,
     44                                                          uint64_t usage, size_t userMetadataSize) {
     45     auto buffer = std::unique_ptr<BufferHubBuffer>(
     46             new BufferHubBuffer(width, height, layerCount, format, usage, userMetadataSize));
     47     return buffer->isValid() ? std::move(buffer) : nullptr;
     48 }
     49 
     50 std::unique_ptr<BufferHubBuffer> BufferHubBuffer::import(const sp<NativeHandle>& token) {
     51     if (token == nullptr || token.get() == nullptr) {
     52         ALOGE("%s: token cannot be nullptr!", __FUNCTION__);
     53         return nullptr;
     54     }
     55 
     56     auto buffer = std::unique_ptr<BufferHubBuffer>(new BufferHubBuffer(token));
     57     return buffer->isValid() ? std::move(buffer) : nullptr;
     58 }
     59 
     60 BufferHubBuffer::BufferHubBuffer(uint32_t width, uint32_t height, uint32_t layerCount,
     61                                  uint32_t format, uint64_t usage, size_t userMetadataSize) {
     62     ATRACE_CALL();
     63     ALOGD("%s: width=%u height=%u layerCount=%u, format=%u "
     64           "usage=%" PRIx64 " mUserMetadataSize=%zu",
     65           __FUNCTION__, width, height, layerCount, format, usage, userMetadataSize);
     66 
     67     sp<IBufferHub> bufferhub = IBufferHub::getService();
     68     if (bufferhub.get() == nullptr) {
     69         ALOGE("%s: BufferHub service not found!", __FUNCTION__);
     70         return;
     71     }
     72 
     73     AHardwareBuffer_Desc aDesc = {width, height,         layerCount,   format,
     74                                   usage, /*stride=*/0UL, /*rfu0=*/0UL, /*rfu1=*/0ULL};
     75     HardwareBufferDescription desc;
     76     memcpy(&desc, &aDesc, sizeof(HardwareBufferDescription));
     77 
     78     BufferHubStatus ret;
     79     sp<IBufferClient> client;
     80     BufferTraits bufferTraits;
     81     IBufferHub::allocateBuffer_cb allocCb = [&](const auto& status, const auto& outClient,
     82                                                 const auto& outTraits) {
     83         ret = status;
     84         client = std::move(outClient);
     85         bufferTraits = std::move(outTraits);
     86     };
     87 
     88     if (!bufferhub->allocateBuffer(desc, static_cast<uint32_t>(userMetadataSize), allocCb).isOk()) {
     89         ALOGE("%s: allocateBuffer transaction failed!", __FUNCTION__);
     90         return;
     91     } else if (ret != BufferHubStatus::NO_ERROR) {
     92         ALOGE("%s: allocateBuffer failed with error %u.", __FUNCTION__, ret);
     93         return;
     94     } else if (client == nullptr) {
     95         ALOGE("%s: allocateBuffer got null BufferClient.", __FUNCTION__);
     96         return;
     97     }
     98 
     99     const int importRet = initWithBufferTraits(bufferTraits);
    100     if (importRet < 0) {
    101         ALOGE("%s: Failed to import buffer: %s", __FUNCTION__, strerror(-importRet));
    102         client->close();
    103     }
    104     mBufferClient = std::move(client);
    105 }
    106 
    107 BufferHubBuffer::BufferHubBuffer(const sp<NativeHandle>& token) {
    108     sp<IBufferHub> bufferhub = IBufferHub::getService();
    109     if (bufferhub.get() == nullptr) {
    110         ALOGE("%s: BufferHub service not found!", __FUNCTION__);
    111         return;
    112     }
    113 
    114     BufferHubStatus ret;
    115     sp<IBufferClient> client;
    116     BufferTraits bufferTraits;
    117     IBufferHub::importBuffer_cb importCb = [&](const auto& status, const auto& outClient,
    118                                                const auto& outTraits) {
    119         ret = status;
    120         client = std::move(outClient);
    121         bufferTraits = std::move(outTraits);
    122     };
    123 
    124     // hidl_handle(native_handle_t*) simply creates a raw pointer reference withouth ownership
    125     // transfer.
    126     if (!bufferhub->importBuffer(hidl_handle(token.get()->handle()), importCb).isOk()) {
    127         ALOGE("%s: importBuffer transaction failed!", __FUNCTION__);
    128         return;
    129     } else if (ret != BufferHubStatus::NO_ERROR) {
    130         ALOGE("%s: importBuffer failed with error %u.", __FUNCTION__, ret);
    131         return;
    132     } else if (client == nullptr) {
    133         ALOGE("%s: importBuffer got null BufferClient.", __FUNCTION__);
    134         return;
    135     }
    136 
    137     const int importRet = initWithBufferTraits(bufferTraits);
    138     if (importRet < 0) {
    139         ALOGE("%s: Failed to import buffer: %s", __FUNCTION__, strerror(-importRet));
    140         client->close();
    141     }
    142     mBufferClient = std::move(client);
    143 }
    144 
    145 BufferHubBuffer::~BufferHubBuffer() {
    146     // Close buffer client to avoid possible race condition: user could first duplicate and hold
    147     // token with the original buffer gone, and then try to import the token. The close function
    148     // will explicitly invalidate the token to avoid this.
    149     if (mBufferClient != nullptr) {
    150         if (!mBufferClient->close().isOk()) {
    151             ALOGE("%s: close BufferClient transaction failed!", __FUNCTION__);
    152         }
    153     }
    154 }
    155 
    156 int BufferHubBuffer::initWithBufferTraits(const BufferTraits& bufferTraits) {
    157     ATRACE_CALL();
    158 
    159     if (bufferTraits.bufferInfo.getNativeHandle() == nullptr) {
    160         ALOGE("%s: missing buffer info handle.", __FUNCTION__);
    161         return -EINVAL;
    162     }
    163 
    164     if (bufferTraits.bufferHandle.getNativeHandle() == nullptr) {
    165         ALOGE("%s: missing gralloc handle.", __FUNCTION__);
    166         return -EINVAL;
    167     }
    168 
    169     // Import fds. Dup fds because hidl_handle owns the fds.
    170     unique_fd ashmemFd(fcntl(bufferTraits.bufferInfo->data[0], F_DUPFD_CLOEXEC, 0));
    171     mMetadata = BufferHubMetadata::import(std::move(ashmemFd));
    172     if (!mMetadata.isValid()) {
    173         ALOGE("%s: Received an invalid metadata.", __FUNCTION__);
    174         return -EINVAL;
    175     }
    176 
    177     mEventFd = BufferHubEventFd(fcntl(bufferTraits.bufferInfo->data[1], F_DUPFD_CLOEXEC, 0));
    178     if (!mEventFd.isValid()) {
    179         ALOGE("%s: Received ad invalid event fd.", __FUNCTION__);
    180         return -EINVAL;
    181     }
    182 
    183     int bufferId = bufferTraits.bufferInfo->data[2];
    184     if (bufferId < 0) {
    185         ALOGE("%s: Received an invalid (negative) id.", __FUNCTION__);
    186         return -EINVAL;
    187     }
    188 
    189     uint32_t clientBitMask;
    190     memcpy(&clientBitMask, &bufferTraits.bufferInfo->data[3], sizeof(clientBitMask));
    191     if (clientBitMask == 0U) {
    192         ALOGE("%s: Received an invalid client state mask.", __FUNCTION__);
    193         return -EINVAL;
    194     }
    195 
    196     uint32_t userMetadataSize;
    197     memcpy(&userMetadataSize, &bufferTraits.bufferInfo->data[4], sizeof(userMetadataSize));
    198     if (mMetadata.userMetadataSize() != userMetadataSize) {
    199         ALOGE("%s: user metadata size not match: expected %u, actual %zu.", __FUNCTION__,
    200               userMetadataSize, mMetadata.userMetadataSize());
    201         return -EINVAL;
    202     }
    203 
    204     size_t metadataSize = static_cast<size_t>(mMetadata.metadataSize());
    205     if (metadataSize < BufferHubDefs::kMetadataHeaderSize) {
    206         ALOGE("%s: metadata too small: %zu", __FUNCTION__, metadataSize);
    207         return -EINVAL;
    208     }
    209 
    210     // Populate shortcuts to the atomics in metadata.
    211     auto metadataHeader = mMetadata.metadataHeader();
    212     mBufferState = &metadataHeader->bufferState;
    213     mFenceState = &metadataHeader->fenceState;
    214     mActiveClientsBitMask = &metadataHeader->activeClientsBitMask;
    215     // The C++ standard recommends (but does not require) that lock-free atomic operations are
    216     // also address-free, that is, suitable for communication between processes using shared
    217     // memory.
    218     LOG_ALWAYS_FATAL_IF(!std::atomic_is_lock_free(mBufferState) ||
    219                                 !std::atomic_is_lock_free(mFenceState) ||
    220                                 !std::atomic_is_lock_free(mActiveClientsBitMask),
    221                         "Atomic variables in ashmen are not lock free.");
    222 
    223     // Import the buffer: We only need to hold on the native_handle_t here so that
    224     // GraphicBuffer instance can be created in future.
    225     mBufferHandle = std::move(bufferTraits.bufferHandle);
    226     memcpy(&mBufferDesc, &bufferTraits.bufferDesc, sizeof(AHardwareBuffer_Desc));
    227 
    228     mId = bufferId;
    229     mClientStateMask = clientBitMask;
    230 
    231     // TODO(b/112012161) Set up shared fences.
    232     ALOGD("%s: id=%d, mBufferState=%" PRIx32 ".", __FUNCTION__, mId,
    233           mBufferState->load(std::memory_order_acquire));
    234     return 0;
    235 }
    236 
    237 int BufferHubBuffer::gain() {
    238     uint32_t currentBufferState = mBufferState->load(std::memory_order_acquire);
    239     if (isClientGained(currentBufferState, mClientStateMask)) {
    240         ALOGV("%s: Buffer is already gained by this client %" PRIx32 ".", __FUNCTION__,
    241               mClientStateMask);
    242         return 0;
    243     }
    244     do {
    245         if (isAnyClientGained(currentBufferState & (~mClientStateMask)) ||
    246             isAnyClientAcquired(currentBufferState)) {
    247             ALOGE("%s: Buffer is in use, id=%d mClientStateMask=%" PRIx32 " state=%" PRIx32 ".",
    248                   __FUNCTION__, mId, mClientStateMask, currentBufferState);
    249             return -EBUSY;
    250         }
    251         // Change the buffer state to gained state, whose value happens to be the same as
    252         // mClientStateMask.
    253     } while (!mBufferState->compare_exchange_weak(currentBufferState, mClientStateMask,
    254                                                   std::memory_order_acq_rel,
    255                                                   std::memory_order_acquire));
    256     // TODO(b/119837586): Update fence state and return GPU fence.
    257     return 0;
    258 }
    259 
    260 int BufferHubBuffer::post() {
    261     uint32_t currentBufferState = mBufferState->load(std::memory_order_acquire);
    262     uint32_t updatedBufferState = (~mClientStateMask) & BufferHubDefs::kHighBitsMask;
    263     do {
    264         if (!isClientGained(currentBufferState, mClientStateMask)) {
    265             ALOGE("%s: Cannot post a buffer that is not gained by this client. buffer_id=%d "
    266                   "mClientStateMask=%" PRIx32 " state=%" PRIx32 ".",
    267                   __FUNCTION__, mId, mClientStateMask, currentBufferState);
    268             return -EBUSY;
    269         }
    270         // Set the producer client buffer state to released, other clients' buffer state to posted.
    271         // Post to all existing and non-existing clients.
    272     } while (!mBufferState->compare_exchange_weak(currentBufferState, updatedBufferState,
    273                                                   std::memory_order_acq_rel,
    274                                                   std::memory_order_acquire));
    275     // TODO(b/119837586): Update fence state and return GPU fence if needed.
    276     return 0;
    277 }
    278 
    279 int BufferHubBuffer::acquire() {
    280     uint32_t currentBufferState = mBufferState->load(std::memory_order_acquire);
    281     if (isClientAcquired(currentBufferState, mClientStateMask)) {
    282         ALOGV("%s: Buffer is already acquired by this client %" PRIx32 ".", __FUNCTION__,
    283               mClientStateMask);
    284         return 0;
    285     }
    286     uint32_t updatedBufferState = 0U;
    287     do {
    288         if (!isClientPosted(currentBufferState, mClientStateMask)) {
    289             ALOGE("%s: Cannot acquire a buffer that is not in posted state. buffer_id=%d "
    290                   "mClientStateMask=%" PRIx32 " state=%" PRIx32 ".",
    291                   __FUNCTION__, mId, mClientStateMask, currentBufferState);
    292             return -EBUSY;
    293         }
    294         // Change the buffer state for this consumer from posted to acquired.
    295         updatedBufferState = currentBufferState ^ mClientStateMask;
    296     } while (!mBufferState->compare_exchange_weak(currentBufferState, updatedBufferState,
    297                                                   std::memory_order_acq_rel,
    298                                                   std::memory_order_acquire));
    299     // TODO(b/119837586): Update fence state and return GPU fence.
    300     return 0;
    301 }
    302 
    303 int BufferHubBuffer::release() {
    304     uint32_t currentBufferState = mBufferState->load(std::memory_order_acquire);
    305     if (isClientReleased(currentBufferState, mClientStateMask)) {
    306         ALOGV("%s: Buffer is already released by this client %" PRIx32 ".", __FUNCTION__,
    307               mClientStateMask);
    308         return 0;
    309     }
    310     uint32_t updatedBufferState = 0U;
    311     do {
    312         updatedBufferState = currentBufferState & (~mClientStateMask);
    313     } while (!mBufferState->compare_exchange_weak(currentBufferState, updatedBufferState,
    314                                                   std::memory_order_acq_rel,
    315                                                   std::memory_order_acquire));
    316     // TODO(b/119837586): Update fence state and return GPU fence if needed.
    317     return 0;
    318 }
    319 
    320 bool BufferHubBuffer::isReleased() const {
    321     return (mBufferState->load(std::memory_order_acquire) &
    322             mActiveClientsBitMask->load(std::memory_order_acquire)) == 0;
    323 }
    324 
    325 bool BufferHubBuffer::isValid() const {
    326     return mBufferHandle.getNativeHandle() != nullptr && mId >= 0 && mClientStateMask != 0U &&
    327             mEventFd.get() >= 0 && mMetadata.isValid() && mBufferClient != nullptr;
    328 }
    329 
    330 sp<NativeHandle> BufferHubBuffer::duplicate() {
    331     if (mBufferClient == nullptr) {
    332         ALOGE("%s: missing BufferClient!", __FUNCTION__);
    333         return nullptr;
    334     }
    335 
    336     hidl_handle token;
    337     BufferHubStatus ret;
    338     IBufferClient::duplicate_cb dupCb = [&](const auto& outToken, const auto& status) {
    339         token = std::move(outToken);
    340         ret = status;
    341     };
    342 
    343     if (!mBufferClient->duplicate(dupCb).isOk()) {
    344         ALOGE("%s: duplicate transaction failed!", __FUNCTION__);
    345         return nullptr;
    346     } else if (ret != BufferHubStatus::NO_ERROR) {
    347         ALOGE("%s: duplicate failed with error %u.", __FUNCTION__, ret);
    348         return nullptr;
    349     } else if (token.getNativeHandle() == nullptr) {
    350         ALOGE("%s: duplicate got null token.", __FUNCTION__);
    351         return nullptr;
    352     }
    353 
    354     return NativeHandle::create(native_handle_clone(token.getNativeHandle()), /*ownsHandle=*/true);
    355 }
    356 
    357 } // namespace android
    358