Welcome to mirror list, hosted at ThFree Co, Russian Federation.

cygwin.com/git/newlib-cygwin.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorChristopher Faylor <me@cgf.cx>2001-09-14 04:49:00 +0400
committerChristopher Faylor <me@cgf.cx>2001-09-14 04:49:00 +0400
commite2e078278cc951344fc5c0a976715bd1fe11f2d8 (patch)
treeafd6de5d0a5ee0c2778c384423f7a59fd38ccd44 /winsup/cygwin/cygheap.cc
parent3e2d8af0b9a8df7b9b1230d35f3e838d3efbfb33 (diff)
* 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.cc38
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);
}