#include "xfs_mount.h"
 #include "xfs_trace.h"
 #include "xfs_log.h"
+#include "xfs_trans.h"
+#include "xfs_buf_item.h"
 #include "xfs_errortag.h"
 #include "xfs_error.h"
 
                bp->b_flags |= XBF_DONE;
        }
 
-       if (bp->b_iodone)
+       if (read)
+               goto out_finish;
+
+       if (bp->b_flags & _XBF_INODES) {
+               xfs_buf_inode_iodone(bp);
+               return;
+       }
+
+       if (bp->b_iodone) {
                (*(bp->b_iodone))(bp);
-       else if (bp->b_flags & XBF_ASYNC)
-               xfs_buf_relse(bp);
-       else
-               complete(&bp->b_iowait);
+               return;
+       }
+
+out_finish:
+       xfs_buf_ioend_finish(bp);
 }
 
 static void
 
 #define XBF_STALE       (1 << 6) /* buffer has been staled, do not find it */
 #define XBF_WRITE_FAIL  (1 << 7) /* async writes have failed on this buffer */
 
-/* flags used only as arguments to access routines */
-#define XBF_TRYLOCK     (1 << 16)/* lock requested, but do not wait */
-#define XBF_UNMAPPED    (1 << 17)/* do not map the buffer */
+/* buffer type flags for write callbacks */
+#define _XBF_INODES     (1 << 16)/* inode buffer */
 
 /* flags used only internally */
 #define _XBF_PAGES      (1 << 20)/* backed by refcounted pages */
 #define _XBF_KMEM       (1 << 21)/* backed by heap memory */
 #define _XBF_DELWRI_Q   (1 << 22)/* buffer on a delwri queue */
 
+/* flags used only as arguments to access routines */
+#define XBF_TRYLOCK     (1 << 30)/* lock requested, but do not wait */
+#define XBF_UNMAPPED    (1 << 31)/* do not map the buffer */
+
 typedef unsigned int xfs_buf_flags_t;
 
 #define XFS_BUF_FLAGS \
        { XBF_DONE,             "DONE" }, \
        { XBF_STALE,            "STALE" }, \
        { XBF_WRITE_FAIL,       "WRITE_FAIL" }, \
-       { XBF_TRYLOCK,          "TRYLOCK" },    /* should never be set */\
-       { XBF_UNMAPPED,         "UNMAPPED" },   /* ditto */\
+       { _XBF_INODES,          "INODES" }, \
        { _XBF_PAGES,           "PAGES" }, \
        { _XBF_KMEM,            "KMEM" }, \
-       { _XBF_DELWRI_Q,        "DELWRI_Q" }
-
+       { _XBF_DELWRI_Q,        "DELWRI_Q" }, \
+       /* The following interface flags should never be set */ \
+       { XBF_TRYLOCK,          "TRYLOCK" }, \
+       { XBF_UNMAPPED,         "UNMAPPED" }
 
 /*
  * Internal state flags.
 #define xfs_buf_islocked(bp) \
        ((bp)->b_sema.count <= 0)
 
+static inline void xfs_buf_relse(xfs_buf_t *bp)
+{
+       xfs_buf_unlock(bp);
+       xfs_buf_rele(bp);
+}
+
 /* Buffer Read and Write Routines */
 extern int xfs_bwrite(struct xfs_buf *bp);
 extern void xfs_buf_ioend(struct xfs_buf *bp);
+static inline void xfs_buf_ioend_finish(struct xfs_buf *bp)
+{
+       if (bp->b_flags & XBF_ASYNC)
+               xfs_buf_relse(bp);
+       else
+               complete(&bp->b_iowait);
+}
+
 extern void __xfs_buf_ioerror(struct xfs_buf *bp, int error,
                xfs_failaddr_t failaddr);
 #define xfs_buf_ioerror(bp, err) __xfs_buf_ioerror((bp), (err), __this_address)
        return atomic_read(&bp->b_pin_count);
 }
 
