summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMiod Vallat <miod@cvs.openbsd.org>2005-12-24 12:01:27 +0000
committerMiod Vallat <miod@cvs.openbsd.org>2005-12-24 12:01:27 +0000
commit1d756a49df11741b7fe6dd62495c3a0a5f6ee729 (patch)
tree3e1f70f620230d0360c3ef197cd14905e2b2847e
parentd580a5c063fae7dde1355e1b4cb0847f837fca78 (diff)
mips64 support code for librthread (_atomic_lock yanked from existing
libpthread code).
-rw-r--r--lib/librthread/arch/mips64/_atomic_lock.c38
-rw-r--r--lib/librthread/arch/mips64/rfork_thread.S79
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)