blob: 05bc7ddfbe934987ae1c2832840f86ecf90feaa5 [file] [log] [blame]
Jiwen 'Steve' Cai4e566ed2018-09-19 22:41:42 -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
17#include <errno.h>
18#include <sys/mman.h>
Fan Xucfbe0742018-11-21 15:03:32 -080019#include <limits>
Jiwen 'Steve' Cai4e566ed2018-09-19 22:41:42 -070020
21#include <cutils/ashmem.h>
22#include <log/log.h>
Jiwen 'Steve' Cai8f51ec62018-08-07 21:50:51 -070023#include <ui/BufferHubMetadata.h>
Jiwen 'Steve' Cai4e566ed2018-09-19 22:41:42 -070024
25namespace android {
Jiwen 'Steve' Cai4e566ed2018-09-19 22:41:42 -070026
27namespace {
28
29static const int kAshmemProt = PROT_READ | PROT_WRITE;
30
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -070031} // namespace
Jiwen 'Steve' Cai4e566ed2018-09-19 22:41:42 -070032
Fan Xucfbe0742018-11-21 15:03:32 -080033using BufferHubDefs::kMetadataHeaderSize;
34using BufferHubDefs::MetadataHeader;
Jiwen 'Steve' Cai4e566ed2018-09-19 22:41:42 -070035
36/* static */
Tianyu Jiang727ede42019-02-01 11:44:51 -080037BufferHubMetadata BufferHubMetadata::create(size_t userMetadataSize) {
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -070038 // The size the of metadata buffer is used as the "width" parameter during allocation. Thus it
39 // cannot overflow uint32_t.
40 if (userMetadataSize >= (std::numeric_limits<uint32_t>::max() - kMetadataHeaderSize)) {
41 ALOGE("BufferHubMetadata::Create: metadata size too big: %zu.", userMetadataSize);
42 return {};
43 }
Jiwen 'Steve' Cai4e566ed2018-09-19 22:41:42 -070044
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -070045 const size_t metadataSize = userMetadataSize + kMetadataHeaderSize;
46 int fd = ashmem_create_region(/*name=*/"BufferHubMetadata", metadataSize);
47 if (fd < 0) {
48 ALOGE("BufferHubMetadata::Create: failed to create ashmem region.");
49 return {};
50 }
Jiwen 'Steve' Cai4e566ed2018-09-19 22:41:42 -070051
Fan Xu069e8382018-11-16 16:28:08 -080052 // Hand over the ownership of the fd to a unique_fd immediately after the successful
53 // return of ashmem_create_region. The ashmemFd is going to own the fd and to prevent fd
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -070054 // leaks during error handling.
Fan Xu069e8382018-11-16 16:28:08 -080055 unique_fd ashmemFd{fd};
Jiwen 'Steve' Cai4e566ed2018-09-19 22:41:42 -070056
Fan Xu069e8382018-11-16 16:28:08 -080057 if (ashmem_set_prot_region(ashmemFd.get(), kAshmemProt) != 0) {
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -070058 ALOGE("BufferHubMetadata::Create: failed to set protect region.");
59 return {};
60 }
Jiwen 'Steve' Cai4e566ed2018-09-19 22:41:42 -070061
Tianyu Jiang727ede42019-02-01 11:44:51 -080062 return BufferHubMetadata::import(std::move(ashmemFd));
Jiwen 'Steve' Cai4e566ed2018-09-19 22:41:42 -070063}
64
65/* static */
Tianyu Jiang727ede42019-02-01 11:44:51 -080066BufferHubMetadata BufferHubMetadata::import(unique_fd ashmemFd) {
Fan Xu069e8382018-11-16 16:28:08 -080067 if (!ashmem_valid(ashmemFd.get())) {
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -070068 ALOGE("BufferHubMetadata::Import: invalid ashmem fd.");
69 return {};
70 }
Jiwen 'Steve' Cai4e566ed2018-09-19 22:41:42 -070071
Fan Xu069e8382018-11-16 16:28:08 -080072 size_t metadataSize = static_cast<size_t>(ashmem_get_size_region(ashmemFd.get()));
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -070073 size_t userMetadataSize = metadataSize - kMetadataHeaderSize;
Jiwen 'Steve' Cai4e566ed2018-09-19 22:41:42 -070074
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -070075 // Note that here the buffer state is mapped from shared memory as an atomic object. The
76 // std::atomic's constructor will not be called so that the original value stored in the memory
77 // region can be preserved.
78 auto metadataHeader = static_cast<MetadataHeader*>(mmap(nullptr, metadataSize, kAshmemProt,
Fan Xu069e8382018-11-16 16:28:08 -080079 MAP_SHARED, ashmemFd.get(),
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -070080 /*offset=*/0));
81 if (metadataHeader == nullptr) {
82 ALOGE("BufferHubMetadata::Import: failed to map region.");
83 return {};
84 }
Jiwen 'Steve' Cai4e566ed2018-09-19 22:41:42 -070085
Fan Xu069e8382018-11-16 16:28:08 -080086 return BufferHubMetadata(userMetadataSize, std::move(ashmemFd), metadataHeader);
Jiwen 'Steve' Cai4e566ed2018-09-19 22:41:42 -070087}
88
Fan Xu069e8382018-11-16 16:28:08 -080089BufferHubMetadata::BufferHubMetadata(size_t userMetadataSize, unique_fd ashmemFd,
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -070090 MetadataHeader* metadataHeader)
91 : mUserMetadataSize(userMetadataSize),
Fan Xu069e8382018-11-16 16:28:08 -080092 mAshmemFd(std::move(ashmemFd)),
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -070093 mMetadataHeader(metadataHeader) {}
Jiwen 'Steve' Cai4e566ed2018-09-19 22:41:42 -070094
95BufferHubMetadata::~BufferHubMetadata() {
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -070096 if (mMetadataHeader != nullptr) {
Tianyu Jiang727ede42019-02-01 11:44:51 -080097 int ret = munmap(mMetadataHeader, metadataSize());
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -070098 ALOGE_IF(ret != 0,
99 "BufferHubMetadata::~BufferHubMetadata: failed to unmap ashmem, error=%d.", errno);
100 mMetadataHeader = nullptr;
101 }
Jiwen 'Steve' Cai4e566ed2018-09-19 22:41:42 -0700102}
103
Jiwen 'Steve' Caiff675b72018-10-09 18:08:29 -0700104} // namespace android