1*ec779b8eSAndroid Build Coastguard Worker /*
2*ec779b8eSAndroid Build Coastguard Worker * Copyright (C) 2020 The Android Open Source Project
3*ec779b8eSAndroid Build Coastguard Worker *
4*ec779b8eSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*ec779b8eSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*ec779b8eSAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*ec779b8eSAndroid Build Coastguard Worker *
8*ec779b8eSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*ec779b8eSAndroid Build Coastguard Worker *
10*ec779b8eSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*ec779b8eSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*ec779b8eSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*ec779b8eSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*ec779b8eSAndroid Build Coastguard Worker * limitations under the License.
15*ec779b8eSAndroid Build Coastguard Worker */
16*ec779b8eSAndroid Build Coastguard Worker #include "media/ShmemCompat.h"
17*ec779b8eSAndroid Build Coastguard Worker
18*ec779b8eSAndroid Build Coastguard Worker #include "binder/MemoryBase.h"
19*ec779b8eSAndroid Build Coastguard Worker #include "binder/MemoryHeapBase.h"
20*ec779b8eSAndroid Build Coastguard Worker #include "media/ShmemUtil.h"
21*ec779b8eSAndroid Build Coastguard Worker
22*ec779b8eSAndroid Build Coastguard Worker namespace android {
23*ec779b8eSAndroid Build Coastguard Worker namespace media {
24*ec779b8eSAndroid Build Coastguard Worker
convertSharedFileRegionToIMemory(const SharedFileRegion & shmem,sp<IMemory> * result)25*ec779b8eSAndroid Build Coastguard Worker bool convertSharedFileRegionToIMemory(const SharedFileRegion& shmem,
26*ec779b8eSAndroid Build Coastguard Worker sp<IMemory>* result) {
27*ec779b8eSAndroid Build Coastguard Worker assert(result != nullptr);
28*ec779b8eSAndroid Build Coastguard Worker
29*ec779b8eSAndroid Build Coastguard Worker if (!validateSharedFileRegion(shmem)) {
30*ec779b8eSAndroid Build Coastguard Worker return false;
31*ec779b8eSAndroid Build Coastguard Worker }
32*ec779b8eSAndroid Build Coastguard Worker
33*ec779b8eSAndroid Build Coastguard Worker // Heap offset and size must be page aligned.
34*ec779b8eSAndroid Build Coastguard Worker const size_t pageSize = getpagesize();
35*ec779b8eSAndroid Build Coastguard Worker const size_t pageMask = ~(pageSize - 1);
36*ec779b8eSAndroid Build Coastguard Worker
37*ec779b8eSAndroid Build Coastguard Worker // OK if this wraps.
38*ec779b8eSAndroid Build Coastguard Worker const uint64_t endOffset = static_cast<uint64_t>(shmem.offset) +
39*ec779b8eSAndroid Build Coastguard Worker static_cast<uint64_t>(shmem.size);
40*ec779b8eSAndroid Build Coastguard Worker
41*ec779b8eSAndroid Build Coastguard Worker // Round down to page boundary.
42*ec779b8eSAndroid Build Coastguard Worker const uint64_t heapStartOffset = shmem.offset & pageMask;
43*ec779b8eSAndroid Build Coastguard Worker // Round up to page boundary.
44*ec779b8eSAndroid Build Coastguard Worker const uint64_t heapEndOffset = (endOffset + pageSize - 1) & pageMask;
45*ec779b8eSAndroid Build Coastguard Worker const uint64_t heapSize = heapEndOffset - heapStartOffset;
46*ec779b8eSAndroid Build Coastguard Worker
47*ec779b8eSAndroid Build Coastguard Worker if (heapStartOffset > std::numeric_limits<size_t>::max() ||
48*ec779b8eSAndroid Build Coastguard Worker heapSize > std::numeric_limits<size_t>::max()) {
49*ec779b8eSAndroid Build Coastguard Worker return false;
50*ec779b8eSAndroid Build Coastguard Worker }
51*ec779b8eSAndroid Build Coastguard Worker
52*ec779b8eSAndroid Build Coastguard Worker uint32_t flags = !shmem.writeable ? IMemoryHeap::READ_ONLY : 0;
53*ec779b8eSAndroid Build Coastguard Worker
54*ec779b8eSAndroid Build Coastguard Worker const sp<MemoryHeapBase> heap =
55*ec779b8eSAndroid Build Coastguard Worker new MemoryHeapBase(shmem.fd.get(), heapSize, flags, heapStartOffset);
56*ec779b8eSAndroid Build Coastguard Worker *result = sp<MemoryBase>::make(heap,
57*ec779b8eSAndroid Build Coastguard Worker shmem.offset - heapStartOffset,
58*ec779b8eSAndroid Build Coastguard Worker shmem.size);
59*ec779b8eSAndroid Build Coastguard Worker return true;
60*ec779b8eSAndroid Build Coastguard Worker }
61*ec779b8eSAndroid Build Coastguard Worker
convertIMemoryToSharedFileRegion(const sp<IMemory> & mem,SharedFileRegion * result)62*ec779b8eSAndroid Build Coastguard Worker bool convertIMemoryToSharedFileRegion(const sp<IMemory>& mem,
63*ec779b8eSAndroid Build Coastguard Worker SharedFileRegion* result) {
64*ec779b8eSAndroid Build Coastguard Worker assert(mem != nullptr);
65*ec779b8eSAndroid Build Coastguard Worker assert(result != nullptr);
66*ec779b8eSAndroid Build Coastguard Worker
67*ec779b8eSAndroid Build Coastguard Worker *result = SharedFileRegion();
68*ec779b8eSAndroid Build Coastguard Worker
69*ec779b8eSAndroid Build Coastguard Worker ssize_t offset;
70*ec779b8eSAndroid Build Coastguard Worker size_t size;
71*ec779b8eSAndroid Build Coastguard Worker
72*ec779b8eSAndroid Build Coastguard Worker sp<IMemoryHeap> heap = mem->getMemory(&offset, &size);
73*ec779b8eSAndroid Build Coastguard Worker if (size > 0) {
74*ec779b8eSAndroid Build Coastguard Worker if (heap == nullptr) {
75*ec779b8eSAndroid Build Coastguard Worker return false;
76*ec779b8eSAndroid Build Coastguard Worker }
77*ec779b8eSAndroid Build Coastguard Worker // Make sure the offset and size do not overflow from int64 boundaries.
78*ec779b8eSAndroid Build Coastguard Worker if (size > std::numeric_limits<int64_t>::max() ||
79*ec779b8eSAndroid Build Coastguard Worker offset > std::numeric_limits<int64_t>::max() ||
80*ec779b8eSAndroid Build Coastguard Worker heap->getOffset() > std::numeric_limits<int64_t>::max() ||
81*ec779b8eSAndroid Build Coastguard Worker static_cast<uint64_t>(heap->getOffset()) +
82*ec779b8eSAndroid Build Coastguard Worker static_cast<uint64_t>(offset)
83*ec779b8eSAndroid Build Coastguard Worker > std::numeric_limits<int64_t>::max()) {
84*ec779b8eSAndroid Build Coastguard Worker return false;
85*ec779b8eSAndroid Build Coastguard Worker }
86*ec779b8eSAndroid Build Coastguard Worker
87*ec779b8eSAndroid Build Coastguard Worker base::unique_fd fd(fcntl(heap->getHeapID(), F_DUPFD_CLOEXEC, 0));
88*ec779b8eSAndroid Build Coastguard Worker if (!fd.ok()) {
89*ec779b8eSAndroid Build Coastguard Worker return false;
90*ec779b8eSAndroid Build Coastguard Worker }
91*ec779b8eSAndroid Build Coastguard Worker result->fd.reset(std::move(fd));
92*ec779b8eSAndroid Build Coastguard Worker result->size = size;
93*ec779b8eSAndroid Build Coastguard Worker result->offset = heap->getOffset() + offset;
94*ec779b8eSAndroid Build Coastguard Worker result->writeable = (heap->getFlags() & IMemoryHeap::READ_ONLY) == 0;
95*ec779b8eSAndroid Build Coastguard Worker }
96*ec779b8eSAndroid Build Coastguard Worker return true;
97*ec779b8eSAndroid Build Coastguard Worker }
98*ec779b8eSAndroid Build Coastguard Worker
convertNullableSharedFileRegionToIMemory(const std::optional<SharedFileRegion> & shmem,sp<IMemory> * result)99*ec779b8eSAndroid Build Coastguard Worker bool convertNullableSharedFileRegionToIMemory(const std::optional<SharedFileRegion>& shmem,
100*ec779b8eSAndroid Build Coastguard Worker sp<IMemory>* result) {
101*ec779b8eSAndroid Build Coastguard Worker assert(result != nullptr);
102*ec779b8eSAndroid Build Coastguard Worker
103*ec779b8eSAndroid Build Coastguard Worker if (!shmem.has_value()) {
104*ec779b8eSAndroid Build Coastguard Worker result->clear();
105*ec779b8eSAndroid Build Coastguard Worker return true;
106*ec779b8eSAndroid Build Coastguard Worker }
107*ec779b8eSAndroid Build Coastguard Worker
108*ec779b8eSAndroid Build Coastguard Worker return convertSharedFileRegionToIMemory(shmem.value(), result);
109*ec779b8eSAndroid Build Coastguard Worker }
110*ec779b8eSAndroid Build Coastguard Worker
convertNullableIMemoryToSharedFileRegion(const sp<IMemory> & mem,std::optional<SharedFileRegion> * result)111*ec779b8eSAndroid Build Coastguard Worker bool convertNullableIMemoryToSharedFileRegion(const sp<IMemory>& mem,
112*ec779b8eSAndroid Build Coastguard Worker std::optional<SharedFileRegion>* result) {
113*ec779b8eSAndroid Build Coastguard Worker assert(result != nullptr);
114*ec779b8eSAndroid Build Coastguard Worker
115*ec779b8eSAndroid Build Coastguard Worker if (mem == nullptr) {
116*ec779b8eSAndroid Build Coastguard Worker result->reset();
117*ec779b8eSAndroid Build Coastguard Worker return true;
118*ec779b8eSAndroid Build Coastguard Worker }
119*ec779b8eSAndroid Build Coastguard Worker
120*ec779b8eSAndroid Build Coastguard Worker result->emplace();
121*ec779b8eSAndroid Build Coastguard Worker return convertIMemoryToSharedFileRegion(mem, &result->value());
122*ec779b8eSAndroid Build Coastguard Worker }
123*ec779b8eSAndroid Build Coastguard Worker
124*ec779b8eSAndroid Build Coastguard Worker } // namespace media
125*ec779b8eSAndroid Build Coastguard Worker } // namespace android
126