diff options
author | Christopher Faylor <me@cgf.cx> | 2001-09-14 00:49:00 +0000 |
---|---|---|
committer | Christopher Faylor <me@cgf.cx> | 2001-09-14 00:49:00 +0000 |
commit | e2e078278cc951344fc5c0a976715bd1fe11f2d8 (patch) | |
tree | afd6de5d0a5ee0c2778c384423f7a59fd38ccd44 /winsup/cygwin/cygheap.cc | |
parent | 3e2d8af0b9a8df7b9b1230d35f3e838d3efbfb33 (diff) | |
download | cygnal-e2e078278cc951344fc5c0a976715bd1fe11f2d8.tar.gz cygnal-e2e078278cc951344fc5c0a976715bd1fe11f2d8.tar.bz2 cygnal-e2e078278cc951344fc5c0a976715bd1fe11f2d8.zip |
* cygheap.cc (dup_now): New function.
(cygheap_setup_for_child): Accept new argument controlling whether to delay
copying of cygheap to shared memory region.
(cygheap_setup_for_child_cleanup): Accept new arguments controlling whether to
copy cygheap at this point.
* cygheap.h: Reflect above changes.
* fork.cc (fork_parent): Break copying of cygheap into two parts when
fork_fixup is required so that the child can see the parent's changes.
(vfork): Do stack cleanup prior to forcing a fork error.
* spawn.cc (spawn_guts): Ditto.
Diffstat (limited to 'winsup/cygwin/cygheap.cc')
-rw-r--r-- | winsup/cygwin/cygheap.cc | 38 |
1 files changed, 28 insertions, 10 deletions
diff --git a/winsup/cygwin/cygheap.cc b/winsup/cygwin/cygheap.cc index 4866a7e8d..c5f6af776 100644 --- a/winsup/cygwin/cygheap.cc +++ b/winsup/cygwin/cygheap.cc @@ -64,8 +64,17 @@ init_cheap () cygheap_max = cygheap + 1; } -void __stdcall -cygheap_setup_for_child (child_info *ci) +static void dup_now (void *, child_info *, unsigned) __attribute__ ((regparm(3))); +static void +dup_now (void *newcygheap, child_info *ci, unsigned n) +{ + if (!VirtualAlloc (newcygheap, n, MEM_COMMIT, PAGE_READWRITE)) + api_fatal ("couldn't allocate new cygwin heap for child, %E"); + memcpy (newcygheap, cygheap, n); +} + +void *__stdcall +cygheap_setup_for_child (child_info *ci, bool dup_later) { void *newcygheap; cygheap_protect->acquire (); @@ -73,20 +82,29 @@ cygheap_setup_for_child (child_info *ci) ci->cygheap_h = CreateFileMapping (INVALID_HANDLE_VALUE, &sec_none, CFMAP_OPTIONS, 0, CYGHEAPSIZE, NULL); newcygheap = MapViewOfFileEx (ci->cygheap_h, MVMAP_OPTIONS, 0, 0, 0, NULL); - if (!VirtualAlloc (newcygheap, n, MEM_COMMIT, PAGE_READWRITE)) - api_fatal ("couldn't allocate new cygwin heap for child, %E"); - memcpy (newcygheap, cygheap, n); - UnmapViewOfFile (newcygheap); - ci->cygheap = cygheap; - ci->cygheap_max = cygheap_max; ProtectHandle1 (ci->cygheap_h, passed_cygheap_h); + if (!dup_later) + dup_now (newcygheap, ci, n); cygheap_protect->release (); - return; + ci->cygheap = cygheap; + ci->cygheap_max = cygheap_max; + return newcygheap; } void __stdcall -cygheap_setup_for_child_cleanup (child_info *ci) +cygheap_setup_for_child_cleanup (void *newcygheap, child_info *ci, + bool dup_it_now) { + if (dup_it_now) + { + /* NOTE: There is an assumption here that cygheap_max has not changed + between the time that cygheap_setup_for_child was called and now. + Make sure that this is a correct assumption. */ + cygheap_protect->acquire (); + dup_now (newcygheap, ci, (char *) cygheap_max - (char *) cygheap); + cygheap_protect->release (); + } + UnmapViewOfFile (newcygheap); ForceCloseHandle1 (ci->cygheap_h, passed_cygheap_h); } |