-static inline void xfs_buf_relse(xfs_buf_t *bp)
-{
-       xfs_buf_unlock(bp);
-       xfs_buf_rele(bp);
-}
-
 static inline int
 xfs_buf_verify_cksum(struct xfs_buf *bp, unsigned long cksum_offset)
 {
 
        return false;
 }
 
-/*
- * This is the iodone() function for buffers which have had callbacks attached
- * to them by xfs_buf_attach_iodone(). We need to iterate the items on the
- * callback list, mark the buffer as having no more callbacks and then push the
- * buffer through IO completion processing.
- */
-void
-xfs_buf_iodone_callbacks(
+static void
+xfs_buf_run_callbacks(
        struct xfs_buf          *bp)
 {
+
        /*
-        * If there is an error, process it. Some errors require us
-        * to run callbacks after failure processing is done so we
-        * detect that and take appropriate action.
+        * If there is an error, process it. Some errors require us to run
+        * callbacks after failure processing is done so we detect that and take
+        * appropriate action.
         */
        if (bp->b_error && xfs_buf_iodone_callback_error(bp))
                return;
        bp->b_log_item = NULL;
        list_del_init(&bp->b_li_list);
        bp->b_iodone = NULL;
+}
+
+/*
+ * This is the iodone() function for buffers which have had callbacks attached
+ * to them by xfs_buf_attach_iodone(). We need to iterate the items on the
+ * callback list, mark the buffer as having no more callbacks and then push the
+ * buffer through IO completion processing.
+ */
+void
+xfs_buf_iodone_callbacks(
+       struct xfs_buf          *bp)
+{
+       xfs_buf_run_callbacks(bp);
        xfs_buf_ioend(bp);
 }
 
+/*
+ * Inode buffer iodone callback function.
+ */
+void
+xfs_buf_inode_iodone(
+       struct xfs_buf          *bp)
+{
+       xfs_buf_run_callbacks(bp);
+       xfs_buf_ioend_finish(bp);
+}
+
+
 /*
  * This is the iodone() function for buffers which have been
  * logged.  It is called when they are eventually flushed out.
 
                              struct xfs_log_item *);
 void   xfs_buf_iodone_callbacks(struct xfs_buf *);
 void   xfs_buf_iodone(struct xfs_buf *, struct xfs_log_item *);
+void   xfs_buf_inode_iodone(struct xfs_buf *);
 bool   xfs_buf_log_check_iovec(struct xfs_log_iovec *iovec);
 
 extern kmem_zone_t     *xfs_buf_item_zone;
 
         * completion on the buffer to remove the inode from the AIL and release
         * the flush lock.
         */
+       bp->b_flags |= _XBF_INODES;
        xfs_buf_attach_iodone(bp, xfs_iflush_done, &iip->ili_item);
 
        /* generate the checksum. */
        xfs_dinode_calc_crc(mp, dip);
 
        ASSERT(!list_empty(&bp->b_li_list));
-       ASSERT(bp->b_iodone != NULL);
        return error;
 }
 
 
        ASSERT(atomic_read(&bip->bli_refcount) > 0);
 
        bip->bli_flags |= XFS_BLI_INODE_BUF;
+       bp->b_flags |= _XBF_INODES;
        xfs_trans_buf_set_type(tp, bp, XFS_BLFT_DINO_BUF);
 }
 
 
        bip->bli_flags |= XFS_BLI_STALE_INODE;
        bip->bli_item.li_cb = xfs_buf_iodone;
+       bp->b_flags |= _XBF_INODES;
        xfs_trans_buf_set_type(tp, bp, XFS_BLFT_DINO_BUF);
 }
 
        ASSERT(atomic_read(&bip->bli_refcount) > 0);
 
        bip->bli_flags |= XFS_BLI_INODE_ALLOC_BUF;
+       bp->b_flags |= _XBF_INODES;
        xfs_trans_buf_set_type(tp, bp, XFS_BLFT_DINO_BUF);
 }