summaryrefslogtreecommitdiffstats
path: root/sys/miscfs
diff options
context:
space:
mode:
authordyson <dyson@FreeBSD.org>1996-07-02 01:40:52 +0000
committerdyson <dyson@FreeBSD.org>1996-07-02 01:40:52 +0000
commit66550c3ffbed3be05696c356e4dbd3ab8b13255d (patch)
tree8e3522041d0828f18acd54c5e72047ec3930c892 /sys/miscfs
parent0e821da241cd152026b20c8292b396df8af5ab89 (diff)
downloadFreeBSD-src-66550c3ffbed3be05696c356e4dbd3ab8b13255d.zip
FreeBSD-src-66550c3ffbed3be05696c356e4dbd3ab8b13255d.tar.gz
Fix a serious problem, with a window where an object lock is needed,
but not there. The extent of the object lock is expanded to be over the range that it is needed. Additionally, clean up the code so that it conforms to better coding style.
Diffstat (limited to 'sys/miscfs')
-rw-r--r--sys/miscfs/procfs/procfs_mem.c154
1 files changed, 97 insertions, 57 deletions
diff --git a/sys/miscfs/procfs/procfs_mem.c b/sys/miscfs/procfs/procfs_mem.c
index 34d7f35..b82d856 100644
--- a/sys/miscfs/procfs/procfs_mem.c
+++ b/sys/miscfs/procfs/procfs_mem.c
@@ -37,7 +37,7 @@
*
* @(#)procfs_mem.c 8.4 (Berkeley) 1/21/94
*
- * $Id: procfs_mem.c,v 1.17 1996/01/25 06:05:38 peter Exp $
+ * $Id: procfs_mem.c,v 1.18 1996/06/11 23:52:27 dyson Exp $
*/
/*
@@ -74,6 +74,10 @@ procfs_rwmem(p, uio)
int error;
int writing;
struct vmspace *vm;
+ int fix_prot = 0;
+ vm_map_t map;
+ vm_object_t object = NULL;
+ vm_offset_t pageno = 0; /* page number */
/*
* if the vmspace is in the midst of being deallocated or the
@@ -84,6 +88,10 @@ procfs_rwmem(p, uio)
if ((p->p_flag & P_WEXIT) || (vm->vm_refcnt < 1))
return EFAULT;
++vm->vm_refcnt;
+ /*
+ * The map we want...
+ */
+ map = &vm->vm_map;
writing = uio->uio_rw == UIO_WRITE;
@@ -92,19 +100,18 @@ procfs_rwmem(p, uio)
* makes things easier. This way is trivial - right?
*/
do {
- vm_map_t map, tmap;
- vm_object_t object;
+ vm_map_t tmap;
vm_offset_t kva = 0;
vm_offset_t uva;
int page_offset; /* offset into page */
- vm_offset_t pageno; /* page number */
vm_map_entry_t out_entry;
vm_prot_t out_prot;
- vm_page_t m;
boolean_t wired, single_use;
vm_pindex_t pindex;
u_int len;
- int fix_prot;
+
+ fix_prot = 0;
+ object = NULL;
uva = (vm_offset_t) uio->uio_offset;
@@ -131,6 +138,7 @@ procfs_rwmem(p, uio)
/* sanity check */
if (!(p->p_flag & P_INMEM)) {
/* aiee! */
+ PRELE(p);
error = EFAULT;
break;
}
@@ -151,33 +159,34 @@ procfs_rwmem(p, uio)
continue;
}
-
- /*
- * The map we want...
- */
- map = &vm->vm_map;
-
/*
* Check the permissions for the area we're interested
* in.
*/
- fix_prot = 0;
- if (writing)
+ if (writing) {
fix_prot = !vm_map_check_protection(map, pageno,
pageno + PAGE_SIZE, VM_PROT_WRITE);
- if (fix_prot) {
- /*
- * If the page is not writable, we make it so.
- * XXX It is possible that a page may *not* be
- * read/executable, if a process changes that!
- * We will assume, for now, that a page is either
- * VM_PROT_ALL, or VM_PROT_READ|VM_PROT_EXECUTE.
- */
- error = vm_map_protect(map, pageno,
+ if (fix_prot) {
+ /*
+ * If the page is not writable, we make it so.
+ * XXX It is possible that a page may *not* be
+ * read/executable, if a process changes that!
+ * We will assume, for now, that a page is either
+ * VM_PROT_ALL, or VM_PROT_READ|VM_PROT_EXECUTE.
+ */
+ error = vm_map_protect(map, pageno,
pageno + PAGE_SIZE, VM_PROT_ALL, 0);
- if (error)
- break;
+ if (error) {
+ /*
+ * We don't have to undo something
+ * that didn't work, so we clear the
+ * flag.
+ */
+ fix_prot = 0;
+ break;
+ }
+ }
}
/*
@@ -188,59 +197,90 @@ procfs_rwmem(p, uio)
*/
tmap = map;
error = vm_map_lookup(&tmap, pageno,
- writing ? VM_PROT_WRITE : VM_PROT_READ,
- &out_entry, &object, &pindex, &out_prot,
- &wired, &single_use);
+ writing ? VM_PROT_WRITE : VM_PROT_READ,
+ &out_entry, &object, &pindex, &out_prot,
+ &wired, &single_use);
+
+ if (error) {
+ /*
+ * Make sure that there is no residue in 'object' from
+ * an error return on vm_map_lookup.
+ */
+ object = NULL;
+ break;
+ }
+
/*
* We're done with tmap now.
+ * But reference the object first, so that we won't loose
+ * it.
*/
- if (!error)
- vm_map_lookup_done(tmap, out_entry);
+ vm_object_reference(object);
+ vm_map_lookup_done(tmap, out_entry);
/*
* Fault the page in...
*/
- if (!error && writing && object->backing_object) {
+ if (writing && object->backing_object) {
+ vm_page_t m;
m = vm_page_lookup(object, pindex);
- if (m == 0)
+ if (m == 0) {
error = vm_fault(map, pageno,
- VM_PROT_WRITE, FALSE);
+ VM_PROT_WRITE, FALSE);
+ break;
+ }
}
/* Find space in kernel_map for the page we're interested in */
- if (!error)
- error = vm_map_find(kernel_map, object,
+ error = vm_map_find(kernel_map, object,
IDX_TO_OFF(pindex), &kva, PAGE_SIZE, 1,
VM_PROT_ALL, VM_PROT_ALL, 0);
+ if (error) {
+ break;
+ }
- if (!error) {
- /*
- * Neither vm_map_lookup() nor vm_map_find() appear
- * to add a reference count to the object, so we do
- * that here and now.
- */
- vm_object_reference(object);
-
- /*
- * Mark the page we just found as pageable.
- */
- error = vm_map_pageable(kernel_map, kva,
+ /*
+ * Mark the page we just found as pageable.
+ */
+ error = vm_map_pageable(kernel_map, kva,
kva + PAGE_SIZE, 0);
-
- /*
- * Now do the i/o move.
- */
- if (!error)
- error = uiomove((caddr_t)(kva + page_offset),
- len, uio);
-
+ if (error) {
vm_map_remove(kernel_map, kva, kva + PAGE_SIZE);
+ object = NULL;
+ break;
}
- if (fix_prot)
+
+ /*
+ * Now do the i/o move.
+ */
+ error = uiomove((caddr_t)(kva + page_offset),
+ len, uio);
+
+ /*
+ * vm_map_remove gets rid of the object reference, so
+ * we need to get rid of our 'object' pointer if there
+ * is subsequently an error.
+ */
+ vm_map_remove(kernel_map, kva, kva + PAGE_SIZE);
+ object = NULL;
+
+ /*
+ * Undo the protection 'damage'.
+ */
+ if (fix_prot) {
vm_map_protect(map, pageno, pageno + PAGE_SIZE,
- VM_PROT_READ|VM_PROT_EXECUTE, 0);
+ VM_PROT_READ|VM_PROT_EXECUTE, 0);
+ fix_prot = 0;
+ }
} while (error == 0 && uio->uio_resid > 0);
+ if (object)
+ vm_object_deallocate(object);
+
+ if (fix_prot)
+ vm_map_protect(map, pageno, pageno + PAGE_SIZE,
+ VM_PROT_READ|VM_PROT_EXECUTE, 0);
+
vmspace_free(vm);
return (error);
}
OpenPOWER on IntegriCloud