212 lines
5.2 KiB
C
212 lines
5.2 KiB
C
/*
|
|
* Copyright 2006 The Android Open Source Project
|
|
*
|
|
* System utilities.
|
|
*/
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <unistd.h>
|
|
#include <string.h>
|
|
#include <sys/mman.h>
|
|
#include <limits.h>
|
|
#include <errno.h>
|
|
#include <assert.h>
|
|
|
|
#define LOG_TAG "minzip"
|
|
#include "Log.h"
|
|
#include "SysUtil.h"
|
|
|
|
/*
|
|
* Having trouble finding a portable way to get this. sysconf(_SC_PAGE_SIZE)
|
|
* seems appropriate, but we don't have that on the device. Some systems
|
|
* have getpagesize(2), though the linux man page has some odd cautions.
|
|
*/
|
|
#define DEFAULT_PAGE_SIZE 4096
|
|
|
|
|
|
/*
|
|
* Create an anonymous shared memory segment large enough to hold "length"
|
|
* bytes. The actual segment may be larger because mmap() operates on
|
|
* page boundaries (usually 4K).
|
|
*/
|
|
static void* sysCreateAnonShmem(size_t length)
|
|
{
|
|
void* ptr;
|
|
|
|
ptr = mmap(NULL, length, PROT_READ | PROT_WRITE,
|
|
MAP_SHARED | MAP_ANON, -1, 0);
|
|
if (ptr == MAP_FAILED) {
|
|
LOGW("mmap(%d, RW, SHARED|ANON) failed: %s\n", (int) length,
|
|
strerror(errno));
|
|
return NULL;
|
|
}
|
|
|
|
return ptr;
|
|
}
|
|
|
|
static int getFileStartAndLength(int fd, off_t *start_, size_t *length_)
|
|
{
|
|
off_t start, end;
|
|
size_t length;
|
|
|
|
assert(start_ != NULL);
|
|
assert(length_ != NULL);
|
|
|
|
start = lseek(fd, 0L, SEEK_CUR);
|
|
end = lseek(fd, 0L, SEEK_END);
|
|
(void) lseek(fd, start, SEEK_SET);
|
|
|
|
if (start == (off_t) -1 || end == (off_t) -1) {
|
|
LOGE("could not determine length of file\n");
|
|
return -1;
|
|
}
|
|
|
|
length = end - start;
|
|
if (length == 0) {
|
|
LOGE("file is empty\n");
|
|
return -1;
|
|
}
|
|
|
|
*start_ = start;
|
|
*length_ = length;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Pull the contents of a file into an new shared memory segment. We grab
|
|
* everything from fd's current offset on.
|
|
*
|
|
* We need to know the length ahead of time so we can allocate a segment
|
|
* of sufficient size.
|
|
*/
|
|
int sysLoadFileInShmem(int fd, MemMapping* pMap)
|
|
{
|
|
off_t start;
|
|
size_t length, actual;
|
|
void* memPtr;
|
|
|
|
assert(pMap != NULL);
|
|
|
|
if (getFileStartAndLength(fd, &start, &length) < 0)
|
|
return -1;
|
|
|
|
memPtr = sysCreateAnonShmem(length);
|
|
if (memPtr == NULL)
|
|
return -1;
|
|
|
|
actual = read(fd, memPtr, length);
|
|
if (actual != length) {
|
|
LOGE("only read %d of %d bytes\n", (int) actual, (int) length);
|
|
sysReleaseShmem(pMap);
|
|
return -1;
|
|
}
|
|
|
|
pMap->baseAddr = pMap->addr = memPtr;
|
|
pMap->baseLength = pMap->length = length;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Map a file (from fd's current offset) into a shared, read-only memory
|
|
* segment. The file offset must be a multiple of the page size.
|
|
*
|
|
* On success, returns 0 and fills out "pMap". On failure, returns a nonzero
|
|
* value and does not disturb "pMap".
|
|
*/
|
|
int sysMapFileInShmem(int fd, MemMapping* pMap)
|
|
{
|
|
off_t start;
|
|
size_t length;
|
|
void* memPtr;
|
|
|
|
assert(pMap != NULL);
|
|
|
|
if (getFileStartAndLength(fd, &start, &length) < 0)
|
|
return -1;
|
|
|
|
memPtr = mmap(NULL, length, PROT_READ, MAP_FILE | MAP_SHARED, fd, start);
|
|
if (memPtr == MAP_FAILED) {
|
|
LOGW("mmap(%d, R, FILE|SHARED, %d, %d) failed: %s\n", (int) length,
|
|
fd, (int) start, strerror(errno));
|
|
return -1;
|
|
}
|
|
|
|
pMap->baseAddr = pMap->addr = memPtr;
|
|
pMap->baseLength = pMap->length = length;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Map part of a file (from fd's current offset) into a shared, read-only
|
|
* memory segment.
|
|
*
|
|
* On success, returns 0 and fills out "pMap". On failure, returns a nonzero
|
|
* value and does not disturb "pMap".
|
|
*/
|
|
int sysMapFileSegmentInShmem(int fd, off_t start, long length,
|
|
MemMapping* pMap)
|
|
{
|
|
off_t dummy;
|
|
size_t fileLength, actualLength;
|
|
off_t actualStart;
|
|
int adjust;
|
|
void* memPtr;
|
|
|
|
assert(pMap != NULL);
|
|
|
|
if (getFileStartAndLength(fd, &dummy, &fileLength) < 0)
|
|
return -1;
|
|
|
|
if (start + length > (long)fileLength) {
|
|
LOGW("bad segment: st=%d len=%ld flen=%d\n",
|
|
(int) start, length, (int) fileLength);
|
|
return -1;
|
|
}
|
|
|
|
/* adjust to be page-aligned */
|
|
adjust = start % DEFAULT_PAGE_SIZE;
|
|
actualStart = start - adjust;
|
|
actualLength = length + adjust;
|
|
|
|
memPtr = mmap(NULL, actualLength, PROT_READ, MAP_FILE | MAP_SHARED,
|
|
fd, actualStart);
|
|
if (memPtr == MAP_FAILED) {
|
|
LOGW("mmap(%d, R, FILE|SHARED, %d, %d) failed: %s\n",
|
|
(int) actualLength, fd, (int) actualStart, strerror(errno));
|
|
return -1;
|
|
}
|
|
|
|
pMap->baseAddr = memPtr;
|
|
pMap->baseLength = actualLength;
|
|
pMap->addr = (char*)memPtr + adjust;
|
|
pMap->length = length;
|
|
|
|
LOGVV("mmap seg (st=%d ln=%d): bp=%p bl=%d ad=%p ln=%d\n",
|
|
(int) start, (int) length,
|
|
pMap->baseAddr, (int) pMap->baseLength,
|
|
pMap->addr, (int) pMap->length);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Release a memory mapping.
|
|
*/
|
|
void sysReleaseShmem(MemMapping* pMap)
|
|
{
|
|
if (pMap->baseAddr == NULL && pMap->baseLength == 0)
|
|
return;
|
|
|
|
if (munmap(pMap->baseAddr, pMap->baseLength) < 0) {
|
|
LOGW("munmap(%p, %d) failed: %s\n",
|
|
pMap->baseAddr, (int)pMap->baseLength, strerror(errno));
|
|
} else {
|
|
LOGV("munmap(%p, %d) succeeded\n", pMap->baseAddr, pMap->baseLength);
|
|
pMap->baseAddr = NULL;
|
|
pMap->baseLength = 0;
|
|
}
|
|
}
|
|
|