* dll_init.cc (reserve_upto): Remove.

(release_upto): Ditto.
(dll_list::reserve_space): New function to reserve space needed by DLL_LOAD
dlls early in the fork process.
(dll_list::load_after_fork): Rewrite to use recursion for tracking reservations
made while trying to make dlls land where they belong.
(dll_list::load_after_fork_impl): Ditto.
(dll_list::alloc): Initialize image base field.
* dll_init.h (struct dll_list): declare new functions.
(dll::image_size): New member.
This commit is contained in:
Christopher Faylor 2011-05-28 20:27:56 +00:00
parent a92339ab63
commit 17a5c8c36e
3 changed files with 34 additions and 6 deletions

View File

@ -1,3 +1,16 @@
2011-05-28 Christopher Faylor <me.cygwin2011@cgf.cx>
* dll_init.cc (reserve_upto): Remove.
(release_upto): Ditto.
(dll_list::reserve_space): New function to reserve space needed by
DLL_LOAD dlls early in the fork process.
(dll_list::load_after_fork): Rewrite to use recursion for tracking
reservations made while trying to make dlls land where they belong.
(dll_list::load_after_fork_impl): Ditto.
(dll_list::alloc): Initialize image base field.
* dll_init.h (struct dll_list): declare new functions.
(dll::image_size): New member.
2011-05-28 Ryan Johnson <ryan.johnson@cs.utoronto.ca> 2011-05-28 Ryan Johnson <ryan.johnson@cs.utoronto.ca>
* dll_init.cc (dll_list::find_by_modname): New function to search the * dll_init.cc (dll_list::find_by_modname): New function to search the

View File

@ -173,6 +173,7 @@ dll_list::alloc (HINSTANCE h, per_process *p, dll_type type)
d->handle = h; d->handle = h;
d->has_dtors = true; d->has_dtors = true;
d->p = p; d->p = p;
d->image_size = ((pefile*)h)->optional_hdr ()->SizeOfImage;
d->ndeps = 0; d->ndeps = 0;
d->deps = NULL; d->deps = NULL;
d->modname = wcsrchr (d->name, L'\\'); d->modname = wcsrchr (d->name, L'\\');
@ -416,21 +417,33 @@ release_upto (const PWCHAR name, DWORD here)
} }
} }
/* Mark one page at "here" as reserved. This may force /* Reserve the chunk of free address space starting _here_ and (usually)
Windows NT to load a DLL elsewhere. */ covering at least _dll_size_ bytes. However, we must take care not
to clobber the dll's target address range because it often overlaps.
*/
static DWORD static DWORD
reserve_at (const PWCHAR name, DWORD here) reserve_at (const PWCHAR name, DWORD here, DWORD dll_base, DWORD dll_size)
{ {
DWORD size; DWORD size;
MEMORY_BASIC_INFORMATION mb; MEMORY_BASIC_INFORMATION mb;
if (!VirtualQuery ((void *) here, &mb, sizeof (mb))) if (!VirtualQuery ((void *) here, &mb, sizeof (mb)))
size = 64 * 1024; api_fatal ("couldn't examine memory at %08lx while mapping %W, %E",
here, name);
if (mb.State != MEM_FREE) if (mb.State != MEM_FREE)
return 0; return 0;
size = mb.RegionSize; size = mb.RegionSize;
// don't clobber the space where we want the dll to land
DWORD end = here + size;
DWORD dll_end = dll_base + dll_size;
if (dll_base < here && dll_end > here)
here = dll_end; // the dll straddles our left edge
else if (dll_base >= here && dll_base < end)
end = dll_base; // the dll overlaps partly or fully to our right
size = end - here;
if (!VirtualAlloc ((void *) here, size, MEM_RESERVE, PAGE_NOACCESS)) if (!VirtualAlloc ((void *) here, size, MEM_RESERVE, PAGE_NOACCESS))
api_fatal ("couldn't allocate memory %p(%d) for '%W' alignment, %E\n", api_fatal ("couldn't allocate memory %p(%d) for '%W' alignment, %E\n",
here, size, name); here, size, name);
@ -508,7 +521,8 @@ dll_list::load_after_fork (HANDLE parent)
can in the child, due to differences in the load ordering. can in the child, due to differences in the load ordering.
Block memory at it's preferred address and try again. */ Block memory at it's preferred address and try again. */
if ((DWORD) h > (DWORD) d->handle) if ((DWORD) h > (DWORD) d->handle)
preferred_block = reserve_at (d->name, (DWORD) h); preferred_block = reserve_at (d->name, (DWORD) h,
(DWORD) d->handle, d->image_size);
} }
} }

View File

@ -52,6 +52,7 @@ struct dll
int count; int count;
bool has_dtors; bool has_dtors;
dll_type type; dll_type type;
DWORD image_size;
long ndeps; long ndeps;
dll** deps; dll** deps;
PWCHAR modname; PWCHAR modname;