M_RES(mp)->tr_attrsetrt.tr_logres * nblks;
 }
 
+/*
+ * Compute an alternate set of log reservation sizes for use exclusively with
+ * minimum log size calculations.
+ */
+static void
+xfs_log_calc_trans_resv_for_minlogblocks(
+       struct xfs_mount        *mp,
+       struct xfs_trans_resv   *resv)
+{
+       unsigned int            rmap_maxlevels = mp->m_rmap_maxlevels;
+
+       /*
+        * In the early days of rmap+reflink, we always set the rmap maxlevels
+        * to 9 even if the AG was small enough that it would never grow to
+        * that height.  Transaction reservation sizes influence the minimum
+        * log size calculation, which influences the size of the log that mkfs
+        * creates.  Use the old value here to ensure that newly formatted
+        * small filesystems will mount on older kernels.
+        */
+       if (xfs_has_rmapbt(mp) && xfs_has_reflink(mp))
+               mp->m_rmap_maxlevels = XFS_OLD_REFLINK_RMAP_MAXLEVELS;
+
+       xfs_trans_resv_calc(mp, resv);
+
+       if (xfs_has_reflink(mp)) {
+               /*
+                * In the early days of reflink, typical log operation counts
+                * were greatly overestimated.
+                */
+               resv->tr_write.tr_logcount = XFS_WRITE_LOG_COUNT_REFLINK;
+               resv->tr_itruncate.tr_logcount =
+                               XFS_ITRUNCATE_LOG_COUNT_REFLINK;
+               resv->tr_qm_dqalloc.tr_logcount = XFS_WRITE_LOG_COUNT_REFLINK;
+       } else if (xfs_has_rmapbt(mp)) {
+               /*
+                * In the early days of non-reflink rmap, the impact of rmapbt
+                * updates on log counts were not taken into account at all.
+                */
+               resv->tr_write.tr_logcount = XFS_WRITE_LOG_COUNT;
+               resv->tr_itruncate.tr_logcount = XFS_ITRUNCATE_LOG_COUNT;
+               resv->tr_qm_dqalloc.tr_logcount = XFS_WRITE_LOG_COUNT;
+       }
+
+       /* Put everything back the way it was.  This goes at the end. */
+       mp->m_rmap_maxlevels = rmap_maxlevels;
+}
+
 /*
  * Iterate over the log space reservation table to figure out and return
  * the maximum one in terms of the pre-calculated values which were done
        struct xfs_mount        *mp,
        struct xfs_trans_res    *max_resp)
 {
-       struct xfs_trans_resv   resv;
+       struct xfs_trans_resv   resv = {};
        struct xfs_trans_res    *resp;
        struct xfs_trans_res    *end_resp;
        unsigned int            i;
 
        attr_space = xfs_log_calc_max_attrsetm_res(mp);
 
-       memcpy(&resv, M_RES(mp), sizeof(struct xfs_trans_resv));
+       xfs_log_calc_trans_resv_for_minlogblocks(mp, &resv);
 
        resp = (struct xfs_trans_res *)&resv;
        end_resp = (struct xfs_trans_res *)(&resv + 1);
 
        struct xfs_mount        *mp,
        struct xfs_trans_resv   *resp)
 {
-       unsigned int            rmap_maxlevels = mp->m_rmap_maxlevels;
-
-       /*
-        * In the early days of rmap+reflink, we always set the rmap maxlevels
-        * to 9 even if the AG was small enough that it would never grow to
-        * that height.  Transaction reservation sizes influence the minimum
-        * log size calculation, which influences the size of the log that mkfs
-        * creates.  Use the old value here to ensure that newly formatted
-        * small filesystems will mount on older kernels.
-        */
-       if (xfs_has_rmapbt(mp) && xfs_has_reflink(mp))
-               mp->m_rmap_maxlevels = XFS_OLD_REFLINK_RMAP_MAXLEVELS;
+       int                     logcount_adj = 0;
 
        /*
         * The following transactions are logged in physical format and
         * require a permanent reservation on space.
         */
        resp->tr_write.tr_logres = xfs_calc_write_reservation(mp);
