diff options
author | Miod Vallat <miod@cvs.openbsd.org> | 2005-12-24 12:01:27 +0000 |
---|---|---|
committer | Miod Vallat <miod@cvs.openbsd.org> | 2005-12-24 12:01:27 +0000 |
commit | 1d756a49df11741b7fe6dd62495c3a0a5f6ee729 (patch) | |
tree | 3e1f70f620230d0360c3ef197cd14905e2b2847e | |
parent | d580a5c063fae7dde1355e1b4cb0847f837fca78 (diff) |
mips64 support code for librthread (_atomic_lock yanked from existing
libpthread code).
-rw-r--r-- | lib/librthread/arch/mips64/_atomic_lock.c | 38 | ||||
-rw-r--r-- | lib/librthread/arch/mips64/rfork_thread.S | 79 |
2 files changed, 117 insertions, 0 deletions
diff --git a/lib/librthread/arch/mips64/_atomic_lock.c b/lib/librthread/arch/mips64/_atomic_lock.c new file mode 100644 index 00000000000..e1a760b6744 --- /dev/null +++ b/lib/librthread/arch/mips64/_atomic_lock.c @@ -0,0 +1,38 @@ +/* $OpenBSD: _atomic_lock.c,v 1.1 2005/12/24 12:01:26 miod Exp $ */ +/* + * Atomic lock for mips + */ + +#include <machine/spinlock.h> + +int +_atomic_lock(volatile _spinlock_lock_t *lock) +{ + _spinlock_lock_t old; + _spinlock_lock_t temp; + + do { + /* + * On a mips2 machine and above, we can use ll/sc. + * Read the lock and tag the cache line with a 'load linked' + * instruction. (Register 17 (LLAddr) will hold the + * physical address of lock for diagnostic purposes); + * (Under pathologically heavy swapping, the physaddr may + * change! XXX) + */ + __asm__("ll %0, %1" : "=r"(old) : "m"(*lock)); + if (old != _SPINLOCK_UNLOCKED) + break; /* already locked */ + /* + * Try and store a 1 at the tagged lock address. If + * anyone else has since written it, the tag on the cache + * line will have been wiped, and temp will be set to zero + * by the 'store conditional' instruction. + */ + temp = _SPINLOCK_LOCKED; + __asm__("sc %0, %1" : "=r"(temp), "=m"(*lock) + : "0"(temp)); + } while (temp == _SPINLOCK_UNLOCKED); + + return (old != _SPINLOCK_UNLOCKED); +} diff --git a/lib/librthread/arch/mips64/rfork_thread.S b/lib/librthread/arch/mips64/rfork_thread.S new file mode 100644 index 00000000000..14db8ab8a20 --- /dev/null +++ b/lib/librthread/arch/mips64/rfork_thread.S @@ -0,0 +1,79 @@ +/* $OpenBSD: rfork_thread.S,v 1.1 2005/12/24 12:01:26 miod Exp $ */ + +/* + * Copyright (c) 2005, Miodrag Vallat + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * + * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR + * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED + * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, + * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + */ + +#include "../../../libc/arch/mips64/SYS.h" + +/* + * int rfork_thread(int flags, void *stack, void (*func)(void *), void *arg); + */ +LEAF(rfork_thread, 32) + /* a0 = flags, a1 = stack, a2 = func, a3 = arg */ + PTR_SUBU sp, 32 + SETUP_GP64(16, _C_LABEL(rfork_thread)) + + move t0, a1 /* stack */ + move t1, a3 /* arg */ + + __DO_SYSCALL(rfork) + bnez a3, 9f + + RESTORE_GP64 + PTR_ADDU sp, 32 + + beqz v0, 1f + + /* + * In parent process: just return. + */ + j ra + nop + +1: + /* + * In child process: switch stack, invoke function, then exit. + */ + + move sp, t0 /* stack */ + move t9, a2 /* func */ + move a0, t1 /* arg */ + jal ra, t9 + move v0, zero + + move a0, zero + __DO_SYSCALL(threxit) + +9: + /* + * System call failure. + */ + LA t9, _C_LABEL(cerror) + + RESTORE_GP64 + PTR_ADDU sp, 32 + jr t9 + nop +END(rfork_thread) |