#include "xfs_shared.h"
 #include "xfs_format.h"
 #include "xfs_log_format.h"
+#include "xfs_da_format.h"
 #include "xfs_trans_resv.h"
 #include "xfs_bit.h"
 #include "xfs_sb.h"
 #include "xfs_refcount.h"
 #include "xfs_bmap.h"
 #include "xfs_inode.h"
+#include "xfs_da_btree.h"
+#include "xfs_attr.h"
 
 /* Dummy defer item ops, since we don't do logging. */
 
        .finish_item    = xfs_bmap_update_finish_item,
        .cancel_item    = xfs_bmap_update_cancel_item,
 };
+
+/* Get an ATTRI. */
+static struct xfs_log_item *
+xfs_attr_create_intent(
+       struct xfs_trans        *tp,
+       struct list_head        *items,
+       unsigned int            count,
+       bool                    sort)
+{
+       return NULL;
+}
+
+/* Abort all pending ATTRs. */
+static void
+xfs_attr_abort_intent(
+       struct xfs_log_item     *intent)
+{
+}
+
+/* Get an ATTRD so we can process all the attrs. */
+static struct xfs_log_item *
+xfs_attr_create_done(
+       struct xfs_trans        *tp,
+       struct xfs_log_item     *intent,
+       unsigned int            count)
+{
+       return NULL;
+}
+
+/* Process an attr. */
+static int
+xfs_attr_finish_item(
+       struct xfs_trans        *tp,
+       struct xfs_log_item     *done,
+       struct list_head        *item,
+       struct xfs_btree_cur    **state)
+{
+       struct xfs_attr_item    *attr;
+       int                     error;
+       struct xfs_delattr_context *dac;
+       struct xfs_da_args      *args;
+       unsigned int            op;
+
+       attr = container_of(item, struct xfs_attr_item, xattri_list);
+       dac = &attr->xattri_dac;
+       args = dac->da_args;
+       op = attr->xattri_op_flags & XFS_ATTR_OP_FLAGS_TYPE_MASK;
+
+       /*
+        * Always reset trans after EAGAIN cycle
+        * since the transaction is new
+        */
+       args->trans = tp;
+
+       switch (op) {
+       case XFS_ATTR_OP_FLAGS_SET:
+               error = xfs_attr_set_iter(dac, &dac->leaf_bp);
+               break;
+       case XFS_ATTR_OP_FLAGS_REMOVE:
+               ASSERT(XFS_IFORK_Q(args->dp));
+               error = xfs_attr_remove_iter(dac);
+               break;
+       default:
+               error = -EFSCORRUPTED;
+               break;
+       }
+
+       if (error != -EAGAIN)
+               kmem_free(attr);
+
+       return error;
+}
+
+/* Cancel an attr */
+static void
+xfs_attr_cancel_item(
+       struct list_head        *item)
+{
+       struct xfs_attr_item    *attr;
+
+       attr = container_of(item, struct xfs_attr_item, xattri_list);
+       kmem_free(attr);
+}
+
+const struct xfs_defer_op_type xfs_attr_defer_type = {
+       .max_items      = 1,
+       .create_intent  = xfs_attr_create_intent,
+       .abort_intent   = xfs_attr_abort_intent,
+       .create_done    = xfs_attr_create_done,
+       .finish_item    = xfs_attr_finish_item,
+       .cancel_item    = xfs_attr_cancel_item,
+};
 
        return (sbp->sb_features_incompat & feature) != 0;
 }
 
-#define XFS_SB_FEAT_INCOMPAT_LOG_ALL 0
+#define XFS_SB_FEAT_INCOMPAT_LOG_XATTRS   (1 << 0)     /* Delayed Attributes */
+#define XFS_SB_FEAT_INCOMPAT_LOG_ALL \
+       (XFS_SB_FEAT_INCOMPAT_LOG_XATTRS)
 #define XFS_SB_FEAT_INCOMPAT_LOG_UNKNOWN       ~XFS_SB_FEAT_INCOMPAT_LOG_ALL
 static inline bool
 xfs_sb_has_incompat_log_feature(
        sbp->sb_features_log_incompat |= features;
 }
 
+static inline bool xfs_sb_version_haslogxattrs(struct xfs_sb *sbp)
+{
+       return xfs_sb_is_v5(sbp) && (sbp->sb_features_log_incompat &
+                XFS_SB_FEAT_INCOMPAT_LOG_XATTRS);
+}
 
 static inline bool
 xfs_is_quota_inode(struct xfs_sb *sbp, xfs_ino_t ino)