2011-11-11 00:46:59 +04:00
|
|
|
/* Memory mapping wrappers.
|
|
|
|
* ARM and MinGW ports contributed by Hideo Okuma and Tomoyuki Yoshimura at
|
|
|
|
* NICT.
|
|
|
|
*/
|
|
|
|
#include "util/mmap.hh"
|
|
|
|
|
2010-09-28 20:26:55 +04:00
|
|
|
#include "util/exception.hh"
|
2011-09-21 20:06:48 +04:00
|
|
|
#include "util/file.hh"
|
2010-09-16 23:53:33 +04:00
|
|
|
|
2010-11-28 05:54:56 +03:00
|
|
|
#include <iostream>
|
|
|
|
|
2010-09-16 23:53:33 +04:00
|
|
|
#include <assert.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <sys/types.h>
|
2011-11-11 00:46:59 +04:00
|
|
|
#include <sys/stat.h>
|
2011-11-17 23:12:19 +04:00
|
|
|
#include <stdlib.h>
|
|
|
|
|
2011-11-11 00:46:59 +04:00
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
#include <windows.h>
|
2011-11-17 23:12:19 +04:00
|
|
|
#include <io.h>
|
2011-11-11 00:46:59 +04:00
|
|
|
#else
|
|
|
|
#include <sys/mman.h>
|
2012-06-28 17:20:34 +04:00
|
|
|
#include <unistd.h>
|
2011-11-11 00:46:59 +04:00
|
|
|
#endif
|
2010-09-16 23:53:33 +04:00
|
|
|
|
|
|
|
namespace util {
|
|
|
|
|
2011-11-11 00:46:59 +04:00
|
|
|
long SizePage() {
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
SYSTEM_INFO si;
|
|
|
|
GetSystemInfo(&si);
|
|
|
|
return si.dwAllocationGranularity;
|
|
|
|
#else
|
|
|
|
return sysconf(_SC_PAGE_SIZE);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void SyncOrThrow(void *start, size_t length) {
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
UTIL_THROW_IF(!::FlushViewOfFile(start, length), ErrnoException, "Failed to sync mmap");
|
|
|
|
#else
|
|
|
|
UTIL_THROW_IF(msync(start, length, MS_SYNC), ErrnoException, "Failed to sync mmap");
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void UnmapOrThrow(void *start, size_t length) {
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
UTIL_THROW_IF(!::UnmapViewOfFile(start), ErrnoException, "Failed to unmap a file");
|
|
|
|
#else
|
|
|
|
UTIL_THROW_IF(munmap(start, length), ErrnoException, "munmap failed");
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2010-09-16 23:53:33 +04:00
|
|
|
scoped_mmap::~scoped_mmap() {
|
|
|
|
if (data_ != (void*)-1) {
|
2011-11-11 00:46:59 +04:00
|
|
|
try {
|
|
|
|
// Thanks Denis Filimonov for pointing out NFS likes msync first.
|
|
|
|
SyncOrThrow(data_, size_);
|
|
|
|
UnmapOrThrow(data_, size_);
|
|
|
|
} catch (const util::ErrnoException &e) {
|
|
|
|
std::cerr << e.what();
|
2010-11-28 05:54:56 +03:00
|
|
|
abort();
|
|
|
|
}
|
2010-09-16 23:53:33 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void scoped_memory::reset(void *data, std::size_t size, Alloc source) {
|
|
|
|
switch(source_) {
|
|
|
|
case MMAP_ALLOCATED:
|
|
|
|
scoped_mmap(data_, size_);
|
|
|
|
break;
|
|
|
|
case ARRAY_ALLOCATED:
|
|
|
|
delete [] reinterpret_cast<char*>(data_);
|
|
|
|
break;
|
|
|
|
case MALLOC_ALLOCATED:
|
|
|
|
free(data_);
|
|
|
|
break;
|
|
|
|
case NONE_ALLOCATED:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
data_ = data;
|
|
|
|
size_ = size;
|
|
|
|
source_ = source;
|
|
|
|
}
|
|
|
|
|
|
|
|
void scoped_memory::call_realloc(std::size_t size) {
|
|
|
|
assert(source_ == MALLOC_ALLOCATED || source_ == NONE_ALLOCATED);
|
|
|
|
void *new_data = realloc(data_, size);
|
|
|
|
if (!new_data) {
|
|
|
|
reset();
|
|
|
|
} else {
|
|
|
|
reset(new_data, size, MALLOC_ALLOCATED);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-11-11 00:46:59 +04:00
|
|
|
void *MapOrThrow(std::size_t size, bool for_write, int flags, bool prefault, int fd, uint64_t offset) {
|
2010-09-16 23:53:33 +04:00
|
|
|
#ifdef MAP_POPULATE // Linux specific
|
|
|
|
if (prefault) {
|
|
|
|
flags |= MAP_POPULATE;
|
|
|
|
}
|
|
|
|
#endif
|
2011-11-11 00:46:59 +04:00
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
int protectC = for_write ? PAGE_READWRITE : PAGE_READONLY;
|
|
|
|
int protectM = for_write ? FILE_MAP_WRITE : FILE_MAP_READ;
|
2012-02-28 22:58:00 +04:00
|
|
|
uint64_t total_size = size + offset;
|
|
|
|
HANDLE hMapping = CreateFileMapping((HANDLE)_get_osfhandle(fd), NULL, protectC, total_size >> 32, static_cast<DWORD>(total_size), NULL);
|
2011-11-11 00:46:59 +04:00
|
|
|
UTIL_THROW_IF(!hMapping, ErrnoException, "CreateFileMapping failed");
|
2012-02-28 22:58:00 +04:00
|
|
|
LPVOID ret = MapViewOfFile(hMapping, protectM, offset >> 32, offset, size);
|
2011-11-11 00:46:59 +04:00
|
|
|
CloseHandle(hMapping);
|
|
|
|
UTIL_THROW_IF(!ret, ErrnoException, "MapViewOfFile failed");
|
|
|
|
#else
|
2010-09-26 20:49:07 +04:00
|
|
|
int protect = for_write ? (PROT_READ | PROT_WRITE) : PROT_READ;
|
2010-09-16 23:53:33 +04:00
|
|
|
void *ret = mmap(NULL, size, protect, flags, fd, offset);
|
2011-11-11 00:46:59 +04:00
|
|
|
UTIL_THROW_IF(ret == MAP_FAILED, ErrnoException, "mmap failed for size " << size << " at offset " << offset);
|
|
|
|
#endif
|
2010-09-16 23:53:33 +04:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-11-28 05:54:56 +03:00
|
|
|
const int kFileFlags =
|
2011-11-11 00:46:59 +04:00
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
0 // MapOrThrow ignores flags on windows
|
|
|
|
#elif defined(MAP_FILE)
|
2010-11-28 05:54:56 +03:00
|
|
|
MAP_FILE | MAP_SHARED
|
|
|
|
#else
|
|
|
|
MAP_SHARED
|
|
|
|
#endif
|
|
|
|
;
|
|
|
|
|
2011-11-11 00:46:59 +04:00
|
|
|
void MapRead(LoadMethod method, int fd, uint64_t offset, std::size_t size, scoped_memory &out) {
|
2010-11-06 03:40:16 +03:00
|
|
|
switch (method) {
|
|
|
|
case LAZY:
|
2010-11-28 05:54:56 +03:00
|
|
|
out.reset(MapOrThrow(size, false, kFileFlags, false, fd, offset), size, scoped_memory::MMAP_ALLOCATED);
|
2010-11-06 03:40:16 +03:00
|
|
|
break;
|
|
|
|
case POPULATE_OR_LAZY:
|
|
|
|
#ifdef MAP_POPULATE
|
|
|
|
case POPULATE_OR_READ:
|
|
|
|
#endif
|
2010-11-28 05:54:56 +03:00
|
|
|
out.reset(MapOrThrow(size, false, kFileFlags, true, fd, offset), size, scoped_memory::MMAP_ALLOCATED);
|
2010-11-06 03:40:16 +03:00
|
|
|
break;
|
|
|
|
#ifndef MAP_POPULATE
|
|
|
|
case POPULATE_OR_READ:
|
|
|
|
#endif
|
|
|
|
case READ:
|
|
|
|
out.reset(malloc(size), size, scoped_memory::MALLOC_ALLOCATED);
|
|
|
|
if (!out.get()) UTIL_THROW(util::ErrnoException, "Allocating " << size << " bytes with malloc");
|
2011-11-11 00:46:59 +04:00
|
|
|
SeekOrThrow(fd, offset);
|
2011-09-21 20:06:48 +04:00
|
|
|
ReadOrThrow(fd, out.get(), size);
|
2010-11-06 03:40:16 +03:00
|
|
|
break;
|
|
|
|
}
|
2010-09-16 23:53:33 +04:00
|
|
|
}
|
|
|
|
|
2012-02-28 22:58:00 +04:00
|
|
|
// Allocates zeroed memory in to.
|
|
|
|
void MapAnonymous(std::size_t size, util::scoped_memory &to) {
|
|
|
|
to.reset();
|
2011-11-11 00:46:59 +04:00
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
2012-02-28 22:58:00 +04:00
|
|
|
to.reset(calloc(1, size), size, scoped_memory::MALLOC_ALLOCATED);
|
2010-09-16 23:53:33 +04:00
|
|
|
#else
|
2012-02-28 22:58:00 +04:00
|
|
|
to.reset(MapOrThrow(size, true,
|
|
|
|
# if defined(MAP_ANONYMOUS)
|
|
|
|
MAP_ANONYMOUS | MAP_PRIVATE // Linux
|
|
|
|
# else
|
2011-11-11 00:46:59 +04:00
|
|
|
MAP_ANON | MAP_PRIVATE // BSD
|
2012-02-28 22:58:00 +04:00
|
|
|
# endif
|
|
|
|
, false, -1, 0), size, scoped_memory::MMAP_ALLOCATED);
|
2010-09-16 23:53:33 +04:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2011-11-11 00:46:59 +04:00
|
|
|
void *MapZeroedWrite(int fd, std::size_t size) {
|
2011-11-17 23:12:19 +04:00
|
|
|
ResizeOrThrow(fd, 0);
|
|
|
|
ResizeOrThrow(fd, size);
|
2011-11-11 00:46:59 +04:00
|
|
|
return MapOrThrow(size, true, kFileFlags, false, fd, 0);
|
|
|
|
}
|
2011-10-23 06:37:47 +04:00
|
|
|
|
2011-11-11 00:46:59 +04:00
|
|
|
void *MapZeroedWrite(const char *name, std::size_t size, scoped_fd &file) {
|
|
|
|
file.reset(CreateOrThrow(name));
|
2010-09-16 23:53:33 +04:00
|
|
|
try {
|
2011-11-11 00:46:59 +04:00
|
|
|
return MapZeroedWrite(file.get(), size);
|
2010-09-16 23:53:33 +04:00
|
|
|
} catch (ErrnoException &e) {
|
|
|
|
e << " in file " << name;
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace util
|