diff options
author | ian <ian@138bc75d-0d04-0410-961f-82ee72b054a4> | 2010-09-27 19:48:20 +0000 |
---|---|---|
committer | ian <ian@138bc75d-0d04-0410-961f-82ee72b054a4> | 2010-09-27 19:48:20 +0000 |
commit | 48b14f5076a521887a2286eb84e6b51de6c50fd8 (patch) | |
tree | 69a38455d6a1f5425700674e4dc6483284eed579 /libgcc/generic-morestack-thread.c | |
parent | f8ee0563dfbffc5c24cc4424fca8cd070ee6121c (diff) | |
download | gcc-48b14f5076a521887a2286eb84e6b51de6c50fd8.tar.gz |
gcc/:
* common.opt (fsplit-stack): New option.
* opts.c (decode_options): Set flag_split_stack to final value.
* target.def (supports_split_stack): New hook.
* gcc.c (STACK_SPLIT_SPEC): Define.
(LINK_COMMAND_SPEC): Use STACK_SPLIT_SPEC.
* doc/invoke.texi (Option Summary): Mention -fsplit-stack.
(Code Gen Options): Document -fsplit-stack.
* doc/extend.texi (Function Attributes): Mention no_split_stack.
(Function Attributes): Document no_split_stack.
* doc/tm.texi.in (Stack Smashing Protection): Add @hook
TARGET_SUPPORTS_SPLIT_STACK.
* doc/tm.texi: Rebuild.
* function.c (thread_prologue_and_epilogue_insns): If
flag_split_stack, add split stack prologue.
* explow.c (allocate_dynamic_stack_space): Support -fsplit-stack.
* varasm.c (saw_no_split_stack): New static variable.
(assemble_start_function): Set saw_no_split_stack if the function
has the no_split_stack attribute.
(file_end_indicate_split_stack): New function.
* output.h (file_end_indicate_split_stack): Declare.
* libgcc-std.ver (GCC_4.6.0): Add -fsplit-stack support variables
and function.
* doc/libgcc.texi (Miscellaneous routines): Document -fsplit-stack
routines.
* config/i386/i386.c (ix86_option_override_internal): Don't set
expand_builtin_va_start to NULL if -fsplit-stack.
(ix86_function_regparm): Reduce local regparm by 1 for 32-bit
-fsplit-stack.
(ix86_va_start): If -fsplit-stack, get overflow pointer from
scratch register set by prologue.
(ix86_code_end): If -fsplit-stack, call
file_end_indicate_split_stack.
(ix86_supports_split_stack): New static function.
(SPLIT_STACK_AVAILABLE): Define.
(split_stack_prologue_scratch_regno): New static function.
(split_stack_fn): New static variable.
(ix86_expand_split_stack_prologue): New function.
(ix86_live_on_entry): New static function.
(ix86_legitimate_address_p): Handle UNSPEC_STACK_CHECK.
(output_pic_addr_const): Likewise.
(i386_asm_output_addr_const_extra): Likewise.
(ix86_expand_call): Change return type to rtx. Return the new
call instruction.
(TARGET_SUPPORTS_SPLIT_STACK): Define.
(TARGET_EXTRA_LIVE_ON_ENTRY): Define.
* config/i386/i386.md (UNSPEC_STACK_CHECK): Define.
(split_stack_prologue, split_stack_return): New insns.
(split_stack_space_check): New insn.
* config/i386/i386.h (struct machine_function): Add
split_stack_varargs_pointer field.
* config/i386/linux.h (TARGET_CAN_SPLIT_STACK): Define.
(TARGET_THREAD_SPLIT_STACK_OFFSET): Define.
* config/i386/linux64.h (TARGET_CAN_SPLIT_STACK): Define.
(TARGET_THREAD_SPLIT_STACK_OFFSET): Define.
* config/i386/i386-protos.h (ix86_expand_split_stack_prologue):
Declare.
(ix86_expand_call): Update declaration.
gcc/c-family/:
* c-common.c (c_common_attribute_table): Add no_split_stack.
(handle_no_split_stack_attribute): New static function.
gcc/testsuite/:
* lib/target-supports.exp (check_effective_target_split_stack):
New procedure.
* gcc.dg/split-1.c: New test.
* gcc.dg/split-2.c: New test.
* gcc.dg/split-3.c: New test.
* gcc.dg/split-4.c: New test.
libgcc/:
* generic-morestack.h: New file.
* generic-morestack.c: New file.
* generic-morestack-thread.c: New file.
* config/i386/morestack.S: New file.
* config/t-stack: New file.
* config/i386/t-stack-i386: New file.
* config.host (i[34567]86-*-linux* and friends): Add t-stack and
i386/t-stack-i386 to tmake_file.
git-svn-id: svn+ssh://gcc.gnu.org/svn/gcc/trunk@164661 138bc75d-0d04-0410-961f-82ee72b054a4
Diffstat (limited to 'libgcc/generic-morestack-thread.c')
-rw-r--r-- | libgcc/generic-morestack-thread.c | 162 |
1 files changed, 162 insertions, 0 deletions
diff --git a/libgcc/generic-morestack-thread.c b/libgcc/generic-morestack-thread.c new file mode 100644 index 00000000000..bc237957ef6 --- /dev/null +++ b/libgcc/generic-morestack-thread.c @@ -0,0 +1,162 @@ +/* Thread library support for -fsplit-stack. */ +/* Copyright (C) 2009, 2010 Free Software Foundation, Inc. + Contributed by Ian Lance Taylor <iant@google.com>. + +This file is part of GCC. + +GCC is free software; you can redistribute it and/or modify it under +the terms of the GNU General Public License as published by the Free +Software Foundation; either version 3, or (at your option) any later +version. + +GCC is distributed in the hope that it will be useful, but WITHOUT ANY +WARRANTY; without even the implied warranty of MERCHANTABILITY or +FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License +for more details. + +Under Section 7 of GPL version 3, you are granted additional +permissions described in the GCC Runtime Library Exception, version +3.1, as published by the Free Software Foundation. + +You should have received a copy of the GNU General Public License and +a copy of the GCC Runtime Library Exception along with this program; +see the files COPYING3 and COPYING.RUNTIME respectively. If not, see +<http://www.gnu.org/licenses/>. */ + +#include "tconfig.h" +#include "tsystem.h" +#include "coretypes.h" +#include "tm.h" + +/* If inhibit_libc is defined, we can not compile this file. The + effect is that people will not be able to use -fsplit-stack. That + is much better than failing the build particularly since people + will want to define inhibit_libc while building a compiler which + can build glibc. */ + +#ifndef inhibit_libc + +#include <pthread.h> + +#include "generic-morestack.h" + +/* We declare the pthread functions we need as weak, so that + libgcc_s.so does not need to be linked against -lpthread. */ + +extern int pthread_once (pthread_once_t *, void (*) (void)) + __attribute__ ((weak)); + +extern int pthread_key_create (pthread_key_t *, void (*) (void *)) + __attribute__ ((weak)); + +extern int pthread_setspecific (pthread_key_t, const void *) + __attribute__ ((weak)); + +/* The key for the list of stack segments to free when the thread + exits. This is created by pthread_key_create. */ + +static pthread_key_t segment_list_key; + +/* Used to only run create_key once. */ + +static pthread_once_t create_key_once = PTHREAD_ONCE_INIT; + +/* Release all the segments for a thread. This is the destructor + function used by pthread_key_create, and is called when a thread + exits. */ + +static void +free_segments (void* arg) +{ + __morestack_release_segments ((struct stack_segment **) arg, 1); +} + +/* Set up the key for the list of segments. This is called via + pthread_once. */ + +static void +create_key (void) +{ + int err; + + err = pthread_key_create (&segment_list_key, free_segments); + if (err != 0) + { + static const char msg[] = "pthread_key_create failed: errno "; + __morestack_fail (msg, sizeof msg - 1, err); + } +} + +/* Pass information from the pthread_create wrapper to + stack_split_initialize_thread. */ + +struct pthread_create_args +{ + void *(*start_routine) (void *); + void *arg; +}; + +/* Initialize a thread. This is called via pthread_create. It calls + a target dependent function to set up any required stack guard. */ + +static void* stack_split_initialize_thread (void *) + __attribute__ ((no_split_stack)); + +static void * +stack_split_initialize_thread (void *varg) +{ + struct pthread_create_args *args = (struct pthread_create_args *) varg; + int err; + void *(*start_routine) (void *); + void *arg; + + __stack_split_initialize (); + + err = pthread_setspecific (segment_list_key, (void *) &__morestack_segments); + if (err != 0) + { + static const char msg[] = "pthread_setspecific failed: errno "; + __morestack_fail (msg, sizeof msg - 1, err); + } + + start_routine = args->start_routine; + arg = args->arg; + free (args); + return (*start_routine) (arg); +} + +/* This function wraps calls to pthread_create to make sure that the + stack guard is initialized for new threads. FIXME: This hack will + not be necessary if glibc supports -fsplit-stack directly. */ + +int __wrap_pthread_create (pthread_t *, const pthread_attr_t *, + void *(*start_routine) (void *), void *) + __attribute__ ((visibility ("hidden"))); + +extern int __real_pthread_create (pthread_t *, const pthread_attr_t *, + void *(*start_routine) (void *), void *) + __attribute__ ((weak)); + +int +__wrap_pthread_create (pthread_t *tid, const pthread_attr_t *attr, + void *(*start_routine) (void *), void *arg) +{ + int err; + struct pthread_create_args* args; + + err = pthread_once (&create_key_once, create_key); + if (err != 0) + { + static const char msg[] = "pthread_once failed: errno "; + __morestack_fail (msg, sizeof msg - 1, err); + } + + args = malloc (sizeof (struct pthread_create_args)); + if (args == NULL) + return EAGAIN; + args->start_routine = start_routine; + args->arg = arg; + return __real_pthread_create (tid, attr, stack_split_initialize_thread, args); +} + +#endif /* !defined (inhibit_libc) */ |