aboutsummaryrefslogtreecommitdiff
path: root/nptl/sysdeps/unix/sysv/linux/ia64/lowlevellock.h
diff options
context:
space:
mode:
authorRoland McGrath <roland@gnu.org>2003-03-11 09:30:37 +0000
committerRoland McGrath <roland@gnu.org>2003-03-11 09:30:37 +0000
commitb33e61633a10a233510f6f49b97b7e2ad15c7311 (patch)
tree9b6025b692435f629bc771dca74dcfb48e41768a /nptl/sysdeps/unix/sysv/linux/ia64/lowlevellock.h
parent5d5d5969b17422e3b1af6f88436e91f32a36fd58 (diff)
downloadglibc-b33e61633a10a233510f6f49b97b7e2ad15c7311.tar
glibc-b33e61633a10a233510f6f49b97b7e2ad15c7311.tar.gz
glibc-b33e61633a10a233510f6f49b97b7e2ad15c7311.tar.bz2
glibc-b33e61633a10a233510f6f49b97b7e2ad15c7311.zip
* sysdeps/generic/dl-sysdep.c (_dl_important_hwcaps): If CNT == 1,
allocate space even for the trailing '/'. Reported by John Reiser <jreiser@BitWagon.com>. * sysdeps/unix/sysv/linux/ia64/sysdep.h (LOAD_ARGS_6, ASM_ARGS_6, ASM_CLOBBERS_6): Define. (ASM_CLOBBERS_5): Use ASM_CLOBBERS_6. * sysdeps/unix/sysv/linux/ia64/clone2.S (__clone2): Reorder arguments to match IA-32 order. * sysdeps/unix/sysv/linux/i386/clone.S: Fix comment.
Diffstat (limited to 'nptl/sysdeps/unix/sysv/linux/ia64/lowlevellock.h')
-rw-r--r--nptl/sysdeps/unix/sysv/linux/ia64/lowlevellock.h251
1 files changed, 251 insertions, 0 deletions
diff --git a/nptl/sysdeps/unix/sysv/linux/ia64/lowlevellock.h b/nptl/sysdeps/unix/sysv/linux/ia64/lowlevellock.h
new file mode 100644
index 0000000000..8fcc7f15ac
--- /dev/null
+++ b/nptl/sysdeps/unix/sysv/linux/ia64/lowlevellock.h
@@ -0,0 +1,251 @@
+/* Copyright (C) 2003 Free Software Foundation, Inc.
+ This file is part of the GNU C Library.
+ Contributed by Jakub Jelinek <jakub@redhat.com>, 2003.
+
+ 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 _LOWLEVELLOCK_H
+#define _LOWLEVELLOCK_H 1
+
+#include <time.h>
+#include <sys/param.h>
+#include <bits/pthreadtypes.h>
+#include <ia64intrin.h>
+
+#define SYS_futex 1230
+#define FUTEX_WAIT 0
+#define FUTEX_WAKE 1
+
+/* Initializer for compatibility lock. */
+#define LLL_MUTEX_LOCK_INITIALIZER (0)
+
+#define lll_futex_clobbers \
+ "out4", "out5", "out6", "out7", \
+ /* Non-stacked integer registers, minus r8, r10, r15. */ \
+ "r2", "r3", "r9", "r11", "r12", "r13", "r14", "r16", "r17", "r18", \
+ "r19", "r20", "r21", "r22", "r23", "r24", "r25", "r26", "r27", \
+ "r28", "r29", "r30", "r31", \
+ /* Predicate registers. */ \
+ "p6", "p7", "p8", "p9", "p10", "p11", "p12", "p13", "p14", "p15", \
+ /* Non-rotating fp registers. */ \
+ "f6", "f7", "f8", "f9", "f10", "f11", "f12", "f13", "f14", "f15", \
+ /* Branch registers. */ \
+ "b6", "b7", \
+ "memory"
+
+#define lll_futex_wait(futex, val) \
+ ({ \
+ register unsigned long int __o0 asm ("out0") \
+ = (unsigned long int) (futex); \
+ register unsigned long int __o1 asm ("out1") = FUTEX_WAIT; \
+ register unsigned long int __o2 asm ("out2") = (unsigned long int) (val);\
+ register unsigned long int __o3 asm ("out3") = 0ul; \
+ register unsigned long int __r8 asm ("r8"); \
+ register unsigned long int __r10 asm ("r10"); \
+ register unsigned long int __r15 asm ("r15") = SYS_futex; \
+ \
+ __asm __volatile ("break %3;;" \
+ : "=r" (__r8), "=r" (__r10), "=r" (__r15) \
+ : "i" (0x100000), "2" (__r15), "r" (__o0), "r" (__o1), \
+ "r" (__o2), "r" (__o3) \
+ : lll_futex_clobbers); \
+ __r10 == -1 ? -__r8 : __r8; \
+ })
+
+
+#define lll_futex_timed_wait(futex, val, timespec) \
+ ({ \
+ register unsigned long int __o0 asm ("out0") \
+ = (unsigned long int) (futex); \
+ register unsigned long int __o1 asm ("out1") = FUTEX_WAIT; \
+ register unsigned long int __o2 asm ("out2") = (unsigned long int) (val);\
+ register unsigned long int __o3 asm ("out3") \
+ = (unsigned long int) (timespec); \
+ register unsigned long int __r8 asm ("r8"); \
+ register unsigned long int __r10 asm ("r10"); \
+ register unsigned long int __r15 asm ("r15") = SYS_futex; \
+ \
+ __asm __volatile ("break %3;;" \
+ : "=r" (__r8), "=r" (__r10), "=r" (__r15) \
+ : "i" (0x100000), "2" (__r15), "r" (__o0), "r" (__o1), \
+ "r" (__o2), "r" (__o3) \
+ : lll_futex_clobbers); \
+ __r10 == -1 ? -__r8 : __r8; \
+ })
+
+
+#define lll_futex_wake(futex, nr) \
+ ({ \
+ register unsigned long int __o0 asm ("out0") \
+ = (unsigned long int) (futex); \
+ register unsigned long int __o1 asm ("out1") = FUTEX_WAKE; \
+ register unsigned long int __o2 asm ("out2") = (unsigned long int) (nr); \
+ register unsigned long int __r8 asm ("r8"); \
+ register unsigned long int __r10 asm ("r10"); \
+ register unsigned long int __r15 asm ("r15") = SYS_futex; \
+ \
+ __asm __volatile ("break %3;;" \
+ : "=r" (__r8), "=r" (__r10), "=r" (__r15) \
+ : "i" (0x100000), "2" (__r15), "r" (__o0), "r" (__o1), \
+ "r" (__o2) \
+ : "out3", lll_futex_clobbers); \
+ __r10 == -1 ? -__r8 : __r8; \
+ })
+
+#define lll_compare_and_swap(futex, oldval, newval) \
+ __sync_val_compare_and_swap_si ((futex), (oldval), (newval))
+
+static inline int
+__attribute__ ((always_inline))
+__lll_mutex_trylock (int *futex)
+{
+ return lll_compare_and_swap (futex, 0, 1) != 0;
+}
+#define lll_mutex_trylock(futex) __lll_mutex_trylock (&(futex))
+
+
+extern void ___lll_mutex_lock (int *, int) attribute_hidden;
+
+
+static inline void
+__attribute__ ((always_inline))
+__lll_mutex_lock (int *futex)
+{
+ int oldval;
+ int val = *futex;
+
+ do
+ oldval = val;
+ while ((val = lll_compare_and_swap (futex, oldval, oldval + 1)) != oldval);
+ if (oldval > 0)
+ ___lll_mutex_lock (futex, oldval + 1);
+}
+#define lll_mutex_lock(futex) __lll_mutex_lock (&(futex))
+
+
+extern int ___lll_mutex_timedlock (int *, const struct timespec *, int)
+ attribute_hidden;
+
+
+static inline int
+__attribute__ ((always_inline))
+__lll_mutex_timedlock (int *futex, const struct timespec *abstime)
+{
+ int oldval;
+ int val = *futex;
+ int result = 0;
+
+ do
+ oldval = val;
+ while ((val = lll_compare_and_swap (futex, oldval, oldval + 1)) != oldval);
+ if (oldval > 0)
+ result = ___lll_mutex_timedlock (futex, abstime, oldval + 1);
+
+ return result;
+}
+#define lll_mutex_timedlock(futex, abstime) \
+ __lll_mutex_timedlock (&(futex), abstime)
+
+
+static inline void
+__attribute__ ((always_inline))
+__lll_mutex_unlock (int *futex)
+{
+ int oldval;
+ int val = *futex;
+
+ do
+ oldval = val;
+ while ((val = lll_compare_and_swap (futex, oldval, 0)) != oldval);
+ if (oldval > 1)
+ lll_futex_wake (futex, 1);
+}
+#define lll_mutex_unlock(futex) __lll_mutex_unlock(&(futex))
+
+#define lll_mutex_islocked(futex) \
+ (futex != 0)
+
+
+/* We have a separate internal lock implementation which is not tied
+ to binary compatibility. We can use the lll_mutex_*. */
+
+/* Type for lock object. */
+typedef int lll_lock_t;
+
+extern int lll_unlock_wake_cb (int *__futex) attribute_hidden;
+
+/* Initializers for lock. */
+#define LLL_LOCK_INITIALIZER (0)
+#define LLL_LOCK_INITIALIZER_LOCKED (1)
+
+#define lll_trylock(futex) lll_mutex_trylock (futex)
+#define lll_lock(futex) lll_mutex_lock (futex)
+#define lll_unlock(futex) lll_mutex_unlock (futex)
+#define lll_islocked(futex) lll_mutex_islocked (futex)
+
+
+/* The kernel notifies a process with uses CLONE_CLEARTID via futex
+ wakeup when the clone terminates. The memory location contains the
+ thread ID while the clone is running and is reset to zero
+ afterwards. */
+static inline void
+__attribute__ ((always_inline))
+__lll_wait_tid (int *ptid)
+{
+ int tid;
+
+ while ((tid = *ptid) != 0)
+ lll_futex_wait (ptid, tid);
+}
+#define lll_wait_tid(tid) __lll_wait_tid(&(tid))
+
+
+extern int ___lll_timedwait_tid (int *, const struct timespec *)
+ attribute_hidden;
+static inline int
+__attribute__ ((always_inline))
+__lll_timedwait_tid (int *ptid, const struct timespec *abstime)
+{
+ if (*ptid == 0)
+ return 0;
+
+ return ___lll_timedwait_tid (ptid, abstime);
+}
+#define lll_timedwait_tid(tid, abstime) __lll_timedwait_tid (&(tid), abstime)
+
+
+/* Conditional variable handling. */
+
+extern void __lll_cond_wait (pthread_cond_t *cond)
+ attribute_hidden;
+extern int __lll_cond_timedwait (pthread_cond_t *cond,
+ const struct timespec *abstime)
+ attribute_hidden;
+extern void __lll_cond_wake (pthread_cond_t *cond)
+ attribute_hidden;
+extern void __lll_cond_broadcast (pthread_cond_t *cond)
+ attribute_hidden;
+
+#define lll_cond_wait(cond) \
+ __lll_cond_wait (cond)
+#define lll_cond_timedwait(cond, abstime) \
+ __lll_cond_timedwait (cond, abstime)
+#define lll_cond_wake(cond) \
+ __lll_cond_wake (cond)
+#define lll_cond_broadcast(cond) \
+ __lll_cond_broadcast (cond)
+
+#endif /* lowlevellock.h */