diff options
| author | Ramsay Jones <ramsay@ramsay1.demon.co.uk> | 2009-03-20 19:51:48 +0000 |
|---|---|---|
| committer | Shawn O. Pearce <spearce@spearce.org> | 2009-03-20 14:39:02 -0700 |
| commit | 79ca2edcd4e7a801400fa33c2c705b8f03a5d7f5 (patch) | |
| tree | d48d51dea1dd8e90bb8c960e68afe72cabc5f6aa /src/win32 | |
| parent | 7a6cf8153799563464d6f1761e55352c327b4122 (diff) | |
| download | libgit2-79ca2edcd4e7a801400fa33c2c705b8f03a5d7f5.tar.gz | |
win32: Add routines to abstract memory-mapped file functions
In particular, the git__mmap() and git__munmap() routines provide
the interface to platform specific memory-mapped file facilities.
We provide implementations for unix and win32, which can be found
in their own sub-directories.
Signed-off-by: Ramsay Jones <ramsay@ramsay1.demon.co.uk>
Signed-off-by: Shawn O. Pearce <spearce@spearce.org>
Diffstat (limited to 'src/win32')
| -rw-r--r-- | src/win32/map.c | 125 |
1 files changed, 125 insertions, 0 deletions
diff --git a/src/win32/map.c b/src/win32/map.c new file mode 100644 index 000000000..7d685c722 --- /dev/null +++ b/src/win32/map.c @@ -0,0 +1,125 @@ + +#include "map.h" +#include <errno.h> + + +static DWORD get_page_size(void) +{ + static DWORD page_size; + SYSTEM_INFO sys; + + if (!page_size) { + GetSystemInfo(&sys); + page_size = sys.dwAllocationGranularity; + } + + return page_size; +} + +int git__mmap(git_map *out, size_t len, int prot, int flags, int fd, off_t offset) +{ + HANDLE fh = (HANDLE)_get_osfhandle(fd); + DWORD page_size = get_page_size(); + DWORD fmap_prot = 0; + DWORD view_prot = 0; + DWORD off_low = 0; + DWORD off_hi = 0; + off_t page_start; + off_t page_offset; + + assert((out != NULL) && (len > 0)); + + if ((out == NULL) || (len == 0)) { + errno = EINVAL; + return GIT_ERROR; + } + + out->data = NULL; + out->len = 0; + out->fmh = NULL; + + if (fh == INVALID_HANDLE_VALUE) { + errno = EBADF; + return GIT_ERROR; + } + + if (prot & GIT_PROT_WRITE) + fmap_prot |= PAGE_READWRITE; + else if (prot & GIT_PROT_READ) + fmap_prot |= PAGE_READONLY; + else { + errno = EINVAL; + return GIT_ERROR; + } + + if (prot & GIT_PROT_WRITE) + view_prot |= FILE_MAP_WRITE; + if (prot & GIT_PROT_READ) + view_prot |= FILE_MAP_READ; + + if (flags & GIT_MAP_FIXED) { + errno = EINVAL; + return GIT_ERROR; + } + + page_start = (offset / page_size) * page_size; + page_offset = offset - page_start; + + if (page_offset != 0) { /* offset must be multiple of page size */ + errno = EINVAL; + return GIT_ERROR; + } + + out->fmh = CreateFileMapping(fh, NULL, fmap_prot, 0, 0, NULL); + if (!out->fmh || out->fmh == INVALID_HANDLE_VALUE) { + /* errno = ? */ + out->fmh = NULL; + return GIT_ERROR; + } + + off_low = (DWORD)(page_start); + if (sizeof(off_t) > 4) + off_hi = (DWORD)(page_start >> 32); + out->data = MapViewOfFile(out->fmh, view_prot, off_hi, off_low, len); + if (!out->data) { + /* errno = ? */ + CloseHandle(out->fmh); + out->fmh = NULL; + return GIT_ERROR; + } + out->len = len; + + return GIT_SUCCESS; +} + +int git__munmap(git_map *map) +{ + assert(map != NULL); + + if (!map) + return GIT_ERROR; + + if (map->data) { + if (!UnmapViewOfFile(map->data)) { + /* errno = ? */ + CloseHandle(map->fmh); + map->data = NULL; + map->fmh = NULL; + return GIT_ERROR; + } + map->data = NULL; + } + + if (map->fmh) { + if (!CloseHandle(map->fmh)) { + /* errno = ? */ + map->fmh = NULL; + return GIT_ERROR; + } + map->fmh = NULL; + } + + return GIT_SUCCESS; +} + + |
