[10073] | 1 | /* $Id: mmapdup.cpp,v 1.3 2003-05-06 12:06:11 sandervl Exp $ */
|
---|
[9946] | 2 |
|
---|
| 3 | /*
|
---|
| 4 | * Win32 Memory mapped file & view classes
|
---|
| 5 | *
|
---|
| 6 | * Copyright 1999-2003 Sander van Leeuwen (sandervl@xs4all.nl)
|
---|
| 7 | *
|
---|
| 8 | * NOTE: Memory mapping DOES NOT work when kernel-mode code causes
|
---|
| 9 | * a pagefault in the memory mapped object. (exceptions aren't
|
---|
| 10 | * dispatched to our exception handler until after the kernel mode
|
---|
| 11 | * call returns (too late))
|
---|
| 12 | *
|
---|
| 13 | * NOTE: Are apps allowed to change the protection flags of memory mapped pages?
|
---|
| 14 | * I'm assuming they aren't for now.
|
---|
| 15 | *
|
---|
| 16 | * TODO: Handles returned should be usable by all apis that accept file handles
|
---|
| 17 | * TODO: Sharing memory mapped files between multiple processes
|
---|
| 18 | * TODO: Memory mapped files with views that extend the file (not 100% correct now)
|
---|
| 19 | * TODO: Suspend all threads when a page is committed (possible that another thread
|
---|
| 20 | * accesses the same memory before the page is read from disk
|
---|
| 21 | * TODO: File maps for new files (must select an initial size)!
|
---|
| 22 | *
|
---|
| 23 | * Project Odin Software License can be found in LICENSE.TXT
|
---|
| 24 | *
|
---|
| 25 | */
|
---|
| 26 | #include <os2win.h>
|
---|
| 27 | #include <stdio.h>
|
---|
| 28 | #include <stdlib.h>
|
---|
| 29 | #include <string.h>
|
---|
[21916] | 30 | #include <win/virtual.h>
|
---|
[9946] | 31 | #include <odincrt.h>
|
---|
| 32 | #include <handlemanager.h>
|
---|
| 33 | #include "mmap.h"
|
---|
| 34 | #include "oslibdos.h"
|
---|
| 35 | #include "oslibmem.h"
|
---|
[21916] | 36 | #include "winimagepeldr.h"
|
---|
| 37 | #include <custombuild.h>
|
---|
[9946] | 38 |
|
---|
| 39 | #define DBG_LOCALLOG DBG_mmapdup
|
---|
| 40 | #include "dbglocal.h"
|
---|
| 41 |
|
---|
| 42 |
|
---|
| 43 | //******************************************************************************
|
---|
| 44 | // Class Win32MemMapDup
|
---|
| 45 | //
|
---|
| 46 | // Duplicate memory mapping class (duplicate map with different protection flags
|
---|
| 47 | // associated with an existing memory map)
|
---|
| 48 | //
|
---|
| 49 | //******************************************************************************
|
---|
[21916] | 50 | Win32MemMapDup::Win32MemMapDup(Win32MemMap *parent, HANDLE hFile, ULONG size,
|
---|
[9946] | 51 | ULONG fdwProtect, LPSTR lpszName) :
|
---|
| 52 | Win32MemMap(hFile, size, fdwProtect, lpszName)
|
---|
| 53 | {
|
---|
| 54 | this->parent = parent;
|
---|
| 55 | hOrgMemFile = -1; //we're a duplicate
|
---|
| 56 | parent->AddRef();
|
---|
| 57 | }
|
---|
| 58 | //******************************************************************************
|
---|
| 59 | //******************************************************************************
|
---|
| 60 | Win32MemMapDup::~Win32MemMapDup()
|
---|
| 61 | {
|
---|
| 62 | parent->Release();
|
---|
| 63 | }
|
---|
| 64 | //******************************************************************************
|
---|
| 65 | //******************************************************************************
|
---|
| 66 | BOOL Win32MemMapDup::Init(DWORD aMSize)
|
---|
| 67 | {
|
---|
| 68 | //copy values from original map
|
---|
[10073] | 69 | mSize = parent->getMapSize();
|
---|
| 70 | //can't use hMemFile or else we'll close this file handle in the memmap dtor!!
|
---|
| 71 | hDupMemFile = parent->getFileHandle();
|
---|
[9946] | 72 |
|
---|
| 73 | //If the parent is a readonly map and we allow write access, then we must
|
---|
| 74 | //override our parent's protection flags
|
---|
| 75 | if((getProtFlags() & PAGE_READWRITE) && (parent->getProtFlags() == PAGE_READONLY))
|
---|
| 76 | {
|
---|
| 77 | parent->setProtFlags(PAGE_READWRITE);
|
---|
| 78 | }
|
---|
| 79 | return TRUE;
|
---|
| 80 | }
|
---|
| 81 | //******************************************************************************
|
---|
| 82 | //******************************************************************************
|
---|
| 83 | BOOL Win32MemMapDup::flushView(ULONG viewaddr, ULONG offset, ULONG cbFlush)
|
---|
| 84 | {
|
---|
| 85 | return parent->flushView(viewaddr, offset, cbFlush);
|
---|
| 86 | }
|
---|
| 87 | //******************************************************************************
|
---|
| 88 | // Win32MemMapDup::mapViewOfFile
|
---|
| 89 | //
|
---|
| 90 | // Map the view identified by addr. Create a view with our parent as the parent
|
---|
| 91 | // map and ourselves as the ower.
|
---|
| 92 | //
|
---|
| 93 | // View parent = memory map that contains the original memory map
|
---|
| 94 | // View owner = duplicate memory map that created this view (can be NULL)
|
---|
| 95 | //
|
---|
| 96 | // Parameters:
|
---|
| 97 | //
|
---|
| 98 | // ULONG size - size of view
|
---|
| 99 | // ULONG offset - offset in memory map
|
---|
| 100 | // ULONG fdwAccess - access flags
|
---|
| 101 | // FILE_MAP_WRITE, FILE_MAP_READ, FILE_MAP_COPY
|
---|
| 102 | // FILE_MAP_ALL_ACCESS
|
---|
| 103 | //
|
---|
| 104 | // Returns:
|
---|
| 105 | // <>NULL - success, view address
|
---|
| 106 | // NULL - failure
|
---|
| 107 | //
|
---|
| 108 | //******************************************************************************
|
---|
| 109 | LPVOID Win32MemMapDup::mapViewOfFile(ULONG size, ULONG offset, ULONG fdwAccess)
|
---|
| 110 | {
|
---|
| 111 | DWORD processId = GetCurrentProcessId();
|
---|
| 112 |
|
---|
| 113 | mapMutex.enter();
|
---|
| 114 | ULONG memFlags = (mProtFlags & (PAGE_READONLY | PAGE_READWRITE | PAGE_WRITECOPY));
|
---|
| 115 | Win32MemMapView *mapview;
|
---|
| 116 |
|
---|
| 117 | //@@@PH: if(fdwAccess & ~(FILE_MAP_WRITE|FILE_MAP_READ|FILE_MAP_COPY))
|
---|
| 118 | // Docs say FILE_MAP_ALL_ACCESS is same as FILE_MAP_WRITE. Doesn't match reality though.
|
---|
| 119 | if(fdwAccess & ~FILE_MAP_ALL_ACCESS)
|
---|
| 120 | goto parmfail;
|
---|
| 121 | if((fdwAccess & FILE_MAP_WRITE) && !(mProtFlags & PAGE_READWRITE))
|
---|
| 122 | goto parmfail;
|
---|
| 123 | if((fdwAccess & FILE_MAP_READ) && !(mProtFlags & (PAGE_READWRITE|PAGE_READONLY)))
|
---|
| 124 | goto parmfail;
|
---|
[21916] | 125 |
|
---|
[9946] | 126 | if (fdwAccess != FILE_MAP_ALL_ACCESS)
|
---|
| 127 | if((fdwAccess & FILE_MAP_COPY) && !(mProtFlags & PAGE_WRITECOPY))
|
---|
| 128 | goto parmfail;
|
---|
| 129 |
|
---|
[10073] | 130 | if(offset+size > mSize && (!(fdwAccess & FILE_MAP_WRITE) || hDupMemFile == -1))
|
---|
[9946] | 131 | goto parmfail;
|
---|
| 132 |
|
---|
| 133 | //SvL: TODO: Doesn't work for multiple views
|
---|
| 134 | if(offset+size > mSize) {
|
---|
| 135 | mSize = offset+size;
|
---|
| 136 | }
|
---|
| 137 |
|
---|
| 138 |
|
---|
| 139 | if(parent->allocateMap() == FALSE) {
|
---|
| 140 | SetLastError(ERROR_NOT_ENOUGH_MEMORY);
|
---|
| 141 | goto fail;
|
---|
| 142 | }
|
---|
| 143 |
|
---|
| 144 | mapview = new Win32MemMapView(parent, offset, (size == 0) ? (mSize - offset) : size, fdwAccess, this);
|
---|
| 145 | if(mapview == NULL) {
|
---|
| 146 | goto fail;
|
---|
| 147 | }
|
---|
| 148 | if(mapview->everythingOk() == FALSE) {
|
---|
| 149 | dprintf(("Win32MemMapDup::mapViewOfFile: !mapview->everythingOk"));
|
---|
| 150 | delete mapview;
|
---|
| 151 | goto fail;
|
---|
| 152 | }
|
---|
| 153 |
|
---|
| 154 | mapMutex.leave();
|
---|
| 155 | SetLastError(ERROR_SUCCESS);
|
---|
| 156 | return mapview->getViewAddr();
|
---|
| 157 |
|
---|
| 158 | parmfail:
|
---|
| 159 | dprintf(("Win32MemMapDup::mapViewOfFile: invalid parameter (ERROR_ACCESS_DENIED)"));
|
---|
| 160 | //NT4 SP6 returns ERROR_ACCESS_DENIED for most invalid parameters
|
---|
| 161 | SetLastError(ERROR_ACCESS_DENIED);
|
---|
| 162 | fail:
|
---|
| 163 | mapMutex.leave();
|
---|
| 164 | return 0;
|
---|
| 165 | }
|
---|
| 166 | //******************************************************************************
|
---|
| 167 | // Win32MemMapDup::unmapViewOfFile
|
---|
| 168 | //
|
---|
| 169 | // Unmap the view identified by addr by calling the parent and cleaning up
|
---|
| 170 | // the duplicate object
|
---|
| 171 | //
|
---|
| 172 | // Parameters:
|
---|
| 173 | //
|
---|
| 174 | // LPVOID addr - view address; doesn't need to be the address
|
---|
| 175 | // returned by MapViewOfFile(Ex) (as MSDN clearly says);
|
---|
| 176 | // can be any address within the view range
|
---|
| 177 | //
|
---|
| 178 | // Returns:
|
---|
| 179 | // TRUE - success
|
---|
| 180 | // FALSE - failure
|
---|
| 181 | //
|
---|
| 182 | //******************************************************************************
|
---|
| 183 | BOOL Win32MemMapDup::unmapViewOfFile(LPVOID addr)
|
---|
| 184 | {
|
---|
| 185 | BOOL ret;
|
---|
| 186 |
|
---|
| 187 | dprintf(("Win32MemMapDup::unmapViewOfFile %x (nrmaps=%d)", addr, nrMappings));
|
---|
| 188 |
|
---|
| 189 | mapMutex.enter();
|
---|
| 190 |
|
---|
| 191 | if(nrMappings == 0)
|
---|
| 192 | goto fail;
|
---|
| 193 |
|
---|
| 194 | ret = parent->unmapViewOfFile(addr);
|
---|
| 195 |
|
---|
| 196 | mapMutex.leave();
|
---|
| 197 |
|
---|
| 198 | SetLastError(ERROR_SUCCESS);
|
---|
| 199 | return TRUE;
|
---|
| 200 | fail:
|
---|
| 201 | mapMutex.leave();
|
---|
| 202 | SetLastError(ERROR_INVALID_ADDRESS);
|
---|
| 203 | return FALSE;
|
---|
| 204 | }
|
---|
| 205 | //******************************************************************************
|
---|
| 206 | // Win32MemMapDup::commitPage
|
---|
| 207 | //
|
---|
| 208 | // Handle a pagefault for a duplicate view
|
---|
| 209 | //
|
---|
| 210 | // Parameters:
|
---|
| 211 | //
|
---|
| 212 | // ULONG ulFaultAddr - exception address
|
---|
| 213 | // ULONG ulOffset - offset in memory map
|
---|
| 214 | // BOOL fWriteAccess - TRUE -> write exception
|
---|
| 215 | // FALSE -> read exception
|
---|
| 216 | //
|
---|
| 217 | // Returns:
|
---|
| 218 | // TRUE - success
|
---|
| 219 | // FALSE - failure
|
---|
| 220 | //
|
---|
| 221 | //******************************************************************************
|
---|
| 222 | BOOL Win32MemMapDup::commitPage(ULONG ulFaultAddr, ULONG offset, BOOL fWriteAccess, int nrpages)
|
---|
| 223 | {
|
---|
[21916] | 224 | if(mProtFlags & PAGE_WRITECOPY)
|
---|
[9971] | 225 | {//this is a COW map, call commitGuardPage
|
---|
| 226 | return commitGuardPage(ulFaultAddr, offset, fWriteAccess);
|
---|
| 227 | }
|
---|
[9946] | 228 | return parent->commitPage(ulFaultAddr, offset, fWriteAccess, nrpages);
|
---|
| 229 | }
|
---|
| 230 | //******************************************************************************
|
---|
| 231 | // Win32MemMapDup::commitGuardPage
|
---|
| 232 | //
|
---|
[9971] | 233 | // Handle a guard page exception for a copy-on-write view
|
---|
[9946] | 234 | //
|
---|
| 235 | // Parameters:
|
---|
| 236 | //
|
---|
| 237 | // ULONG ulFaultAddr - exception address
|
---|
| 238 | // ULONG ulOffset - offset in memory map
|
---|
| 239 | // BOOL fWriteAccess - TRUE -> write exception
|
---|
| 240 | // FALSE -> read exception
|
---|
| 241 | //
|
---|
| 242 | // Returns:
|
---|
| 243 | // TRUE - success
|
---|
| 244 | // FALSE - failure
|
---|
| 245 | //
|
---|
| 246 | //******************************************************************************
|
---|
[21916] | 247 | BOOL Win32MemMapDup::commitGuardPage(ULONG ulFaultAddr, ULONG ulOffset,
|
---|
[9946] | 248 | BOOL fWriteAccess)
|
---|
| 249 | {
|
---|
| 250 | return parent->commitGuardPage(ulFaultAddr, ulOffset, fWriteAccess);
|
---|
| 251 | }
|
---|
| 252 | //******************************************************************************
|
---|
| 253 | // Win32MemMapDup::invalidatePages
|
---|
| 254 | //
|
---|
| 255 | // Invalidate map pages. (called by WriteFile)
|
---|
| 256 | //
|
---|
| 257 | // Parameters:
|
---|
| 258 | //
|
---|
| 259 | // ULONG offset - offset in memory map
|
---|
| 260 | // ULONG size - invalid range size
|
---|
| 261 | //
|
---|
| 262 | // Returns:
|
---|
| 263 | // TRUE - success
|
---|
| 264 | // FALSE - failure
|
---|
| 265 | //
|
---|
| 266 | //******************************************************************************
|
---|
| 267 | BOOL Win32MemMapDup::invalidatePages(ULONG offset, ULONG size)
|
---|
| 268 | {
|
---|
| 269 | return parent->invalidatePages(offset, size);
|
---|
| 270 | }
|
---|
| 271 | //******************************************************************************
|
---|
| 272 | //******************************************************************************
|
---|