xref: /aosp_15_r20/frameworks/av/media/libshmem/ShmemCompat.cpp (revision ec779b8e0859a360c3d303172224686826e6e0e1)
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