xfs_defer_drain_free(&xg->xg_intents_drain);
 #ifdef __KERNEL__
-       kfree(xg->xg_busy_extents);
+       if (xfs_group_has_extent_busy(xg->xg_mount, xg->xg_type))
+               kfree(xg->xg_busy_extents);
 #endif
 
        if (uninit)
        xg->xg_type = type;
 
 #ifdef __KERNEL__
-       xg->xg_busy_extents = xfs_extent_busy_alloc();
-       if (!xg->xg_busy_extents)
-               return -ENOMEM;
+       if (xfs_group_has_extent_busy(mp, type)) {
+               xg->xg_busy_extents = xfs_extent_busy_alloc();
+               if (!xg->xg_busy_extents)
+                       return -ENOMEM;
+       }
        spin_lock_init(&xg->xg_state_lock);
        xfs_hooks_init(&xg->xg_rmap_update_hooks);
 #endif
 out_drain:
        xfs_defer_drain_free(&xg->xg_intents_drain);
 #ifdef __KERNEL__
-       kfree(xg->xg_busy_extents);
+       if (xfs_group_has_extent_busy(xg->xg_mount, xg->xg_type))
+               kfree(xg->xg_busy_extents);
 #endif
        return error;
 }
 
        list_sort(NULL, list, xfs_extent_busy_ag_cmp);
 }
 
+/*
+ * Zoned RTGs don't need to track busy extents, as the actual block freeing only
+ * happens by a zone reset, which forces out all transactions that touched the
+ * to be reset zone first.
+ */
+#define xfs_group_has_extent_busy(mp, type) \
+       ((type) == XG_TYPE_AG || !xfs_has_zoned((mp)))
+
 #endif /* __XFS_EXTENT_BUSY_H__ */