diff options
author | Corinna Vinschen <corinna@vinschen.de> | 2011-05-20 11:23:11 +0400 |
---|---|---|
committer | Corinna Vinschen <corinna@vinschen.de> | 2011-05-20 11:23:11 +0400 |
commit | 89d3c72d510b4e254b38b15865d921d45b26d80a (patch) | |
tree | 8be3d82b93084d305beae6c87019bb8d192e4b44 /winsup/cygwin/fork.cc | |
parent | 660302eb67bc951f9a4e543f4b5d02450eea10e2 (diff) |
* child_info.h (CURR_CHILD_INFO_MAGIC): Update.
(class child_info_fork): Remove stacksize, add stackaddr and guardsize
members.
* dcrt0.cc (child_info_fork::alloc_stack_hard_way): Partial rewrite
to regenerate the stack exactly as in the parent.
(child_info_fork::alloc_stack): Set stackaddr to 0, rather than
stacksize.
(dll_crt0_1): Check for stackaddr before changing the stack addresses
in the TEB.
* fork.cc (frok::child): Check for stackaddr here.
(frok::parent): Set ch.stackaddr and ch.guardsize if not called from
the main thread.
* init.cc (dll_entry): Replace pointer to NT_TIB with pointer to TEB.
Fix incorrectly changed address test before removing _my_tls.
Set StackLimit to NULL on Windows 2000. Explain why.
* miscfuncs.cc (struct thread_wrapper_arg): Store stackbase rather
than stacksize, store commitaddr, remove guardsize. Store all pointers
as char * for easier address arithmetic.
(thread_wrapper): Rewrite to remove OS stack before calling thread
function. Add lots of comments to explain what we do.
(CygwinCreateThread): Reserve our own stack in case we got no
application stack. Add comments.
* ntdll.h (struct _TEB): Extend defintion up to DeallocationStack
member.
* thread.cc (pthread_attr::pthread_attr): Use "(size_t) -1"
rather then 0xffffffff.
* wincap.h (wincaps::has_stack_size_param_is_a_reservation): New
element.
* wincap.cc: Implement above element throughout.
Diffstat (limited to 'winsup/cygwin/fork.cc')
-rw-r--r-- | winsup/cygwin/fork.cc | 34 |
1 files changed, 29 insertions, 5 deletions
diff --git a/winsup/cygwin/fork.cc b/winsup/cygwin/fork.cc index 8701cdfa3..4a800b551 100644 --- a/winsup/cygwin/fork.cc +++ b/winsup/cygwin/fork.cc @@ -25,6 +25,7 @@ details. */ #include "tls_pbuf.h" #include "dll_init.h" #include "cygmalloc.h" +#include "ntdll.h" #define NPIDS_HELD 4 @@ -178,7 +179,7 @@ frok::child (volatile char * volatile here) /* If we've played with the stack, stacksize != 0. That means that fork() was invoked from other than the main thread. Make sure that the threadinfo information is properly set up. */ - if (fork_info->stacksize) + if (fork_info->stackaddr) { _main_tls = &_my_tls; _main_tls->init_thread (NULL, NULL); @@ -327,10 +328,33 @@ frok::parent (volatile char * volatile stack_here) ch.forker_finished = forker_finished; ch.stackbottom = _tlsbase; - ch.stacktop = (void *) stack_here; - ch.stacksize = (char *) ch.stackbottom - (char *) stack_here; - debug_printf ("stack - bottom %p, top %p, size %d", - ch.stackbottom, ch.stacktop, ch.stacksize); + ch.stacktop = (void *) _tlstop; + ch.stackaddr = 0; + ch.guardsize = 0; + if (&_my_tls != _main_tls) + { + /* We have not been started from the main thread. Fetch the + information required to set up the thread stack identically + in the child. */ + PTEB teb = NtCurrentTeb (); + if (!teb->DeallocationStack) + { + /* Pthread with application-provided stack. Don't set up a + PAGE_GUARD page. guardsize == -1 is used in alloc_stack_hard_way + to recognize this type of stack. */ + ch.stackaddr = _my_tls.tid->attr.stackaddr; + ch.guardsize = (size_t) -1; + } + else + { + ch.stackaddr = teb->DeallocationStack; + /* If it's a pthread, fetch guardsize from thread attributes. */ + if (_my_tls.tid) + ch.guardsize = _my_tls.tid->attr.guardsize; + } + } + debug_printf ("stack - bottom %p, top %p, addr %p, guardsize %p", + ch.stackbottom, ch.stacktop, ch.stackaddr, ch.guardsize); PROCESS_INFORMATION pi; STARTUPINFOW si; |