summaryrefslogtreecommitdiff
path: root/sys/kern
diff options
context:
space:
mode:
authorBob Beck <beck@cvs.openbsd.org>2012-11-17 23:08:23 +0000
committerBob Beck <beck@cvs.openbsd.org>2012-11-17 23:08:23 +0000
commitd61772fd2a28c4d68abf20354d6a30324357f557 (patch)
tree821e69a0d17cde66bcfa6018f8e5d8a090937825 /sys/kern
parent49175dac6e732dd877e6aa295f301ab21e422c75 (diff)
Don't map a buffer (and potentially sleep) when invalidating it in vinvalbuf.
This fixes a problem where we could sleep for kva and then our pointers would not be valid on the next pass through the loop. We do this by adding buf_acquire_nomap() - which can be used to busy up the buffer without changing its mapped or unmapped state. We do not need to have the buffer mapped to invalidate it, so it is sufficient to acquire it for that. In the case where we write the buffer, we do map the buffer, and potentially sleep.
Diffstat (limited to 'sys/kern')
-rw-r--r--sys/kern/vfs_biomem.c19
-rw-r--r--sys/kern/vfs_subr.c5
2 files changed, 21 insertions, 3 deletions
diff --git a/sys/kern/vfs_biomem.c b/sys/kern/vfs_biomem.c
index 2a1e342c62b..c0149d83624 100644
--- a/sys/kern/vfs_biomem.c
+++ b/sys/kern/vfs_biomem.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: vfs_biomem.c,v 1.18 2011/09/19 14:48:04 beck Exp $ */
+/* $OpenBSD: vfs_biomem.c,v 1.19 2012/11/17 23:08:22 beck Exp $ */
/*
* Copyright (c) 2007 Artur Grabowski <art@openbsd.org>
*
@@ -105,6 +105,23 @@ buf_acquire_unmapped(struct buf *bp)
splx(s);
}
+/*
+ * Acquire a buf but do not map it. Preserve any mapping it did have.
+ */
+void
+buf_acquire_nomap(struct buf *bp)
+{
+ splassert(IPL_BIO);
+ SET(bp->b_flags, B_BUSY);
+ if (bp->b_data == NULL)
+ SET(bp->b_flags, B_NOTMAPPED);
+ else {
+ TAILQ_REMOVE(&buf_valist, bp, b_valist);
+ bcstats.kvaslots_avail--;
+ bcstats.busymapped++;
+ }
+}
+
void
buf_map(struct buf *bp)
{
diff --git a/sys/kern/vfs_subr.c b/sys/kern/vfs_subr.c
index 9e1a850b931..3a6ff0773c4 100644
--- a/sys/kern/vfs_subr.c
+++ b/sys/kern/vfs_subr.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: vfs_subr.c,v 1.199 2012/10/01 00:08:43 guenther Exp $ */
+/* $OpenBSD: vfs_subr.c,v 1.200 2012/11/17 23:08:22 beck Exp $ */
/* $NetBSD: vfs_subr.c,v 1.53 1996/04/22 01:39:13 christos Exp $ */
/*
@@ -1863,17 +1863,18 @@ loop:
break;
}
bremfree(bp);
- buf_acquire(bp);
/*
* XXX Since there are no node locks for NFS, I believe
* there is a slight chance that a delayed write will
* occur while sleeping just above, so check for it.
*/
if ((bp->b_flags & B_DELWRI) && (flags & V_SAVE)) {
+ buf_acquire(bp);
splx(s);
(void) VOP_BWRITE(bp);
goto loop;
}
+ buf_acquire_nomap(bp);
bp->b_flags |= B_INVAL;
brelse(bp);
}