summaryrefslogtreecommitdiff
path: root/nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386
diff options
context:
space:
mode:
Diffstat (limited to 'nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386')
-rw-r--r--nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386/Implies1
-rw-r--r--nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386/createthread_arch.c52
-rw-r--r--nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386/sysdep-cancel.h186
-rw-r--r--nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386/tcb-offsets.sym8
-rw-r--r--nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386/tls.h61
5 files changed, 308 insertions, 0 deletions
diff --git a/nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386/Implies b/nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386/Implies
new file mode 100644
index 0000000000..772b20e20f
--- /dev/null
+++ b/nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386/Implies
@@ -0,0 +1 @@
+unix/sysv/solaris2/kopensolaris-gnu/x86
diff --git a/nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386/createthread_arch.c b/nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386/createthread_arch.c
new file mode 100644
index 0000000000..4241f9182c
--- /dev/null
+++ b/nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386/createthread_arch.c
@@ -0,0 +1,52 @@
+/* Copyright (C) 2008 Free Software Foundation, Inc.
+ This file is part of the GNU C Library.
+ Contributed by David Bartley <dtbartle@csclub.uwaterloo.ca>, 2008.
+
+ The GNU C Library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ The GNU C Library 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
+ Lesser General Public License for more details.
+
+ You should have received a copy of the GNU Lesser General Public
+ License along with the GNU C Library; if not, write to the Free
+ Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
+ 02111-1307 USA. */
+
+#define LWPGS_SEL 0x01C3
+
+static int
+create_thread_arch (ucontext_t *ctx, struct pthread *pd,
+ const struct pthread_attr *attr, STACK_VARIABLES_PARMS)
+{
+ /* Clone the segment registers (except gs - see below). */
+ ucontext_t _ctx;
+ _ctx.uc_flags = UC_CPU;
+ if (getcontext(&_ctx) != 0)
+ return errno;
+ ctx->uc_mcontext.gregs[CS] = _ctx.uc_mcontext.gregs[CS];
+ ctx->uc_mcontext.gregs[DS] = _ctx.uc_mcontext.gregs[DS];
+ ctx->uc_mcontext.gregs[ES] = _ctx.uc_mcontext.gregs[ES];
+ ctx->uc_mcontext.gregs[FS] = _ctx.uc_mcontext.gregs[FS];
+ ctx->uc_mcontext.gregs[SS] = _ctx.uc_mcontext.gregs[SS];
+
+ /* Setup the stack (note that it grows down). */
+ uint32_t *stack_ptr = (uint32_t *)((uintptr_t)((uint32_t *)stackaddr - 1) &
+ ~(STACK_ALIGN - 1)) + 1;
+ *--stack_ptr = (uint32_t)pd; /* arg 1 */
+ *--stack_ptr = 0; /* return addr (thread_start never returns) */
+ ctx->uc_mcontext.gregs[UESP] = (greg_t)stack_ptr;
+ ctx->uc_mcontext.gregs[EBP] = 0;
+ ctx->uc_mcontext.gregs[EIP] = (greg_t)start_thread;
+ ctx->uc_flags |= UC_CPU;
+
+ /* This is a hack to get the kernel to set gs for us. */
+ ctx->uc_mcontext.gregs[GS] = (greg_t)LWPGS_SEL;
+ ctx->uc_mcontext.gregs[ESP] = (greg_t)pd;
+
+ return 0;
+}
diff --git a/nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386/sysdep-cancel.h b/nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386/sysdep-cancel.h
new file mode 100644
index 0000000000..480283a22e
--- /dev/null
+++ b/nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386/sysdep-cancel.h
@@ -0,0 +1,186 @@
+/* Copyright (C) 2002, 2003, 2004, 2005, 2006, 2008
+ Free Software Foundation, Inc.
+ This file is part of the GNU C Library.
+ Contributed by Jakub Jelinek <jakub@redhat.com>, 2002.
+
+ The GNU C Library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ The GNU C Library 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
+ Lesser General Public License for more details.
+
+ You should have received a copy of the GNU Lesser General Public
+ License along with the GNU C Library; if not, write to the Free
+ Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
+ 02111-1307 USA. */
+
+#include <sysdep.h>
+#include <tls.h>
+#ifndef __ASSEMBLER__
+# include <pthreadP.h>
+#endif
+
+/* XXX: We don't support _NOERRNO or _ERRVAL varietites. */
+#undef PSEUDO_ERRNO
+#undef PSEUDO_SUBCALL_NOERRNO
+#undef PSEUDO_ERRVAL
+#undef PSEUDO_SUBCALL_NOERRNO
+
+/* This is needed so that we don't clobber %edx in the cancellation calls. */
+#ifdef SYSCALL_64BIT_RETURN
+# define SYSCALL_64BIT_PUSH_ASM pushl %edx; cfi_adjust_cfa_offset (4);
+# define SYSCALL_64BIT_POP_ASM popl %edx; cfi_adjust_cfa_offset (-4);
+#else
+# define SYSCALL_64BIT_PUSH_ASM
+# define SYSCALL_64BIT_POP_ASM
+#endif
+
+#ifdef SYSCALL_RESTARTABLE
+# define DO_RESTART_CANCEL \
+ cmpl $ERESTART, %eax; \
+ je L(restart_cancel);
+#else
+# define DO_RESTART_CANCEL
+#endif
+
+#if !defined NOT_IN_libc || defined IS_IN_libpthread || defined IS_IN_librt
+
+/* Note that CDISABLE is an internal function, so we need to
+ pass-by-register (we pass the argument in %eax). */
+
+# undef PSEUDO
+# define PSEUDO(name, syscall_name, args) \
+ .text; \
+ ENTRY (name) \
+ cmpl $0, %gs:MULTIPLE_THREADS_OFFSET; \
+ jne L(pseudo_cancel); \
+ .type __##syscall_name##_nocancel,@function; \
+ .globl __##syscall_name##_nocancel; \
+ __##syscall_name##_nocancel: \
+ L(restart): \
+ DO_CALL (syscall_name, args); \
+ jnb 2f; \
+ DO_RESTART \
+ jmp SYSCALL_ERROR_LABEL; \
+2: ret; \
+ .size __##syscall_name##_nocancel,.-__##syscall_name##_nocancel; \
+ L(pseudo_cancel): \
+ L(restart_cancel): \
+ CENABLE; \
+ movl %eax, %ecx; \
+ DO_CALL (syscall_name, args); \
+ jnb 3f; \
+ DO_RESTART_CANCEL \
+ pushl %eax; cfi_adjust_cfa_offset (4); \
+ SYSCALL_64BIT_PUSH_ASM \
+ movl %ecx, %eax; \
+ CDISABLE; \
+ SYSCALL_64BIT_POP_ASM \
+ popl %eax; cfi_adjust_cfa_offset (-4); \
+ jmp SYSCALL_ERROR_LABEL; \
+3: \
+ pushl %eax; cfi_adjust_cfa_offset (4); \
+ SYSCALL_64BIT_PUSH_ASM \
+ movl %ecx, %eax; \
+ CDISABLE; \
+ SYSCALL_64BIT_POP_ASM \
+ popl %eax; cfi_adjust_cfa_offset (-4); \
+ L(pseudo_end):
+
+# undef PSEUDO_SUBCALL
+# define PSEUDO_SUBCALL(name, syscall_name, subcall_name, args) \
+ .text; \
+ ENTRY (name) \
+ cmpl $0, %gs:MULTIPLE_THREADS_OFFSET; \
+ jne L(pseudo_cancel); \
+ .type __##subcall_name##_nocancel,@function; \
+ .globl __##subcall_name##_nocancel; \
+ __##subcall_name##_nocancel: \
+ movl 0(%esp), %ecx; \
+ movl %ecx, -4(%esp); \
+ addl $-4, %esp; \
+ movl $SYS_ify (SUB_##subcall_name), 4(%esp); \
+ L(restart): \
+ DO_CALL (syscall_name, args); \
+ jnb 2f; \
+ DO_RESTART \
+ movl %ecx, 4(%esp); \
+ addl $-4, %esp; \
+ jmp SYSCALL_ERROR_LABEL; \
+2: \
+ addl $4, %esp; \
+ movl %ecx, 0(%esp); \
+ ret; \
+ .size __##subcall_name##_nocancel,.-__##subcall_name##_nocancel; \
+ L(pseudo_cancel): \
+ CENABLE; \
+ movl %eax, %ecx; \
+ movl 0(%esp), %edx; \
+ movl %edx, -4(%esp); \
+ addl $-4, %esp; \
+ movl $SYS_ify (SUB_##subcall_name), 4(%esp); \
+ L(restart_cancel): \
+ DO_CALL (syscall_name, args); \
+ jnb 3f; \
+ DO_RESTART_CANCEL \
+ pushl %eax; cfi_adjust_cfa_offset (4); \
+ SYSCALL_64BIT_PUSH_ASM \
+ movl %ecx, %eax; \
+ CDISABLE; \
+ SYSCALL_64BIT_POP_ASM \
+ popl %eax; cfi_adjust_cfa_offset (-4); \
+ movl 0(%esp), %ecx; \
+ movl %ecx, 4(%esp); \
+ addl $4, %esp; \
+ jmp SYSCALL_ERROR_LABEL; \
+3: \
+ pushl %eax; cfi_adjust_cfa_offset (4); \
+ SYSCALL_64BIT_PUSH_ASM \
+ movl %ecx, %eax; \
+ CDISABLE; \
+ SYSCALL_64BIT_POP_ASM \
+ popl %eax; cfi_adjust_cfa_offset (-4); \
+ movl 0(%esp), %ecx; \
+ movl %ecx, 4(%esp); \
+ addl $4, %esp; \
+ L(pseudo_end):
+
+
+# define SAVE_OLDTYPE movl %eax, %ecx;
+# ifdef IS_IN_libpthread
+# define CENABLE call __pthread_enable_asynccancel;
+# define CDISABLE call __pthread_disable_asynccancel
+# elif !defined NOT_IN_libc
+# define CENABLE call __libc_enable_asynccancel;
+# define CDISABLE call __libc_disable_asynccancel
+# elif defined IS_IN_librt
+# define CENABLE call __librt_enable_asynccancel;
+# define CDISABLE call __librt_disable_asynccancel
+# else
+# error Unsupported library
+# endif
+
+# ifndef __ASSEMBLER__
+# define SINGLE_THREAD_P \
+ __builtin_expect (THREAD_GETMEM (THREAD_SELF, \
+ header.multiple_threads) == 0, 1)
+# else
+# define SINGLE_THREAD_P cmpl $0, %gs:MULTIPLE_THREADS_OFFSET
+# endif
+
+#elif !defined __ASSEMBLER__
+
+# define SINGLE_THREAD_P (1)
+# define NO_CANCELLATION 1
+
+#endif
+
+#ifndef __ASSEMBLER__
+# define RTLD_SINGLE_THREAD_P \
+ __builtin_expect (THREAD_GETMEM (THREAD_SELF, \
+ header.multiple_threads) == 0, 1)
+#endif
diff --git a/nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386/tcb-offsets.sym b/nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386/tcb-offsets.sym
new file mode 100644
index 0000000000..4db5d95e11
--- /dev/null
+++ b/nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386/tcb-offsets.sym
@@ -0,0 +1,8 @@
+#include <sysdep.h>
+#include <tls.h>
+
+MULTIPLE_THREADS_OFFSET offsetof (tcbhead_t, multiple_threads)
+#ifdef NEED_DL_SYSINFO
+SYSINFO_OFFSET offsetof (tcbhead_t, sysinfo)
+#endif
+POINTER_GUARD offsetof (tcbhead_t, pointer_guard)
diff --git a/nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386/tls.h b/nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386/tls.h
new file mode 100644
index 0000000000..6bacdd859e
--- /dev/null
+++ b/nptl/sysdeps/unix/sysv/solaris2/kopensolaris-gnu/i386/tls.h
@@ -0,0 +1,61 @@
+/* Definition for thread-local data handling. OpenSolaris/i386 version.
+ Copyright (C) 2002, 2003, 2004, 2008 Free Software Foundation, Inc.
+ This file is part of the GNU C Library.
+ OpenSolaris bits contributed by David Bartley
+ <dtbartle@csclub.uwaterloo.ca>, 2008.
+
+ The GNU C Library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ The GNU C Library 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
+ Lesser General Public License for more details.
+
+ You should have received a copy of the GNU Lesser General Public
+ License along with the GNU C Library; if not, write to the Free
+ Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
+ 02111-1307 USA. */
+
+#ifndef _OPENSOLARIS_TLS_H
+#define _OPENSOLARIS_TLS_H 1
+
+/* __thread doesn't work for statically-linked executables. */
+#if !defined (SHARED)
+# undef HAVE___THREAD
+#endif
+
+/* We can use most of the code from the nptl tls.h. */
+#include_next <tls.h>
+
+#ifndef __ASSEMBLER__
+
+/* OpenSolaris has lwp_private in order to set the thread data. */
+# include <sys/lwp.h>
+# include <sys/syscall.h>
+
+# undef TLS_INIT_TP
+# define TLS_INIT_TP(thrdescr, secondcall) \
+ ({ void *_thrdescr = (thrdescr); \
+ tcbhead_t *_head = _thrdescr; \
+ sysret_t ret; \
+ \
+ _head->tcb = _thrdescr; \
+ /* For now the thread descriptor is at the same address. */ \
+ _head->self = _thrdescr; \
+ /* New syscall handling support. */ \
+ INIT_SYSINFO; \
+ \
+ (__systemcall (&ret, SYS_lwp_private, _LWP_SETPRIVATE, _LWP_GSBASE, \
+ (uintptr_t)_thrdescr) != 0) \
+ ? "lwp_private failed when setting up thread-local storage\n" : NULL; \
+ })
+
+/* TODO: Figure out what to define this as. */
+# undef DB_THREAD_SELF
+
+# endif /* __ASSEMBLER__ */
+
+#endif /* _OPENSOLARIS_TLS_H */