summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/arch/hppa64/conf/RAMDISK8
-rw-r--r--sys/arch/hppa64/hppa64/autoconf.c9
-rw-r--r--sys/arch/hppa64/hppa64/machdep.c4
-rw-r--r--sys/arch/hppa64/hppa64/mutex.c23
-rw-r--r--sys/arch/hppa64/hppa64/pmap.c32
-rw-r--r--sys/arch/hppa64/hppa64/trap.c8
-rw-r--r--sys/arch/hppa64/include/atomic.h237
-rw-r--r--sys/arch/hppa64/include/mutex.h34
8 files changed, 290 insertions, 65 deletions
diff --git a/sys/arch/hppa64/conf/RAMDISK b/sys/arch/hppa64/conf/RAMDISK
index d73cb721a40..8626998b771 100644
--- a/sys/arch/hppa64/conf/RAMDISK
+++ b/sys/arch/hppa64/conf/RAMDISK
@@ -1,4 +1,4 @@
-# $OpenBSD: RAMDISK,v 1.21 2014/10/10 05:43:35 deraadt Exp $
+# $OpenBSD: RAMDISK,v 1.22 2014/10/12 20:39:46 miod Exp $
machine hppa64
maxusers 4
@@ -26,11 +26,11 @@ mainbus0 at root
mem* at mainbus0 flags 0x00 # /dev/*mem and memory controller
pdc0 at mainbus0 # PDC/IODC wrapper for boot console
-power0 at mainbus0 # power/fail manager (iv 30)
+#power0 at mainbus0 # power/fail manager (iv 30)
cpu* at mainbus0 # HP PA-RISC cpu
-plut0 at mainbus0 # Astro Runway-Ropes, MIOC and IOA
+astro0 at mainbus0 # Astro Runway-Ropes, MIOC and IOA
-elroy* at plut0
+elroy* at astro0
pci* at elroy?
ppb* at pci?
diff --git a/sys/arch/hppa64/hppa64/autoconf.c b/sys/arch/hppa64/hppa64/autoconf.c
index 23b06d05dc0..0df825000b6 100644
--- a/sys/arch/hppa64/hppa64/autoconf.c
+++ b/sys/arch/hppa64/hppa64/autoconf.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: autoconf.c,v 1.21 2014/09/15 19:08:21 miod Exp $ */
+/* $OpenBSD: autoconf.c,v 1.22 2014/10/12 20:39:46 miod Exp $ */
/*
* Copyright (c) 1998-2005 Michael Shalayeff
@@ -232,8 +232,9 @@ print_devpath(const char *label, struct pz_device *pz)
for (i = 1; i < 6 && pz->pz_layers[i]; i++)
printf(".%x", pz->pz_layers[i]);
- printf(" class=%d flags=%b hpa=%p spa=%p io=%p\n", pz->pz_class,
- pz->pz_flags, PZF_BITS, pz->pz_hpa, pz->pz_spa, pz->pz_iodc_io);
+ printf(" class=%d flags=%b hpa=0x%08x spa=0x%08x io=0x%08x\n",
+ pz->pz_class, pz->pz_flags, PZF_BITS, pz->pz_hpa, pz->pz_spa,
+ pz->pz_iodc_io);
}
u_int32_t pdc_rt[16 / 4 * sizeof(struct pdc_pat_pci_rt)] PDC_ALIGNMENT;
@@ -374,7 +375,7 @@ pdc_getirt(int *pn)
return (NULL);
}
-printf("num %ld ", pdc_pat_io_num.num);
+printf("num %d ", pdc_pat_io_num.num);
*pn = num = pdc_pat_io_num.num;
if (num > sizeof(pdc_rt) / sizeof(*rt)) {
printf("\nPCI IRT is too big %d\n", num);
diff --git a/sys/arch/hppa64/hppa64/machdep.c b/sys/arch/hppa64/hppa64/machdep.c
index 7eb043828ec..c3437abd796 100644
--- a/sys/arch/hppa64/hppa64/machdep.c
+++ b/sys/arch/hppa64/hppa64/machdep.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: machdep.c,v 1.73 2014/09/19 18:21:14 kettenis Exp $ */
+/* $OpenBSD: machdep.c,v 1.74 2014/10/12 20:39:46 miod Exp $ */
/*
* Copyright (c) 2005 Michael Shalayeff
@@ -378,7 +378,7 @@ cpu_startup(void)
printf("%s%s\n", version, cpu_model);
printf("real mem = %lu (%luMB)\n", ptoa((psize_t)physmem),
ptoa((psize_t)physmem) / 1024 / 1024);
- printf("rsvd mem = %u (%uKB)\n", ptoa(resvmem), ptoa(resvmem) / 1024);
+ printf("rsvd mem = %lu (%luKB)\n", ptoa(resvmem), ptoa(resvmem) / 1024);
/*
* Allocate a submap for exec arguments. This map effectively
diff --git a/sys/arch/hppa64/hppa64/mutex.c b/sys/arch/hppa64/hppa64/mutex.c
index a6206e0817f..1e24878eafd 100644
--- a/sys/arch/hppa64/hppa64/mutex.c
+++ b/sys/arch/hppa64/hppa64/mutex.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: mutex.c,v 1.8 2012/06/05 11:43:41 jsing Exp $ */
+/* $OpenBSD: mutex.c,v 1.9 2014/10/12 20:39:46 miod Exp $ */
/*
* Copyright (c) 2004 Artur Grabowski <art@openbsd.org>
@@ -28,6 +28,7 @@
#include <sys/param.h>
#include <sys/mutex.h>
#include <sys/systm.h>
+#include <sys/atomic.h>
#include <machine/intr.h>
@@ -36,11 +37,12 @@
static inline int
try_lock(struct mutex *mtx)
{
- volatile int *lock = &mtx->mtx_lock;
+ volatile int *lock = (int *)(((vaddr_t)mtx->mtx_lock + 0xf) & ~0xf);
volatile register_t ret = 0;
+ /* Note: lock must be 16-byte aligned. */
asm volatile (
- "ldcw,co 0(%2), %0"
+ "ldcw,co 0(%2), %0"
: "=&r" (ret), "+m" (lock)
: "r" (lock)
);
@@ -49,9 +51,12 @@ try_lock(struct mutex *mtx)
}
void
-mtx_init(struct mutex *mtx, int wantipl)
+__mtx_init(struct mutex *mtx, int wantipl)
{
- mtx->mtx_lock = MUTEX_UNLOCKED;
+ mtx->mtx_lock[0] = 1;
+ mtx->mtx_lock[1] = 1;
+ mtx->mtx_lock[2] = 1;
+ mtx->mtx_lock[3] = 1;
mtx->mtx_wantipl = wantipl;
mtx->mtx_oldipl = IPL_NONE;
}
@@ -65,6 +70,7 @@ mtx_enter(struct mutex *mtx)
if (mtx->mtx_wantipl != IPL_NONE)
s = splraise(mtx->mtx_wantipl);
if (try_lock(mtx)) {
+ membar_enter();
if (mtx->mtx_wantipl != IPL_NONE)
mtx->mtx_oldipl = s;
mtx->mtx_owner = curcpu();
@@ -86,6 +92,7 @@ mtx_enter_try(struct mutex *mtx)
if (mtx->mtx_wantipl != IPL_NONE)
s = splraise(mtx->mtx_wantipl);
if (try_lock(mtx)) {
+ membar_enter();
if (mtx->mtx_wantipl != IPL_NONE)
mtx->mtx_oldipl = s;
mtx->mtx_owner = curcpu();
@@ -112,8 +119,12 @@ mtx_leave(struct mutex *mtx)
#endif
s = mtx->mtx_oldipl;
mtx->mtx_owner = NULL;
+ membar_exit();
- mtx->mtx_lock = MUTEX_UNLOCKED;
+ mtx->mtx_lock[0] = 1;
+ mtx->mtx_lock[1] = 1;
+ mtx->mtx_lock[2] = 1;
+ mtx->mtx_lock[3] = 1;
if (mtx->mtx_wantipl != IPL_NONE)
splx(s);
diff --git a/sys/arch/hppa64/hppa64/pmap.c b/sys/arch/hppa64/hppa64/pmap.c
index 26a59f7684d..0b066cd6439 100644
--- a/sys/arch/hppa64/hppa64/pmap.c
+++ b/sys/arch/hppa64/hppa64/pmap.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: pmap.c,v 1.23 2012/06/03 13:28:40 jsing Exp $ */
+/* $OpenBSD: pmap.c,v 1.24 2014/10/12 20:39:46 miod Exp $ */
/*
* Copyright (c) 2005 Michael Shalayeff
@@ -17,7 +17,9 @@
* OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
+#ifndef SMALL_KERNEL
#define PMAPDEBUG
+#endif
#include <sys/param.h>
#include <sys/systm.h>
@@ -253,7 +255,7 @@ static __inline void
pmap_pte_set(volatile pt_entry_t *pde, vaddr_t va, pt_entry_t pte)
{
DPRINTF(PDB_FOLLOW|PDB_VP,
- ("pmap_pte_set(%p, 0x%lx, 0x%lx)\n", pde, va, pte));
+ ("pmap_pte_set(%p, 0x%lx, 0x%lx)\n", pde, va, (long)pte));
pde[(va & PTE_MASK) >> PTE_SHIFT] = pte;
}
@@ -309,8 +311,8 @@ pmap_dump_table(pa_space_t space, vaddr_t sva)
if (!(pte = pmap_pte_get(pde, va)))
continue;
- printf("0x%08lx-0x%08lx:%b\n",
- va, PTE_PAGE(pte), PTE_GETBITS(pte), PTE_BITS);
+ printf("0x%08lx-0x%08llx:%lb\n",
+ va, PTE_PAGE(pte), (long)PTE_GETBITS(pte), PTE_BITS);
}
}
@@ -460,7 +462,7 @@ printf("pa 0x%lx tpa 0x%lx\n", pa, tpa);
epde = pde + (PTE_MASK >> PTE_SHIFT) + 1;
if (pa + (PTE_MASK + (1 << PTE_SHIFT)) > tpa)
epde = pde + ((tpa & PTE_MASK) >> PTE_SHIFT);
-printf("pde %p epde %p pte 0x%lx\n", pde, epde, pte);
+printf("pde %p epde %p pte 0x%lx\n", pde, epde, (long)pte);
for (pde += (pa & PTE_MASK) >> PTE_SHIFT; pde < epde;)
*pde++ = pte;
pa += PTE_MASK + (1 << PTE_SHIFT);
@@ -480,6 +482,7 @@ pmap_bootstrap(vaddr_t vstart)
DPRINTF(PDB_FOLLOW|PDB_INIT, ("pmap_bootstrap(0x%lx)\n", vstart));
+ uvmexp.pagesize = PAGE_SIZE;
uvm_setpagesize();
hppa_prot[UVM_PROT_NONE] = PTE_ORDER|PTE_ACC_NONE;
@@ -536,7 +539,7 @@ pmap_bootstrap(vaddr_t vstart)
eaddr = physmem - atop(round_page(MSGBUFSIZE));
resvphysmem = atop(addr);
- DPRINTF(PDB_INIT, ("physmem: 0x%lx - 0x%lx\n", resvphysmem, eaddr));
+ DPRINTF(PDB_INIT, ("physmem: 0x%x - 0x%lx\n", resvphysmem, eaddr));
uvm_page_physload(0, physmem, resvphysmem, eaddr, 0);
}
@@ -580,7 +583,7 @@ pmap_init(void)
#ifdef PMAP_STEAL_MEMORY
vaddr_t
-pmap_steal_memory(vsize_t size, vaddr_t *vstartp, vaddr_t *vendp)
+pmap_steal_memory(vsize_t size, vaddr_t *vstartp, vaddr_t *vendp, int zero)
{
vaddr_t va;
int npg;
@@ -602,7 +605,8 @@ pmap_steal_memory(vsize_t size, vaddr_t *vstartp, vaddr_t *vendp)
vm_physmem[0].end -= npg;
vm_physmem[0].avail_end -= npg;
va = ptoa(vm_physmem[0].avail_end);
- bzero((void *)va, size);
+ if (zero)
+ bzero((void *)va, size);
DPRINTF(PDB_FOLLOW|PDB_PHYS, ("pmap_steal_memory: 0x%lx\n", va));
@@ -637,7 +641,7 @@ pmap_growkernel(vaddr_t kva)
} else {
paddr_t pa;
- pa = pmap_steal_memory(PAGE_SIZE, NULL, NULL);
+ pa = pmap_steal_memory(PAGE_SIZE, NULL, NULL, 1);
if (pa)
panic("pmap_growkernel: out of memory");
pmap_pde_set(pmap_kernel(), va, pa);
@@ -739,7 +743,7 @@ pmap_destroy(struct pmap *pmap)
npv = pv->pv_next;
if (pv->pv_pmap == pmap) {
#ifdef PMAPDEBUG
- printf(" 0x%x", pv->pv_va);
+ printf(" 0x%lx", pv->pv_va);
#endif
pmap_remove(pmap, pv->pv_va,
pv->pv_va + PAGE_SIZE);
@@ -818,7 +822,7 @@ pmap_enter(struct pmap *pmap, vaddr_t va, paddr_t pa, vm_prot_t prot, int flags)
if ((pte = pmap_pte_get(pde, va))) {
DPRINTF(PDB_ENTER,
- ("pmap_enter: remapping 0x%lx -> 0x%lx\n", pte, pa));
+ ("pmap_enter: remapping 0x%lx -> 0x%lx\n", (long)pte, pa));
pmap_pte_flush(pmap, va, pte);
if (wired && !(pte & PTE_WIRED))
@@ -979,7 +983,7 @@ pmap_write_protect(struct pmap *pmap, vaddr_t sva, vaddr_t eva, vm_prot_t prot)
DPRINTF(PDB_PMAP,
("pmap_write_protect: va=0x%lx pte=0x%lx\n",
- sva, pte));
+ sva, (long)pte));
/*
* Determine if mapping is changing.
* If not, nothing to do.
@@ -1078,7 +1082,7 @@ pmap_changebit(struct vm_page *pg, pt_entry_t set, pt_entry_t clear)
pt_entry_t res;
DPRINTF(PDB_FOLLOW|PDB_BITS,
- ("pmap_changebit(%p, %lx, %lx)\n", pg, set, clear));
+ ("pmap_changebit(%p, %lx, %lx)\n", pg, (long)set, (long)clear));
simple_lock(&pg->mdpage.pvh_lock);
res = pg->mdpage.pvh_attrs = 0;
@@ -1121,7 +1125,7 @@ pmap_testbit(struct vm_page *pg, pt_entry_t bit)
struct pv_entry *pve;
pt_entry_t pte;
- DPRINTF(PDB_FOLLOW|PDB_BITS, ("pmap_testbit(%p, %lx)\n", pg, bit));
+ DPRINTF(PDB_FOLLOW|PDB_BITS, ("pmap_testbit(%p, %lx)\n", pg, (long)bit));
simple_lock(&pg->mdpage.pvh_lock);
for(pve = pg->mdpage.pvh_list; !(pg->mdpage.pvh_attrs & bit) && pve;
diff --git a/sys/arch/hppa64/hppa64/trap.c b/sys/arch/hppa64/hppa64/trap.c
index 857268e43ca..afa646c7370 100644
--- a/sys/arch/hppa64/hppa64/trap.c
+++ b/sys/arch/hppa64/hppa64/trap.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: trap.c,v 1.39 2014/05/11 00:12:44 guenther Exp $ */
+/* $OpenBSD: trap.c,v 1.40 2014/10/12 20:39:46 miod Exp $ */
/*
* Copyright (c) 2005 Michael Shalayeff
@@ -250,7 +250,7 @@ trap(int type, struct trapframe *frame)
}
#else
if (type == T_DATALIGN)
- panic ("trap: %s at 0x%x", tts, va);
+ panic ("trap: %s at 0x%lx", tts, va);
else
panic ("trap: no debugger for \"%s\" (%d)", tts, type);
#endif
@@ -378,7 +378,7 @@ trap(int type, struct trapframe *frame)
sv.sival_int = va;
trapsignal(p, SIGILL, type & ~T_USER, ILL_ILLTRP, sv);
} else
- panic("trap: %s @ 0x%x:0x%x for 0x%x:0x%x irr 0x%08x",
+ panic("trap: %s @ 0x%lx:0x%lx for 0x%x:0x%lx irr 0x%08x",
tts, frame->tf_iisq[0], frame->tf_iioq[0],
space, va, opcode);
break;
@@ -635,7 +635,7 @@ syscall(struct trapframe *frame)
#ifdef DIAGNOSTIC
if (curcpu()->ci_cpl != oldcpl) {
printf("WARNING: SPL (0x%x) NOT LOWERED ON "
- "syscall(0x%x, 0x%x, 0x%x, 0x%x...) EXIT, PID %d\n",
+ "syscall(0x%x, 0x%lx, 0x%lx, 0x%lx...) EXIT, PID %d\n",
curcpu()->ci_cpl, code, args[0], args[1], args[2],
p->p_pid);
curcpu()->ci_cpl = oldcpl;
diff --git a/sys/arch/hppa64/include/atomic.h b/sys/arch/hppa64/include/atomic.h
index 9cfe677db9c..e1c122ea09e 100644
--- a/sys/arch/hppa64/include/atomic.h
+++ b/sys/arch/hppa64/include/atomic.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: atomic.h,v 1.6 2014/03/29 18:09:29 guenther Exp $ */
+/* $OpenBSD: atomic.h,v 1.7 2014/10/12 20:39:46 miod Exp $ */
/* Public Domain */
@@ -7,41 +7,219 @@
#if defined(_KERNEL)
-#include <sys/mutex.h>
+typedef volatile u_int __cpu_simple_lock_t __attribute__((__aligned__(16)));
+
+#define __SIMPLELOCK_LOCKED 0
+#define __SIMPLELOCK_UNLOCKED 1
+
+static inline void
+__cpu_simple_lock_init(__cpu_simple_lock_t *l)
+{
+ *l = __SIMPLELOCK_UNLOCKED;
+}
+
+static inline unsigned int
+__cpu_simple_lock_ldcws(__cpu_simple_lock_t *l)
+{
+ unsigned int o;
+
+ asm volatile("ldcws 0(%2), %0" : "=&r" (o), "+m" (l) : "r" (l));
+
+ return (o);
+}
+
+static inline void
+__cpu_simple_lock(__cpu_simple_lock_t *l)
+{
+ while (__cpu_simple_lock_ldcws(l) == __SIMPLELOCK_UNLOCKED)
+ ;
+}
+
+static inline int
+__cpu_simple_lock_try(__cpu_simple_lock_t *l)
+{
+ return (__cpu_simple_lock_ldcws(l) == __SIMPLELOCK_UNLOCKED);
+}
+
+static inline void
+__cpu_simple_unlock(__cpu_simple_lock_t *l)
+{
+ *l = __SIMPLELOCK_UNLOCKED;
+}
#ifdef MULTIPROCESSOR
-extern struct mutex mtx_atomic;
-#define ATOMIC_LOCK mtx_enter(&mtx_atomic)
-#define ATOMIC_UNLOCK mtx_leave(&mtx_atomic)
+extern __cpu_simple_lock_t atomic_lock;
+#define ATOMIC_LOCK __cpu_simple_lock(&atomic_lock);
+#define ATOMIC_UNLOCK __cpu_simple_unlock(&atomic_lock);
#else
#define ATOMIC_LOCK
#define ATOMIC_UNLOCK
#endif
-static __inline void
-atomic_setbits_int(volatile unsigned int *uip, unsigned int v)
+static inline register_t
+atomic_enter(void)
{
register_t eiem;
__asm volatile("mfctl %%cr15, %0": "=r" (eiem));
__asm volatile("mtctl %r0, %cr15");
ATOMIC_LOCK;
- *uip |= v;
+
+ return (eiem);
+}
+
+static inline void
+atomic_leave(register_t eiem)
+{
ATOMIC_UNLOCK;
__asm volatile("mtctl %0, %%cr15":: "r" (eiem));
}
+static inline unsigned int
+_atomic_cas_uint(volatile unsigned int *uip, unsigned int o, unsigned int n)
+{
+ register_t eiem;
+ unsigned int rv;
+
+ eiem = atomic_enter();
+ rv = *uip;
+ if (rv == o)
+ *uip = n;
+ atomic_leave(eiem);
+
+ return (rv);
+}
+#define atomic_cas_uint(_p, _o, _n) _atomic_cas_uint((_p), (_o), (_n))
+
+static inline unsigned long
+_atomic_cas_ulong(volatile unsigned long *uip, unsigned long o, unsigned long n)
+{
+ register_t eiem;
+ unsigned long rv;
+
+ eiem = atomic_enter();
+ rv = *uip;
+ if (rv == o)
+ *uip = n;
+ atomic_leave(eiem);
+
+ return (rv);
+}
+#define atomic_cas_ulong(_p, _o, _n) _atomic_cas_ulong((_p), (_o), (_n))
+
+static inline void *
+_atomic_cas_ptr(volatile void *uip, void *o, void *n)
+{
+ register_t eiem;
+ void * volatile *uipp = (void * volatile *)uip;
+ void *rv;
+
+ eiem = atomic_enter();
+ rv = *uipp;
+ if (rv == o)
+ *uipp = n;
+ atomic_leave(eiem);
+
+ return (rv);
+}
+#define atomic_cas_ptr(_p, _o, _n) _atomic_cas_ptr((_p), (_o), (_n))
+
+static inline unsigned int
+_atomic_swap_uint(volatile unsigned int *uip, unsigned int n)
+{
+ register_t eiem;
+ unsigned int rv;
+
+ eiem = atomic_enter();
+ rv = *uip;
+ *uip = n;
+ atomic_leave(eiem);
+
+ return (rv);
+}
+#define atomic_swap_uint(_p, _n) _atomic_swap_uint((_p), (_n))
+
+static inline unsigned long
+_atomic_swap_ulong(volatile unsigned long *uip, unsigned long n)
+{
+ register_t eiem;
+ unsigned long rv;
+
+ eiem = atomic_enter();
+ rv = *uip;
+ *uip = n;
+ atomic_leave(eiem);
+
+ return (rv);
+}
+#define atomic_swap_ulong(_p, _n) _atomic_swap_ulong((_p), (_n))
+
+static inline void *
+_atomic_swap_ptr(volatile void *uip, void *n)
+{
+ register_t eiem;
+ void * volatile *uipp = (void * volatile *)uip;
+ void *rv;
+
+ eiem = atomic_enter();
+ rv = *uipp;
+ *uipp = n;
+ atomic_leave(eiem);
+
+ return (rv);
+}
+#define atomic_swap_ptr(_p, _n) _atomic_swap_ptr((_p), (_n))
+
+static __inline unsigned int
+_atomic_add_int_nv(volatile unsigned int *uip, unsigned int v)
+{
+ register_t eiem;
+ unsigned int rv;
+
+ eiem = atomic_enter();
+ rv = *uip + v;
+ *uip = rv;
+ atomic_leave(eiem);
+
+ return (rv);
+}
+#define atomic_add_int_nv(_uip, _v) _atomic_add_int_nv((_uip), (_v))
+#define atomic_sub_int_nv(_uip, _v) _atomic_add_int_nv((_uip), 0 - (_v))
+
+static __inline unsigned long
+_atomic_add_long_nv(volatile unsigned long *uip, unsigned long v)
+{
+ register_t eiem;
+ unsigned long rv;
+
+ eiem = atomic_enter();
+ rv = *uip + v;
+ *uip = rv;
+ atomic_leave(eiem);
+
+ return (rv);
+}
+#define atomic_add_long_nv(_uip, _v) _atomic_add_long_nv((_uip), (_v))
+#define atomic_sub_long_nv(_uip, _v) _atomic_add_long_nv((_uip), 0 - (_v))
+
+static __inline void
+atomic_setbits_int(volatile unsigned int *uip, unsigned int v)
+{
+ register_t eiem;
+
+ eiem = atomic_enter();
+ *uip |= v;
+ atomic_leave(eiem);
+}
+
static __inline void
atomic_clearbits_int(volatile unsigned int *uip, unsigned int v)
{
register_t eiem;
- __asm volatile("mfctl %%cr15, %0": "=r" (eiem));
- __asm volatile("mtctl %r0, %cr15");
- ATOMIC_LOCK;
+ eiem = atomic_enter();
*uip &= ~v;
- ATOMIC_UNLOCK;
- __asm volatile("mtctl %0, %%cr15":: "r" (eiem));
+ atomic_leave(eiem);
}
static __inline void
@@ -49,12 +227,9 @@ atomic_setbits_long(volatile unsigned long *uip, unsigned long v)
{
register_t eiem;
- __asm volatile("mfctl %%cr15, %0": "=r" (eiem));
- __asm volatile("mtctl %r0, %cr15");
- ATOMIC_LOCK;
+ eiem = atomic_enter();
*uip |= v;
- ATOMIC_UNLOCK;
- __asm volatile("mtctl %0, %%cr15":: "r" (eiem));
+ atomic_leave(eiem);
}
static __inline void
@@ -62,13 +237,29 @@ atomic_clearbits_long(volatile unsigned long *uip, unsigned long v)
{
register_t eiem;
- __asm volatile("mfctl %%cr15, %0": "=r" (eiem));
- __asm volatile("mtctl %r0, %cr15");
- ATOMIC_LOCK;
+ eiem = atomic_enter();
*uip &= ~v;
- ATOMIC_UNLOCK;
- __asm volatile("mtctl %0, %%cr15":: "r" (eiem));
+ atomic_leave(eiem);
+}
+
+/*
+ * Although the PA-RISC 2.0 architecture allows an implementation to
+ * be weakly ordered, all PA-RISC processers to date implement a
+ * strong memory ordering model. So all we need is a compiler
+ * barrier.
+ */
+
+static inline void
+__insn_barrier(void)
+{
+ __asm volatile("" : : : "memory");
}
+#define membar_enter() __insn_barrier()
+#define membar_exit() __insn_barrier()
+#define membar_producer() __insn_barrier()
+#define membar_consumer() __insn_barrier()
+#define membar_sync() __insn_barrier()
+
#endif /* defined(_KERNEL) */
#endif /* _MACHINE_ATOMIC_H_ */
diff --git a/sys/arch/hppa64/include/mutex.h b/sys/arch/hppa64/include/mutex.h
index 6f20f5cb85b..a4b19bfa63e 100644
--- a/sys/arch/hppa64/include/mutex.h
+++ b/sys/arch/hppa64/include/mutex.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: mutex.h,v 1.2 2012/06/05 11:43:41 jsing Exp $ */
+/* $OpenBSD: mutex.h,v 1.3 2014/10/12 20:39:46 miod Exp $ */
/*
* Copyright (c) 2004 Artur Grabowski <art@openbsd.org>
@@ -28,28 +28,46 @@
#ifndef _MACHINE_MUTEX_H_
#define _MACHINE_MUTEX_H_
-#define MUTEX_LOCKED 0
-#define MUTEX_UNLOCKED 1
+#define MUTEX_LOCKED { 0, 0, 0, 0 }
+#define MUTEX_UNLOCKED { 1, 1, 1, 1 }
+/* Note: mtx_lock must be 16-byte aligned. */
struct mutex {
- volatile int mtx_lock;
+ volatile int mtx_lock[4];
int mtx_wantipl;
int mtx_oldipl;
void *mtx_owner;
};
-void mtx_init(struct mutex *, int);
+/*
+ * To prevent lock ordering problems with the kernel lock, we need to
+ * make sure we block all interrupts that can grab the kernel lock.
+ * The simplest way to achieve this is to make sure mutexes always
+ * raise the interrupt priority level to the highest level that has
+ * interrupts that grab the kernel lock.
+ */
+#ifdef MULTIPROCESSOR
+#define __MUTEX_IPL(ipl) \
+ (((ipl) > IPL_NONE && (ipl) < IPL_AUDIO) ? IPL_AUDIO : (ipl))
+#else
+#define __MUTEX_IPL(ipl) (ipl)
+#endif
+
+#define MUTEX_INITIALIZER(ipl) { MUTEX_UNLOCKED, __MUTEX_IPL((ipl)), 0, NULL }
-#define MUTEX_INITIALIZER(ipl) { MUTEX_UNLOCKED, (ipl), 0, NULL }
+void __mtx_init(struct mutex *, int);
+#define mtx_init(mtx, ipl) __mtx_init((mtx), __MUTEX_IPL((ipl)))
#ifdef DIAGNOSTIC
#define MUTEX_ASSERT_LOCKED(mtx) do { \
- if ((mtx)->mtx_lock != MUTEX_LOCKED) \
+ if ((mtx)->mtx_lock[0] == 1 && (mtx)->mtx_lock[1] == 1 && \
+ (mtx)->mtx_lock[2] == 1 && (mtx)->mtx_lock[3] == 1) \
panic("mutex %p not held in %s", (mtx), __func__); \
} while (0)
#define MUTEX_ASSERT_UNLOCKED(mtx) do { \
- if ((mtx)->mtx_lock[0] != MUTEX_UNLOCKED) \
+ if ((mtx)->mtx_lock[0] != 1 && (mtx)->mtx_lock[1] != 1 && \
+ (mtx)->mtx_lock[2] != 1 && (mtx)->mtx_lock[3] != 1) \
panic("mutex %p held in %s", (mtx), __func__); \
} while (0)
#else