-       if (xfs_has_reflink(mp))
-               resp->tr_write.tr_logcount = XFS_WRITE_LOG_COUNT_REFLINK;
-       else
-               resp->tr_write.tr_logcount = XFS_WRITE_LOG_COUNT;
+       resp->tr_write.tr_logcount = XFS_WRITE_LOG_COUNT;
        resp->tr_write.tr_logflags |= XFS_TRANS_PERM_LOG_RES;
 
        resp->tr_itruncate.tr_logres = xfs_calc_itruncate_reservation(mp);
-       if (xfs_has_reflink(mp))
-               resp->tr_itruncate.tr_logcount =
-                               XFS_ITRUNCATE_LOG_COUNT_REFLINK;
-       else
-               resp->tr_itruncate.tr_logcount = XFS_ITRUNCATE_LOG_COUNT;
+       resp->tr_itruncate.tr_logcount = XFS_ITRUNCATE_LOG_COUNT;
        resp->tr_itruncate.tr_logflags |= XFS_TRANS_PERM_LOG_RES;
 
        resp->tr_rename.tr_logres = xfs_calc_rename_reservation(mp);
        resp->tr_growrtalloc.tr_logflags |= XFS_TRANS_PERM_LOG_RES;
 
        resp->tr_qm_dqalloc.tr_logres = xfs_calc_qm_dqalloc_reservation(mp);
-       if (xfs_has_reflink(mp))
-               resp->tr_qm_dqalloc.tr_logcount = XFS_WRITE_LOG_COUNT_REFLINK;
-       else
-               resp->tr_qm_dqalloc.tr_logcount = XFS_WRITE_LOG_COUNT;
+       resp->tr_qm_dqalloc.tr_logcount = XFS_WRITE_LOG_COUNT;
        resp->tr_qm_dqalloc.tr_logflags |= XFS_TRANS_PERM_LOG_RES;
 
        /*
        resp->tr_growrtzero.tr_logres = xfs_calc_growrtzero_reservation(mp);
        resp->tr_growrtfree.tr_logres = xfs_calc_growrtfree_reservation(mp);
 
-       /* Put everything back the way it was.  This goes at the end. */
-       mp->m_rmap_maxlevels = rmap_maxlevels;
+       /*
+        * Add one logcount for BUI items that appear with rmap or reflink,
+        * one logcount for refcount intent items, and one logcount for rmap
+        * intent items.
+        */
+       if (xfs_has_reflink(mp) || xfs_has_rmapbt(mp))
+               logcount_adj++;
+       if (xfs_has_reflink(mp))
+               logcount_adj++;
+       if (xfs_has_rmapbt(mp))
+               logcount_adj++;
+
+       resp->tr_itruncate.tr_logcount += logcount_adj;
+       resp->tr_write.tr_logcount += logcount_adj;
+       resp->tr_qm_dqalloc.tr_logcount += logcount_adj;
 }
 
 #define        XFS_DEFAULT_LOG_COUNT           1
 #define        XFS_DEFAULT_PERM_LOG_COUNT      2
 #define        XFS_ITRUNCATE_LOG_COUNT         2
-#define        XFS_ITRUNCATE_LOG_COUNT_REFLINK 8
 #define XFS_INACTIVE_LOG_COUNT         2
 #define        XFS_CREATE_LOG_COUNT            2
 #define        XFS_CREATE_TMPFILE_LOG_COUNT    2
 #define        XFS_LINK_LOG_COUNT              2
 #define        XFS_RENAME_LOG_COUNT            2
 #define        XFS_WRITE_LOG_COUNT             2
-#define        XFS_WRITE_LOG_COUNT_REFLINK     8
 #define        XFS_ADDAFORK_LOG_COUNT          2
 #define        XFS_ATTRINVAL_LOG_COUNT         1
 #define        XFS_ATTRSET_LOG_COUNT           3
 #define        XFS_ATTRRM_LOG_COUNT            3
 
+/*
+ * Original log operation counts were overestimated in the early days of
+ * reflink.  These are retained here purely for minimum log size calculations
+ * and must not be used for runtime reservations.
+ */
+#define        XFS_ITRUNCATE_LOG_COUNT_REFLINK 8
+#define        XFS_WRITE_LOG_COUNT_REFLINK     8
+
 void xfs_trans_resv_calc(struct xfs_mount *mp, struct xfs_trans_resv *resp);
 uint xfs_allocfree_log_count(struct xfs_mount *mp, uint num_ops);