Mercurial > hg > xemacs-beta
annotate src/insdel.c @ 5908:6174848f3e6c
Use parse_integer() in read_atom(); support bases with ratios like integers
src/ChangeLog addition:
2015-05-08 Aidan Kehoe <kehoea@parhasard.net>
* data.c (init_errors_once_early):
Move the Qunsupported_type here from numbers.c, so it's available
when the majority of our types are not supported.
* general-slots.h: Add it here, too.
* number.c: Remove the definition of Qunsupported_type from here.
* lread.c (read_atom):
Check if the first character could reflect a rational, if so, call
parse_integer(), don't check the syntax of the other
characters. This allows us to accept the non-ASCII digit
characters too.
If that worked partially, but not completely, and the next char is
a slash, try to parse as a ratio.
If that fails, try isfloat_string(), but only if the first
character could plausibly be part of a float.
Otherwise, treat as a symbol.
* lread.c (read_rational):
Rename from read_integer. Handle ratios with the same radix
specification as was used for integers.
* lread.c (read1):
Rename read_integer in this function. Support the Common Lisp
#NNNrMMM syntax for parsing a number MMM of arbitrary radix NNN.
man/ChangeLog addition:
2015-05-08 Aidan Kehoe <kehoea@parhasard.net>
* lispref/numbers.texi (Numbers):
Describe the newly-supported arbitrary-base syntax for rationals
(integers and ratios). Describe that ratios can take the same base
specification as integers, something also new.
tests/ChangeLog addition:
2015-05-08 Aidan Kehoe <kehoea@parhasard.net>
* automated/lisp-reader-tests.el:
Check the arbitrary-base integer reader syntax support, just
added. Check the reader base support for ratios, just added.
Check the non-ASCII-digit support in the reader, just added.
author | Aidan Kehoe <kehoea@parhasard.net> |
---|---|
date | Sat, 09 May 2015 00:40:57 +0100 |
parents | 77d7b77909c2 |
children |
rev | line source |
---|---|
428 | 1 /* Buffer insertion/deletion and gap motion for XEmacs. |
2 Copyright (C) 1985, 1986, 1991, 1992, 1993, 1994, 1995 | |
3 Free Software Foundation, Inc. | |
4 Copyright (C) 1995 Sun Microsystems, Inc. | |
5050
6f2158fa75ed
Fix quick-build, use asserts() in place of ABORT()
Ben Wing <ben@xemacs.org>
parents:
4976
diff
changeset
|
5 Copyright (C) 2001, 2002, 2003, 2004, 2010 Ben Wing. |
428 | 6 |
7 This file is part of XEmacs. | |
8 | |
5402
308d34e9f07d
Changed bulk of GPLv2 or later files identified by script
Mats Lidell <matsl@xemacs.org>
parents:
5169
diff
changeset
|
9 XEmacs is free software: you can redistribute it and/or modify it |
428 | 10 under the terms of the GNU General Public License as published by the |
5402
308d34e9f07d
Changed bulk of GPLv2 or later files identified by script
Mats Lidell <matsl@xemacs.org>
parents:
5169
diff
changeset
|
11 Free Software Foundation, either version 3 of the License, or (at your |
308d34e9f07d
Changed bulk of GPLv2 or later files identified by script
Mats Lidell <matsl@xemacs.org>
parents:
5169
diff
changeset
|
12 option) any later version. |
428 | 13 |
14 XEmacs is distributed in the hope that it will be useful, but WITHOUT | |
15 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | |
16 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License | |
17 for more details. | |
18 | |
19 You should have received a copy of the GNU General Public License | |
5402
308d34e9f07d
Changed bulk of GPLv2 or later files identified by script
Mats Lidell <matsl@xemacs.org>
parents:
5169
diff
changeset
|
20 along with XEmacs. If not, see <http://www.gnu.org/licenses/>. */ |
428 | 21 |
22 /* Synched up with: Mule 2.0, FSF 19.30. Diverges significantly. */ | |
23 | |
24 /* This file has been Mule-ized. */ | |
25 | |
853 | 26 /* Original file from FSF, 1991. |
27 Some changes for extents, c. 1991 by unknown Lucid author. | |
28 Completely rewritten December 1994, for Mule implementation by Ben Wing; | |
29 all buffer modification code ripped out of other files and consolidated | |
30 here. | |
31 Indirect buffers written c. 1997? by Hrvoje Niksic. | |
32 */ | |
428 | 33 |
34 #include <config.h> | |
35 #include "lisp.h" | |
36 | |
37 #include "buffer.h" | |
38 #include "device.h" | |
39 #include "frame.h" | |
40 #include "extents.h" | |
41 #include "insdel.h" | |
42 #include "lstream.h" | |
43 #include "redisplay.h" | |
44 #include "line-number.h" | |
45 | |
46 /* Various macros modelled along the lines of those in buffer.h. | |
47 Purposefully omitted from buffer.h because files other than this | |
48 one should not be using them. */ | |
49 | |
50 /* Address of beginning of buffer. This is an lvalue because | |
51 BUFFER_ALLOC needs it to be. */ | |
52 #define BUF_BEG_ADDR(buf) ((buf)->text->beg) | |
53 | |
54 /* Set the address of beginning of buffer. */ | |
55 #define SET_BUF_BEG_ADDR(buf, addr) do { (buf)->text->beg = (addr); } while (0) | |
56 | |
57 /* Gap size. */ | |
58 #define BUF_GAP_SIZE(buf) ((buf)->text->gap_size + 0) | |
59 #define BUF_END_GAP_SIZE(buf) ((buf)->text->end_gap_size + 0) | |
60 /* Set gap size. */ | |
61 #define SET_BUF_GAP_SIZE(buf, value) \ | |
62 do { (buf)->text->gap_size = (value); } while (0) | |
63 #define SET_BUF_END_GAP_SIZE(buf, value) \ | |
64 do { (buf)->text->end_gap_size = (value); } while (0) | |
65 | |
826 | 66 #define BUF_GPT_ADDR(buf) (BUF_BEG_ADDR (buf) + BYTE_BUF_GPT (buf) - 1) |
428 | 67 |
68 /* Set gap location. */ | |
2367 | 69 #define SET_BOTH_BUF_GPT(buf, cval, bval) \ |
70 do \ | |
71 { \ | |
72 (buf)->text->gpt = (bval); \ | |
73 (buf)->text->bufgpt = (cval); \ | |
74 } while (0) | |
428 | 75 |
76 /* Set end of buffer. */ | |
2367 | 77 #define SET_BOTH_BUF_Z(buf, cval, bval) \ |
428 | 78 do \ |
79 { \ | |
2367 | 80 (buf)->text->z = (bval); \ |
81 (buf)->text->bufz = (cval); \ | |
428 | 82 } while (0) |
83 | |
84 /* Under Mule, we maintain two sentinels in the buffer: one at the | |
85 beginning of the gap, and one at the end of the buffer. This | |
86 allows us to move forward, examining bytes looking for the | |
87 end of a character, and not worry about running off the end. | |
88 We do not need corresponding sentinels when moving backwards | |
89 because we do not have to look past the beginning of a character | |
90 to find the beginning of the character. | |
91 | |
92 Every time we change the beginning of the gap, we have to | |
93 call SET_GAP_SENTINEL(). | |
94 | |
95 Every time we change the total size (characters plus gap) | |
96 of the buffer, we have to call SET_END_SENTINEL(). | |
97 */ | |
98 | |
99 | |
100 #ifdef MULE | |
101 # define GAP_CAN_HOLD_SIZE_P(buf, len) (BUF_GAP_SIZE (buf) >= (len) + 1) | |
102 # define SET_GAP_SENTINEL(buf) (*BUF_GPT_ADDR (buf) = 0) | |
103 # define BUF_END_SENTINEL_SIZE 1 | |
104 # define SET_END_SENTINEL(buf) \ | |
826 | 105 (*(BUF_BEG_ADDR (buf) + BUF_GAP_SIZE (buf) + BYTE_BUF_Z (buf) - 1) = 0) |
428 | 106 #else |
107 # define GAP_CAN_HOLD_SIZE_P(buf, len) (BUF_GAP_SIZE (buf) >= (len)) | |
108 # define SET_GAP_SENTINEL(buf) | |
109 # define BUF_END_SENTINEL_SIZE 0 | |
110 # define SET_END_SENTINEL(buf) | |
111 #endif | |
112 | |
113 | |
114 /************************************************************************/ | |
115 /* point and marker adjustment */ | |
116 /************************************************************************/ | |
117 | |
118 /* just_set_point() is the only place `PT' is an lvalue in all of emacs. | |
119 This function is called from set_buffer_point(), which is the function | |
120 that the SET_PT and BUF_SET_PT macros expand into, and from the | |
121 routines below that insert and delete text. (This is in cases where | |
122 the point marker logically doesn't move but PT (being a byte index) | |
123 needs to get adjusted.) */ | |
124 | |
125 /* Set point to a specified value. This is used only when the value | |
126 of point changes due to an insert or delete; it does not represent | |
127 a conceptual change in point as a marker. In particular, point is | |
128 not crossing any interval boundaries, so there's no need to use the | |
129 usual SET_PT macro. In fact it would be incorrect to do so, because | |
130 either the old or the new value of point is out of synch with the | |
131 current set of intervals. */ | |
132 | |
133 /* This gets called more than enough to make the function call | |
134 overhead a significant factor so we've turned it into a macro. */ | |
665 | 135 #define JUST_SET_POINT(buf, charbpos, ind) \ |
428 | 136 do \ |
137 { \ | |
665 | 138 buf->bufpt = (charbpos); \ |
428 | 139 buf->pt = (ind); \ |
140 } while (0) | |
141 | |
142 /* Set a buffer's point. */ | |
143 | |
144 void | |
665 | 145 set_buffer_point (struct buffer *buf, Charbpos charbpos, Bytebpos bytpos) |
428 | 146 { |
826 | 147 assert (bytpos >= BYTE_BUF_BEGV (buf) && bytpos <= BYTE_BUF_ZV (buf)); |
148 if (bytpos == BYTE_BUF_PT (buf)) | |
428 | 149 return; |
665 | 150 JUST_SET_POINT (buf, charbpos, bytpos); |
428 | 151 MARK_POINT_CHANGED; |
152 assert (MARKERP (buf->point_marker)); | |
665 | 153 XMARKER (buf->point_marker)->membpos = |
154 bytebpos_to_membpos (buf, bytpos); | |
428 | 155 |
156 /* FSF makes sure that PT is not being set within invisible text. | |
157 However, this is the wrong place for that check. The check | |
158 should happen only at the next redisplay. */ | |
159 | |
160 /* Some old coder said: | |
161 | |
162 "If there were to be hooks which were run when point entered/left an | |
163 extent, this would be the place to put them. | |
164 | |
165 However, it's probably the case that such hooks should be implemented | |
166 using a post-command-hook instead, to avoid running the hooks as a | |
167 result of intermediate motion inside of save-excursions, for example." | |
168 | |
169 I definitely agree with this. PT gets moved all over the place | |
170 and it would be a Bad Thing for any hooks to get called, both for | |
171 the reason above and because many callers are not prepared for | |
172 a GC within this function. --ben | |
173 */ | |
174 } | |
175 | |
176 /* Do the correct marker-like adjustment on MPOS (see below). FROM, TO, | |
177 and AMOUNT are as in adjust_markers(). If MPOS doesn't need to be | |
178 adjusted, nothing will happen. */ | |
665 | 179 Membpos |
180 do_marker_adjustment (Membpos mpos, Membpos from, | |
181 Membpos to, Bytecount amount) | |
428 | 182 { |
183 if (amount > 0) | |
184 { | |
185 if (mpos > to && mpos < to + amount) | |
186 mpos = to + amount; | |
187 } | |
188 else | |
189 { | |
190 if (mpos > from + amount && mpos <= from) | |
191 mpos = from + amount; | |
192 } | |
193 if (mpos > from && mpos <= to) | |
194 mpos += amount; | |
195 return mpos; | |
196 } | |
197 | |
198 /* Do the following: | |
199 | |
200 (1) Add `amount' to the position of every marker in the current buffer | |
201 whose current position is between `from' (exclusive) and `to' (inclusive). | |
202 | |
203 (2) Also, any markers past the outside of that interval, in the direction | |
204 of adjustment, are first moved back to the near end of the interval | |
205 and then adjusted by `amount'. | |
206 | |
207 This function is called in two different cases: when a region of | |
208 characters adjacent to the gap is moved, causing the gap to shift | |
209 to the other side of the region (in this case, `from' and `to' | |
210 point to the old position of the region and there should be no | |
211 markers affected by (2) because they would be inside the gap), | |
212 or when a region of characters adjacent to the gap is wiped out, | |
213 causing the gap to increase to include the region (in this case, | |
214 `from' and `to' are the same, both pointing to the boundary | |
215 between the gap and the deleted region, and there are no markers | |
216 affected by (1)). | |
217 | |
218 The reason for the use of exclusive and inclusive is that markers at | |
219 the gap always sit at the beginning, not at the end. | |
220 */ | |
221 | |
222 static void | |
665 | 223 adjust_markers (struct buffer *buf, Membpos from, Membpos to, |
428 | 224 Bytecount amount) |
225 { | |
440 | 226 Lisp_Marker *m; |
428 | 227 |
228 for (m = BUF_MARKERS (buf); m; m = marker_next (m)) | |
665 | 229 m->membpos = do_marker_adjustment (m->membpos, from, to, amount); |
428 | 230 } |
231 | |
232 /* Adjust markers whose insertion-type is t | |
233 for an insertion of AMOUNT characters at POS. */ | |
234 | |
235 static void | |
665 | 236 adjust_markers_for_insert (struct buffer *buf, Membpos ind, Bytecount amount) |
428 | 237 { |
440 | 238 Lisp_Marker *m; |
428 | 239 |
240 for (m = BUF_MARKERS (buf); m; m = marker_next (m)) | |
241 { | |
665 | 242 if (m->insertion_type && m->membpos == ind) |
243 m->membpos += amount; | |
428 | 244 } |
245 } | |
246 | |
247 | |
248 /************************************************************************/ | |
249 /* Routines for dealing with the gap */ | |
250 /************************************************************************/ | |
251 | |
252 /* maximum amount of memory moved in a single chunk. Increasing this | |
253 value improves gap-motion efficiency but decreases QUIT responsiveness | |
254 time. Was 32000 but today's processors are faster and files are | |
255 bigger. --ben */ | |
256 #define GAP_MOVE_CHUNK 300000 | |
257 | |
2367 | 258 /* Move the gap to CPOS/BPOS, which is less than the current GPT. */ |
428 | 259 |
260 static void | |
2367 | 261 gap_left (struct buffer *buf, Charbpos cpos, Bytebpos bpos) |
428 | 262 { |
867 | 263 Ibyte *to, *from; |
428 | 264 Bytecount i; |
665 | 265 Bytebpos new_s1; |
428 | 266 struct buffer *mbuf; |
267 Lisp_Object bufcons; | |
268 | |
269 from = BUF_GPT_ADDR (buf); | |
270 to = from + BUF_GAP_SIZE (buf); | |
826 | 271 new_s1 = BYTE_BUF_GPT (buf); |
428 | 272 |
273 /* Now copy the characters. To move the gap down, | |
274 copy characters up. */ | |
275 | |
276 while (1) | |
277 { | |
278 /* I gets number of characters left to copy. */ | |
2367 | 279 i = new_s1 - bpos; |
428 | 280 if (i == 0) |
281 break; | |
282 /* If a quit is requested, stop copying now. | |
2367 | 283 Change BPOS to be where we have actually moved the gap to. */ |
428 | 284 if (QUITP) |
285 { | |
2367 | 286 bpos = new_s1; |
287 cpos = bytebpos_to_charbpos (buf, bpos); | |
428 | 288 break; |
289 } | |
290 /* Move at most GAP_MOVE_CHUNK chars before checking again for a quit. */ | |
291 if (i > GAP_MOVE_CHUNK) | |
292 i = GAP_MOVE_CHUNK; | |
440 | 293 |
2367 | 294 if (i >= 10) /* was 128 but memmove() should be extremely efficient |
295 nowadays */ | |
428 | 296 { |
297 new_s1 -= i; | |
440 | 298 from -= i; |
299 to -= i; | |
428 | 300 memmove (to, from, i); |
301 } | |
302 else | |
303 { | |
304 new_s1 -= i; | |
305 while (--i >= 0) | |
306 *--to = *--from; | |
307 } | |
308 } | |
309 | |
2367 | 310 /* Adjust markers, and buffer data structure, to put the gap at BPOS. |
311 BPOS is where the loop above stopped, which may be what was specified | |
428 | 312 or may be where a quit was detected. */ |
313 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
314 { | |
2367 | 315 adjust_markers (mbuf, bpos, BYTE_BUF_GPT (mbuf), BUF_GAP_SIZE (mbuf)); |
428 | 316 } |
317 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
318 { | |
2367 | 319 adjust_extents (wrap_buffer (mbuf), bpos, BYTE_BUF_GPT (mbuf), |
428 | 320 BUF_GAP_SIZE (mbuf)); |
321 } | |
2367 | 322 SET_BOTH_BUF_GPT (buf, cpos, bpos); |
428 | 323 SET_GAP_SENTINEL (buf); |
324 #ifdef ERROR_CHECK_EXTENTS | |
325 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
326 { | |
771 | 327 sledgehammer_extent_check (wrap_buffer (mbuf)); |
428 | 328 } |
329 #endif | |
330 QUIT; | |
331 } | |
332 | |
333 static void | |
2367 | 334 gap_right (struct buffer *buf, Charbpos cpos, Bytebpos bpos) |
428 | 335 { |
867 | 336 Ibyte *to, *from; |
428 | 337 Bytecount i; |
665 | 338 Bytebpos new_s1; |
428 | 339 struct buffer *mbuf; |
340 Lisp_Object bufcons; | |
341 | |
342 to = BUF_GPT_ADDR (buf); | |
343 from = to + BUF_GAP_SIZE (buf); | |
826 | 344 new_s1 = BYTE_BUF_GPT (buf); |
428 | 345 |
346 /* Now copy the characters. To move the gap up, | |
347 copy characters down. */ | |
348 | |
349 while (1) | |
350 { | |
351 /* I gets number of characters left to copy. */ | |
2367 | 352 i = bpos - new_s1; |
428 | 353 if (i == 0) |
354 break; | |
355 /* If a quit is requested, stop copying now. | |
2367 | 356 Change BPOS to be where we have actually moved the gap to. */ |
428 | 357 if (QUITP) |
358 { | |
2367 | 359 bpos = new_s1; |
360 cpos = bytebpos_to_charbpos (buf, bpos); | |
428 | 361 break; |
362 } | |
363 /* Move at most GAP_MOVE_CHUNK chars before checking again for a quit. */ | |
364 if (i > GAP_MOVE_CHUNK) | |
365 i = GAP_MOVE_CHUNK; | |
440 | 366 |
2367 | 367 if (i >= 10) /* was 128 but memmove() should be extremely efficient |
368 nowadays */ | |
428 | 369 { |
370 new_s1 += i; | |
371 memmove (to, from, i); | |
440 | 372 from += i; |
373 to += i; | |
428 | 374 } |
375 else | |
376 { | |
377 new_s1 += i; | |
378 while (--i >= 0) | |
379 *to++ = *from++; | |
380 } | |
381 } | |
382 | |
383 { | |
384 int gsize = BUF_GAP_SIZE (buf); | |
385 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
386 { | |
2367 | 387 adjust_markers (mbuf, BYTE_BUF_GPT (mbuf) + gsize, bpos + gsize, - gsize); |
428 | 388 } |
389 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
390 { | |
826 | 391 adjust_extents (wrap_buffer (mbuf), BYTE_BUF_GPT (mbuf) + gsize, |
2367 | 392 bpos + gsize, - gsize); |
428 | 393 } |
2367 | 394 SET_BOTH_BUF_GPT (buf, cpos, bpos); |
428 | 395 SET_GAP_SENTINEL (buf); |
396 #ifdef ERROR_CHECK_EXTENTS | |
397 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
398 { | |
771 | 399 sledgehammer_extent_check (wrap_buffer (mbuf)); |
428 | 400 } |
401 #endif | |
402 } | |
2367 | 403 if (bpos == BYTE_BUF_Z (buf)) |
428 | 404 { |
405 /* merge gap with end gap */ | |
406 | |
407 SET_BUF_GAP_SIZE (buf, BUF_GAP_SIZE (buf) + BUF_END_GAP_SIZE (buf)); | |
408 SET_BUF_END_GAP_SIZE (buf, 0); | |
409 SET_END_SENTINEL (buf); | |
410 } | |
411 | |
412 QUIT; | |
413 } | |
414 | |
2367 | 415 /* Move gap to position `bpos'. |
428 | 416 Note that this can quit! */ |
417 | |
418 static void | |
2367 | 419 move_gap (struct buffer *buf, Charbpos cpos, Bytebpos bpos) |
428 | 420 { |
5050
6f2158fa75ed
Fix quick-build, use asserts() in place of ABORT()
Ben Wing <ben@xemacs.org>
parents:
4976
diff
changeset
|
421 assert (BUF_BEG_ADDR (buf)); |
2367 | 422 if (bpos < BYTE_BUF_GPT (buf)) |
423 gap_left (buf, cpos, bpos); | |
424 else if (bpos > BYTE_BUF_GPT (buf)) | |
425 gap_right (buf, cpos, bpos); | |
428 | 426 } |
427 | |
428 /* Merge the end gap into the gap */ | |
429 | |
430 static void | |
431 merge_gap_with_end_gap (struct buffer *buf) | |
432 { | |
433 Lisp_Object tem; | |
2367 | 434 Charbpos real_gap_loc_char; |
435 Bytebpos real_gap_loc_byte; | |
428 | 436 Bytecount old_gap_size; |
437 Bytecount increment; | |
438 | |
439 increment = BUF_END_GAP_SIZE (buf); | |
440 SET_BUF_END_GAP_SIZE (buf, 0); | |
441 | |
442 if (increment > 0) | |
443 { | |
444 /* Prevent quitting in move_gap. */ | |
445 tem = Vinhibit_quit; | |
446 Vinhibit_quit = Qt; | |
447 | |
2367 | 448 real_gap_loc_char = BUF_GPT (buf); |
449 real_gap_loc_byte = BYTE_BUF_GPT (buf); | |
428 | 450 old_gap_size = BUF_GAP_SIZE (buf); |
451 | |
452 /* Pretend the end gap is the gap */ | |
2367 | 453 SET_BOTH_BUF_GPT (buf, BUF_Z (buf) + BUF_GAP_SIZE (buf), |
454 BYTE_BUF_Z (buf) + BUF_GAP_SIZE (buf)); | |
428 | 455 SET_BUF_GAP_SIZE (buf, increment); |
456 | |
457 /* Move the new gap down to be consecutive with the end of the old one. | |
458 This adjusts the markers properly too. */ | |
2367 | 459 gap_left (buf, real_gap_loc_char + old_gap_size, |
460 real_gap_loc_byte + old_gap_size); | |
428 | 461 |
462 /* Now combine the two into one large gap. */ | |
463 SET_BUF_GAP_SIZE (buf, BUF_GAP_SIZE (buf) + old_gap_size); | |
2367 | 464 SET_BOTH_BUF_GPT (buf, real_gap_loc_char, real_gap_loc_byte); |
428 | 465 SET_GAP_SENTINEL (buf); |
466 | |
467 /* We changed the total size of the buffer (including gap), | |
468 so we need to fix up the end sentinel. */ | |
469 SET_END_SENTINEL (buf); | |
470 | |
471 Vinhibit_quit = tem; | |
472 } | |
473 } | |
474 | |
475 /* Make the gap INCREMENT bytes longer. */ | |
476 | |
477 static void | |
478 make_gap (struct buffer *buf, Bytecount increment) | |
479 { | |
867 | 480 Ibyte *result; |
428 | 481 Lisp_Object tem; |
2367 | 482 Charbpos real_gap_loc_char; |
483 Bytebpos real_gap_loc_byte; | |
428 | 484 Bytecount old_gap_size; |
485 | |
486 /* If we have to get more space, get enough to last a while. We use | |
487 a geometric progression that saves on realloc space. */ | |
826 | 488 increment += 2000 + ((BYTE_BUF_Z (buf) - BYTE_BUF_BEG (buf)) / 8); |
489 /* Make sure the gap is always aligned properly in case we're using a | |
490 16-bit or 32-bit fixed-width format. (Other sizes should already be | |
491 aligned in such a case.) */ | |
492 increment = MAX_ALIGN_SIZE (increment); | |
428 | 493 |
494 if (increment > BUF_END_GAP_SIZE (buf)) | |
495 { | |
496 /* Don't allow a buffer size that won't fit in an int | |
497 even if it will fit in a Lisp integer. | |
498 That won't work because so many places use `int'. */ | |
499 | |
500 if (BUF_Z (buf) - BUF_BEG (buf) + BUF_GAP_SIZE (buf) + increment | |
5581
56144c8593a8
Mechanically change INT to FIXNUM in our sources.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5402
diff
changeset
|
501 > MOST_POSITIVE_FIXNUM) |
563 | 502 out_of_memory ("Maximum buffer size exceeded", Qunbound); |
428 | 503 |
504 result = BUFFER_REALLOC (buf->text->beg, | |
826 | 505 BYTE_BUF_Z (buf) - BYTE_BUF_BEG (buf) + |
428 | 506 BUF_GAP_SIZE (buf) + increment + |
507 BUF_END_SENTINEL_SIZE); | |
508 if (result == 0) | |
509 memory_full (); | |
510 | |
511 SET_BUF_BEG_ADDR (buf, result); | |
512 } | |
513 else | |
514 increment = BUF_END_GAP_SIZE (buf); | |
515 | |
516 /* Prevent quitting in move_gap. */ | |
517 tem = Vinhibit_quit; | |
518 Vinhibit_quit = Qt; | |
519 | |
2367 | 520 real_gap_loc_char = BUF_GPT (buf); |
521 real_gap_loc_byte = BYTE_BUF_GPT (buf); | |
428 | 522 old_gap_size = BUF_GAP_SIZE (buf); |
523 | |
524 /* Call the newly allocated space a gap at the end of the whole space. */ | |
2367 | 525 SET_BOTH_BUF_GPT (buf, BUF_Z (buf) + BUF_GAP_SIZE (buf), |
526 BYTE_BUF_Z (buf) + BUF_GAP_SIZE (buf)); | |
428 | 527 SET_BUF_GAP_SIZE (buf, increment); |
528 | |
529 SET_BUF_END_GAP_SIZE (buf, 0); | |
530 | |
531 /* Move the new gap down to be consecutive with the end of the old one. | |
532 This adjusts the markers properly too. */ | |
2367 | 533 gap_left (buf, real_gap_loc_char + old_gap_size, |
534 real_gap_loc_byte + old_gap_size); | |
428 | 535 |
536 /* Now combine the two into one large gap. */ | |
537 SET_BUF_GAP_SIZE (buf, BUF_GAP_SIZE (buf) + old_gap_size); | |
2367 | 538 SET_BOTH_BUF_GPT (buf, real_gap_loc_char, real_gap_loc_byte); |
428 | 539 SET_GAP_SENTINEL (buf); |
540 | |
541 /* We changed the total size of the buffer (including gap), | |
542 so we need to fix up the end sentinel. */ | |
543 SET_END_SENTINEL (buf); | |
544 | |
545 Vinhibit_quit = tem; | |
546 } | |
547 | |
548 | |
549 /************************************************************************/ | |
550 /* Before/after-change processing */ | |
551 /************************************************************************/ | |
552 | |
553 /* Those magic changes ... */ | |
554 | |
555 static void | |
665 | 556 buffer_signal_changed_region (struct buffer *buf, Charbpos start, |
557 Charbpos end) | |
428 | 558 { |
559 /* The changed region is recorded as the number of unchanged | |
560 characters from the beginning and from the end of the | |
561 buffer. This obviates much of the need of shifting the | |
562 region around to compensate for insertions and deletions. | |
563 */ | |
564 if (buf->changes->begin_unchanged < 0 || | |
565 buf->changes->begin_unchanged > start - BUF_BEG (buf)) | |
566 buf->changes->begin_unchanged = start - BUF_BEG (buf); | |
567 if (buf->changes->end_unchanged < 0 || | |
568 buf->changes->end_unchanged > BUF_Z (buf) - end) | |
569 buf->changes->end_unchanged = BUF_Z (buf) - end; | |
570 } | |
571 | |
572 void | |
5879
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
573 buffer_extent_signal_changed_region (struct buffer *buf, Bytebpos start, |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
574 Bytebpos end) |
428 | 575 { |
576 if (buf->changes->begin_extent_unchanged < 0 || | |
5879
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
577 buf->changes->begin_extent_unchanged > start - BYTE_BUF_BEG (buf)) |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
578 buf->changes->begin_extent_unchanged = start - BYTE_BUF_BEG (buf); |
428 | 579 if (buf->changes->end_extent_unchanged < 0 || |
5879
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
580 buf->changes->end_extent_unchanged > BYTE_BUF_Z (buf) - end) |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
581 buf->changes->end_extent_unchanged = BYTE_BUF_Z (buf) - end; |
428 | 582 } |
583 | |
584 void | |
585 buffer_reset_changes (struct buffer *buf) | |
586 { | |
587 buf->changes->begin_unchanged = -1; | |
588 buf->changes->end_unchanged = -1; | |
589 buf->changes->begin_extent_unchanged = -1; | |
590 buf->changes->end_extent_unchanged = -1; | |
591 buf->changes->newline_was_deleted = 0; | |
592 } | |
593 | |
594 static void | |
665 | 595 signal_after_change (struct buffer *buf, Charbpos start, Charbpos orig_end, |
596 Charbpos new_end); | |
428 | 597 |
598 | |
599 /* Call the after-change-functions according to the changes made so far | |
600 and treat all further changes as single until the outermost | |
601 multiple change exits. This is called when the outermost multiple | |
602 change exits and when someone is trying to make a change that violates | |
603 the constraints specified in begin_multiple_change(), typically | |
604 when nested multiple-change sessions occur. (There are smarter ways of | |
605 dealing with nested multiple changes, but these rarely occur so there's | |
606 probably no point in it.) */ | |
607 | |
608 /* #### This needs to keep track of what actually changed and only | |
609 call the after-change functions on that region. */ | |
610 | |
611 static void | |
612 cancel_multiple_change (struct buffer *buf) | |
613 { | |
614 /* This function can GC */ | |
615 /* Call the after-change-functions except when they've already been | |
616 called or when there were no changes made to the buffer at all. */ | |
617 if (buf->text->changes->mc_begin != 0 && | |
618 buf->text->changes->mc_begin_signaled) | |
619 { | |
665 | 620 Charbpos real_mc_begin = buf->text->changes->mc_begin; |
428 | 621 buf->text->changes->mc_begin = 0; |
622 | |
623 signal_after_change (buf, real_mc_begin, buf->text->changes->mc_orig_end, | |
624 buf->text->changes->mc_new_end); | |
625 } | |
626 else | |
627 { | |
628 buf->text->changes->mc_begin = 0; | |
629 } | |
630 } | |
631 | |
632 /* this is an unwind_protect, to ensure that the after-change-functions | |
633 get called even in a non-local exit. */ | |
634 | |
635 static Lisp_Object | |
636 multiple_change_finish_up (Lisp_Object buffer) | |
637 { | |
638 struct buffer *buf = XBUFFER (buffer); | |
639 | |
640 /* #### I don't know whether or not it should even be possible to | |
641 get here with a dead buffer (though given how it is called I can | |
642 see how it might be). In any case, there isn't time before 19.14 | |
643 to find out. */ | |
644 if (!BUFFER_LIVE_P (buf)) | |
645 return Qnil; | |
646 | |
647 /* This function can GC */ | |
648 buf->text->changes->in_multiple_change = 0; /* do this first so that | |
649 errors in the after-change | |
650 functions don't mess things | |
651 up. */ | |
652 cancel_multiple_change (buf); | |
653 return Qnil; | |
654 } | |
655 | |
656 /* Call this function when you're about to make a number of buffer changes | |
657 that should be considered a single change. (e.g. `replace-match' calls | |
658 this.) You need to specify the START and END of the region that is | |
659 going to be changed so that the before-change-functions are called | |
660 with the correct arguments. The after-change region is calculated | |
661 automatically, however, and if changes somehow or other happen outside | |
662 of the specified region, that will also be handled correctly. | |
663 | |
664 begin_multiple_change() returns a number (actually a specpdl depth) | |
438 | 665 that you must pass to end_multiple_change() when you are done. |
666 | |
667 FSF Emacs 20 implements a similar feature, accessible from Lisp | |
668 through a `combine-after-change-calls' special form, which is | |
669 essentially equivalent to this function. We should consider | |
670 whether we want to introduce a similar Lisp form. */ | |
428 | 671 |
672 int | |
665 | 673 begin_multiple_change (struct buffer *buf, Charbpos start, Charbpos end) |
428 | 674 { |
675 /* This function can GC */ | |
676 int count = -1; | |
677 if (buf->text->changes->in_multiple_change) | |
678 { | |
679 if (buf->text->changes->mc_begin != 0 && | |
680 (start < buf->text->changes->mc_begin || | |
681 end > buf->text->changes->mc_new_end)) | |
682 cancel_multiple_change (buf); | |
683 } | |
684 else | |
685 { | |
686 Lisp_Object buffer; | |
687 | |
688 buf->text->changes->mc_begin = start; | |
689 buf->text->changes->mc_orig_end = buf->text->changes->mc_new_end = end; | |
690 buf->text->changes->mc_begin_signaled = 0; | |
691 count = specpdl_depth (); | |
793 | 692 buffer = wrap_buffer (buf); |
428 | 693 record_unwind_protect (multiple_change_finish_up, buffer); |
694 } | |
695 buf->text->changes->in_multiple_change++; | |
696 /* We don't call before-change-functions until signal_before_change() | |
697 is called, in case there is a read-only or other error. */ | |
698 return count; | |
699 } | |
700 | |
701 void | |
702 end_multiple_change (struct buffer *buf, int count) | |
703 { | |
704 assert (buf->text->changes->in_multiple_change > 0); | |
705 buf->text->changes->in_multiple_change--; | |
706 if (!buf->text->changes->in_multiple_change) | |
771 | 707 unbind_to (count); |
428 | 708 } |
709 | |
710 static int inside_change_hook; | |
711 | |
712 static Lisp_Object | |
713 change_function_restore (Lisp_Object buffer) | |
714 { | |
715 /* We should first reset the variable and then change the buffer, | |
716 because Fset_buffer() can throw. */ | |
717 inside_change_hook = 0; | |
438 | 718 if (XBUFFER (buffer) != current_buffer) |
719 Fset_buffer (buffer); | |
428 | 720 return Qnil; |
721 } | |
722 | |
723 static int in_first_change; | |
724 | |
725 static Lisp_Object | |
726 first_change_hook_restore (Lisp_Object buffer) | |
727 { | |
728 in_first_change = 0; | |
729 Fset_buffer (buffer); | |
730 return Qnil; | |
731 } | |
732 | |
733 /* Signal an initial modification to the buffer. */ | |
734 | |
735 static void | |
736 signal_first_change (struct buffer *buf) | |
737 { | |
738 /* This function can GC */ | |
793 | 739 Lisp_Object buffer = wrap_buffer (current_buffer); |
740 | |
428 | 741 |
742 if (!in_first_change) | |
743 { | |
744 if (!NILP (symbol_value_in_buffer (Qfirst_change_hook, buffer))) | |
745 { | |
746 int speccount = specpdl_depth (); | |
747 record_unwind_protect (first_change_hook_restore, buffer); | |
748 set_buffer_internal (buf); | |
749 in_first_change = 1; | |
853 | 750 run_hook_trapping_problems |
1333 | 751 (Qchange, Qfirst_change_hook, |
853 | 752 INHIBIT_EXISTING_PERMANENT_DISPLAY_OBJECT_DELETION); |
771 | 753 unbind_to (speccount); |
428 | 754 } |
755 } | |
756 } | |
757 | |
758 /* Signal a change to the buffer immediately before it happens. | |
759 START and END are the bounds of the text to be changed. */ | |
760 | |
761 static void | |
665 | 762 signal_before_change (struct buffer *buf, Charbpos start, Charbpos end) |
428 | 763 { |
764 /* This function can GC */ | |
765 struct buffer *mbuf; | |
766 Lisp_Object bufcons; | |
767 | |
768 if (!inside_change_hook) | |
769 { | |
770 Lisp_Object buffer; | |
438 | 771 int speccount; |
428 | 772 |
773 /* Are we in a multiple-change session? */ | |
774 if (buf->text->changes->in_multiple_change && | |
775 buf->text->changes->mc_begin != 0) | |
776 { | |
777 /* If we're violating the constraints of the session, | |
778 call the after-change-functions as necessary for the | |
779 changes already made and treat further changes as | |
780 single. */ | |
781 if (start < buf->text->changes->mc_begin || | |
782 end > buf->text->changes->mc_new_end) | |
783 cancel_multiple_change (buf); | |
784 /* Do nothing if this is not the first change in the session. */ | |
785 else if (buf->text->changes->mc_begin_signaled) | |
786 return; | |
787 else | |
788 { | |
789 /* First time through; call the before-change-functions | |
790 specifying the entire region to be changed. (Note that | |
791 we didn't call before-change-functions in | |
792 begin_multiple_change() because the buffer might be | |
793 read-only, etc.) */ | |
794 start = buf->text->changes->mc_begin; | |
795 end = buf->text->changes->mc_new_end; | |
796 } | |
797 } | |
798 | |
799 /* If buffer is unmodified, run a special hook for that case. */ | |
800 if (BUF_SAVE_MODIFF (buf) >= BUF_MODIFF (buf)) | |
801 { | |
802 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
803 { | |
804 signal_first_change (mbuf); | |
805 } | |
806 } | |
807 | |
808 /* Now in any case run the before-change-functions if any. */ | |
438 | 809 speccount = specpdl_depth (); |
810 record_unwind_protect (change_function_restore, Fcurrent_buffer ()); | |
811 inside_change_hook = 1; | |
428 | 812 |
813 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
814 { | |
793 | 815 buffer = wrap_buffer (mbuf); |
428 | 816 if (!NILP (symbol_value_in_buffer (Qbefore_change_functions, buffer)) |
817 /* Obsolete, for compatibility */ | |
818 || !NILP (symbol_value_in_buffer (Qbefore_change_function, buffer))) | |
819 { | |
820 set_buffer_internal (buf); | |
853 | 821 va_run_hook_with_args_trapping_problems |
1333 | 822 (Qchange, Qbefore_change_functions, 2, |
5581
56144c8593a8
Mechanically change INT to FIXNUM in our sources.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5402
diff
changeset
|
823 make_fixnum (start), make_fixnum (end), |
853 | 824 INHIBIT_EXISTING_PERMANENT_DISPLAY_OBJECT_DELETION); |
428 | 825 /* Obsolete, for compatibility */ |
853 | 826 va_run_hook_with_args_trapping_problems |
1333 | 827 (Qchange, Qbefore_change_function, 2, |
5581
56144c8593a8
Mechanically change INT to FIXNUM in our sources.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5402
diff
changeset
|
828 make_fixnum (start), make_fixnum (end), |
853 | 829 INHIBIT_EXISTING_PERMANENT_DISPLAY_OBJECT_DELETION); |
428 | 830 } |
831 } | |
832 | |
438 | 833 /* Make sure endpoints remain valid. before-change-functions |
834 might have modified the buffer. */ | |
835 if (start < BUF_BEGV (buf)) start = BUF_BEGV (buf); | |
836 if (start > BUF_ZV (buf)) start = BUF_ZV (buf); | |
837 if (end < BUF_BEGV (buf)) end = BUF_BEGV (buf); | |
838 if (end > BUF_ZV (buf)) end = BUF_ZV (buf); | |
839 | |
5879
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
840 { |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
841 Bytexpos byte_start = charbpos_to_bytebpos (buf, start); |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
842 Bytexpos byte_end = charbpos_to_bytebpos (buf, end); |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
843 |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
844 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
845 { |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
846 report_extent_modification (wrap_buffer (mbuf), byte_start, |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
847 byte_end, 0); |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
848 } |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
849 } |
771 | 850 unbind_to (speccount); |
428 | 851 |
852 /* Only now do we indicate that the before-change-functions have | |
853 been called, in case some function throws out. */ | |
854 buf->text->changes->mc_begin_signaled = 1; | |
855 } | |
856 } | |
857 | |
858 /* Signal a change immediately after it happens. | |
665 | 859 START is the charbpos of the start of the changed text. |
860 ORIG_END is the charbpos of the end of the before-changed text. | |
861 NEW_END is the charbpos of the end of the after-changed text. | |
428 | 862 */ |
863 | |
864 static void | |
665 | 865 signal_after_change (struct buffer *buf, Charbpos start, Charbpos orig_end, |
866 Charbpos new_end) | |
428 | 867 { |
868 /* This function can GC */ | |
869 struct buffer *mbuf; | |
870 Lisp_Object bufcons; | |
871 | |
872 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
873 { | |
874 /* always do this. */ | |
875 buffer_signal_changed_region (mbuf, start, new_end); | |
876 } | |
826 | 877 #ifdef USE_C_FONT_LOCK |
428 | 878 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) |
879 { | |
880 /* #### This seems inefficient. Wouldn't it be better to just | |
881 keep one cache per base buffer? */ | |
882 font_lock_maybe_update_syntactic_caches (mbuf, start, orig_end, new_end); | |
883 } | |
826 | 884 #endif /* USE_C_FONT_LOCK */ |
428 | 885 |
886 if (!inside_change_hook) | |
887 { | |
888 Lisp_Object buffer; | |
438 | 889 int speccount; |
428 | 890 |
891 if (buf->text->changes->in_multiple_change && | |
892 buf->text->changes->mc_begin != 0) | |
893 { | |
894 assert (start >= buf->text->changes->mc_begin && | |
895 start <= buf->text->changes->mc_new_end); | |
896 assert (orig_end >= buf->text->changes->mc_begin && | |
897 orig_end <= buf->text->changes->mc_new_end); | |
898 buf->text->changes->mc_new_end += new_end - orig_end; | |
899 return; /* after-change-functions signalled when all changes done */ | |
900 } | |
901 | |
438 | 902 speccount = specpdl_depth (); |
903 record_unwind_protect (change_function_restore, Fcurrent_buffer ()); | |
904 inside_change_hook = 1; | |
428 | 905 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) |
906 { | |
793 | 907 buffer = wrap_buffer (mbuf); |
428 | 908 |
909 if (!NILP (symbol_value_in_buffer (Qafter_change_functions, buffer)) | |
910 /* Obsolete, for compatibility */ | |
911 || !NILP (symbol_value_in_buffer (Qafter_change_function, buffer))) | |
912 { | |
913 set_buffer_internal (buf); | |
914 /* The actual after-change functions take slightly | |
915 different arguments than what we were passed. */ | |
853 | 916 va_run_hook_with_args_trapping_problems |
1333 | 917 (Qchange, Qafter_change_functions, 3, |
5581
56144c8593a8
Mechanically change INT to FIXNUM in our sources.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5402
diff
changeset
|
918 make_fixnum (start), make_fixnum (new_end), |
56144c8593a8
Mechanically change INT to FIXNUM in our sources.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5402
diff
changeset
|
919 make_fixnum (orig_end - start), |
853 | 920 INHIBIT_EXISTING_PERMANENT_DISPLAY_OBJECT_DELETION); |
428 | 921 /* Obsolete, for compatibility */ |
853 | 922 va_run_hook_with_args_trapping_problems |
1333 | 923 (Qchange, Qafter_change_function, 3, |
5581
56144c8593a8
Mechanically change INT to FIXNUM in our sources.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5402
diff
changeset
|
924 make_fixnum (start), make_fixnum (new_end), |
56144c8593a8
Mechanically change INT to FIXNUM in our sources.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5402
diff
changeset
|
925 make_fixnum (orig_end - start), |
853 | 926 INHIBIT_EXISTING_PERMANENT_DISPLAY_OBJECT_DELETION); |
428 | 927 } |
928 } | |
929 | |
438 | 930 /* Make sure endpoints remain valid. after-change-functions |
931 might have modified the buffer. */ | |
932 if (start < BUF_BEGV (buf)) start = BUF_BEGV (buf); | |
933 if (start > BUF_ZV (buf)) start = BUF_ZV (buf); | |
934 if (new_end < BUF_BEGV (buf)) new_end = BUF_BEGV (buf); | |
935 if (new_end > BUF_ZV (buf)) new_end = BUF_ZV (buf); | |
936 if (orig_end < BUF_BEGV (buf)) orig_end = BUF_BEGV (buf); | |
937 if (orig_end > BUF_ZV (buf)) orig_end = BUF_ZV (buf); | |
938 | |
5879
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
939 { |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
940 Bytexpos byte_start = charbpos_to_bytebpos (buf, start); |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
941 Bytexpos byte_new_end = charbpos_to_bytebpos (buf, new_end); |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
942 |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
943 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
944 { |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
945 buffer = wrap_buffer (mbuf); |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
946 report_extent_modification (buffer, byte_start, byte_new_end, 1); |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
947 } |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
948 unbind_to (speccount); /* sets inside_change_hook back to 0 */ |
77d7b77909c2
Move extents.c to working in byte positions only; fix a bug, extent_detach()
Aidan Kehoe <kehoea@parhasard.net>
parents:
5776
diff
changeset
|
949 } |
428 | 950 } |
951 } | |
952 | |
953 /* Call this if you're about to change the region of BUFFER from START | |
954 to END. This checks the read-only properties of the region, calls | |
955 the necessary modification hooks, and warns the next redisplay that | |
956 it should pay attention to that area. */ | |
957 | |
958 static void | |
665 | 959 prepare_to_modify_buffer (struct buffer *buf, Charbpos start, Charbpos end, |
428 | 960 int lockit) |
961 { | |
962 /* This function can GC */ | |
963 /* dmoore - This function can also kill the buffer buf, the current | |
964 buffer, and do anything it pleases. So if you call it, be | |
965 careful. */ | |
966 struct buffer *mbuf; | |
967 Lisp_Object buffer, bufcons; | |
968 struct gcpro gcpro1; | |
969 | |
970 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
971 { | |
853 | 972 check_allowed_operation (OPERATION_MODIFY_BUFFER_TEXT, |
973 wrap_buffer (mbuf), Qnil); | |
428 | 974 barf_if_buffer_read_only (mbuf, start, end); |
975 } | |
976 | |
977 /* if this is the first modification, see about locking the buffer's | |
978 file */ | |
793 | 979 buffer = wrap_buffer (buf); |
428 | 980 GCPRO1 (buffer); |
981 if (!NILP (buf->filename) && lockit && | |
982 BUF_SAVE_MODIFF (buf) >= BUF_MODIFF (buf)) | |
983 { | |
758 | 984 #ifdef CLASH_DETECTION |
985 if (!NILP (buf->file_truename)) | |
986 /* Make binding buffer-file-name to nil effective. */ | |
987 lock_file (buf->file_truename); | |
988 #else | |
428 | 989 /* At least warn if this file has changed on disk since it was visited.*/ |
990 if (NILP (Fverify_visited_file_modtime (buffer)) | |
991 && !NILP (Ffile_exists_p (buf->filename))) | |
992 call1_in_buffer (buf, intern ("ask-user-about-supersession-threat"), | |
993 buf->filename); | |
994 #endif /* not CLASH_DETECTION */ | |
995 } | |
996 UNGCPRO; | |
997 | |
998 /* #### dmoore - is this reasonable in case of buf being killed above? */ | |
999 if (!BUFFER_LIVE_P (buf)) | |
1000 return; | |
1001 | |
1002 signal_before_change (buf, start, end); | |
1003 | |
1004 #ifdef REGION_CACHE_NEEDS_WORK | |
1005 if (buf->newline_cache) | |
1006 invalidate_region_cache (buf, | |
1007 buf->newline_cache, | |
1008 start - BUF_BEG (buf), BUF_Z (buf) - end); | |
1009 if (buf->width_run_cache) | |
1010 invalidate_region_cache (buf, | |
1011 buf->width_run_cache, | |
1012 start - BUF_BEG (buf), BUF_Z (buf) - end); | |
1013 #endif | |
1014 | |
1015 #if 0 /* FSFmacs */ | |
1016 Vdeactivate_mark = Qt; | |
1017 #endif | |
1018 | |
1019 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
1020 { | |
1021 mbuf->point_before_scroll = Qnil; | |
1022 } | |
1023 } | |
1024 | |
1025 | |
1026 /************************************************************************/ | |
1027 /* Insertion of strings */ | |
1028 /************************************************************************/ | |
1029 | |
1030 void | |
867 | 1031 fixup_internal_substring (const Ibyte *nonreloc, Lisp_Object reloc, |
428 | 1032 Bytecount offset, Bytecount *len) |
1033 { | |
1034 assert ((nonreloc && NILP (reloc)) || (!nonreloc && STRINGP (reloc))); | |
1035 | |
1036 if (*len < 0) | |
1037 { | |
1038 if (nonreloc) | |
442 | 1039 *len = strlen ((const char *) nonreloc) - offset; |
428 | 1040 else |
1041 *len = XSTRING_LENGTH (reloc) - offset; | |
1042 } | |
800 | 1043 #ifdef ERROR_CHECK_TEXT |
428 | 1044 assert (*len >= 0); |
1045 if (STRINGP (reloc)) | |
1046 { | |
1047 assert (offset >= 0 && offset <= XSTRING_LENGTH (reloc)); | |
1048 assert (offset + *len <= XSTRING_LENGTH (reloc)); | |
1049 } | |
1050 #endif | |
1051 } | |
1052 | |
5776
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1053 /* Insert a string into BUF at Charbpos POS. The string data comes from one |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1054 of two sources: constant, non-relocatable data (specified in NONRELOC), |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1055 or a Lisp string object (specified in RELOC), which is relocatable and |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1056 may have extent data that needs to be copied into the buffer. OFFSET and |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1057 LENGTH specify the substring of the data that is actually to be inserted. |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1058 As a special case, if POS is -1, insert the string at point and move |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1059 point to the end of the string. CCLEN is the character count of the data |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1060 to be inserted, and can be -1 to indicate that buffer_insert_string_1 () |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1061 should work this out itself with bytecount_to_charcount(). |
428 | 1062 |
1063 Normally, markers at the insertion point end up before the | |
1064 inserted string. If INSDEL_BEFORE_MARKERS is set in flags, however, | |
1065 they end up after the string. | |
1066 | |
1067 INSDEL_NO_LOCKING is kludgy and is used when insert-file-contents is | |
1068 visiting a new file; it inhibits the locking checks normally done | |
1069 before modifying a buffer. Similar checks were already done | |
1070 in the higher-level Lisp functions calling insert-file-contents. */ | |
1071 | |
1072 Charcount | |
665 | 1073 buffer_insert_string_1 (struct buffer *buf, Charbpos pos, |
867 | 1074 const Ibyte *nonreloc, Lisp_Object reloc, |
428 | 1075 Bytecount offset, Bytecount length, |
5776
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1076 Charcount cclen, int flags) |
428 | 1077 { |
1078 /* This function can GC */ | |
1079 struct gcpro gcpro1; | |
826 | 1080 Bytebpos bytepos; |
1081 Bytecount length_in_buffer; | |
428 | 1082 int move_point = 0; |
1083 struct buffer *mbuf; | |
1084 Lisp_Object bufcons; | |
1085 | |
1086 /* Defensive steps just in case a buffer gets deleted and a calling | |
1087 function doesn't notice it. */ | |
1088 if (!BUFFER_LIVE_P (buf)) | |
1089 return 0; | |
1090 | |
1091 fixup_internal_substring (nonreloc, reloc, offset, &length); | |
1092 | |
1093 if (pos == -1) | |
1094 { | |
1095 pos = BUF_PT (buf); | |
1096 move_point = 1; | |
1097 } | |
1098 | |
1099 #ifdef I18N3 | |
1100 /* #### See the comment in print_internal(). If this buffer is marked | |
1101 as translatable, then Fgettext() should be called on obj if it | |
1102 is a string. */ | |
1103 #endif | |
1104 | |
1105 /* Make sure that point-max won't exceed the size of an emacs int. */ | |
5581
56144c8593a8
Mechanically change INT to FIXNUM in our sources.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5402
diff
changeset
|
1106 if ((length + BUF_Z (buf)) > MOST_POSITIVE_FIXNUM) |
563 | 1107 out_of_memory ("Maximum buffer size exceeded", Qunbound); |
428 | 1108 |
1109 /* theoretically not necessary -- caller should GCPRO. | |
1110 #### buffer_insert_from_buffer_1() doesn't! */ | |
1111 GCPRO1 (reloc); | |
1112 | |
1113 prepare_to_modify_buffer (buf, pos, pos, !(flags & INSDEL_NO_LOCKING)); | |
1114 | |
1115 /* Defensive steps in case the before-change-functions fuck around */ | |
1116 if (!BUFFER_LIVE_P (buf)) | |
1117 { | |
1118 UNGCPRO; | |
1119 /* Bad bad pre-change function. */ | |
1120 return 0; | |
1121 } | |
1122 | |
1123 /* Make args be valid again. prepare_to_modify_buffer() might have | |
1124 modified the buffer. */ | |
1125 if (pos < BUF_BEGV (buf)) | |
1126 pos = BUF_BEGV (buf); | |
1127 if (pos > BUF_ZV (buf)) | |
1128 pos = BUF_ZV (buf); | |
1129 | |
826 | 1130 bytepos = charbpos_to_bytebpos (buf, pos); |
771 | 1131 |
5776
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1132 if (cclen < 0) |
771 | 1133 { |
5776
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1134 /* string may have been relocated up to this point */ |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1135 if (STRINGP (reloc)) |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1136 { |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1137 cclen = string_offset_byte_to_char_len (reloc, offset, length); |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1138 nonreloc = XSTRING_DATA (reloc); |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1139 } |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1140 else |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1141 cclen = bytecount_to_charcount (nonreloc + offset, length); |
771 | 1142 } |
1143 else | |
5776
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1144 { |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1145 text_checking_assert (cclen > 0 && cclen |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1146 == (STRINGP (reloc) ? |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1147 string_offset_byte_to_char_len (reloc, offset, |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1148 length) |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1149 : bytecount_to_charcount (nonreloc + offset, |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1150 length))); |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1151 } |
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1152 |
826 | 1153 /* &&#### Here we check if the text can't fit into the format of the buffer, |
1154 and if so convert it to another format (either default or 32-bit-fixed, | |
1155 according to some flag; if no flag, use default). */ | |
1156 | |
1157 length_in_buffer = copy_text_between_formats (nonreloc + offset, length, | |
1158 FORMAT_DEFAULT, | |
1159 STRINGP (reloc) ? reloc : Qnil, | |
1160 NULL, 0, | |
1161 BUF_FORMAT (buf), | |
1162 wrap_buffer (buf), | |
1163 NULL); | |
428 | 1164 |
826 | 1165 if (bytepos != BYTE_BUF_GPT (buf)) |
428 | 1166 /* #### if debug-on-quit is invoked and the user changes the |
1167 buffer, bad things can happen. This is a rampant problem | |
1168 in Emacs. */ | |
2367 | 1169 move_gap (buf, pos, bytepos); /* may QUIT */ |
826 | 1170 if (! GAP_CAN_HOLD_SIZE_P (buf, length_in_buffer)) |
428 | 1171 { |
826 | 1172 if (BUF_END_GAP_SIZE (buf) >= length_in_buffer) |
428 | 1173 merge_gap_with_end_gap (buf); |
1174 else | |
826 | 1175 make_gap (buf, length_in_buffer - BUF_GAP_SIZE (buf)); |
428 | 1176 } |
1177 | |
826 | 1178 /* At this point, no more QUITting or processing of Lisp code. Buffer is |
1179 in a consistent state. Following code puts buffer in an inconsistent | |
1180 state and can be considered a "critical section". */ | |
1181 | |
428 | 1182 insert_invalidate_line_number_cache (buf, pos, nonreloc + offset, length); |
1183 | |
1184 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
1185 { | |
1186 record_insert (mbuf, pos, cclen); | |
1187 } | |
1188 | |
1189 BUF_MODIFF (buf)++; | |
1190 MARK_BUFFERS_CHANGED; | |
1191 | |
826 | 1192 /* string may have been relocated up to this point #### if string is |
1193 modified during quit processing, bad things can happen. */ | |
428 | 1194 if (STRINGP (reloc)) |
1195 nonreloc = XSTRING_DATA (reloc); | |
1196 | |
853 | 1197 memcpy (BUF_GPT_ADDR (buf), nonreloc + offset, length); |
1198 | |
826 | 1199 copy_text_between_formats (nonreloc + offset, length, FORMAT_DEFAULT, |
1200 STRINGP (reloc) ? reloc : Qnil, | |
1201 BUF_GPT_ADDR (buf), length_in_buffer, | |
1202 BUF_FORMAT (buf), wrap_buffer (buf), NULL); | |
1203 | |
1204 SET_BUF_GAP_SIZE (buf, BUF_GAP_SIZE (buf) - length_in_buffer); | |
2367 | 1205 SET_BOTH_BUF_GPT (buf, BUF_GPT (buf) + cclen, |
1206 BYTE_BUF_GPT (buf) + length_in_buffer); | |
428 | 1207 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) |
1208 { | |
826 | 1209 SET_BOTH_BUF_ZV (mbuf, BUF_ZV (mbuf) + cclen, |
1210 BYTE_BUF_ZV (mbuf) + length_in_buffer); | |
428 | 1211 } |
826 | 1212 SET_BOTH_BUF_Z (buf, BUF_Z (buf) + cclen, BYTE_BUF_Z (buf) + length_in_buffer); |
428 | 1213 SET_GAP_SENTINEL (buf); |
771 | 1214 |
1215 | |
428 | 1216 #ifdef MULE |
826 | 1217 buffer_mule_signal_inserted_region (buf, pos, length_in_buffer, cclen); |
1218 /* Update our count of ASCII, 8-bit and 16-bit chars and the | |
1219 entirely-one-byte flag */ | |
1220 { | |
867 | 1221 const Ibyte *ptr = nonreloc + offset; |
1222 const Ibyte *ptrend = ptr + length; | |
826 | 1223 |
1224 while (ptr < ptrend) | |
1225 { | |
867 | 1226 Ichar ch = itext_ichar (ptr); |
1227 if (ichar_ascii_p (ch)) | |
826 | 1228 buf->text->num_ascii_chars++; |
867 | 1229 if (ichar_8_bit_fixed_p (ch, wrap_buffer (buf))) |
826 | 1230 buf->text->num_8_bit_fixed_chars++; |
867 | 1231 if (ichar_16_bit_fixed_p (ch, wrap_buffer (buf))) |
826 | 1232 buf->text->num_16_bit_fixed_chars++; |
867 | 1233 INC_IBYTEPTR (ptr); |
826 | 1234 } |
1235 | |
1236 buf->text->entirely_one_byte_p = | |
1237 (BUF_FORMAT (buf) == FORMAT_8_BIT_FIXED || | |
1238 (BUF_FORMAT (buf) == FORMAT_DEFAULT && BUF_Z (buf) == BYTE_BUF_Z (buf))); | |
1239 } | |
428 | 1240 #endif |
1241 | |
1242 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
1243 { | |
826 | 1244 process_extents_for_insertion (wrap_buffer (mbuf), bytepos, |
1245 length_in_buffer); | |
428 | 1246 } |
1247 | |
1248 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
1249 { | |
826 | 1250 /* We know the gap is at BYTEPOS so the cast is OK. */ |
1251 adjust_markers_for_insert (mbuf, (Membpos) bytepos, length_in_buffer); | |
428 | 1252 } |
1253 | |
1254 /* Point logically doesn't move, but may need to be adjusted because | |
1255 it's a byte index. point-marker doesn't change because it's a | |
1256 memory index. */ | |
1257 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
1258 { | |
826 | 1259 if (BYTE_BUF_PT (mbuf) > bytepos) |
428 | 1260 JUST_SET_POINT (mbuf, BUF_PT (mbuf) + cclen, |
826 | 1261 BYTE_BUF_PT (mbuf) + length_in_buffer); |
428 | 1262 } |
1263 | |
1264 /* Well, point might move. */ | |
1265 if (move_point) | |
826 | 1266 BYTE_BUF_SET_PT (buf, bytepos + length_in_buffer); |
428 | 1267 |
1268 if (STRINGP (reloc)) | |
1269 { | |
1270 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
1271 { | |
826 | 1272 splice_in_string_extents (reloc, mbuf, bytepos, length, offset); |
428 | 1273 } |
1274 } | |
1275 | |
1276 if (flags & INSDEL_BEFORE_MARKERS) | |
1277 { | |
1278 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
1279 { | |
826 | 1280 /* bytepos - 1 is correct because the FROM argument is exclusive. |
665 | 1281 I formerly used DEC_BYTEBPOS() but that caused problems at the |
428 | 1282 beginning of the buffer. */ |
826 | 1283 adjust_markers (mbuf, bytepos - 1, bytepos, length_in_buffer); |
428 | 1284 } |
1285 } | |
1286 | |
826 | 1287 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) |
1288 { | |
3250 | 1289 signal_syntax_cache_extent_adjust (mbuf); |
826 | 1290 } |
1291 | |
428 | 1292 signal_after_change (buf, pos, pos, pos + cclen); |
1293 | |
1294 UNGCPRO; | |
1295 | |
1296 return cclen; | |
1297 } | |
1298 | |
1299 | |
1300 /* The following functions are interfaces onto the above function, | |
1301 for inserting particular sorts of data. In all the functions, | |
1302 BUF and POS specify the buffer and location where the insertion is | |
1303 to take place. (If POS is -1, text is inserted at point and point | |
1304 moves forward past the text.) FLAGS is as above. */ | |
1305 | |
1306 Charcount | |
665 | 1307 buffer_insert_raw_string_1 (struct buffer *buf, Charbpos pos, |
867 | 1308 const Ibyte *nonreloc, Bytecount length, |
428 | 1309 int flags) |
1310 { | |
1311 /* This function can GC */ | |
1312 return buffer_insert_string_1 (buf, pos, nonreloc, Qnil, 0, length, | |
5776
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1313 -1, flags); |
428 | 1314 } |
1315 | |
1316 Charcount | |
665 | 1317 buffer_insert_lisp_string_1 (struct buffer *buf, Charbpos pos, Lisp_Object str, |
428 | 1318 int flags) |
1319 { | |
1320 /* This function can GC */ | |
1321 return buffer_insert_string_1 (buf, pos, 0, str, 0, | |
5776
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1322 XSTRING_LENGTH (str), -1, flags); |
428 | 1323 } |
1324 | |
1325 /* Insert the null-terminated string S (in external format). */ | |
1326 | |
1327 Charcount | |
4953
304aebb79cd3
function renamings to track names of char typedefs
Ben Wing <ben@xemacs.org>
parents:
3250
diff
changeset
|
1328 buffer_insert_ascstring_1 (struct buffer *buf, Charbpos pos, const Ascbyte *s, |
428 | 1329 int flags) |
1330 { | |
1331 /* This function can GC */ | |
4953
304aebb79cd3
function renamings to track names of char typedefs
Ben Wing <ben@xemacs.org>
parents:
3250
diff
changeset
|
1332 const CIbyte *translated = GETTEXT (s); |
304aebb79cd3
function renamings to track names of char typedefs
Ben Wing <ben@xemacs.org>
parents:
3250
diff
changeset
|
1333 ASSERT_ASCTEXT_ASCII (s); |
867 | 1334 return buffer_insert_string_1 (buf, pos, (const Ibyte *) translated, Qnil, |
5776
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1335 0, strlen (translated), -1, flags); |
428 | 1336 } |
1337 | |
1338 Charcount | |
867 | 1339 buffer_insert_emacs_char_1 (struct buffer *buf, Charbpos pos, Ichar ch, |
428 | 1340 int flags) |
1341 { | |
1342 /* This function can GC */ | |
867 | 1343 Ibyte str[MAX_ICHAR_LEN]; |
1344 Bytecount len = set_itext_ichar (str, ch); | |
5776
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1345 return buffer_insert_string_1 (buf, pos, str, Qnil, 0, len, -1, flags); |
428 | 1346 } |
1347 | |
1348 Charcount | |
665 | 1349 buffer_insert_c_char_1 (struct buffer *buf, Charbpos pos, char c, |
428 | 1350 int flags) |
1351 { | |
1352 /* This function can GC */ | |
867 | 1353 return buffer_insert_emacs_char_1 (buf, pos, (Ichar) (unsigned char) c, |
428 | 1354 flags); |
1355 } | |
1356 | |
1357 Charcount | |
665 | 1358 buffer_insert_from_buffer_1 (struct buffer *buf, Charbpos pos, |
1359 struct buffer *buf2, Charbpos pos2, | |
428 | 1360 Charcount length, int flags) |
1361 { | |
1362 /* This function can GC */ | |
1363 Lisp_Object str = make_string_from_buffer (buf2, pos2, length); | |
1364 return buffer_insert_string_1 (buf, pos, 0, str, 0, | |
5776
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1365 XSTRING_LENGTH (str), -1, flags); |
428 | 1366 } |
1367 | |
1368 | |
1369 /************************************************************************/ | |
1370 /* Deletion of ranges */ | |
1371 /************************************************************************/ | |
1372 | |
1373 /* Delete characters in buffer from FROM up to (but not including) TO. */ | |
1374 | |
1375 void | |
665 | 1376 buffer_delete_range (struct buffer *buf, Charbpos from, Charbpos to, int flags) |
428 | 1377 { |
1378 /* This function can GC */ | |
1379 Charcount numdel; | |
826 | 1380 Bytebpos byte_from, byte_to; |
1381 Bytecount byte_numdel; | |
428 | 1382 EMACS_INT shortage; |
1383 struct buffer *mbuf; | |
1384 Lisp_Object bufcons; | |
826 | 1385 int do_move_gap = 0; |
428 | 1386 |
1387 /* Defensive steps just in case a buffer gets deleted and a calling | |
1388 function doesn't notice it. */ | |
1389 if (!BUFFER_LIVE_P (buf)) | |
1390 return; | |
1391 | |
1392 /* Make args be valid */ | |
1393 if (from < BUF_BEGV (buf)) | |
1394 from = BUF_BEGV (buf); | |
1395 if (to > BUF_ZV (buf)) | |
1396 to = BUF_ZV (buf); | |
1397 if ((numdel = to - from) <= 0) | |
1398 return; | |
1399 | |
1400 prepare_to_modify_buffer (buf, from, to, !(flags & INSDEL_NO_LOCKING)); | |
1401 | |
1402 /* Defensive steps in case the before-change-functions fuck around */ | |
1403 if (!BUFFER_LIVE_P (buf)) | |
1404 /* Bad bad pre-change function. */ | |
1405 return; | |
1406 | |
1407 /* Make args be valid again. prepare_to_modify_buffer() might have | |
1408 modified the buffer. */ | |
1409 if (from < BUF_BEGV (buf)) | |
1410 from = BUF_BEGV (buf); | |
1411 if (to > BUF_ZV (buf)) | |
1412 to = BUF_ZV (buf); | |
1413 if ((numdel = to - from) <= 0) | |
1414 return; | |
1415 | |
826 | 1416 byte_from = charbpos_to_bytebpos (buf, from); |
1417 byte_to = charbpos_to_bytebpos (buf, to); | |
1418 byte_numdel = byte_to - byte_from; | |
1419 | |
1420 if (to == BUF_Z (buf) && | |
1421 byte_from > BYTE_BUF_GPT (buf)) | |
1422 /* avoid moving the gap just to delete from the bottom. */ | |
1423 do_move_gap = 0; | |
1424 else | |
1425 { | |
1426 /* Make sure the gap is somewhere in or next to what we are deleting. */ | |
1427 /* NOTE: Can QUIT! */ | |
1428 if (byte_to < BYTE_BUF_GPT (buf)) | |
2367 | 1429 gap_left (buf, to, byte_to); |
826 | 1430 if (byte_from > BYTE_BUF_GPT (buf)) |
2367 | 1431 gap_right (buf, from, byte_from); |
826 | 1432 do_move_gap = 1; |
1433 } | |
1434 | |
1435 /* At this point, no more QUITting or processing of Lisp code. Buffer is | |
1436 in a consistent state. Following code puts buffer in an inconsistent | |
1437 state and can be considered a "critical section". */ | |
1438 | |
1439 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
1440 { | |
1441 record_delete (mbuf, from, numdel); | |
1442 } | |
1443 BUF_MODIFF (buf)++; | |
1444 MARK_BUFFERS_CHANGED; | |
1445 | |
1446 /* We used to do the following before the gap move. But that might QUIT, | |
1447 and (as a result of this) the gap code always leaves the buffer in | |
1448 a consistent state. Therefore, it's totally safe to do these operations | |
1449 now, and just as well not before, as we're making state changes | |
1450 related to the deletion. */ | |
1451 | |
428 | 1452 /* Redisplay needs to know if a newline was in the deleted region. |
1453 If we've already marked the changed region as having a deleted | |
1454 newline there is no use in performing the check. */ | |
1455 if (!buf->changes->newline_was_deleted) | |
1456 { | |
1457 scan_buffer (buf, '\n', from, to, 1, &shortage, 1); | |
1458 if (!shortage) | |
1459 { | |
1460 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
1461 { | |
1462 mbuf->changes->newline_was_deleted = 1; | |
1463 } | |
1464 } | |
1465 } | |
1466 | |
1467 delete_invalidate_line_number_cache (buf, from, to); | |
1468 | |
826 | 1469 #ifdef MULE |
1470 /* Update our count of ASCII, 8-bit and 16-bit chars and the | |
1471 entirely-one-byte flag */ | |
1472 { | |
1473 Bytebpos i; | |
428 | 1474 |
826 | 1475 for (i = byte_from; i < byte_to; i = next_bytebpos (buf, i)) |
1476 { | |
867 | 1477 Ichar ch = BYTE_BUF_FETCH_CHAR (buf, i); |
1478 if (ichar_ascii_p (ch)) | |
826 | 1479 buf->text->num_ascii_chars--; |
867 | 1480 if (ichar_8_bit_fixed_p (ch, wrap_buffer (buf))) |
826 | 1481 buf->text->num_8_bit_fixed_chars--; |
867 | 1482 if (ichar_16_bit_fixed_p (ch, wrap_buffer (buf))) |
826 | 1483 buf->text->num_16_bit_fixed_chars--; |
1484 } | |
1485 } | |
1486 #endif /* MULE */ | |
428 | 1487 |
826 | 1488 /* #### Point used to be modified here, but this causes problems |
1489 with MULE, as point is used to calculate bytebpos's, and if the | |
1490 offset in byte_numdel causes point to move to a non first-byte | |
1491 location, causing some other function to throw an assertion | |
1492 in ASSERT_VALID_BYTEBPOS. I've moved the code to right after | |
1493 the other movements and adjustments, but before the gap is | |
1494 moved. -- jh 970813 */ | |
428 | 1495 |
826 | 1496 /* Detach any extents that are completely within the range [FROM, TO], |
1497 if the extents are detachable. | |
1498 | |
1499 This must come AFTER record_delete(), so that the appropriate extents | |
1500 will be present to be recorded, and BEFORE the gap size is increased, | |
1501 as otherwise we will be confused about where the extents end. */ | |
1502 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
1503 { | |
1504 process_extents_for_deletion (wrap_buffer (mbuf), byte_from, byte_to, 0); | |
428 | 1505 } |
1506 | |
826 | 1507 /* Relocate all markers pointing into the new, larger gap to |
1508 point at the end of the text before the gap. */ | |
1509 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
1510 { | |
1511 adjust_markers (mbuf, | |
1512 (byte_to + BUF_GAP_SIZE (mbuf)), | |
1513 (byte_to + BUF_GAP_SIZE (mbuf)), | |
1514 (- byte_numdel - | |
1515 (do_move_gap ? BUF_GAP_SIZE (mbuf) : 0))); | |
1516 } | |
1517 | |
1518 /* Relocate any extent endpoints just like markers. */ | |
1519 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
1520 { | |
1521 adjust_extents_for_deletion (wrap_buffer (mbuf), byte_from, byte_to, | |
1522 BUF_GAP_SIZE (mbuf), | |
1523 byte_numdel, | |
1524 do_move_gap ? BUF_GAP_SIZE (mbuf) : 0); | |
1525 } | |
1526 | |
1527 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
1528 { | |
1529 /* Relocate point as if it were a marker. */ | |
1530 if (byte_from < BYTE_BUF_PT (mbuf)) | |
1531 { | |
1532 if (BYTE_BUF_PT (mbuf) < byte_to) | |
1533 JUST_SET_POINT (mbuf, from, byte_from); | |
1534 else | |
1535 JUST_SET_POINT (mbuf, BUF_PT (mbuf) - numdel, | |
1536 BYTE_BUF_PT (mbuf) - byte_numdel); | |
1537 } | |
1538 } | |
1539 | |
1540 if (do_move_gap) | |
1541 SET_BUF_GAP_SIZE (buf, BUF_GAP_SIZE (buf) + byte_numdel); | |
1542 else | |
1543 SET_BUF_END_GAP_SIZE (buf, BUF_END_GAP_SIZE (buf) + byte_numdel); | |
1544 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
1545 { | |
1546 SET_BOTH_BUF_ZV (mbuf, BUF_ZV (mbuf) - numdel, | |
1547 BYTE_BUF_ZV (mbuf) - byte_numdel); | |
1548 } | |
1549 SET_BOTH_BUF_Z (buf, BUF_Z (buf) - numdel, BYTE_BUF_Z (buf) - byte_numdel); | |
1550 if (do_move_gap) | |
2367 | 1551 SET_BOTH_BUF_GPT (buf, from, byte_from); |
826 | 1552 SET_GAP_SENTINEL (buf); |
1553 | |
428 | 1554 #ifdef MULE |
826 | 1555 buffer_mule_signal_deleted_region (buf, from, to, byte_from, byte_to); |
1556 buf->text->entirely_one_byte_p = | |
1557 (BUF_FORMAT (buf) == FORMAT_8_BIT_FIXED || | |
1558 (BUF_FORMAT (buf) == FORMAT_DEFAULT && BUF_Z (buf) == BYTE_BUF_Z (buf))); | |
428 | 1559 #endif |
1560 | |
1561 #ifdef ERROR_CHECK_EXTENTS | |
1562 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
1563 { | |
771 | 1564 sledgehammer_extent_check (wrap_buffer (mbuf)); |
428 | 1565 } |
1566 #endif | |
1567 | |
826 | 1568 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) |
1569 { | |
3250 | 1570 signal_syntax_cache_extent_adjust (mbuf); |
826 | 1571 } |
1572 | |
1573 /* &&#### Here we consider converting the buffer from default to | |
1574 8-bit-fixed if is entirely 8-bit-fixed chars and has been that way for | |
1575 a long time, e.g. 20 minutes. And if the buffer just switched to all | |
1576 8-bit-fixed chars, start the timer. */ | |
428 | 1577 signal_after_change (buf, from, to, from); |
1578 } | |
1579 | |
1580 | |
1581 /************************************************************************/ | |
1582 /* Replacement of characters */ | |
1583 /************************************************************************/ | |
1584 | |
1585 /* Replace the character at POS in buffer B with CH. */ | |
1586 | |
1587 void | |
867 | 1588 buffer_replace_char (struct buffer *buf, Charbpos pos, Ichar ch, |
428 | 1589 int not_real_change, int force_lock_check) |
1590 { | |
1591 /* This function can GC */ | |
867 | 1592 Ibyte newstr[MAX_ICHAR_LEN]; |
826 | 1593 Bytecount newlen; |
867 | 1594 Ichar oldch; |
428 | 1595 |
1596 /* Defensive steps just in case a buffer gets deleted and a calling | |
1597 function doesn't notice it. */ | |
1598 if (!BUFFER_LIVE_P (buf)) | |
1599 return; | |
1600 | |
867 | 1601 newlen = set_itext_ichar_fmt (newstr, ch, BUF_FORMAT (buf), |
826 | 1602 wrap_buffer (buf)); |
1603 oldch = BUF_FETCH_CHAR (buf, pos); | |
867 | 1604 if (ichar_fits_in_format (ch, BUF_FORMAT (buf), wrap_buffer (buf)) && |
1605 newlen == ichar_len_fmt (oldch, BUF_FORMAT (buf))) | |
428 | 1606 { |
1607 struct buffer *mbuf; | |
1608 Lisp_Object bufcons; | |
1609 | |
1610 /* then we can just replace the text. */ | |
1611 prepare_to_modify_buffer (buf, pos, pos + 1, | |
1612 !not_real_change || force_lock_check); | |
1613 /* Defensive steps in case the before-change-functions fuck around */ | |
1614 if (!BUFFER_LIVE_P (buf)) | |
1615 /* Bad bad pre-change function. */ | |
1616 return; | |
1617 | |
1618 /* Make args be valid again. prepare_to_modify_buffer() might have | |
1619 modified the buffer. */ | |
1620 if (pos < BUF_BEGV (buf)) | |
1621 pos = BUF_BEGV (buf); | |
1622 if (pos >= BUF_ZV (buf)) | |
1623 pos = BUF_ZV (buf) - 1; | |
1624 if (pos < BUF_BEGV (buf)) | |
1625 /* no more characters in buffer! */ | |
1626 return; | |
1627 | |
1628 if (BUF_FETCH_CHAR (buf, pos) == '\n') | |
1629 { | |
1630 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
1631 { | |
1632 mbuf->changes->newline_was_deleted = 1; | |
1633 } | |
1634 } | |
1635 MARK_BUFFERS_CHANGED; | |
1636 if (!not_real_change) | |
1637 { | |
1638 MAP_INDIRECT_BUFFERS (buf, mbuf, bufcons) | |
1639 { | |
1640 record_change (mbuf, pos, 1); | |
1641 } | |
1642 BUF_MODIFF (buf)++; | |
1643 } | |
826 | 1644 |
1645 #ifdef MULE | |
867 | 1646 if (ichar_ascii_p (oldch)) |
826 | 1647 buf->text->num_ascii_chars--; |
867 | 1648 if (ichar_8_bit_fixed_p (oldch, wrap_buffer (buf))) |
826 | 1649 buf->text->num_8_bit_fixed_chars--; |
867 | 1650 if (ichar_16_bit_fixed_p (oldch, wrap_buffer (buf))) |
826 | 1651 buf->text->num_16_bit_fixed_chars--; |
867 | 1652 if (ichar_ascii_p (ch)) |
826 | 1653 buf->text->num_ascii_chars++; |
867 | 1654 if (ichar_8_bit_fixed_p (ch, wrap_buffer (buf))) |
826 | 1655 buf->text->num_8_bit_fixed_chars++; |
867 | 1656 if (ichar_16_bit_fixed_p (ch, wrap_buffer (buf))) |
826 | 1657 buf->text->num_16_bit_fixed_chars++; |
1658 #endif /* MULE */ | |
1659 | |
428 | 1660 memcpy (BUF_BYTE_ADDRESS (buf, pos), newstr, newlen); |
1661 | |
1662 signal_after_change (buf, pos, pos + 1, pos + 1); | |
1663 | |
1664 /* We do not have to adjust the Mule data; we just replaced a | |
1665 character with another of the same number of bytes. */ | |
1666 } | |
1667 else | |
1668 { | |
1669 /* | |
1670 * Must implement as deletion followed by insertion. | |
1671 * | |
1672 * Make a note to move point forward later in the one situation | |
1673 * where it is needed, a delete/insert one position behind | |
1674 * point. Point will drift backward by one position and stay | |
1675 * there otherwise. | |
1676 */ | |
1677 int movepoint = (pos == BUF_PT (buf) - 1); | |
1678 | |
1679 buffer_delete_range (buf, pos, pos + 1, 0); | |
1680 /* Defensive steps in case the before-change-functions fuck around */ | |
1681 if (!BUFFER_LIVE_P (buf)) | |
1682 /* Bad bad pre-change function. */ | |
1683 return; | |
1684 | |
1685 /* Make args be valid again. prepare_to_modify_buffer() might have | |
1686 modified the buffer. */ | |
1687 if (pos < BUF_BEGV (buf)) | |
1688 pos = BUF_BEGV (buf); | |
1689 if (pos >= BUF_ZV (buf)) | |
1690 pos = BUF_ZV (buf) - 1; | |
1691 if (pos < BUF_BEGV (buf)) | |
1692 /* no more characters in buffer! */ | |
1693 return; | |
1694 /* | |
1695 * -1 as the pos argument means to move point forward with the | |
1696 * insertion, which we must do if the deletion moved point | |
1697 * backward so that it now equals the insertion point. | |
1698 */ | |
1699 buffer_insert_string_1 (buf, (movepoint ? -1 : pos), | |
5776
65d65b52d608
Pass character count from coding systems to buffer insertion code.
Aidan Kehoe <kehoea@parhasard.net>
parents:
5581
diff
changeset
|
1700 newstr, Qnil, 0, newlen, -1, 0); |
428 | 1701 } |
1702 } | |
1703 | |
1704 | |
1705 /************************************************************************/ | |
1706 /* Other functions */ | |
1707 /************************************************************************/ | |
1708 | |
1709 /* Make a string from a buffer. This needs to take into account the gap, | |
1710 and add any necessary extents from the buffer. */ | |
1711 | |
1712 static Lisp_Object | |
665 | 1713 make_string_from_buffer_1 (struct buffer *buf, Charbpos pos, Charcount length, |
428 | 1714 int no_extents) |
1715 { | |
1716 /* This function can GC */ | |
826 | 1717 Bytebpos bytepos = charbpos_to_bytebpos (buf, pos); |
1718 Bytecount bytelen = charbpos_to_bytebpos (buf, pos + length) - bytepos; | |
1719 Bytecount needed = copy_buffer_text_out (buf, bytepos, bytelen, NULL, 0, | |
1720 FORMAT_DEFAULT, Qnil, NULL); | |
1721 Lisp_Object val = make_uninit_string (needed); | |
428 | 1722 |
1723 struct gcpro gcpro1; | |
1724 GCPRO1 (val); | |
1725 | |
1726 if (!no_extents) | |
826 | 1727 add_string_extents (val, buf, bytepos, bytelen); |
1728 copy_buffer_text_out (buf, bytepos, bytelen, XSTRING_DATA (val), needed, | |
1729 FORMAT_DEFAULT, Qnil, NULL); | |
771 | 1730 init_string_ascii_begin (val); |
1731 sledgehammer_check_ascii_begin (val); | |
1732 | |
428 | 1733 UNGCPRO; |
1734 return val; | |
1735 } | |
1736 | |
1737 Lisp_Object | |
665 | 1738 make_string_from_buffer (struct buffer *buf, Charbpos pos, Charcount length) |
428 | 1739 { |
1740 return make_string_from_buffer_1 (buf, pos, length, 0); | |
1741 } | |
1742 | |
1743 Lisp_Object | |
665 | 1744 make_string_from_buffer_no_extents (struct buffer *buf, Charbpos pos, |
428 | 1745 Charcount length) |
1746 { | |
1747 return make_string_from_buffer_1 (buf, pos, length, 1); | |
1748 } | |
1749 | |
1750 void | |
665 | 1751 barf_if_buffer_read_only (struct buffer *buf, Charbpos from, Charbpos to) |
428 | 1752 { |
1753 Lisp_Object buffer; | |
1754 Lisp_Object iro; | |
1755 | |
793 | 1756 buffer = wrap_buffer (buf); |
428 | 1757 back: |
1758 iro = (buf == current_buffer ? Vinhibit_read_only : | |
1759 symbol_value_in_buffer (Qinhibit_read_only, buffer)); | |
1760 if (!LISTP (iro)) | |
1761 return; | |
1762 if (NILP (iro) && !NILP (buf->read_only)) | |
1763 { | |
1764 Fsignal (Qbuffer_read_only, (list1 (buffer))); | |
1765 goto back; | |
1766 } | |
1767 if (from > 0) | |
1768 { | |
1769 if (to < 0) | |
1770 to = from; | |
1771 verify_extent_modification (buffer, | |
665 | 1772 charbpos_to_bytebpos (buf, from), |
1773 charbpos_to_bytebpos (buf, to), | |
428 | 1774 iro); |
1775 } | |
1776 } | |
1777 | |
1778 | |
1779 /************************************************************************/ | |
1780 /* initialization */ | |
1781 /************************************************************************/ | |
1782 | |
1783 void | |
1784 reinit_vars_of_insdel (void) | |
1785 { | |
1786 inside_change_hook = 0; | |
1787 in_first_change = 0; | |
1788 } | |
1789 | |
1790 void | |
1791 vars_of_insdel (void) | |
1792 { | |
1793 } | |
1794 | |
1795 void | |
1796 init_buffer_text (struct buffer *b) | |
1797 { | |
1798 if (!b->base_buffer) | |
1799 { | |
1800 SET_BUF_GAP_SIZE (b, 20); | |
1801 BUFFER_ALLOC (b->text->beg, BUF_GAP_SIZE (b) + BUF_END_SENTINEL_SIZE); | |
1802 if (! BUF_BEG_ADDR (b)) | |
1803 memory_full (); | |
1804 | |
1805 SET_BUF_END_GAP_SIZE (b, 0); | |
2367 | 1806 SET_BOTH_BUF_GPT (b, 1, 1); |
428 | 1807 SET_BOTH_BUF_Z (b, 1, 1); |
1808 SET_GAP_SENTINEL (b); | |
1809 SET_END_SENTINEL (b); | |
2367 | 1810 |
428 | 1811 #ifdef MULE |
2367 | 1812 b->text->entirely_one_byte_p = 1; |
428 | 1813 |
2367 | 1814 #ifdef OLD_BYTE_CHAR |
1815 b->text->mule_bufmin = b->text->mule_bufmax = 1; | |
1816 b->text->mule_bytmin = b->text->mule_bytmax = 1; | |
1817 #endif | |
428 | 1818 |
2367 | 1819 b->text->cached_charpos = 1; |
1820 b->text->cached_bytepos = 1; | |
1821 | |
826 | 1822 /* &&#### Set to FORMAT_8_BIT_FIXED when that code is working */ |
1823 BUF_FORMAT (b) = FORMAT_DEFAULT; | |
428 | 1824 #endif /* MULE */ |
1825 b->text->line_number_cache = Qnil; | |
1826 | |
1827 BUF_MODIFF (b) = 1; | |
1828 BUF_SAVE_MODIFF (b) = 1; | |
1829 | |
1830 JUST_SET_POINT (b, 1, 1); | |
1831 SET_BOTH_BUF_BEGV (b, 1, 1); | |
1832 SET_BOTH_BUF_ZV (b, 1, 1); | |
1833 | |
1834 b->text->changes = xnew_and_zero (struct buffer_text_change_data); | |
1835 } | |
1836 else | |
1837 { | |
826 | 1838 JUST_SET_POINT (b, BUF_PT (b->base_buffer), BYTE_BUF_PT (b->base_buffer)); |
428 | 1839 SET_BOTH_BUF_BEGV (b, BUF_BEGV (b->base_buffer), |
826 | 1840 BYTE_BUF_BEGV (b->base_buffer)); |
428 | 1841 SET_BOTH_BUF_ZV (b, BUF_ZV (b->base_buffer), |
826 | 1842 BYTE_BUF_ZV (b->base_buffer)); |
428 | 1843 } |
1844 | |
1845 b->changes = xnew_and_zero (struct each_buffer_change_data); | |
1846 BUF_FACECHANGE (b) = 1; | |
1847 | |
1848 #ifdef REGION_CACHE_NEEDS_WORK | |
1849 b->newline_cache = 0; | |
1850 b->width_run_cache = 0; | |
1851 b->width_table = Qnil; | |
1852 #endif | |
1853 } | |
1854 | |
1855 void | |
1856 uninit_buffer_text (struct buffer *b) | |
1857 { | |
1858 if (!b->base_buffer) | |
1859 { | |
1860 BUFFER_FREE (b->text->beg); | |
4976
16112448d484
Rename xfree(FOO, TYPE) -> xfree(FOO)
Ben Wing <ben@xemacs.org>
parents:
4953
diff
changeset
|
1861 xfree (b->text->changes); |
5169
6c6d78781d59
cleanup of code related to xfree(), better KKCC backtrace capabilities, document XD_INLINE_LISP_OBJECT_BLOCK_PTR, fix some memory leaks, other code cleanup
Ben Wing <ben@xemacs.org>
parents:
5050
diff
changeset
|
1862 b->text->changes = 0; |
428 | 1863 } |
4976
16112448d484
Rename xfree(FOO, TYPE) -> xfree(FOO)
Ben Wing <ben@xemacs.org>
parents:
4953
diff
changeset
|
1864 xfree (b->changes); |
5169
6c6d78781d59
cleanup of code related to xfree(), better KKCC backtrace capabilities, document XD_INLINE_LISP_OBJECT_BLOCK_PTR, fix some memory leaks, other code cleanup
Ben Wing <ben@xemacs.org>
parents:
5050
diff
changeset
|
1865 b->changes = 0; |
428 | 1866 |
1867 #ifdef REGION_CACHE_NEEDS_WORK | |
1868 if (b->newline_cache) | |
1869 { | |
1870 free_region_cache (b->newline_cache); | |
1871 b->newline_cache = 0; | |
1872 } | |
1873 if (b->width_run_cache) | |
1874 { | |
1875 free_region_cache (b->width_run_cache); | |
1876 b->width_run_cache = 0; | |
1877 } | |
1878 b->width_table = Qnil; | |
1879 #endif | |
1880 } |