/*
  * Test whether it is appropriate to check an inode for and free post EOF
- * blocks. The 'force' parameter determines whether we should also consider
- * regular files that are marked preallocated or append-only.
+ * blocks.
  */
 bool
 xfs_can_free_eofblocks(
-       struct xfs_inode        *ip,
-       bool                    force)
+       struct xfs_inode        *ip)
 {
        struct xfs_bmbt_irec    imap;
        struct xfs_mount        *mp = ip->i_mount;
                return false;
 
        /*
-        * Do not free real preallocated or append-only files unless the file
-        * has delalloc blocks and we are forced to remove them.
+        * Only free real extents for inodes with persistent preallocations or
+        * the append-only flag.
         */
        if (ip->i_diflags & (XFS_DIFLAG_PREALLOC | XFS_DIFLAG_APPEND))
-               if (!force || ip->i_delayed_blks == 0)
+               if (ip->i_delayed_blks == 0)
                        return false;
 
        /*
        /* Wait on dio to ensure i_size has settled. */
        inode_dio_wait(VFS_I(ip));
 
+       /*
+        * For preallocated files only free delayed allocations.
+        *
+        * Note that this means we also leave speculative preallocations in
+        * place for preallocated files.
+        */
+       if (ip->i_diflags & (XFS_DIFLAG_PREALLOC | XFS_DIFLAG_APPEND)) {
+               if (ip->i_delayed_blks) {
+                       xfs_bmap_punch_delalloc_range(ip,
+                               round_up(XFS_ISIZE(ip), mp->m_sb.sb_blocksize),
+                               LLONG_MAX);
+               }
+               xfs_inode_clear_eofblocks_tag(ip);
+               return 0;
+       }
+
        error = xfs_trans_alloc(mp, &M_RES(mp)->tr_itruncate, 0, 0, 0, &tp);
        if (error) {
                ASSERT(xfs_is_shutdown(mp));
         * Trim eofblocks to avoid shifting uninitialized post-eof preallocation
         * into the accessible region of the file.
         */
-       if (xfs_can_free_eofblocks(ip, true)) {
+       if (xfs_can_free_eofblocks(ip)) {
                error = xfs_free_eofblocks(ip);
                if (error)
                        return error;
 
                                xfs_off_t len);
 
 /* EOF block manipulation functions */
-bool   xfs_can_free_eofblocks(struct xfs_inode *ip, bool force);
+bool   xfs_can_free_eofblocks(struct xfs_inode *ip);
 int    xfs_free_eofblocks(struct xfs_inode *ip);
 
 int    xfs_swap_extents(struct xfs_inode *ip, struct xfs_inode *tip,
 
        }
        *lockflags |= XFS_IOLOCK_EXCL;
 
-       if (xfs_can_free_eofblocks(ip, false))
+       if (xfs_can_free_eofblocks(ip))
                return xfs_free_eofblocks(ip);
 
        /* inode could be preallocated or append-only */
 
        if (!xfs_ilock_nowait(ip, XFS_IOLOCK_EXCL))
                return 0;
 
-       if (xfs_can_free_eofblocks(ip, false)) {
+       if (xfs_can_free_eofblocks(ip)) {
                /*
                 * Check if the inode is being opened, written and closed
                 * frequently and we have delayed allocation blocks outstanding
 
        /*
         * This file isn't being freed, so check if there are post-eof blocks
-        * to free.  @force is true because we are evicting an inode from the
-        * cache.  Post-eof blocks must be freed, lest we end up with broken
-        * free space accounting.
+        * to free.
         *
         * Note: don't bother with iolock here since lockdep complains about
         * acquiring it in reclaim context. We have the only reference to the
         * inode at this point anyways.
         */
-       return xfs_can_free_eofblocks(ip, true);
+       return xfs_can_free_eofblocks(ip);
 }
 
 /*
 
        if (VFS_I(ip)->i_nlink != 0) {
                /*
-                * force is true because we are evicting an inode from the
-                * cache. Post-eof blocks must be freed, lest we end up with
-                * broken free space accounting.
-                *
                 * Note: don't bother with iolock here since lockdep complains
                 * about acquiring it in reclaim context. We have the only
                 * reference to the inode at this point anyways.
                 */
-               if (xfs_can_free_eofblocks(ip, true))
+               if (xfs_can_free_eofblocks(ip))
                        error = xfs_free_eofblocks(ip);
 
                goto out;