1 /*
2 * Copyright (C) 2008 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 #define LOG_TAG "MemoryHeapPmem"
18
19 #include <stdlib.h>
20 #include <stdint.h>
21 #include <unistd.h>
22 #include <fcntl.h>
23 #include <errno.h>
24 #include <sys/types.h>
25 #include <sys/stat.h>
26 #include <sys/ioctl.h>
27
28 #include <cutils/log.h>
29
30 #include <binder/MemoryHeapPmem.h>
31 #include <binder/MemoryHeapBase.h>
32
33 #if HAVE_ANDROID_OS
34 #include <linux/android_pmem.h>
35 #endif
36
37 namespace android {
38
39 // ---------------------------------------------------------------------------
40
MemoryPmem(const sp<MemoryHeapPmem> & heap)41 MemoryHeapPmem::MemoryPmem::MemoryPmem(const sp<MemoryHeapPmem>& heap)
42 : BnMemory(), mClientHeap(heap)
43 {
44 }
45
~MemoryPmem()46 MemoryHeapPmem::MemoryPmem::~MemoryPmem() {
47 if (mClientHeap != NULL) {
48 mClientHeap->remove(this);
49 }
50 }
51
52 // ---------------------------------------------------------------------------
53
54 class SubRegionMemory : public MemoryHeapPmem::MemoryPmem {
55 public:
56 SubRegionMemory(const sp<MemoryHeapPmem>& heap, ssize_t offset, size_t size);
57 virtual ~SubRegionMemory();
58 virtual sp<IMemoryHeap> getMemory(ssize_t* offset, size_t* size) const;
59 private:
60 friend class MemoryHeapPmem;
61 void revoke();
62 size_t mSize;
63 ssize_t mOffset;
64 };
65
SubRegionMemory(const sp<MemoryHeapPmem> & heap,ssize_t offset,size_t size)66 SubRegionMemory::SubRegionMemory(const sp<MemoryHeapPmem>& heap,
67 ssize_t offset, size_t size)
68 : MemoryHeapPmem::MemoryPmem(heap), mSize(size), mOffset(offset)
69 {
70 #ifndef NDEBUG
71 void* const start_ptr = (void*)(intptr_t(getHeap()->base()) + offset);
72 memset(start_ptr, 0xda, size);
73 #endif
74
75 #if HAVE_ANDROID_OS
76 if (size > 0) {
77 const size_t pagesize = getpagesize();
78 size = (size + pagesize-1) & ~(pagesize-1);
79 int our_fd = heap->heapID();
80 struct pmem_region sub = { offset, size };
81 int err = ioctl(our_fd, PMEM_MAP, &sub);
82 LOGE_IF(err<0, "PMEM_MAP failed (%s), "
83 "mFD=%d, sub.offset=%lu, sub.size=%lu",
84 strerror(errno), our_fd, sub.offset, sub.len);
85 }
86 #endif
87 }
88
getMemory(ssize_t * offset,size_t * size) const89 sp<IMemoryHeap> SubRegionMemory::getMemory(ssize_t* offset, size_t* size) const
90 {
91 if (offset) *offset = mOffset;
92 if (size) *size = mSize;
93 return getHeap();
94 }
95
~SubRegionMemory()96 SubRegionMemory::~SubRegionMemory()
97 {
98 revoke();
99 }
100
101
revoke()102 void SubRegionMemory::revoke()
103 {
104 // NOTE: revoke() doesn't need to be protected by a lock because it
105 // can only be called from MemoryHeapPmem::revoke(), which means
106 // that we can't be in ~SubRegionMemory(), or in ~SubRegionMemory(),
107 // which means MemoryHeapPmem::revoke() wouldn't have been able to
108 // promote() it.
109
110 #if HAVE_ANDROID_OS
111 if (mSize != 0) {
112 const sp<MemoryHeapPmem>& heap(getHeap());
113 int our_fd = heap->heapID();
114 struct pmem_region sub;
115 sub.offset = mOffset;
116 sub.len = mSize;
117 int err = ioctl(our_fd, PMEM_UNMAP, &sub);
118 LOGE_IF(err<0, "PMEM_UNMAP failed (%s), "
119 "mFD=%d, sub.offset=%lu, sub.size=%lu",
120 strerror(errno), our_fd, sub.offset, sub.len);
121 mSize = 0;
122 }
123 #endif
124 }
125
126 // ---------------------------------------------------------------------------
127
MemoryHeapPmem(const sp<MemoryHeapBase> & pmemHeap,uint32_t flags)128 MemoryHeapPmem::MemoryHeapPmem(const sp<MemoryHeapBase>& pmemHeap,
129 uint32_t flags)
130 : HeapInterface(), MemoryHeapBase()
131 {
132 char const * const device = pmemHeap->getDevice();
133 #if HAVE_ANDROID_OS
134 if (device) {
135 int fd = open(device, O_RDWR | (flags & NO_CACHING ? O_SYNC : 0));
136 LOGE_IF(fd<0, "couldn't open %s (%s)", device, strerror(errno));
137 if (fd >= 0) {
138 int err = ioctl(fd, PMEM_CONNECT, pmemHeap->heapID());
139 if (err < 0) {
140 LOGE("PMEM_CONNECT failed (%s), mFD=%d, sub-fd=%d",
141 strerror(errno), fd, pmemHeap->heapID());
142 close(fd);
143 } else {
144 // everything went well...
145 mParentHeap = pmemHeap;
146 MemoryHeapBase::init(fd,
147 pmemHeap->getBase(),
148 pmemHeap->getSize(),
149 pmemHeap->getFlags() | flags,
150 device);
151 }
152 }
153 }
154 #else
155 mParentHeap = pmemHeap;
156 MemoryHeapBase::init(
157 dup(pmemHeap->heapID()),
158 pmemHeap->getBase(),
159 pmemHeap->getSize(),
160 pmemHeap->getFlags() | flags,
161 device);
162 #endif
163 }
164
~MemoryHeapPmem()165 MemoryHeapPmem::~MemoryHeapPmem()
166 {
167 }
168
mapMemory(size_t offset,size_t size)169 sp<IMemory> MemoryHeapPmem::mapMemory(size_t offset, size_t size)
170 {
171 sp<MemoryPmem> memory = createMemory(offset, size);
172 if (memory != 0) {
173 Mutex::Autolock _l(mLock);
174 mAllocations.add(memory);
175 }
176 return memory;
177 }
178
createMemory(size_t offset,size_t size)179 sp<MemoryHeapPmem::MemoryPmem> MemoryHeapPmem::createMemory(
180 size_t offset, size_t size)
181 {
182 sp<SubRegionMemory> memory;
183 if (heapID() > 0)
184 memory = new SubRegionMemory(this, offset, size);
185 return memory;
186 }
187
slap()188 status_t MemoryHeapPmem::slap()
189 {
190 #if HAVE_ANDROID_OS
191 size_t size = getSize();
192 const size_t pagesize = getpagesize();
193 size = (size + pagesize-1) & ~(pagesize-1);
194 int our_fd = getHeapID();
195 struct pmem_region sub = { 0, size };
196 int err = ioctl(our_fd, PMEM_MAP, &sub);
197 LOGE_IF(err<0, "PMEM_MAP failed (%s), "
198 "mFD=%d, sub.offset=%lu, sub.size=%lu",
199 strerror(errno), our_fd, sub.offset, sub.len);
200 return -errno;
201 #else
202 return NO_ERROR;
203 #endif
204 }
205
unslap()206 status_t MemoryHeapPmem::unslap()
207 {
208 #if HAVE_ANDROID_OS
209 size_t size = getSize();
210 const size_t pagesize = getpagesize();
211 size = (size + pagesize-1) & ~(pagesize-1);
212 int our_fd = getHeapID();
213 struct pmem_region sub = { 0, size };
214 int err = ioctl(our_fd, PMEM_UNMAP, &sub);
215 LOGE_IF(err<0, "PMEM_UNMAP failed (%s), "
216 "mFD=%d, sub.offset=%lu, sub.size=%lu",
217 strerror(errno), our_fd, sub.offset, sub.len);
218 return -errno;
219 #else
220 return NO_ERROR;
221 #endif
222 }
223
revoke()224 void MemoryHeapPmem::revoke()
225 {
226 SortedVector< wp<MemoryPmem> > allocations;
227
228 { // scope for lock
229 Mutex::Autolock _l(mLock);
230 allocations = mAllocations;
231 }
232
233 ssize_t count = allocations.size();
234 for (ssize_t i=0 ; i<count ; i++) {
235 sp<MemoryPmem> memory(allocations[i].promote());
236 if (memory != 0)
237 memory->revoke();
238 }
239 }
240
remove(const wp<MemoryPmem> & memory)241 void MemoryHeapPmem::remove(const wp<MemoryPmem>& memory)
242 {
243 Mutex::Autolock _l(mLock);
244 mAllocations.remove(memory);
245 }
246
247 // ---------------------------------------------------------------------------
248 }; // namespace android
249