1 /*
2  * Copyright (C) 2022 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 <cutils/ashmem.h>
18 #include <sys/mman.h>
19 #include "allocator.h"
20 
21 union Params {
22   struct {
23     uint32_t capacity;
24   } data;
25   uint8_t array[0];
Params()26   Params() : data{0} {}
Params(uint32_t size)27   Params(uint32_t size)
28       : data{size} {}
29 };
30 
31 
32 namespace {
33 
34 struct HandleAshmem : public native_handle_t {
HandleAshmem__anon8535dd720211::HandleAshmem35   HandleAshmem(int ashmemFd, size_t size)
36     : native_handle_t(cHeader),
37     mFds{ ashmemFd },
38     mInts{ int (size & 0xFFFFFFFF), int((uint64_t(size) >> 32) & 0xFFFFFFFF), kMagic } {}
39 
ashmemFd__anon8535dd720211::HandleAshmem40   int ashmemFd() const { return mFds.mAshmem; }
size__anon8535dd720211::HandleAshmem41   size_t size() const {
42     return size_t(unsigned(mInts.mSizeLo))
43         | size_t(uint64_t(unsigned(mInts.mSizeHi)) << 32);
44   }
45 
46   static bool isValid(const native_handle_t * const o);
47 
48 protected:
49   struct {
50     int mAshmem;
51   } mFds;
52   struct {
53     int mSizeLo;
54     int mSizeHi;
55     int mMagic;
56   } mInts;
57 
58 private:
59   enum {
60     kMagic = 'ahm\x00',
61     numFds = sizeof(mFds) / sizeof(int),
62     numInts = sizeof(mInts) / sizeof(int),
63     version = sizeof(native_handle_t)
64   };
65   const static native_handle_t cHeader;
66 };
67 
68 const native_handle_t HandleAshmem::cHeader = {
69   HandleAshmem::version,
70   HandleAshmem::numFds,
71   HandleAshmem::numInts,
72   {}
73 };
74 
isValid(const native_handle_t * const o)75 bool HandleAshmem::isValid(const native_handle_t * const o) {
76   if (!o || memcmp(o, &cHeader, sizeof(cHeader))) {
77     return false;
78   }
79   const HandleAshmem *other = static_cast<const HandleAshmem*>(o);
80   return other->mInts.mMagic == kMagic;
81 }
82 
83 class AllocationAshmem {
84 private:
AllocationAshmem(int ashmemFd,size_t capacity,bool res)85   AllocationAshmem(int ashmemFd, size_t capacity, bool res)
86     : mHandle(ashmemFd, capacity),
87       mInit(res) {}
88 
89 public:
Alloc(size_t size)90   static AllocationAshmem *Alloc(size_t size) {
91     constexpr static const char *kAllocationTag = "bufferpool_test";
92     int ashmemFd = ashmem_create_region(kAllocationTag, size);
93     return new AllocationAshmem(ashmemFd, size, ashmemFd >= 0);
94   }
95 
~AllocationAshmem()96   ~AllocationAshmem() {
97     if (mInit) {
98       native_handle_close(&mHandle);
99     }
100   }
101 
handle()102   const HandleAshmem *handle() {
103     return &mHandle;
104   }
105 
106 private:
107   HandleAshmem mHandle;
108   bool mInit;
109   // TODO: mapping and map fd
110 };
111 
112 struct AllocationDtor {
AllocationDtor__anon8535dd720211::AllocationDtor113   AllocationDtor(const std::shared_ptr<AllocationAshmem> &alloc)
114       : mAlloc(alloc) {}
115 
operator ()__anon8535dd720211::AllocationDtor116   void operator()(BufferPoolAllocation *poolAlloc) { delete poolAlloc; }
117 
118   const std::shared_ptr<AllocationAshmem> mAlloc;
119 };
120 
121 }
122 
init()123 void IpcMutex::init() {
124   pthread_mutexattr_t mattr;
125   pthread_mutexattr_init(&mattr);
126   pthread_mutexattr_setpshared(&mattr, PTHREAD_PROCESS_SHARED);
127   pthread_mutex_init(&lock, &mattr);
128   pthread_mutexattr_destroy(&mattr);
129 
130   pthread_condattr_t cattr;
131   pthread_condattr_init(&cattr);
132   pthread_condattr_setpshared(&cattr, PTHREAD_PROCESS_SHARED);
133   pthread_cond_init(&cond, &cattr);
134   pthread_condattr_destroy(&cattr);
135 }
136 
Import(void * pMutex)137 IpcMutex *IpcMutex::Import(void *pMutex) {
138   return reinterpret_cast<IpcMutex *>(pMutex);
139 }
140 
141 
allocate(const std::vector<uint8_t> & params,std::shared_ptr<BufferPoolAllocation> * alloc,size_t * allocSize)142 BufferPoolStatus TestBufferPoolAllocator::allocate(
143     const std::vector<uint8_t> &params,
144     std::shared_ptr<BufferPoolAllocation> *alloc,
145     size_t *allocSize) {
146   Params ashmemParams;
147   memcpy(&ashmemParams, params.data(), std::min(sizeof(Params), params.size()));
148 
149   std::shared_ptr<AllocationAshmem> ashmemAlloc =
150       std::shared_ptr<AllocationAshmem>(
151           AllocationAshmem::Alloc(ashmemParams.data.capacity));
152   if (ashmemAlloc) {
153     BufferPoolAllocation *ptr = new BufferPoolAllocation(ashmemAlloc->handle());
154     if (ptr) {
155       *alloc = std::shared_ptr<BufferPoolAllocation>(ptr, AllocationDtor(ashmemAlloc));
156       if (*alloc) {
157           *allocSize = ashmemParams.data.capacity;
158           return ResultStatus::OK;
159       }
160       delete ptr;
161       return ResultStatus::NO_MEMORY;
162     }
163   }
164   return ResultStatus::CRITICAL_ERROR;
165 }
166 
compatible(const std::vector<uint8_t> & newParams,const std::vector<uint8_t> & oldParams)167 bool TestBufferPoolAllocator::compatible(const std::vector<uint8_t> &newParams,
168                                         const std::vector<uint8_t> &oldParams) {
169   size_t newSize = newParams.size();
170   size_t oldSize = oldParams.size();
171   if (newSize == oldSize) {
172     for (size_t i = 0; i < newSize; ++i) {
173       if (newParams[i] != oldParams[i]) {
174         return false;
175       }
176     }
177     return true;
178   }
179   return false;
180 }
181 
Fill(const native_handle_t * handle,const unsigned char val)182 bool TestBufferPoolAllocator::Fill(const native_handle_t *handle, const unsigned char val) {
183   if (!HandleAshmem::isValid(handle)) {
184     return false;
185   }
186   const HandleAshmem *o = static_cast<const HandleAshmem*>(handle);
187   unsigned char *ptr = (unsigned char *)mmap(
188       NULL, o->size(), PROT_READ|PROT_WRITE, MAP_SHARED, o->ashmemFd(), 0);
189 
190   if (ptr != MAP_FAILED) {
191     for (size_t i = 0; i < o->size(); ++i) {
192       ptr[i] = val;
193     }
194     munmap(ptr, o->size());
195     return true;
196   }
197   return false;
198 }
199 
Verify(const native_handle_t * handle,const unsigned char val)200 bool TestBufferPoolAllocator::Verify(const native_handle_t *handle, const unsigned char val) {
201   if (!HandleAshmem::isValid(handle)) {
202     return false;
203   }
204   const HandleAshmem *o = static_cast<const HandleAshmem*>(handle);
205   unsigned char *ptr = (unsigned char *)mmap(
206       NULL, o->size(), PROT_READ, MAP_SHARED, o->ashmemFd(), 0);
207 
208   if (ptr != MAP_FAILED) {
209     bool res = true;
210     for (size_t i = 0; i < o->size(); ++i) {
211       if (ptr[i] != val) {
212         res = false;
213         break;
214       }
215     }
216     munmap(ptr, o->size());
217     return res;
218   }
219   return false;
220 }
221 
MapMemoryForMutex(const native_handle_t * handle,void ** mem)222 bool TestBufferPoolAllocator::MapMemoryForMutex(const native_handle_t *handle, void **mem) {
223   if (!HandleAshmem::isValid(handle)) {
224     return false;
225   }
226   const HandleAshmem *o = static_cast<const HandleAshmem*>(handle);
227   *mem = mmap(
228       NULL, o->size(), PROT_READ|PROT_WRITE, MAP_SHARED, o->ashmemFd(), 0);
229   if (*mem == MAP_FAILED || *mem == nullptr) {
230     return false;
231   }
232   return true;
233 }
234 
UnmapMemoryForMutex(void * mem)235 bool TestBufferPoolAllocator::UnmapMemoryForMutex(void *mem) {
236   munmap(mem, sizeof(IpcMutex));
237   return true;
238 }
239 
getTestAllocatorParams(std::vector<uint8_t> * params)240 void getTestAllocatorParams(std::vector<uint8_t> *params) {
241   constexpr static int kAllocationSize = 1024 * 10;
242   Params ashmemParams(kAllocationSize);
243 
244   params->assign(ashmemParams.array, ashmemParams.array + sizeof(ashmemParams));
245 }
246 
getIpcMutexParams(std::vector<uint8_t> * params)247 void getIpcMutexParams(std::vector<uint8_t> *params) {
248   Params ashmemParams(sizeof(IpcMutex));
249 
250   params->assign(ashmemParams.array, ashmemParams.array + sizeof(ashmemParams));
251 }
252