]> git.eshelyaron.com Git - emacs.git/commitdiff
Fix aborts and text corruption in 'replace-buffer-contents'
authorEli Zaretskii <eliz@gnu.org>
Thu, 13 Mar 2025 20:19:14 +0000 (22:19 +0200)
committerEshel Yaron <me@eshelyaron.com>
Sat, 15 Mar 2025 15:28:41 +0000 (16:28 +0100)
* src/insdel.c (replace_range): Fix a thinko.  Fix commentary.
(Bug#76997)

(cherry picked from commit 89441e12e2a25d43d1d5567ac356a7ecb8193063)

src/insdel.c

index 3707342d2c4c7cc0d02e8bbbd274e0700261df88..9b770725971e6a60d2bdb76dec5923e68f94782e 100644 (file)
@@ -1409,7 +1409,11 @@ adjust_after_insert (ptrdiff_t from, ptrdiff_t from_byte,
   adjust_after_replace (from, from_byte, Qnil, newlen, len_byte);
 }
 \f
-/* Replace the text from character positions FROM to TO with NEW,
+/* Replace the text from character positions FROM to TO with NEW.
+   NEW could either be a string, the replacement text, or a vector
+   [BUFFER BEG END], where BUFFER is the buffer with the replacement
+   text and BEG and END are buffer positions in BUFFER that give the
+   replacement text beginning and end.
    If PREPARE, call prepare_to_modify_buffer.
    If INHERIT, the newly inserted text should inherit text properties
    from the surrounding non-deleted text.
@@ -1419,9 +1423,7 @@ adjust_after_insert (ptrdiff_t from, ptrdiff_t from_byte,
 /* Note that this does not yet handle markers quite right.
    Also it needs to record a single undo-entry that does a replacement
    rather than a separate delete and insert.
-   That way, undo will also handle markers properly.
-
-   But if MARKERS is 0, don't relocate markers.  */
+   That way, undo will also handle markers properly.  */
 
 void
 replace_range (ptrdiff_t from, ptrdiff_t to, Lisp_Object new,
@@ -1504,9 +1506,19 @@ replace_range (ptrdiff_t from, ptrdiff_t to, Lisp_Object new,
          insend_bytes = insend;
        }
       insbytes = insend_bytes - insbeg_bytes;
+      /* Move gap out of the replacement text, to arrange for
+         replacement text to be contiguous in the source buffer, so that
+         we could copy it in one go.  */
       if (insbuf->text->gpt_byte > insbeg_bytes
          && insbuf->text->gpt_byte < insend_bytes)
-       move_gap_both (insbeg, insbeg_bytes);
+       {
+         struct buffer *old = current_buffer;
+         if (insbuf != old)
+           set_buffer_internal (insbuf);
+         move_gap_both (insbeg, insbeg_bytes);
+         if (insbuf != old)
+           set_buffer_internal (old);
+       }
       insbeg_ptr = BUF_BYTE_ADDRESS (insbuf, insbeg_bytes);
       eassert (insbuf->text->gpt_byte <= insbeg_bytes
               || insbuf->text->gpt_byte >= insend_bytes);