* dll_init.cc (dll_list::find_by_modname): New function to search the dll list
for a module name only (no path). (dll_list::alloc): Initialize newly-added members of struct dll. (dll_list::append): New function to factor out the append operation (used by dll_list::topsort). (dll_list::populate_deps): New function to identify dll dependencies. (dll_list::topsort): New function to sort the dll list topologically by dependencies. (dll_list::topsort_visit): New helper function for the above. * dll_init.h (dll::ndeps): New class member. (dll::deps): Ditto. (dll::modname): Ditto. (dll_list::find_by_modname): New function related to topsort. (dll_list::populate_all_deps): Ditto. (dll_list::populate_deps): Ditto. (dll_list::topsort): Ditto. (dll_list::topsort_visit): Ditto. (dll_list::append): Ditto. (pefile): New struct allowing simple introspection of dll images. * fork.cc (fork): Topologically sort the dll list before forking * child_info.h (CURR_CHILD_INFO_MAGIC): Refresh. (child_info::refresh_cygheap): New function. * spawn.cc (spawn_guts): Call refresh_cygheap before creating a new process to ensure that cygheap_max is up-to-date. * fork.cc (frok::parent): Ditto.
This commit is contained in:
parent
3a88e09e7f
commit
977ad5434c
|
@ -1,3 +1,34 @@
|
||||||
|
2011-05-30 Ryan Johnson <ryan.johnson@cs.utoronto.ca>
|
||||||
|
|
||||||
|
* dll_init.cc (dll_list::find_by_modname): New function to search the
|
||||||
|
dll list for a module name only (no path).
|
||||||
|
(dll_list::alloc): Initialize newly-added members of struct dll.
|
||||||
|
(dll_list::append): New function to factor out the append operation
|
||||||
|
(used by dll_list::topsort).
|
||||||
|
(dll_list::populate_deps): New function to identify dll dependencies.
|
||||||
|
(dll_list::topsort): New function to sort the dll list topologically by
|
||||||
|
dependencies.
|
||||||
|
(dll_list::topsort_visit): New helper function for the above.
|
||||||
|
* dll_init.h (dll::ndeps): New class member.
|
||||||
|
(dll::deps): Ditto.
|
||||||
|
(dll::modname): Ditto.
|
||||||
|
(dll_list::find_by_modname): New function related to topsort.
|
||||||
|
(dll_list::populate_all_deps): Ditto.
|
||||||
|
(dll_list::populate_deps): Ditto.
|
||||||
|
(dll_list::topsort): Ditto.
|
||||||
|
(dll_list::topsort_visit): Ditto.
|
||||||
|
(dll_list::append): Ditto.
|
||||||
|
(pefile): New struct allowing simple introspection of dll images.
|
||||||
|
* fork.cc (fork): Topologically sort the dll list before forking
|
||||||
|
|
||||||
|
2011-05-30 Christopher Faylor <me.cygwin2011@cgf.cx>
|
||||||
|
|
||||||
|
* child_info.h (CURR_CHILD_INFO_MAGIC): Refresh.
|
||||||
|
(child_info::refresh_cygheap): New function.
|
||||||
|
* spawn.cc (spawn_guts): Call refresh_cygheap before creating a new
|
||||||
|
process to ensure that cygheap_max is up-to-date.
|
||||||
|
* fork.cc (frok::parent): Ditto.
|
||||||
|
|
||||||
2011-05-30 Christopher Faylor <me.cygwin2011@cgf.cx>
|
2011-05-30 Christopher Faylor <me.cygwin2011@cgf.cx>
|
||||||
|
|
||||||
* cygheap.cc (cygheap_dummy): Rename from cygheap_at_start.
|
* cygheap.cc (cygheap_dummy): Rename from cygheap_at_start.
|
||||||
|
|
|
@ -38,7 +38,7 @@ enum child_status
|
||||||
#define EXEC_MAGIC_SIZE sizeof(child_info)
|
#define EXEC_MAGIC_SIZE sizeof(child_info)
|
||||||
|
|
||||||
/* Change this value if you get a message indicating that it is out-of-sync. */
|
/* Change this value if you get a message indicating that it is out-of-sync. */
|
||||||
#define CURR_CHILD_INFO_MAGIC 0x76ca2aaeU
|
#define CURR_CHILD_INFO_MAGIC 0xeef5640dU
|
||||||
|
|
||||||
/* NOTE: Do not make gratuitous changes to the names or organization of the
|
/* NOTE: Do not make gratuitous changes to the names or organization of the
|
||||||
below class. The layout is checksummed to determine compatibility between
|
below class. The layout is checksummed to determine compatibility between
|
||||||
|
@ -65,6 +65,7 @@ public:
|
||||||
child_info (unsigned, child_info_types, bool);
|
child_info (unsigned, child_info_types, bool);
|
||||||
child_info (): subproc_ready (NULL), parent (NULL) {}
|
child_info (): subproc_ready (NULL), parent (NULL) {}
|
||||||
~child_info ();
|
~child_info ();
|
||||||
|
void refresh_cygheap () { cygheap_max = ::cygheap_max; }
|
||||||
void ready (bool);
|
void ready (bool);
|
||||||
bool sync (int, HANDLE&, DWORD) __attribute__ ((regparm (3)));
|
bool sync (int, HANDLE&, DWORD) __attribute__ ((regparm (3)));
|
||||||
DWORD proc_retry (HANDLE) __attribute__ ((regparm (2)));
|
DWORD proc_retry (HANDLE) __attribute__ ((regparm (2)));
|
||||||
|
|
|
@ -116,6 +116,18 @@ dll_list::operator[] (const PWCHAR name)
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Look for a dll based on is short name only (no path) */
|
||||||
|
dll *
|
||||||
|
dll_list::find_by_modname (const PWCHAR name)
|
||||||
|
{
|
||||||
|
dll *d = &start;
|
||||||
|
while ((d = d->next) != NULL)
|
||||||
|
if (!wcscasecmp (name, d->modname))
|
||||||
|
return d;
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
#define RETRIES 1000
|
#define RETRIES 1000
|
||||||
|
|
||||||
/* Allocate space for a dll struct. */
|
/* Allocate space for a dll struct. */
|
||||||
|
@ -161,15 +173,14 @@ 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->ndeps = 0;
|
||||||
|
d->deps = NULL;
|
||||||
|
d->modname = wcsrchr (d->name, L'\\');
|
||||||
|
if (d->modname)
|
||||||
|
d->modname++;
|
||||||
d->image_size = ((pefile*)h)->optional_hdr ()->SizeOfImage;
|
d->image_size = ((pefile*)h)->optional_hdr ()->SizeOfImage;
|
||||||
d->type = type;
|
d->type = type;
|
||||||
if (end == NULL)
|
append (d);
|
||||||
end = &start; /* Point to "end" of dll chain. */
|
|
||||||
end->next = d; /* Standard linked list stuff. */
|
|
||||||
d->next = NULL;
|
|
||||||
d->prev = end;
|
|
||||||
end = d;
|
|
||||||
tot++;
|
|
||||||
if (type == DLL_LOAD)
|
if (type == DLL_LOAD)
|
||||||
loaded_dlls++;
|
loaded_dlls++;
|
||||||
}
|
}
|
||||||
|
@ -178,6 +189,119 @@ dll_list::alloc (HINSTANCE h, per_process *p, dll_type type)
|
||||||
return d;
|
return d;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
dll_list::append (dll* d)
|
||||||
|
{
|
||||||
|
if (end == NULL)
|
||||||
|
end = &start; /* Point to "end" of dll chain. */
|
||||||
|
end->next = d; /* Standard linked list stuff. */
|
||||||
|
d->next = NULL;
|
||||||
|
d->prev = end;
|
||||||
|
end = d;
|
||||||
|
tot++;
|
||||||
|
}
|
||||||
|
|
||||||
|
void dll_list::populate_deps (dll* d)
|
||||||
|
{
|
||||||
|
WCHAR wmodname[NT_MAX_PATH];
|
||||||
|
pefile* pef = (pefile*) d->handle;
|
||||||
|
PIMAGE_DATA_DIRECTORY dd = pef->idata_dir (IMAGE_DIRECTORY_ENTRY_IMPORT);
|
||||||
|
/* Annoyance: calling crealloc with a NULL pointer will use the
|
||||||
|
wrong heap and crash, so we have to replicate some code */
|
||||||
|
long maxdeps = 4;
|
||||||
|
d->deps = (dll**) cmalloc (HEAP_2_DLL, maxdeps*sizeof (dll*));
|
||||||
|
d->ndeps = 0;
|
||||||
|
for (PIMAGE_IMPORT_DESCRIPTOR id=
|
||||||
|
(PIMAGE_IMPORT_DESCRIPTOR) pef->rva (dd->VirtualAddress);
|
||||||
|
dd->Size && id->Name;
|
||||||
|
id++)
|
||||||
|
{
|
||||||
|
char* modname = pef->rva (id->Name);
|
||||||
|
sys_mbstowcs (wmodname, NT_MAX_PATH, modname);
|
||||||
|
if (dll* dep = find_by_modname (wmodname))
|
||||||
|
{
|
||||||
|
if (d->ndeps >= maxdeps)
|
||||||
|
{
|
||||||
|
maxdeps = 2*(1+maxdeps);
|
||||||
|
d->deps = (dll**) crealloc (d->deps, maxdeps*sizeof (dll*));
|
||||||
|
}
|
||||||
|
d->deps[d->ndeps++] = dep;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* add one to differentiate no deps from unknown */
|
||||||
|
d->ndeps++;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void
|
||||||
|
dll_list::topsort ()
|
||||||
|
{
|
||||||
|
/* Anything to do? */
|
||||||
|
if (!end)
|
||||||
|
return;
|
||||||
|
|
||||||
|
/* make sure we have all the deps available */
|
||||||
|
dll* d = &start;
|
||||||
|
while ((d = d->next))
|
||||||
|
if (!d->ndeps)
|
||||||
|
populate_deps (d);
|
||||||
|
|
||||||
|
/* unlink head and tail pointers so the sort can rebuild the list */
|
||||||
|
d = start.next;
|
||||||
|
start.next = end = NULL;
|
||||||
|
topsort_visit (d, true);
|
||||||
|
|
||||||
|
/* clear node markings made by the sort */
|
||||||
|
d = &start;
|
||||||
|
while ((d = d->next))
|
||||||
|
{
|
||||||
|
debug_printf ("%W", d->modname);
|
||||||
|
for (int i=1; i < -d->ndeps; i++)
|
||||||
|
debug_printf ("-> %W", d->deps[i-1]->modname);
|
||||||
|
|
||||||
|
/* It would be really nice to be able to keep this information
|
||||||
|
around for next time, but we don't have an easy way to
|
||||||
|
invalidate cached dependencies when a module unloads. */
|
||||||
|
d->ndeps = 0;
|
||||||
|
cfree (d->deps);
|
||||||
|
d->deps = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* A recursive in-place topological sort. The result is ordered so that
|
||||||
|
dependencies of a dll appear before it in the list.
|
||||||
|
|
||||||
|
NOTE: this algorithm is guaranteed to terminate with a "partial
|
||||||
|
order" of dlls but does not do anything smart about cycles: an
|
||||||
|
arbitrary dependent dll will necessarily appear first. Perhaps not
|
||||||
|
surprisingly, Windows ships several dlls containing dependency
|
||||||
|
cycles, including SspiCli/RPCRT4.dll and a lovely tangle involving
|
||||||
|
USP10/LPK/GDI32/USER32.dll). Fortunately, we don't care about
|
||||||
|
Windows DLLs here, and cygwin dlls should behave better */
|
||||||
|
void
|
||||||
|
dll_list::topsort_visit (dll* d, bool seek_tail)
|
||||||
|
{
|
||||||
|
/* Recurse to the end of the dll chain, then visit nodes as we
|
||||||
|
unwind. We do this because once we start visiting nodes we can no
|
||||||
|
longer trust any _next_ pointers.
|
||||||
|
|
||||||
|
We "mark" visited nodes (to avoid revisiting them) by negating
|
||||||
|
ndeps (undone once the sort completes). */
|
||||||
|
if (seek_tail && d->next)
|
||||||
|
topsort_visit (d->next, true);
|
||||||
|
|
||||||
|
if (d->ndeps > 0)
|
||||||
|
{
|
||||||
|
d->ndeps = -d->ndeps;
|
||||||
|
for (long i=1; i < -d->ndeps; i++)
|
||||||
|
topsort_visit (d->deps[i-1], false);
|
||||||
|
|
||||||
|
append (d);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
dll *
|
dll *
|
||||||
dll_list::find (void *retaddr)
|
dll_list::find (void *retaddr)
|
||||||
{
|
{
|
||||||
|
|
|
@ -52,6 +52,9 @@ struct dll
|
||||||
int count;
|
int count;
|
||||||
bool has_dtors;
|
bool has_dtors;
|
||||||
dll_type type;
|
dll_type type;
|
||||||
|
long ndeps;
|
||||||
|
dll** deps;
|
||||||
|
PWCHAR modname;
|
||||||
DWORD image_size;
|
DWORD image_size;
|
||||||
WCHAR name[1];
|
WCHAR name[1];
|
||||||
void detach ();
|
void detach ();
|
||||||
|
@ -85,6 +88,13 @@ public:
|
||||||
void detach (void *);
|
void detach (void *);
|
||||||
void init ();
|
void init ();
|
||||||
void load_after_fork (HANDLE);
|
void load_after_fork (HANDLE);
|
||||||
|
dll *find_by_modname (const PWCHAR name);
|
||||||
|
void populate_all_deps ();
|
||||||
|
void populate_deps (dll* d);
|
||||||
|
void topsort ();
|
||||||
|
void topsort_visit (dll* d, bool goto_tail);
|
||||||
|
void append (dll* d);
|
||||||
|
|
||||||
dll *inext ()
|
dll *inext ()
|
||||||
{
|
{
|
||||||
while ((hold = hold->next))
|
while ((hold = hold->next))
|
||||||
|
|
|
@ -397,6 +397,7 @@ frok::parent (volatile char * volatile stack_here)
|
||||||
/* Remove impersonation */
|
/* Remove impersonation */
|
||||||
cygheap->user.deimpersonate ();
|
cygheap->user.deimpersonate ();
|
||||||
fix_impersonation = true;
|
fix_impersonation = true;
|
||||||
|
ch.refresh_cygheap ();
|
||||||
|
|
||||||
while (1)
|
while (1)
|
||||||
{
|
{
|
||||||
|
@ -601,7 +602,6 @@ extern "C" int
|
||||||
fork ()
|
fork ()
|
||||||
{
|
{
|
||||||
frok grouped;
|
frok grouped;
|
||||||
/* No cygheap allocation beyond this point. */
|
|
||||||
|
|
||||||
debug_printf ("entering");
|
debug_printf ("entering");
|
||||||
grouped.load_dlls = 0;
|
grouped.load_dlls = 0;
|
||||||
|
@ -635,6 +635,11 @@ fork ()
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Put the dll list in topological dependency ordering, in
|
||||||
|
hopes that the child will have a better shot at loading dlls
|
||||||
|
properly if it only has to deal with one at a time. */
|
||||||
|
dlls.topsort ();
|
||||||
|
|
||||||
ischild = !!setjmp (grouped.ch.jmp);
|
ischild = !!setjmp (grouped.ch.jmp);
|
||||||
|
|
||||||
volatile char * volatile esp;
|
volatile char * volatile esp;
|
||||||
|
|
|
@ -560,6 +560,7 @@ spawn_guts (const char *prog_arg, const char *const *argv,
|
||||||
|| cygheap->fdtab.need_fixup_before ()))
|
|| cygheap->fdtab.need_fixup_before ()))
|
||||||
c_flags |= CREATE_SUSPENDED;
|
c_flags |= CREATE_SUSPENDED;
|
||||||
|
|
||||||
|
ch.refresh_cygheap ();
|
||||||
/* When ruid != euid we create the new process under the current original
|
/* When ruid != euid we create the new process under the current original
|
||||||
account and impersonate in child, this way maintaining the different
|
account and impersonate in child, this way maintaining the different
|
||||||
effective vs. real ids.
|
effective vs. real ids.
|
||||||
|
|
Loading…
Reference in New Issue