blob: e2b1fff6222c69323243d84227f85274c9e9f9cb [file] [log] [blame]
The Android Open Source Projectf6c38712009-03-03 19:28:47 -08001/*
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 */
The Android Open Source Project99409882009-03-18 22:20:24 -070016
The Android Open Source Projectf6c38712009-03-03 19:28:47 -080017/*
18 * System utilities.
19 */
20#include "DexFile.h"
21#include "SysUtil.h"
22
23#include <stdlib.h>
24#include <stdio.h>
25#include <unistd.h>
26#include <string.h>
The Android Open Source Projectf6c38712009-03-03 19:28:47 -080027#ifdef HAVE_POSIX_FILEMAP
Andy McFadden64896a22010-05-04 15:41:22 -070028# include <sys/mman.h>
The Android Open Source Projectf6c38712009-03-03 19:28:47 -080029#endif
The Android Open Source Projectf6c38712009-03-03 19:28:47 -080030#include <limits.h>
31#include <errno.h>
32
Andy McFadden64896a22010-05-04 15:41:22 -070033#include <JNIHelp.h> // TEMP_FAILURE_RETRY may or may not be in unistd
The Android Open Source Projectf6c38712009-03-03 19:28:47 -080034
35
36/*
37 * Create an anonymous shared memory segment large enough to hold "length"
38 * bytes. The actual segment may be larger because mmap() operates on
39 * page boundaries (usually 4K).
40 */
41static void* sysCreateAnonShmem(size_t length)
42{
43#ifdef HAVE_POSIX_FILEMAP
44 void* ptr;
45
46 ptr = mmap(NULL, length, PROT_READ | PROT_WRITE,
47 MAP_SHARED | MAP_ANON, -1, 0);
48 if (ptr == MAP_FAILED) {
49 LOGW("mmap(%d, RW, SHARED|ANON) failed: %s\n", (int) length,
50 strerror(errno));
51 return NULL;
52 }
53
54 return ptr;
55#else
56 LOGE("sysCreateAnonShmem not implemented.\n");
57 return NULL;
58#endif
59}
60
The Android Open Source Project99409882009-03-18 22:20:24 -070061/*
62 * Create a private anonymous storage area.
63 */
64int sysCreatePrivateMap(size_t length, MemMapping* pMap)
65{
66 void* memPtr;
67
68 memPtr = sysCreateAnonShmem(length);
69 if (memPtr == NULL)
70 return -1;
71
72 pMap->addr = pMap->baseAddr = memPtr;
73 pMap->length = pMap->baseLength = length;
74 return 0;
75}
76
77/*
78 * Determine the current offset and remaining length of the open file.
79 */
The Android Open Source Projectf6c38712009-03-03 19:28:47 -080080static int getFileStartAndLength(int fd, off_t *start_, size_t *length_)
81{
82 off_t start, end;
83 size_t length;
84
85 assert(start_ != NULL);
86 assert(length_ != NULL);
87
88 start = lseek(fd, 0L, SEEK_CUR);
89 end = lseek(fd, 0L, SEEK_END);
90 (void) lseek(fd, start, SEEK_SET);
91
92 if (start == (off_t) -1 || end == (off_t) -1) {
93 LOGE("could not determine length of file\n");
94 return -1;
95 }
96
97 length = end - start;
98 if (length == 0) {
99 LOGE("file is empty\n");
100 return -1;
101 }
102
103 *start_ = start;
104 *length_ = length;
105
106 return 0;
107}
108
109/*
110 * Pull the contents of a file into an new shared memory segment. We grab
111 * everything from fd's current offset on.
112 *
113 * We need to know the length ahead of time so we can allocate a segment
114 * of sufficient size.
115 */
116int sysLoadFileInShmem(int fd, MemMapping* pMap)
117{
118#ifdef HAVE_POSIX_FILEMAP
119 off_t start;
120 size_t length, actual;
121 void* memPtr;
122
123 assert(pMap != NULL);
124
125 if (getFileStartAndLength(fd, &start, &length) < 0)
126 return -1;
127
128 memPtr = sysCreateAnonShmem(length);
129 if (memPtr == NULL)
130 return -1;
131
132 actual = read(fd, memPtr, length);
133 if (actual != length) {
134 LOGE("only read %d of %d bytes\n", (int) actual, (int) length);
135 sysReleaseShmem(pMap);
136 return -1;
137 }
138
139 pMap->baseAddr = pMap->addr = memPtr;
140 pMap->baseLength = pMap->length = length;
141
142 return 0;
143#else
144 LOGE("sysLoadFileInShmem not implemented.\n");
145 return -1;
146#endif
147}
148
Andy McFadden518925b2009-11-23 16:12:39 -0800149#ifndef HAVE_POSIX_FILEMAP
150int sysFakeMapFile(int fd, MemMapping* pMap)
151{
152 /* No MMAP, just fake it by copying the bits.
153 For Win32 we could use MapViewOfFile if really necessary
154 (see libs/utils/FileMap.cpp).
155 */
156 off_t start;
157 size_t length;
158 void* memPtr;
159
160 assert(pMap != NULL);
161
162 if (getFileStartAndLength(fd, &start, &length) < 0)
163 return -1;
164
165 memPtr = malloc(length);
166 if (read(fd, memPtr, length) < 0) {
167 LOGW("read(fd=%d, start=%d, length=%d) failed: %s\n", (int) length,
168 fd, (int) start, strerror(errno));
169 return -1;
170 }
171
172 pMap->baseAddr = pMap->addr = memPtr;
173 pMap->baseLength = pMap->length = length;
174
175 return 0;
176}
177#endif
178
The Android Open Source Projectf6c38712009-03-03 19:28:47 -0800179/*
180 * Map a file (from fd's current offset) into a shared, read-only memory
Andy McFadden96516932009-10-28 17:39:02 -0700181 * segment. The file offset must be a multiple of the system page size.
The Android Open Source Projectf6c38712009-03-03 19:28:47 -0800182 *
183 * On success, returns 0 and fills out "pMap". On failure, returns a nonzero
184 * value and does not disturb "pMap".
185 */
Andy McFaddenb5ebe472009-11-16 16:14:54 -0800186int sysMapFileInShmemReadOnly(int fd, MemMapping* pMap)
The Android Open Source Projectf6c38712009-03-03 19:28:47 -0800187{
188#ifdef HAVE_POSIX_FILEMAP
189 off_t start;
190 size_t length;
191 void* memPtr;
192
193 assert(pMap != NULL);
194
195 if (getFileStartAndLength(fd, &start, &length) < 0)
196 return -1;
197
Andy McFaddenb5ebe472009-11-16 16:14:54 -0800198 memPtr = mmap(NULL, length, PROT_READ, MAP_FILE | MAP_SHARED, fd, start);
The Android Open Source Projectf6c38712009-03-03 19:28:47 -0800199 if (memPtr == MAP_FAILED) {
Andy McFaddenb5ebe472009-11-16 16:14:54 -0800200 LOGW("mmap(%d, RO, FILE|SHARED, %d, %d) failed: %s\n", (int) length,
The Android Open Source Projectf6c38712009-03-03 19:28:47 -0800201 fd, (int) start, strerror(errno));
202 return -1;
203 }
204
205 pMap->baseAddr = pMap->addr = memPtr;
206 pMap->baseLength = pMap->length = length;
207
208 return 0;
209#else
Andy McFadden518925b2009-11-23 16:12:39 -0800210 return sysFakeMapFile(fd, pMap);
Andy McFaddenb5ebe472009-11-16 16:14:54 -0800211#endif
212}
213
214/*
215 * Map a file (from fd's current offset) into a private, read-write memory
216 * segment that will be marked read-only (a/k/a "writable read-only"). The
217 * file offset must be a multiple of the system page size.
218 *
219 * In some cases the mapping will be fully writable (e.g. for files on
220 * FAT filesystems).
221 *
222 * On success, returns 0 and fills out "pMap". On failure, returns a nonzero
223 * value and does not disturb "pMap".
224 */
225int sysMapFileInShmemWritableReadOnly(int fd, MemMapping* pMap)
226{
227#ifdef HAVE_POSIX_FILEMAP
228 off_t start;
229 size_t length;
230 void* memPtr;
231
232 assert(pMap != NULL);
233
234 if (getFileStartAndLength(fd, &start, &length) < 0)
235 return -1;
236
237 memPtr = mmap(NULL, length, PROT_READ | PROT_WRITE, MAP_FILE | MAP_PRIVATE,
238 fd, start);
239 if (memPtr == MAP_FAILED) {
240 LOGW("mmap(%d, R/W, FILE|PRIVATE, %d, %d) failed: %s\n", (int) length,
241 fd, (int) start, strerror(errno));
242 return -1;
243 }
244 if (mprotect(memPtr, length, PROT_READ) < 0) {
245 /* this fails with EACCESS on FAT filesystems, e.g. /sdcard */
246 int err = errno;
247 LOGV("mprotect(%p, %d, PROT_READ) failed: %s\n",
248 memPtr, length, strerror(err));
249 LOGD("mprotect(RO) failed (%d), file will remain read-write\n", err);
250 }
251
252 pMap->baseAddr = pMap->addr = memPtr;
253 pMap->baseLength = pMap->length = length;
254
255 return 0;
256#else
Andy McFadden518925b2009-11-23 16:12:39 -0800257 return sysFakeMapFile(fd, pMap);
Andy McFaddenb5ebe472009-11-16 16:14:54 -0800258#endif
259}
260
The Android Open Source Projectf6c38712009-03-03 19:28:47 -0800261/*
Andy McFadden8911f7a2010-04-23 16:34:52 -0700262 * Map part of a file into a shared, read-only memory segment. The "start"
263 * offset is absolute, not relative.
The Android Open Source Projectf6c38712009-03-03 19:28:47 -0800264 *
265 * On success, returns 0 and fills out "pMap". On failure, returns a nonzero
266 * value and does not disturb "pMap".
267 */
Andy McFadden8911f7a2010-04-23 16:34:52 -0700268int sysMapFileSegmentInShmem(int fd, off_t start, size_t length,
The Android Open Source Projectf6c38712009-03-03 19:28:47 -0800269 MemMapping* pMap)
270{
271#ifdef HAVE_POSIX_FILEMAP
Andy McFadden8911f7a2010-04-23 16:34:52 -0700272 size_t actualLength;
The Android Open Source Projectf6c38712009-03-03 19:28:47 -0800273 off_t actualStart;
274 int adjust;
275 void* memPtr;
276
277 assert(pMap != NULL);
278
The Android Open Source Projectf6c38712009-03-03 19:28:47 -0800279 /* adjust to be page-aligned */
Andy McFadden64896a22010-05-04 15:41:22 -0700280 adjust = start % SYSTEM_PAGE_SIZE;
The Android Open Source Projectf6c38712009-03-03 19:28:47 -0800281 actualStart = start - adjust;
282 actualLength = length + adjust;
283
284 memPtr = mmap(NULL, actualLength, PROT_READ, MAP_FILE | MAP_SHARED,
285 fd, actualStart);
286 if (memPtr == MAP_FAILED) {
287 LOGW("mmap(%d, R, FILE|SHARED, %d, %d) failed: %s\n",
288 (int) actualLength, fd, (int) actualStart, strerror(errno));
289 return -1;
290 }
291
292 pMap->baseAddr = memPtr;
293 pMap->baseLength = actualLength;
294 pMap->addr = (char*)memPtr + adjust;
295 pMap->length = length;
296
297 LOGVV("mmap seg (st=%d ln=%d): bp=%p bl=%d ad=%p ln=%d\n",
298 (int) start, (int) length,
299 pMap->baseAddr, (int) pMap->baseLength,
300 pMap->addr, (int) pMap->length);
301
302 return 0;
303#else
304 LOGE("sysMapFileSegmentInShmem not implemented.\n");
305 return -1;
306#endif
307}
308
309/*
Andy McFadden96516932009-10-28 17:39:02 -0700310 * Change the access rights on one or more pages to read-only or read-write.
311 *
312 * Returns 0 on success.
313 */
314int sysChangeMapAccess(void* addr, size_t length, int wantReadWrite,
315 MemMapping* pMap)
316{
Andy McFadden518925b2009-11-23 16:12:39 -0800317#ifdef HAVE_POSIX_FILEMAP
Andy McFadden96516932009-10-28 17:39:02 -0700318 /*
319 * Verify that "addr" is part of this mapping file.
320 */
321 if (addr < pMap->baseAddr ||
322 (u1*)addr >= (u1*)pMap->baseAddr + pMap->baseLength)
323 {
324 LOGE("Attempted to change %p; map is %p - %p\n",
325 addr, pMap->baseAddr, (u1*)pMap->baseAddr + pMap->baseLength);
326 return -1;
327 }
328
329 /*
330 * Align "addr" to a page boundary and adjust "length" appropriately.
331 * (The address must be page-aligned, the length doesn't need to be,
332 * but we do need to ensure we cover the same range.)
333 */
334 u1* alignAddr = (u1*) ((int) addr & ~(SYSTEM_PAGE_SIZE-1));
335 size_t alignLength = length + ((u1*) addr - alignAddr);
336
337 //LOGI("%p/%zd --> %p/%zd\n", addr, length, alignAddr, alignLength);
338 int prot = wantReadWrite ? (PROT_READ|PROT_WRITE) : (PROT_READ);
339 if (mprotect(alignAddr, alignLength, prot) != 0) {
340 int err = errno;
Andy McFaddenb5ebe472009-11-16 16:14:54 -0800341 LOGV("mprotect (%p,%zd,%d) failed: %s\n",
Andy McFadden96516932009-10-28 17:39:02 -0700342 alignAddr, alignLength, prot, strerror(errno));
343 return (errno != 0) ? errno : -1;
344 }
Andy McFadden518925b2009-11-23 16:12:39 -0800345#endif
Andy McFadden96516932009-10-28 17:39:02 -0700346
Andy McFadden518925b2009-11-23 16:12:39 -0800347 /* for "fake" mapping, no need to do anything */
Andy McFadden96516932009-10-28 17:39:02 -0700348 return 0;
349}
350
351/*
The Android Open Source Projectf6c38712009-03-03 19:28:47 -0800352 * Release a memory mapping.
353 */
354void sysReleaseShmem(MemMapping* pMap)
355{
356#ifdef HAVE_POSIX_FILEMAP
357 if (pMap->baseAddr == NULL && pMap->baseLength == 0)
358 return;
359
360 if (munmap(pMap->baseAddr, pMap->baseLength) < 0) {
361 LOGW("munmap(%p, %d) failed: %s\n",
362 pMap->baseAddr, (int)pMap->baseLength, strerror(errno));
363 } else {
364 LOGV("munmap(%p, %d) succeeded\n", pMap->baseAddr, pMap->baseLength);
365 pMap->baseAddr = NULL;
366 pMap->baseLength = 0;
367 }
368#else
369 /* Free the bits allocated by sysMapFileInShmem. */
370 if (pMap->baseAddr != NULL) {
371 free(pMap->baseAddr);
372 pMap->baseAddr = NULL;
373 }
374 pMap->baseLength = 0;
375#endif
376}
377
378/*
379 * Make a copy of a MemMapping.
380 */
381void sysCopyMap(MemMapping* dst, const MemMapping* src)
382{
383 memcpy(dst, src, sizeof(MemMapping));
384}
385
Andy McFadden64896a22010-05-04 15:41:22 -0700386/*
387 * Write until all bytes have been written.
388 *
389 * Returns 0 on success, or an errno value on failure.
390 */
391int sysWriteFully(int fd, const void* buf, size_t count, const char* logMsg)
392{
393 while (count != 0) {
394 ssize_t actual = TEMP_FAILURE_RETRY(write(fd, buf, count));
395 if (actual < 0) {
396 int err = errno;
397 LOGE("%s: write failed: %s\n", logMsg, strerror(err));
398 return err;
399 } else if (actual != (ssize_t) count) {
400 LOGD("%s: partial write (will retry): (%d of %zd)\n",
401 logMsg, (int) actual, count);
402 buf = (const void*) (((const u1*) buf) + actual);
403 }
404 count -= actual;
405 }
406
407 return 0;
408}