Merge changes I7718b1ba,I648adc35
* changes: Only initialize the global stack protector once. Make getpid work before TLS has been initialized.
This commit is contained in:
commit
4f62c5b311
|
@ -28,14 +28,24 @@
|
||||||
|
|
||||||
#include "libc_init_common.h"
|
#include "libc_init_common.h"
|
||||||
|
|
||||||
|
#include "private/KernelArgumentBlock.h"
|
||||||
#include "private/bionic_auxv.h"
|
#include "private/bionic_auxv.h"
|
||||||
#include "private/bionic_globals.h"
|
#include "private/bionic_globals.h"
|
||||||
#include "private/KernelArgumentBlock.h"
|
#include "private/bionic_ssp.h"
|
||||||
#include "pthread_internal.h"
|
#include "pthread_internal.h"
|
||||||
|
|
||||||
extern "C" int __set_tls(void* ptr);
|
extern "C" int __set_tls(void* ptr);
|
||||||
extern "C" int __set_tid_address(int* tid_address);
|
extern "C" int __set_tid_address(int* tid_address);
|
||||||
|
|
||||||
|
// Declared in "private/bionic_ssp.h".
|
||||||
|
uintptr_t __stack_chk_guard = 0;
|
||||||
|
|
||||||
|
void __libc_init_global_stack_chk_guard(KernelArgumentBlock& args) {
|
||||||
|
// AT_RANDOM is a pointer to 16 bytes of randomness on the stack.
|
||||||
|
// Take the first 4/8 for the -fstack-protector implementation.
|
||||||
|
__stack_chk_guard = *reinterpret_cast<uintptr_t*>(args.getauxval(AT_RANDOM));
|
||||||
|
}
|
||||||
|
|
||||||
// Setup for the main thread. For dynamic executables, this is called by the
|
// Setup for the main thread. For dynamic executables, this is called by the
|
||||||
// linker _before_ libc is mapped in memory. This means that all writes to
|
// linker _before_ libc is mapped in memory. This means that all writes to
|
||||||
// globals from this function will apply to linker-private copies and will not
|
// globals from this function will apply to linker-private copies and will not
|
||||||
|
@ -78,7 +88,8 @@ void __libc_init_main_thread(KernelArgumentBlock& args) {
|
||||||
// TODO: the main thread's sched_policy and sched_priority need to be queried.
|
// TODO: the main thread's sched_policy and sched_priority need to be queried.
|
||||||
|
|
||||||
// The TLS stack guard is set from the global, so ensure that we've initialized the global
|
// The TLS stack guard is set from the global, so ensure that we've initialized the global
|
||||||
// before we initialize the TLS.
|
// before we initialize the TLS. Dynamic executables will initialize their copy of the global
|
||||||
|
// stack protector from the one in the main thread's TLS.
|
||||||
__libc_init_global_stack_chk_guard(args);
|
__libc_init_global_stack_chk_guard(args);
|
||||||
|
|
||||||
__init_thread(&main_thread);
|
__init_thread(&main_thread);
|
||||||
|
|
|
@ -45,7 +45,6 @@
|
||||||
#include "private/WriteProtected.h"
|
#include "private/WriteProtected.h"
|
||||||
#include "private/bionic_auxv.h"
|
#include "private/bionic_auxv.h"
|
||||||
#include "private/bionic_globals.h"
|
#include "private/bionic_globals.h"
|
||||||
#include "private/bionic_ssp.h"
|
|
||||||
#include "private/bionic_tls.h"
|
#include "private/bionic_tls.h"
|
||||||
#include "private/libc_logging.h"
|
#include "private/libc_logging.h"
|
||||||
#include "private/thread_private.h"
|
#include "private/thread_private.h"
|
||||||
|
@ -62,15 +61,6 @@ const char* __progname;
|
||||||
// Declared in <unistd.h>.
|
// Declared in <unistd.h>.
|
||||||
char** environ;
|
char** environ;
|
||||||
|
|
||||||
// Declared in "private/bionic_ssp.h".
|
|
||||||
uintptr_t __stack_chk_guard = 0;
|
|
||||||
|
|
||||||
void __libc_init_global_stack_chk_guard(KernelArgumentBlock& args) {
|
|
||||||
// AT_RANDOM is a pointer to 16 bytes of randomness on the stack.
|
|
||||||
// Take the first 4/8 for the -fstack-protector implementation.
|
|
||||||
__stack_chk_guard = *reinterpret_cast<uintptr_t*>(args.getauxval(AT_RANDOM));
|
|
||||||
}
|
|
||||||
|
|
||||||
#if defined(__i386__)
|
#if defined(__i386__)
|
||||||
__LIBC_HIDDEN__ void* __libc_sysinfo = nullptr;
|
__LIBC_HIDDEN__ void* __libc_sysinfo = nullptr;
|
||||||
|
|
||||||
|
@ -91,7 +81,6 @@ void __libc_init_globals(KernelArgumentBlock& args) {
|
||||||
// Initialize libc globals that are needed in both the linker and in libc.
|
// Initialize libc globals that are needed in both the linker and in libc.
|
||||||
// In dynamic binaries, this is run at least twice for different copies of the
|
// In dynamic binaries, this is run at least twice for different copies of the
|
||||||
// globals, once for the linker's copy and once for the one in libc.so.
|
// globals, once for the linker's copy and once for the one in libc.so.
|
||||||
__libc_init_global_stack_chk_guard(args);
|
|
||||||
__libc_auxv = args.auxv;
|
__libc_auxv = args.auxv;
|
||||||
__libc_globals.initialize();
|
__libc_globals.initialize();
|
||||||
__libc_globals.mutate([&args](libc_globals* globals) {
|
__libc_globals.mutate([&args](libc_globals* globals) {
|
||||||
|
|
|
@ -52,6 +52,7 @@
|
||||||
#include "libc_init_common.h"
|
#include "libc_init_common.h"
|
||||||
|
|
||||||
#include "private/bionic_globals.h"
|
#include "private/bionic_globals.h"
|
||||||
|
#include "private/bionic_ssp.h"
|
||||||
#include "private/bionic_tls.h"
|
#include "private/bionic_tls.h"
|
||||||
#include "private/KernelArgumentBlock.h"
|
#include "private/KernelArgumentBlock.h"
|
||||||
|
|
||||||
|
@ -74,6 +75,10 @@ __attribute__((constructor)) static void __libc_preinit() {
|
||||||
// __libc_init_common() will change the TLS area so the old one won't be accessible anyway.
|
// __libc_init_common() will change the TLS area so the old one won't be accessible anyway.
|
||||||
*args_slot = NULL;
|
*args_slot = NULL;
|
||||||
|
|
||||||
|
// The linker has initialized its copy of the global stack_chk_guard, and filled in the main
|
||||||
|
// thread's TLS slot with that value. Initialize the local global stack guard with its value.
|
||||||
|
__stack_chk_guard = reinterpret_cast<uintptr_t>(tls[TLS_SLOT_STACK_GUARD]);
|
||||||
|
|
||||||
__libc_init_globals(*args);
|
__libc_init_globals(*args);
|
||||||
__libc_init_common(*args);
|
__libc_init_common(*args);
|
||||||
|
|
||||||
|
|
|
@ -39,6 +39,7 @@
|
||||||
#include "libc_init_common.h"
|
#include "libc_init_common.h"
|
||||||
#include "pthread_internal.h"
|
#include "pthread_internal.h"
|
||||||
|
|
||||||
|
#include "private/bionic_globals.h"
|
||||||
#include "private/bionic_page.h"
|
#include "private/bionic_page.h"
|
||||||
#include "private/bionic_tls.h"
|
#include "private/bionic_tls.h"
|
||||||
#include "private/KernelArgumentBlock.h"
|
#include "private/KernelArgumentBlock.h"
|
||||||
|
@ -85,6 +86,7 @@ __noreturn void __libc_init(void* raw_args,
|
||||||
__libc_init_main_thread(args);
|
__libc_init_main_thread(args);
|
||||||
|
|
||||||
// Initializing the globals requires TLS to be available for errno.
|
// Initializing the globals requires TLS to be available for errno.
|
||||||
|
__init_thread_stack_guard(__get_thread());
|
||||||
__libc_init_globals(args);
|
__libc_init_globals(args);
|
||||||
|
|
||||||
__libc_init_AT_SECURE(args);
|
__libc_init_AT_SECURE(args);
|
||||||
|
|
|
@ -56,6 +56,10 @@ void __init_tls(pthread_internal_t* thread) {
|
||||||
// Slot 0 must point to itself. The x86 Linux kernel reads the TLS from %fs:0.
|
// Slot 0 must point to itself. The x86 Linux kernel reads the TLS from %fs:0.
|
||||||
thread->tls[TLS_SLOT_SELF] = thread->tls;
|
thread->tls[TLS_SLOT_SELF] = thread->tls;
|
||||||
thread->tls[TLS_SLOT_THREAD_ID] = thread;
|
thread->tls[TLS_SLOT_THREAD_ID] = thread;
|
||||||
|
__init_thread_stack_guard(thread);
|
||||||
|
}
|
||||||
|
|
||||||
|
void __init_thread_stack_guard(pthread_internal_t* thread) {
|
||||||
// GCC looks in the TLS for the stack guard on x86, so copy it there from our global.
|
// GCC looks in the TLS for the stack guard on x86, so copy it there from our global.
|
||||||
thread->tls[TLS_SLOT_STACK_GUARD] = reinterpret_cast<void*>(__stack_chk_guard);
|
thread->tls[TLS_SLOT_STACK_GUARD] = reinterpret_cast<void*>(__stack_chk_guard);
|
||||||
}
|
}
|
||||||
|
|
|
@ -114,6 +114,7 @@ class pthread_internal_t {
|
||||||
|
|
||||||
__LIBC_HIDDEN__ int __init_thread(pthread_internal_t* thread);
|
__LIBC_HIDDEN__ int __init_thread(pthread_internal_t* thread);
|
||||||
__LIBC_HIDDEN__ void __init_tls(pthread_internal_t* thread);
|
__LIBC_HIDDEN__ void __init_tls(pthread_internal_t* thread);
|
||||||
|
__LIBC_HIDDEN__ void __init_thread_stack_guard(pthread_internal_t* thread);
|
||||||
__LIBC_HIDDEN__ void __init_alternate_signal_stack(pthread_internal_t*);
|
__LIBC_HIDDEN__ void __init_alternate_signal_stack(pthread_internal_t*);
|
||||||
|
|
||||||
__LIBC_HIDDEN__ pthread_t __pthread_internal_add(pthread_internal_t* thread);
|
__LIBC_HIDDEN__ pthread_t __pthread_internal_add(pthread_internal_t* thread);
|
||||||
|
@ -123,7 +124,13 @@ __LIBC_HIDDEN__ void __pthread_internal_remove_and_free(pthread_i
|
||||||
|
|
||||||
// Make __get_thread() inlined for performance reason. See http://b/19825434.
|
// Make __get_thread() inlined for performance reason. See http://b/19825434.
|
||||||
static inline __always_inline pthread_internal_t* __get_thread() {
|
static inline __always_inline pthread_internal_t* __get_thread() {
|
||||||
return reinterpret_cast<pthread_internal_t*>(__get_tls()[TLS_SLOT_THREAD_ID]);
|
void** tls = __get_tls();
|
||||||
|
if (__predict_true(tls)) {
|
||||||
|
return reinterpret_cast<pthread_internal_t*>(tls[TLS_SLOT_THREAD_ID]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// This happens when called during libc initialization before TLS has been initialized.
|
||||||
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
__LIBC_HIDDEN__ void pthread_key_clean_all(void);
|
__LIBC_HIDDEN__ void pthread_key_clean_all(void);
|
||||||
|
|
|
@ -44,7 +44,6 @@ struct libc_globals {
|
||||||
__LIBC_HIDDEN__ extern WriteProtected<libc_globals> __libc_globals;
|
__LIBC_HIDDEN__ extern WriteProtected<libc_globals> __libc_globals;
|
||||||
|
|
||||||
class KernelArgumentBlock;
|
class KernelArgumentBlock;
|
||||||
__LIBC_HIDDEN__ void __libc_init_global_stack_chk_guard(KernelArgumentBlock& args);
|
|
||||||
__LIBC_HIDDEN__ void __libc_init_malloc(libc_globals* globals);
|
__LIBC_HIDDEN__ void __libc_init_malloc(libc_globals* globals);
|
||||||
__LIBC_HIDDEN__ void __libc_init_setjmp_cookie(libc_globals* globals, KernelArgumentBlock& args);
|
__LIBC_HIDDEN__ void __libc_init_setjmp_cookie(libc_globals* globals, KernelArgumentBlock& args);
|
||||||
__LIBC_HIDDEN__ void __libc_init_vdso(libc_globals* globals, KernelArgumentBlock& args);
|
__LIBC_HIDDEN__ void __libc_init_vdso(libc_globals* globals, KernelArgumentBlock& args);
|
||||||
|
|
Loading…
Reference in New Issue