summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/uvm/uvm_device.c60
1 files changed, 22 insertions, 38 deletions
diff --git a/sys/uvm/uvm_device.c b/sys/uvm/uvm_device.c
index c383b94dbb9..1013f46d20a 100644
--- a/sys/uvm/uvm_device.c
+++ b/sys/uvm/uvm_device.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: uvm_device.c,v 1.31 2009/05/08 13:50:15 ariane Exp $ */
+/* $OpenBSD: uvm_device.c,v 1.32 2009/05/12 20:49:56 oga Exp $ */
/* $NetBSD: uvm_device.c,v 1.30 2000/11/25 06:27:59 chs Exp $ */
/*
@@ -55,28 +55,25 @@
* we keep a list of active device objects in the system.
*/
-LIST_HEAD(udv_list_struct, uvm_device);
-static struct udv_list_struct udv_list;
-static simple_lock_data_t udv_lock;
+LIST_HEAD(udvlist, uvm_device) udv_list = LIST_HEAD_INITIALIZER(udv_list);
+struct mutex udv_lock = MUTEX_INITIALIZER(IPL_NONE);
/*
* functions
*/
-void udv_init(void);
void udv_reference(struct uvm_object *);
void udv_detach(struct uvm_object *);
-int udv_fault(struct uvm_faultinfo *, vaddr_t,
- vm_page_t *, int, int, vm_fault_t,
- vm_prot_t, int);
boolean_t udv_flush(struct uvm_object *, voff_t, voff_t, int);
+int udv_fault(struct uvm_faultinfo *, vaddr_t, vm_page_t *,
+ int, int, vm_fault_t, vm_prot_t, int);
/*
* master pager structure
*/
struct uvm_pagerops uvm_deviceops = {
- udv_init,
+ NULL, /* lock and list already initialized */
udv_reference,
udv_detach,
udv_fault,
@@ -88,20 +85,6 @@ struct uvm_pagerops uvm_deviceops = {
*/
/*
- * udv_init
- *
- * init pager private data structures.
- */
-
-void
-udv_init(void)
-{
-
- LIST_INIT(&udv_list);
- simple_lock_init(&udv_lock);
-}
-
-/*
* udv_attach
*
* get a VM object that is associated with a device. allocate a new
@@ -163,7 +146,7 @@ udv_attach(void *arg, vm_prot_t accessprot, voff_t off, vsize_t size)
* first, attempt to find it on the main list
*/
- simple_lock(&udv_lock);
+ mtx_enter(&udv_lock);
LIST_FOREACH(lcv, &udv_list, u_list) {
if (device == lcv->u_device)
break;
@@ -177,19 +160,19 @@ udv_attach(void *arg, vm_prot_t accessprot, voff_t off, vsize_t size)
/*
* if someone else has a hold on it, sleep and start
- * over again.
+ * over again. Else, we need the HOLD flag so we
+ * don't have to re-order locking here.
*/
-
if (lcv->u_flags & UVM_DEVICE_HOLD) {
lcv->u_flags |= UVM_DEVICE_WANTED;
- UVM_UNLOCK_AND_WAIT(lcv, &udv_lock, FALSE,
- "udv_attach",0);
+ msleep(lcv, &udv_lock, PVM | PNORELOCK,
+ "udv_attach", 0);
continue;
}
/* we are now holding it */
lcv->u_flags |= UVM_DEVICE_HOLD;
- simple_unlock(&udv_lock);
+ mtx_leave(&udv_lock);
/*
* bump reference count, unhold, return.
@@ -199,11 +182,11 @@ udv_attach(void *arg, vm_prot_t accessprot, voff_t off, vsize_t size)
lcv->u_obj.uo_refs++;
simple_unlock(&lcv->u_obj.vmobjlock);
- simple_lock(&udv_lock);
+ mtx_enter(&udv_lock);
if (lcv->u_flags & UVM_DEVICE_WANTED)
wakeup(lcv);
lcv->u_flags &= ~(UVM_DEVICE_WANTED|UVM_DEVICE_HOLD);
- simple_unlock(&udv_lock);
+ mtx_leave(&udv_lock);
return(&lcv->u_obj);
}
@@ -211,10 +194,10 @@ udv_attach(void *arg, vm_prot_t accessprot, voff_t off, vsize_t size)
* did not find it on main list. need to malloc a new one.
*/
- simple_unlock(&udv_lock);
+ mtx_leave(&udv_lock);
/* NOTE: we could sleep in the following malloc() */
udv = malloc(sizeof(*udv), M_TEMP, M_WAITOK);
- simple_lock(&udv_lock);
+ mtx_enter(&udv_lock);
/*
* now we have to double check to make sure no one added it
@@ -232,7 +215,7 @@ udv_attach(void *arg, vm_prot_t accessprot, voff_t off, vsize_t size)
*/
if (lcv) {
- simple_unlock(&udv_lock);
+ mtx_leave(&udv_lock);
free(udv, M_TEMP);
continue;
}
@@ -250,7 +233,7 @@ udv_attach(void *arg, vm_prot_t accessprot, voff_t off, vsize_t size)
udv->u_flags = 0;
udv->u_device = device;
LIST_INSERT_HEAD(&udv_list, udv, u_list);
- simple_unlock(&udv_lock);
+ mtx_leave(&udv_lock);
return(&udv->u_obj);
}
/*NOTREACHED*/
@@ -308,13 +291,14 @@ again:
/*
* is it being held? if so, wait until others are done.
+ * It is probably about to be referenced.
*/
- simple_lock(&udv_lock);
+ mtx_enter(&udv_lock);
if (udv->u_flags & UVM_DEVICE_HOLD) {
udv->u_flags |= UVM_DEVICE_WANTED;
simple_unlock(&uobj->vmobjlock);
- UVM_UNLOCK_AND_WAIT(udv, &udv_lock, FALSE, "udv_detach",0);
+ msleep(udv, &udv_lock, PVM | PNORELOCK, "udv_detach", 0);
goto again;
}
@@ -325,7 +309,7 @@ again:
LIST_REMOVE(udv, u_list);
if (udv->u_flags & UVM_DEVICE_WANTED)
wakeup(udv);
- simple_unlock(&udv_lock);
+ mtx_leave(&udv_lock);
simple_unlock(&uobj->vmobjlock);
free(udv, M_TEMP);
UVMHIST_LOG(maphist," <- done, freed uobj=%p", uobj,0,0,0);