{
        int ret, err;
        handle_t *handle;
+       bool freeze_protected = false;
+
+       /*
+        * Trying to sb_start_intwrite() in a running transaction
+        * can result in a deadlock. Further, running transactions
+        * are already protected from freezing.
+        */
+       if (!ext4_journal_current_handle()) {
+               sb_start_intwrite(dquot->dq_sb);
+               freeze_protected = true;
+       }
 
        handle = ext4_journal_start(dquot_to_inode(dquot), EXT4_HT_QUOTA,
                                    EXT4_QUOTA_DEL_BLOCKS(dquot->dq_sb));
        if (IS_ERR(handle)) {
                /* Release dquot anyway to avoid endless cycle in dqput() */
                dquot_release(dquot);
+               if (freeze_protected)
+                       sb_end_intwrite(dquot->dq_sb);
                return PTR_ERR(handle);
        }
        ret = dquot_release(dquot);
        err = ext4_journal_stop(handle);
        if (!ret)
                ret = err;
+
+       if (freeze_protected)
+               sb_end_intwrite(dquot->dq_sb);
+
        return ret;
 }