Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/dev As discussed in
details:   https://anonhg.NetBSD.org/src/rev/86066e10164b
branches:  trunk
changeset: 806026:86066e10164b
user:      bouyer <bouyer%NetBSD.org@localhost>
date:      Wed Jan 28 15:08:12 2015 +0000
description:
As discussed in
http://mail-index.netbsd.org/tech-kern/2015/01/24/msg018339.html
don't bump v_numoutput if we need to vn_lock() the vnode before queuing
the corresponding I/O, because this may deadlock with genfs_do_putpages()
when called with the vnode locked (as can happen with fsync(2)).
Instead bump is just before the last VOP_STRATEGY(), or before calling
nestiobuf_done().
Thanks to Taylor R Campbell for review.
diffstat:
 sys/dev/vnd.c |  40 +++++++++++++++++++++++++++++-----------
 1 files changed, 29 insertions(+), 11 deletions(-)
diffs (82 lines):
diff -r 6f10be2aa4f9 -r 86066e10164b sys/dev/vnd.c
--- a/sys/dev/vnd.c     Wed Jan 28 14:00:58 2015 +0000
+++ b/sys/dev/vnd.c     Wed Jan 28 15:08:12 2015 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: vnd.c,v 1.239 2015/01/02 19:42:06 christos Exp $       */
+/*     $NetBSD: vnd.c,v 1.240 2015/01/28 15:08:12 bouyer Exp $ */
 
 /*-
  * Copyright (c) 1996, 1997, 1998, 2008 The NetBSD Foundation, Inc.
@@ -91,7 +91,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: vnd.c,v 1.239 2015/01/02 19:42:06 christos Exp $");
+__KERNEL_RCSID(0, "$NetBSD: vnd.c,v 1.240 2015/01/28 15:08:12 bouyer Exp $");
 
 #if defined(_KERNEL_OPT)
 #include "opt_vnd.h"
@@ -795,15 +795,10 @@
        size_t resid, sz;
        off_t bn, offset;
        struct vnode *vp;
+       struct buf *nbp = NULL;
 
        flags = obp->b_flags;
 
-       if (!(flags & B_READ)) {
-               vp = bp->b_vp;
-               mutex_enter(vp->v_interlock);
-               vp->v_numoutput++;
-               mutex_exit(vp->v_interlock);
-       }
 
        /* convert to a byte offset within the file. */
        bn = obp->b_rawblkno * vnd->sc_dkdev.dk_label->d_secsize;
@@ -820,9 +815,8 @@
         */
        error = 0;
        bp->b_resid = bp->b_bcount;
-       for (offset = 0, resid = bp->b_resid; resid;
+       for (offset = 0, resid = bp->b_resid; /* true */;
            resid -= sz, offset += sz) {
-               struct buf *nbp;
                daddr_t nbn;
                int off, nra;
 
@@ -875,10 +869,34 @@
                            nbp->vb_buf.b_flags, nbp->vb_buf.b_data,
                            nbp->vb_buf.b_bcount);
 #endif
+               if (resid == sz) {
+                       break;
+               }
                VOP_STRATEGY(vp, nbp);
                bn += sz;
        }
-       nestiobuf_done(bp, skipped, error);
+       if (!(flags & B_READ)) {
+               struct vnode *w_vp;
+               /*
+                * this is the last nested buf, account for
+                * the parent buf write too.
+                * This has to be done last, so that
+                * fsync won't wait for this write which
+                * has to chance to complete before all nested bufs
+                * have been queued. But it has to be done
+                * before the last VOP_STRATEGY() 
+                * or the call to nestiobuf_done().
+                */
+               w_vp = bp->b_vp;
+               mutex_enter(w_vp->v_interlock);
+               w_vp->v_numoutput++;
+               mutex_exit(w_vp->v_interlock);
+       }
+       KASSERT(skipped != 0 || nbp != NULL);
+       if (skipped) 
+               nestiobuf_done(bp, skipped, error);
+       else 
+               VOP_STRATEGY(vp, nbp);
 }
 
 static void
Home |
Main Index |
Thread Index |
Old Index