This source file includes following definitions.
- init_and_cache_system_name
- init_editfns
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- region_limit
- DEFUN
- DEFUN
- DEFUN
- overlays_around
- find_field
- DEFUN
- DEFUN
- DEFUN
- bol
- DEFUN
- DEFUN
- eol
- DEFUN
- DEFUN
- save_excursion_save
- save_excursion_restore
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- DEFUN
- general_insert_function
- insert1
- make_buffer_string
- make_buffer_string_both
- update_buffer_properties
- DEFUN
- set_bit
- bit_is_set
- buffer_chars_equal
- compareseq_early_abort
- subst_char_in_region_unwind
- subst_char_in_region_unwind_1
- check_translation
- labeled_restrictions_add
- labeled_restrictions_remove
- labeled_restrictions_get_bound
- labeled_restrictions_peek_label
- labeled_restrictions_push
- labeled_restrictions_pop
- labeled_restrictions_remove_in_current_buffer
- unwind_reset_outermost_restriction
- reset_outermost_restrictions
- labeled_restrictions_save
- labeled_restrictions_restore
- unwind_labeled_narrow_to_region
- labeled_narrow_to_region
- DEFUN
- DEFUN
- save_restriction_save_1
- save_restriction_restore_1
- save_restriction_save
- save_restriction_restore
- DEFUN
- DEFUN
- str2num
- styled_format
- transpose_markers
- syms_of_editfns
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21 #include <config.h>
22 #include <sys/types.h>
23 #include <stdio.h>
24
25 #ifdef HAVE_PWD_H
26 #include <pwd.h>
27 #include <grp.h>
28 #endif
29
30 #include <unistd.h>
31
32 #ifdef HAVE_SYS_UTSNAME_H
33 #include <sys/utsname.h>
34 #endif
35
36 #ifdef HAVE_ANDROID
37 #include "android.h"
38 #endif
39
40 #include "lisp.h"
41
42 #include <float.h>
43 #include <limits.h>
44 #include <math.h>
45
46 #include <c-ctype.h>
47 #include <intprops.h>
48 #include <stdlib.h>
49 #include <verify.h>
50
51 #include "composite.h"
52 #include "intervals.h"
53 #include "systime.h"
54 #include "character.h"
55 #include "buffer.h"
56 #include "window.h"
57 #include "blockinput.h"
58
59 #ifdef WINDOWSNT
60 # include "w32common.h"
61 #endif
62
63 #ifdef HAVE_TREE_SITTER
64 #include "treesit.h"
65 #endif
66
67 static void update_buffer_properties (ptrdiff_t, ptrdiff_t);
68 static Lisp_Object styled_format (ptrdiff_t, Lisp_Object *, bool);
69
70
71
72 static Lisp_Object cached_system_name;
73
74 static void
75 init_and_cache_system_name (void)
76 {
77 init_system_name ();
78 cached_system_name = Vsystem_name;
79 }
80
81 void
82 init_editfns (void)
83 {
84 const char *user_name;
85 register char *p;
86 struct passwd *pw;
87 Lisp_Object tem;
88
89
90 init_and_cache_system_name ();
91
92 pw = getpwuid (getuid ());
93 #ifdef MSDOS
94
95
96
97 Vuser_real_login_name = build_string (pw ? pw->pw_name : "root");
98 #else
99 Vuser_real_login_name = build_string (pw ? pw->pw_name : "unknown");
100 #endif
101
102
103
104 user_name = getenv ("LOGNAME");
105 if (!user_name)
106 #ifdef WINDOWSNT
107 user_name = getenv ("USERNAME");
108 #else
109 user_name = getenv ("USER");
110 #endif
111 if (!user_name)
112 {
113 pw = getpwuid (geteuid ());
114 user_name = pw ? pw->pw_name : "unknown";
115 }
116 Vuser_login_name = build_string (user_name);
117
118
119
120 tem = Fstring_equal (Vuser_login_name, Vuser_real_login_name);
121 if (! NILP (tem))
122 tem = Vuser_login_name;
123 else
124 {
125 uid_t euid = geteuid ();
126 tem = INT_TO_INTEGER (euid);
127 }
128 Vuser_full_name = Fuser_full_name (tem);
129
130 p = getenv ("NAME");
131 if (p)
132 Vuser_full_name = build_string (p);
133 else if (NILP (Vuser_full_name))
134 Vuser_full_name = build_string ("unknown");
135
136 #if defined HAVE_SYS_UTSNAME_H
137 {
138 struct utsname uts;
139 uname (&uts);
140 Voperating_system_release = build_string (uts.release);
141 }
142 #elif defined WINDOWSNT
143 Voperating_system_release = build_string (w32_version_string ());
144 #else
145 Voperating_system_release = Qnil;
146 #endif
147 }
148
149 DEFUN ("char-to-string", Fchar_to_string, Schar_to_string, 1, 1, 0,
150 doc:
151 )
152 (Lisp_Object character)
153 {
154 int c, len;
155 unsigned char str[MAX_MULTIBYTE_LENGTH];
156
157 CHECK_CHARACTER (character);
158 c = XFIXNAT (character);
159
160 len = CHAR_STRING (c, str);
161 return make_string_from_bytes ((char *) str, 1, len);
162 }
163
164 DEFUN ("byte-to-string", Fbyte_to_string, Sbyte_to_string, 1, 1, 0,
165 doc: )
166 (Lisp_Object byte)
167 {
168 unsigned char b;
169 CHECK_FIXNUM (byte);
170 if (XFIXNUM (byte) < 0 || XFIXNUM (byte) > 255)
171 error ("Invalid byte");
172 b = XFIXNUM (byte);
173 return make_unibyte_string ((char *) &b, 1);
174 }
175
176 DEFUN ("string-to-char", Fstring_to_char, Sstring_to_char, 1, 1, 0,
177 doc: )
178 (Lisp_Object string)
179 {
180 CHECK_STRING (string);
181
182
183 return make_fixnum (STRING_MULTIBYTE (string)
184 ? STRING_CHAR (SDATA (string))
185 : SREF (string, 0));
186 }
187
188 DEFUN ("point", Fpoint, Spoint, 0, 0, 0,
189 doc:
190 )
191 (void)
192 {
193 Lisp_Object temp;
194 XSETFASTINT (temp, PT);
195 return temp;
196 }
197
198 DEFUN ("point-marker", Fpoint_marker, Spoint_marker, 0, 0, 0,
199 doc: )
200 (void)
201 {
202 return build_marker (current_buffer, PT, PT_BYTE);
203 }
204
205 DEFUN ("goto-char", Fgoto_char, Sgoto_char, 1, 1,
206 "(goto-char--read-natnum-interactive \"Go to char: \")",
207 doc:
208
209
210
211
212
213
214 )
215 (register Lisp_Object position)
216 {
217 if (MARKERP (position))
218 set_point_from_marker (position);
219 else if (FIXNUMP (position))
220 SET_PT (clip_to_bounds (BEGV, XFIXNUM (position), ZV));
221 else
222 wrong_type_argument (Qinteger_or_marker_p, position);
223 return position;
224 }
225
226
227
228
229
230
231 static Lisp_Object
232 region_limit (bool beginningp)
233 {
234 Lisp_Object m;
235
236 if (!NILP (Vtransient_mark_mode)
237 && NILP (Vmark_even_if_inactive)
238 && NILP (BVAR (current_buffer, mark_active)))
239 xsignal0 (Qmark_inactive);
240
241 m = Fmarker_position (BVAR (current_buffer, mark));
242 if (NILP (m))
243 error ("The mark is not set now, so there is no region");
244
245
246 return make_fixnum ((PT < XFIXNAT (m)) == beginningp
247 ? PT
248 : clip_to_bounds (BEGV, XFIXNAT (m), ZV));
249 }
250
251 DEFUN ("region-beginning", Fregion_beginning, Sregion_beginning, 0, 0, 0,
252 doc: )
253 (void)
254 {
255 return region_limit (1);
256 }
257
258 DEFUN ("region-end", Fregion_end, Sregion_end, 0, 0, 0,
259 doc: )
260 (void)
261 {
262 return region_limit (0);
263 }
264
265 DEFUN ("mark-marker", Fmark_marker, Smark_marker, 0, 0, 0,
266 doc:
267
268 )
269 (void)
270 {
271 return BVAR (current_buffer, mark);
272 }
273
274
275
276
277
278
279
280
281
282 static ptrdiff_t
283 overlays_around (ptrdiff_t pos, Lisp_Object *vec, ptrdiff_t len)
284 {
285
286
287
288
289 return overlays_in (pos - 1, pos + 1, false, &vec, &len,
290 true, false, NULL);
291 }
292
293 DEFUN ("get-pos-property", Fget_pos_property, Sget_pos_property, 2, 3, 0,
294 doc:
295
296
297
298
299
300 )
301 (Lisp_Object position, register Lisp_Object prop, Lisp_Object object)
302 {
303 CHECK_FIXNUM_COERCE_MARKER (position);
304
305 if (NILP (object))
306 XSETBUFFER (object, current_buffer);
307 else if (WINDOWP (object))
308 object = XWINDOW (object)->contents;
309
310 if (!BUFFERP (object))
311
312
313
314 return Fget_text_property (position, prop, object);
315 else
316 {
317 EMACS_INT posn = XFIXNUM (position);
318 ptrdiff_t noverlays;
319 Lisp_Object *overlay_vec, tem;
320 struct buffer *obuf = current_buffer;
321 USE_SAFE_ALLOCA;
322
323 set_buffer_temp (XBUFFER (object));
324
325
326 Lisp_Object overlay_vecbuf[40];
327 noverlays = ARRAYELTS (overlay_vecbuf);
328 overlay_vec = overlay_vecbuf;
329 noverlays = overlays_around (posn, overlay_vec, noverlays);
330
331
332
333 if (ARRAYELTS (overlay_vecbuf) < noverlays)
334 {
335 SAFE_ALLOCA_LISP (overlay_vec, noverlays);
336 noverlays = overlays_around (posn, overlay_vec, noverlays);
337 }
338 noverlays = sort_overlays (overlay_vec, noverlays, NULL);
339
340 set_buffer_temp (obuf);
341
342
343 while (--noverlays >= 0)
344 {
345 Lisp_Object ol = overlay_vec[noverlays];
346 tem = Foverlay_get (ol, prop);
347 if (!NILP (tem))
348 {
349
350 if ((OVERLAY_START (ol) == posn
351 && OVERLAY_FRONT_ADVANCE_P (ol))
352 || (OVERLAY_END (ol) == posn
353 && ! OVERLAY_REAR_ADVANCE_P (ol))
354 || OVERLAY_START (ol) > posn
355 || OVERLAY_END (ol) < posn)
356 ;
357 else
358 {
359 SAFE_FREE ();
360 return tem;
361 }
362 }
363 }
364 SAFE_FREE ();
365
366 {
367 int stickiness = text_property_stickiness (prop, position, object);
368 if (stickiness > 0)
369 return Fget_text_property (position, prop, object);
370 else if (stickiness < 0
371 && XFIXNUM (position) > BUF_BEGV (XBUFFER (object)))
372 return Fget_text_property (make_fixnum (XFIXNUM (position) - 1),
373 prop, object);
374 else
375 return Qnil;
376 }
377 }
378 }
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400 static void
401 find_field (Lisp_Object pos, Lisp_Object merge_at_boundary,
402 Lisp_Object beg_limit,
403 ptrdiff_t *beg, Lisp_Object end_limit, ptrdiff_t *end)
404 {
405
406 Lisp_Object before_field, after_field;
407
408 bool at_field_start = 0;
409
410 bool at_field_end = 0;
411
412 if (NILP (pos))
413 XSETFASTINT (pos, PT);
414 else
415 CHECK_FIXNUM_COERCE_MARKER (pos);
416
417 after_field
418 = get_char_property_and_overlay (pos, Qfield, Qnil, NULL);
419 before_field
420 = (XFIXNAT (pos) > BEGV
421 ? get_char_property_and_overlay (make_fixnum (XFIXNUM (pos) - 1),
422 Qfield, Qnil, NULL)
423
424
425 : after_field);
426
427
428
429
430
431
432
433 if (NILP (merge_at_boundary))
434 {
435 Lisp_Object field = Fget_pos_property (pos, Qfield, Qnil);
436 if (!EQ (field, after_field))
437 at_field_end = 1;
438 if (!EQ (field, before_field))
439 at_field_start = 1;
440 if (NILP (field) && at_field_start && at_field_end)
441
442
443
444
445 at_field_end = at_field_start = 0;
446 }
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470 if (beg)
471 {
472 if (at_field_start)
473
474
475 *beg = XFIXNAT (pos);
476 else
477
478 {
479 Lisp_Object p = pos;
480 if (!NILP (merge_at_boundary) && EQ (before_field, Qboundary))
481
482 p = Fprevious_single_char_property_change (p, Qfield, Qnil,
483 beg_limit);
484
485 p = Fprevious_single_char_property_change (p, Qfield, Qnil,
486 beg_limit);
487 *beg = NILP (p) ? BEGV : XFIXNAT (p);
488 }
489 }
490
491 if (end)
492 {
493 if (at_field_end)
494
495
496 *end = XFIXNAT (pos);
497 else
498
499 {
500 if (!NILP (merge_at_boundary) && EQ (after_field, Qboundary))
501
502 pos = Fnext_single_char_property_change (pos, Qfield, Qnil,
503 end_limit);
504
505 pos = Fnext_single_char_property_change (pos, Qfield, Qnil,
506 end_limit);
507 *end = NILP (pos) ? ZV : XFIXNAT (pos);
508 }
509 }
510 }
511
512
513 DEFUN ("delete-field", Fdelete_field, Sdelete_field, 0, 1, 0,
514 doc:
515
516 )
517 (Lisp_Object pos)
518 {
519 ptrdiff_t beg, end;
520 find_field (pos, Qnil, Qnil, &beg, Qnil, &end);
521 if (beg != end)
522 del_range (beg, end);
523 return Qnil;
524 }
525
526 DEFUN ("field-string", Ffield_string, Sfield_string, 0, 1, 0,
527 doc:
528
529 )
530 (Lisp_Object pos)
531 {
532 ptrdiff_t beg, end;
533 find_field (pos, Qnil, Qnil, &beg, Qnil, &end);
534 return make_buffer_string (beg, end, 1);
535 }
536
537 DEFUN ("field-string-no-properties", Ffield_string_no_properties, Sfield_string_no_properties, 0, 1, 0,
538 doc:
539
540 )
541 (Lisp_Object pos)
542 {
543 ptrdiff_t beg, end;
544 find_field (pos, Qnil, Qnil, &beg, Qnil, &end);
545 return make_buffer_string (beg, end, 0);
546 }
547
548 DEFUN ("field-beginning", Ffield_beginning, Sfield_beginning, 0, 3, 0,
549 doc:
550
551
552
553
554
555 )
556 (Lisp_Object pos, Lisp_Object escape_from_edge, Lisp_Object limit)
557 {
558 ptrdiff_t beg;
559 find_field (pos, escape_from_edge, limit, &beg, Qnil, 0);
560 return make_fixnum (beg);
561 }
562
563 DEFUN ("field-end", Ffield_end, Sfield_end, 0, 3, 0,
564 doc:
565
566
567
568
569
570 )
571 (Lisp_Object pos, Lisp_Object escape_from_edge, Lisp_Object limit)
572 {
573 ptrdiff_t end;
574 find_field (pos, escape_from_edge, Qnil, 0, limit, &end);
575 return make_fixnum (end);
576 }
577
578 DEFUN ("constrain-to-field", Fconstrain_to_field, Sconstrain_to_field, 2, 5, 0,
579 doc:
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605 )
606 (Lisp_Object new_pos, Lisp_Object old_pos, Lisp_Object escape_from_edge,
607 Lisp_Object only_in_line, Lisp_Object inhibit_capture_property)
608 {
609
610 ptrdiff_t orig_point = 0;
611 bool fwd;
612 Lisp_Object prev_old, prev_new;
613
614 if (NILP (new_pos))
615
616 {
617 orig_point = PT;
618 XSETFASTINT (new_pos, PT);
619 }
620
621 CHECK_FIXNUM_COERCE_MARKER (new_pos);
622 CHECK_FIXNUM_COERCE_MARKER (old_pos);
623
624 fwd = (XFIXNUM (new_pos) > XFIXNUM (old_pos));
625
626 prev_old = make_fixnum (XFIXNUM (old_pos) - 1);
627 prev_new = make_fixnum (XFIXNUM (new_pos) - 1);
628
629 if (NILP (Vinhibit_field_text_motion)
630 && !BASE_EQ (new_pos, old_pos)
631 && (!NILP (Fget_char_property (new_pos, Qfield, Qnil))
632 || !NILP (Fget_char_property (old_pos, Qfield, Qnil))
633
634
635
636
637 || (XFIXNAT (new_pos) > BEGV
638 && !NILP (Fget_char_property (prev_new, Qfield, Qnil)))
639 || (XFIXNAT (old_pos) > BEGV
640 && !NILP (Fget_char_property (prev_old, Qfield, Qnil))))
641 && (NILP (inhibit_capture_property)
642
643
644
645 || (NILP (Fget_pos_property (old_pos, inhibit_capture_property, Qnil))
646 && (XFIXNAT (old_pos) <= BEGV
647 || NILP (Fget_char_property
648 (old_pos, inhibit_capture_property, Qnil))
649 || NILP (Fget_char_property
650 (prev_old, inhibit_capture_property, Qnil))))))
651
652
653 {
654 ptrdiff_t counted;
655 Lisp_Object field_bound;
656
657 if (fwd)
658 field_bound = Ffield_end (old_pos, escape_from_edge, new_pos);
659 else
660 field_bound = Ffield_beginning (old_pos, escape_from_edge, new_pos);
661
662 if (
663
664
665
666 ((XFIXNAT (field_bound) < XFIXNAT (new_pos)) ? fwd : !fwd)
667
668
669
670
671 && (NILP (only_in_line)
672
673
674
675 || (find_newline (XFIXNAT (new_pos), -1,
676 XFIXNAT (field_bound), -1,
677 fwd ? -1 : 1, &counted, NULL, 1),
678 counted == 0)))
679
680 new_pos = field_bound;
681
682 if (orig_point && XFIXNAT (new_pos) != orig_point)
683
684 SET_PT (XFIXNAT (new_pos));
685 }
686
687 return new_pos;
688 }
689
690
691 static ptrdiff_t
692 bol (Lisp_Object n, ptrdiff_t *out_count)
693 {
694 ptrdiff_t bytepos, charpos, count;
695
696 if (NILP (n))
697 count = 0;
698 else if (FIXNUMP (n))
699 count = clip_to_bounds (-BUF_BYTES_MAX, XFIXNUM (n) - 1, BUF_BYTES_MAX);
700 else
701 {
702 CHECK_INTEGER (n);
703 count = NILP (Fnatnump (n)) ? -BUF_BYTES_MAX : BUF_BYTES_MAX;
704 }
705 if (out_count)
706 *out_count = count;
707 scan_newline_from_point (count, &charpos, &bytepos);
708 return charpos;
709 }
710
711 DEFUN ("pos-bol", Fpos_bol, Spos_bol, 0, 1, 0,
712 doc:
713
714
715
716
717
718
719
720
721 )
722 (Lisp_Object n)
723 {
724 return make_fixnum (bol (n, NULL));
725 }
726
727 DEFUN ("line-beginning-position",
728 Fline_beginning_position, Sline_beginning_position, 0, 1, 0,
729 doc:
730
731
732
733
734
735
736
737
738 )
739 (Lisp_Object n)
740 {
741 ptrdiff_t count, charpos = bol (n, &count);
742
743 return Fconstrain_to_field (make_fixnum (charpos), make_fixnum (PT),
744 count != 0 ? Qt : Qnil,
745 Qt, Qnil);
746 }
747
748 static ptrdiff_t
749 eol (Lisp_Object n)
750 {
751 ptrdiff_t count;
752
753 if (NILP (n))
754 count = 1;
755 else if (FIXNUMP (n))
756 count = clip_to_bounds (-BUF_BYTES_MAX, XFIXNUM (n), BUF_BYTES_MAX);
757 else
758 {
759 CHECK_INTEGER (n);
760 count = NILP (Fnatnump (n)) ? -BUF_BYTES_MAX : BUF_BYTES_MAX;
761 }
762 return find_before_next_newline (PT, 0, count - (count <= 0),
763 NULL);
764 }
765
766 DEFUN ("pos-eol", Fpos_eol, Spos_eol, 0, 1, 0,
767 doc:
768
769
770
771
772
773
774
775 )
776 (Lisp_Object n)
777 {
778 return make_fixnum (eol (n));
779 }
780
781 DEFUN ("line-end-position", Fline_end_position, Sline_end_position, 0, 1, 0,
782 doc:
783
784
785
786
787
788
789
790
791
792
793
794 )
795 (Lisp_Object n)
796 {
797
798 return Fconstrain_to_field (make_fixnum (eol (n)), make_fixnum (PT),
799 Qnil, Qt, Qnil);
800 }
801
802
803
804 void
805 save_excursion_save (union specbinding *pdl)
806 {
807 eassert (pdl->unwind_excursion.kind == SPECPDL_UNWIND_EXCURSION);
808 pdl->unwind_excursion.marker = Fpoint_marker ();
809
810 pdl->unwind_excursion.window
811 = (BASE_EQ (XWINDOW (selected_window)->contents, Fcurrent_buffer ())
812 ? selected_window : Qnil);
813 }
814
815
816
817 void
818 save_excursion_restore (Lisp_Object marker, Lisp_Object window)
819 {
820 Lisp_Object buffer = Fmarker_buffer (marker);
821
822
823 if (NILP (buffer))
824 return;
825
826 Fset_buffer (buffer);
827
828
829 Fgoto_char (marker);
830 unchain_marker (XMARKER (marker));
831
832
833
834
835 if (WINDOWP (window) && !BASE_EQ (window, selected_window))
836 {
837
838 Lisp_Object contents = XWINDOW (window)->contents;
839 if (BUFFERP (contents) && XBUFFER (contents) == current_buffer)
840 Fset_window_point (window, make_fixnum (PT));
841 }
842 }
843
844 DEFUN ("save-excursion", Fsave_excursion, Ssave_excursion, 0, UNEVALLED, 0,
845 doc:
846
847
848
849
850
851
852
853
854
855
856
857 )
858 (Lisp_Object args)
859 {
860 register Lisp_Object val;
861 specpdl_ref count = SPECPDL_INDEX ();
862
863 record_unwind_protect_excursion ();
864
865 val = Fprogn (args);
866 return unbind_to (count, val);
867 }
868
869 DEFUN ("save-current-buffer", Fsave_current_buffer, Ssave_current_buffer, 0, UNEVALLED, 0,
870 doc:
871
872 )
873 (Lisp_Object args)
874 {
875 specpdl_ref count = SPECPDL_INDEX ();
876
877 record_unwind_current_buffer ();
878 return unbind_to (count, Fprogn (args));
879 }
880
881 DEFUN ("buffer-size", Fbuffer_size, Sbuffer_size, 0, 1, 0,
882 doc:
883
884
885
886
887
888
889
890 )
891 (Lisp_Object buffer)
892 {
893 if (NILP (buffer))
894 return make_fixnum (Z - BEG);
895 else
896 {
897 CHECK_BUFFER (buffer);
898 return make_fixnum (BUF_Z (XBUFFER (buffer))
899 - BUF_BEG (XBUFFER (buffer)));
900 }
901 }
902
903 DEFUN ("point-min", Fpoint_min, Spoint_min, 0, 0, 0,
904 doc:
905 )
906 (void)
907 {
908 Lisp_Object temp;
909 XSETFASTINT (temp, BEGV);
910 return temp;
911 }
912
913 DEFUN ("point-min-marker", Fpoint_min_marker, Spoint_min_marker, 0, 0, 0,
914 doc:
915 )
916 (void)
917 {
918 return build_marker (current_buffer, BEGV, BEGV_BYTE);
919 }
920
921 DEFUN ("point-max", Fpoint_max, Spoint_max, 0, 0, 0,
922 doc:
923
924 )
925 (void)
926 {
927 Lisp_Object temp;
928 XSETFASTINT (temp, ZV);
929 return temp;
930 }
931
932 DEFUN ("point-max-marker", Fpoint_max_marker, Spoint_max_marker, 0, 0, 0,
933 doc:
934
935 )
936 (void)
937 {
938 return build_marker (current_buffer, ZV, ZV_BYTE);
939 }
940
941 DEFUN ("gap-position", Fgap_position, Sgap_position, 0, 0, 0,
942 doc:
943 )
944 (void)
945 {
946 Lisp_Object temp;
947 XSETFASTINT (temp, GPT);
948 return temp;
949 }
950
951 DEFUN ("gap-size", Fgap_size, Sgap_size, 0, 0, 0,
952 doc:
953 )
954 (void)
955 {
956 Lisp_Object temp;
957 XSETFASTINT (temp, GAP_SIZE);
958 return temp;
959 }
960
961 DEFUN ("position-bytes", Fposition_bytes, Sposition_bytes, 1, 1, 0,
962 doc:
963 )
964 (Lisp_Object position)
965 {
966 EMACS_INT pos = fix_position (position);
967 if (! (BEG <= pos && pos <= Z))
968 return Qnil;
969 return make_fixnum (CHAR_TO_BYTE (pos));
970 }
971
972 DEFUN ("byte-to-position", Fbyte_to_position, Sbyte_to_position, 1, 1, 0,
973 doc:
974 )
975 (Lisp_Object bytepos)
976 {
977 ptrdiff_t pos_byte;
978
979 CHECK_FIXNUM (bytepos);
980 pos_byte = XFIXNUM (bytepos);
981 if (pos_byte < BEG_BYTE || pos_byte > Z_BYTE)
982 return Qnil;
983 if (Z != Z_BYTE)
984
985
986
987
988 while (!CHAR_HEAD_P (FETCH_BYTE (pos_byte)))
989 pos_byte--;
990 return make_fixnum (BYTE_TO_CHAR (pos_byte));
991 }
992
993 DEFUN ("following-char", Ffollowing_char, Sfollowing_char, 0, 0, 0,
994 doc:
995 )
996 (void)
997 {
998 Lisp_Object temp;
999 if (PT >= ZV)
1000 XSETFASTINT (temp, 0);
1001 else
1002 XSETFASTINT (temp, FETCH_CHAR (PT_BYTE));
1003 return temp;
1004 }
1005
1006 DEFUN ("preceding-char", Fprevious_char, Sprevious_char, 0, 0, 0,
1007 doc:
1008 )
1009 (void)
1010 {
1011 Lisp_Object temp;
1012 if (PT <= BEGV)
1013 XSETFASTINT (temp, 0);
1014 else if (!NILP (BVAR (current_buffer, enable_multibyte_characters)))
1015 {
1016 ptrdiff_t pos = PT_BYTE;
1017 pos -= prev_char_len (pos);
1018 XSETFASTINT (temp, FETCH_CHAR (pos));
1019 }
1020 else
1021 XSETFASTINT (temp, FETCH_BYTE (PT_BYTE - 1));
1022 return temp;
1023 }
1024
1025 DEFUN ("bobp", Fbobp, Sbobp, 0, 0, 0,
1026 doc:
1027 )
1028 (void)
1029 {
1030 if (PT == BEGV)
1031 return Qt;
1032 return Qnil;
1033 }
1034
1035 DEFUN ("eobp", Feobp, Seobp, 0, 0, 0,
1036 doc:
1037 )
1038 (void)
1039 {
1040 if (PT == ZV)
1041 return Qt;
1042 return Qnil;
1043 }
1044
1045 DEFUN ("bolp", Fbolp, Sbolp, 0, 0, 0,
1046 doc: )
1047 (void)
1048 {
1049 if (PT == BEGV || FETCH_BYTE (PT_BYTE - 1) == '\n')
1050 return Qt;
1051 return Qnil;
1052 }
1053
1054 DEFUN ("eolp", Feolp, Seolp, 0, 0, 0,
1055 doc:
1056 )
1057 (void)
1058 {
1059 if (PT == ZV || FETCH_BYTE (PT_BYTE) == '\n')
1060 return Qt;
1061 return Qnil;
1062 }
1063
1064 DEFUN ("char-after", Fchar_after, Schar_after, 0, 1, 0,
1065 doc:
1066
1067 )
1068 (Lisp_Object pos)
1069 {
1070 register ptrdiff_t pos_byte;
1071
1072 if (NILP (pos))
1073 {
1074 pos_byte = PT_BYTE;
1075 if (pos_byte < BEGV_BYTE || pos_byte >= ZV_BYTE)
1076 return Qnil;
1077 }
1078 else if (MARKERP (pos))
1079 {
1080 pos_byte = marker_byte_position (pos);
1081 if (pos_byte < BEGV_BYTE || pos_byte >= ZV_BYTE)
1082 return Qnil;
1083 }
1084 else
1085 {
1086 EMACS_INT p = fix_position (pos);
1087 if (! (BEGV <= p && p < ZV))
1088 return Qnil;
1089
1090 pos_byte = CHAR_TO_BYTE (p);
1091 }
1092
1093 return make_fixnum (FETCH_CHAR (pos_byte));
1094 }
1095
1096 DEFUN ("char-before", Fchar_before, Schar_before, 0, 1, 0,
1097 doc:
1098
1099 )
1100 (Lisp_Object pos)
1101 {
1102 register Lisp_Object val;
1103 register ptrdiff_t pos_byte;
1104
1105 if (NILP (pos))
1106 {
1107 pos_byte = PT_BYTE;
1108 XSETFASTINT (pos, PT);
1109 }
1110
1111 if (MARKERP (pos))
1112 {
1113 pos_byte = marker_byte_position (pos);
1114
1115 if (pos_byte <= BEGV_BYTE || pos_byte > ZV_BYTE)
1116 return Qnil;
1117 }
1118 else
1119 {
1120 EMACS_INT p = fix_position (pos);
1121
1122 if (! (BEGV < p && p <= ZV))
1123 return Qnil;
1124
1125 pos_byte = CHAR_TO_BYTE (p);
1126 }
1127
1128 if (!NILP (BVAR (current_buffer, enable_multibyte_characters)))
1129 {
1130 pos_byte -= prev_char_len (pos_byte);
1131 XSETFASTINT (val, FETCH_CHAR (pos_byte));
1132 }
1133 else
1134 {
1135 pos_byte--;
1136 XSETFASTINT (val, FETCH_BYTE (pos_byte));
1137 }
1138 return val;
1139 }
1140
1141 DEFUN ("user-login-name", Fuser_login_name, Suser_login_name, 0, 1, 0,
1142 doc:
1143
1144
1145
1146
1147
1148 )
1149 (Lisp_Object uid)
1150 {
1151 struct passwd *pw;
1152 uid_t id;
1153
1154
1155
1156
1157 if (NILP (Vuser_login_name))
1158 init_editfns ();
1159
1160 if (NILP (uid))
1161 return Vuser_login_name;
1162
1163 CONS_TO_INTEGER (uid, uid_t, id);
1164 block_input ();
1165 pw = getpwuid (id);
1166 unblock_input ();
1167 return (pw ? build_string (pw->pw_name) : Qnil);
1168 }
1169
1170 DEFUN ("user-real-login-name", Fuser_real_login_name, Suser_real_login_name,
1171 0, 0, 0,
1172 doc:
1173
1174 )
1175 (void)
1176 {
1177
1178
1179
1180 if (NILP (Vuser_login_name))
1181 init_editfns ();
1182 return Vuser_real_login_name;
1183 }
1184
1185 DEFUN ("user-uid", Fuser_uid, Suser_uid, 0, 0, 0,
1186 doc: )
1187 (void)
1188 {
1189 uid_t euid = geteuid ();
1190 return INT_TO_INTEGER (euid);
1191 }
1192
1193 DEFUN ("user-real-uid", Fuser_real_uid, Suser_real_uid, 0, 0, 0,
1194 doc: )
1195 (void)
1196 {
1197 uid_t uid = getuid ();
1198 return INT_TO_INTEGER (uid);
1199 }
1200
1201 DEFUN ("group-name", Fgroup_name, Sgroup_name, 1, 1, 0,
1202 doc:
1203
1204 )
1205 (Lisp_Object gid)
1206 {
1207 struct group *gr;
1208 gid_t id;
1209
1210 if (!NUMBERP (gid) && !CONSP (gid))
1211 error ("Invalid GID specification");
1212 CONS_TO_INTEGER (gid, gid_t, id);
1213 block_input ();
1214 gr = getgrgid (id);
1215 unblock_input ();
1216 return gr ? build_string (gr->gr_name) : Qnil;
1217 }
1218
1219 DEFUN ("group-gid", Fgroup_gid, Sgroup_gid, 0, 0, 0,
1220 doc: )
1221 (void)
1222 {
1223 gid_t egid = getegid ();
1224 return INT_TO_INTEGER (egid);
1225 }
1226
1227 DEFUN ("group-real-gid", Fgroup_real_gid, Sgroup_real_gid, 0, 0, 0,
1228 doc: )
1229 (void)
1230 {
1231 gid_t gid = getgid ();
1232 return INT_TO_INTEGER (gid);
1233 }
1234
1235 DEFUN ("user-full-name", Fuser_full_name, Suser_full_name, 0, 1, 0,
1236 doc:
1237
1238
1239
1240
1241
1242
1243
1244
1245
1246
1247 )
1248 (Lisp_Object uid)
1249 {
1250 struct passwd *pw;
1251 register char *p, *q;
1252 Lisp_Object full;
1253
1254 if (NILP (uid))
1255 return Vuser_full_name;
1256 else if (NUMBERP (uid))
1257 {
1258 uid_t u;
1259 CONS_TO_INTEGER (uid, uid_t, u);
1260 block_input ();
1261 pw = getpwuid (u);
1262 unblock_input ();
1263 }
1264 else if (STRINGP (uid))
1265 {
1266 block_input ();
1267 pw = getpwnam (SSDATA (uid));
1268 unblock_input ();
1269 }
1270 else
1271 error ("Invalid UID specification");
1272
1273 if (!pw)
1274 return Qnil;
1275
1276 #if defined HAVE_ANDROID && !defined ANDROID_STUBIFY
1277 p = android_user_full_name (pw);
1278 #else
1279 p = USER_FULL_NAME;
1280 #endif
1281
1282
1283 q = strchr (p, ',');
1284 full = make_string (p, q ? q - p : strlen (p));
1285
1286 #ifdef AMPERSAND_FULL_NAME
1287 p = SSDATA (full);
1288 q = strchr (p, '&');
1289
1290 if (q)
1291 {
1292 Lisp_Object login = Fuser_login_name (INT_TO_INTEGER (pw->pw_uid));
1293 if (!NILP (login))
1294 {
1295 USE_SAFE_ALLOCA;
1296 char *r = SAFE_ALLOCA (strlen (p) + SBYTES (login) + 1);
1297 memcpy (r, p, q - p);
1298 char *s = lispstpcpy (&r[q - p], login);
1299 r[q - p] = upcase ((unsigned char) r[q - p]);
1300 strcpy (s, q + 1);
1301 full = build_string (r);
1302 SAFE_FREE ();
1303 }
1304 }
1305 #endif
1306
1307 return full;
1308 }
1309
1310 DEFUN ("system-name", Fsystem_name, Ssystem_name, 0, 0, 0,
1311 doc: )
1312 (void)
1313 {
1314 if (EQ (Vsystem_name, cached_system_name))
1315 init_and_cache_system_name ();
1316 return Vsystem_name;
1317 }
1318
1319 DEFUN ("emacs-pid", Femacs_pid, Semacs_pid, 0, 0, 0,
1320 doc: )
1321 (void)
1322 {
1323 pid_t pid = getpid ();
1324 return INT_TO_INTEGER (pid);
1325 }
1326
1327
1328
1329
1330
1331
1332
1333 static void
1334 general_insert_function (void (*insert_func)
1335 (const char *, ptrdiff_t),
1336 void (*insert_from_string_func)
1337 (Lisp_Object, ptrdiff_t, ptrdiff_t,
1338 ptrdiff_t, ptrdiff_t, bool),
1339 bool inherit, ptrdiff_t nargs, Lisp_Object *args)
1340 {
1341 ptrdiff_t argnum;
1342 Lisp_Object val;
1343
1344 for (argnum = 0; argnum < nargs; argnum++)
1345 {
1346 val = args[argnum];
1347 if (CHARACTERP (val))
1348 {
1349 int c = XFIXNAT (val);
1350 unsigned char str[MAX_MULTIBYTE_LENGTH];
1351 int len;
1352
1353 if (!NILP (BVAR (current_buffer, enable_multibyte_characters)))
1354 len = CHAR_STRING (c, str);
1355 else
1356 {
1357 str[0] = CHAR_TO_BYTE8 (c);
1358 len = 1;
1359 }
1360 (*insert_func) ((char *) str, len);
1361 }
1362 else if (STRINGP (val))
1363 {
1364 (*insert_from_string_func) (val, 0, 0,
1365 SCHARS (val),
1366 SBYTES (val),
1367 inherit);
1368 }
1369 else
1370 wrong_type_argument (Qchar_or_string_p, val);
1371 }
1372 }
1373
1374 void
1375 insert1 (Lisp_Object arg)
1376 {
1377 Finsert (1, &arg);
1378 }
1379
1380
1381 DEFUN ("insert", Finsert, Sinsert, 0, MANY, 0,
1382 doc:
1383
1384
1385
1386
1387
1388
1389
1390
1391
1392
1393
1394
1395
1396
1397 )
1398 (ptrdiff_t nargs, Lisp_Object *args)
1399 {
1400 general_insert_function (insert, insert_from_string, 0, nargs, args);
1401 return Qnil;
1402 }
1403
1404 DEFUN ("insert-and-inherit", Finsert_and_inherit, Sinsert_and_inherit,
1405 0, MANY, 0,
1406 doc:
1407
1408
1409
1410
1411
1412
1413
1414
1415
1416 )
1417 (ptrdiff_t nargs, Lisp_Object *args)
1418 {
1419 general_insert_function (insert_and_inherit, insert_from_string, 1,
1420 nargs, args);
1421 return Qnil;
1422 }
1423
1424 DEFUN ("insert-before-markers", Finsert_before_markers, Sinsert_before_markers, 0, MANY, 0,
1425 doc:
1426
1427
1428
1429
1430
1431
1432
1433
1434
1435
1436
1437 )
1438 (ptrdiff_t nargs, Lisp_Object *args)
1439 {
1440 general_insert_function (insert_before_markers,
1441 insert_from_string_before_markers, 0,
1442 nargs, args);
1443 return Qnil;
1444 }
1445
1446 DEFUN ("insert-before-markers-and-inherit", Finsert_and_inherit_before_markers,
1447 Sinsert_and_inherit_before_markers, 0, MANY, 0,
1448 doc:
1449
1450
1451
1452
1453
1454
1455
1456 )
1457 (ptrdiff_t nargs, Lisp_Object *args)
1458 {
1459 general_insert_function (insert_before_markers_and_inherit,
1460 insert_from_string_before_markers, 1,
1461 nargs, args);
1462 return Qnil;
1463 }
1464
1465 DEFUN ("insert-char", Finsert_char, Sinsert_char, 1, 3,
1466 "(list (read-char-by-name \"Insert character (Unicode name or hex): \")\
1467 (prefix-numeric-value current-prefix-arg)\
1468 t))",
1469 doc:
1470
1471
1472
1473
1474
1475
1476
1477
1478
1479
1480
1481
1482
1483
1484
1485
1486
1487
1488
1489
1490
1491
1492
1493 )
1494 (Lisp_Object character, Lisp_Object count, Lisp_Object inherit)
1495 {
1496 int i, stringlen;
1497 register ptrdiff_t n;
1498 int c, len;
1499 unsigned char str[MAX_MULTIBYTE_LENGTH];
1500 char string[4000];
1501
1502 CHECK_CHARACTER (character);
1503 if (NILP (count))
1504 XSETFASTINT (count, 1);
1505 else
1506 CHECK_FIXNUM (count);
1507 c = XFIXNAT (character);
1508
1509 if (!NILP (BVAR (current_buffer, enable_multibyte_characters)))
1510 len = CHAR_STRING (c, str);
1511 else
1512 str[0] = c, len = 1;
1513 if (XFIXNUM (count) <= 0)
1514 return Qnil;
1515 if (BUF_BYTES_MAX / len < XFIXNUM (count))
1516 buffer_overflow ();
1517 n = XFIXNUM (count) * len;
1518 stringlen = min (n, sizeof string - sizeof string % len);
1519 for (i = 0; i < stringlen; i++)
1520 string[i] = str[i % len];
1521 while (n > stringlen)
1522 {
1523 maybe_quit ();
1524 if (!NILP (inherit))
1525 insert_and_inherit (string, stringlen);
1526 else
1527 insert (string, stringlen);
1528 n -= stringlen;
1529 }
1530 if (!NILP (inherit))
1531 insert_and_inherit (string, n);
1532 else
1533 insert (string, n);
1534 return Qnil;
1535 }
1536
1537 DEFUN ("insert-byte", Finsert_byte, Sinsert_byte, 2, 3, 0,
1538 doc:
1539
1540
1541
1542
1543
1544
1545
1546
1547 )
1548 (Lisp_Object byte, Lisp_Object count, Lisp_Object inherit)
1549 {
1550 CHECK_FIXNUM (byte);
1551 if (XFIXNUM (byte) < 0 || XFIXNUM (byte) > 255)
1552 args_out_of_range_3 (byte, make_fixnum (0), make_fixnum (255));
1553 if (XFIXNUM (byte) >= 128
1554 && ! NILP (BVAR (current_buffer, enable_multibyte_characters)))
1555 XSETFASTINT (byte, BYTE8_TO_CHAR (XFIXNUM (byte)));
1556 return Finsert_char (byte, count, inherit);
1557 }
1558
1559
1560
1561
1562
1563
1564
1565
1566
1567
1568
1569
1570
1571
1572
1573
1574
1575 Lisp_Object
1576 make_buffer_string (ptrdiff_t start, ptrdiff_t end, bool props)
1577 {
1578 ptrdiff_t start_byte = CHAR_TO_BYTE (start);
1579 ptrdiff_t end_byte = CHAR_TO_BYTE (end);
1580
1581 return make_buffer_string_both (start, start_byte, end, end_byte, props);
1582 }
1583
1584
1585
1586
1587
1588
1589
1590
1591
1592
1593
1594
1595
1596
1597
1598
1599 Lisp_Object
1600 make_buffer_string_both (ptrdiff_t start, ptrdiff_t start_byte,
1601 ptrdiff_t end, ptrdiff_t end_byte, bool props)
1602 {
1603 Lisp_Object result, tem, tem1;
1604 ptrdiff_t beg0, end0, beg1, end1, size;
1605
1606 if (start_byte < GPT_BYTE && GPT_BYTE < end_byte)
1607 {
1608
1609 beg0 = start_byte;
1610 end0 = GPT_BYTE;
1611 beg1 = GPT_BYTE + GAP_SIZE - BEG_BYTE;
1612 end1 = end_byte + GAP_SIZE - BEG_BYTE;
1613 }
1614 else
1615 {
1616
1617 beg0 = start_byte;
1618 end0 = end_byte;
1619 beg1 = -1;
1620 end1 = -1;
1621 }
1622
1623 if (! NILP (BVAR (current_buffer, enable_multibyte_characters)))
1624 result = make_uninit_multibyte_string (end - start, end_byte - start_byte);
1625 else
1626 result = make_uninit_string (end - start);
1627
1628 size = end0 - beg0;
1629 memcpy (SDATA (result), BYTE_POS_ADDR (beg0), size);
1630 if (beg1 != -1)
1631 memcpy (SDATA (result) + size, BEG_ADDR + beg1, end1 - beg1);
1632
1633
1634 if (props)
1635 {
1636 update_buffer_properties (start, end);
1637
1638 tem = Fnext_property_change (make_fixnum (start), Qnil, make_fixnum (end));
1639 tem1 = Ftext_properties_at (make_fixnum (start), Qnil);
1640
1641 if (XFIXNUM (tem) != end || !NILP (tem1))
1642 copy_intervals_to_string (result, current_buffer, start,
1643 end - start);
1644 }
1645
1646 return result;
1647 }
1648
1649
1650
1651
1652 static void
1653 update_buffer_properties (ptrdiff_t start, ptrdiff_t end)
1654 {
1655
1656
1657 if (!NILP (Vbuffer_access_fontify_functions))
1658 {
1659
1660
1661 if (!NILP (Vbuffer_access_fontified_property))
1662 {
1663 Lisp_Object tem
1664 = Ftext_property_any (make_fixnum (start), make_fixnum (end),
1665 Vbuffer_access_fontified_property,
1666 Qnil, Qnil);
1667 if (NILP (tem))
1668 return;
1669 }
1670
1671 CALLN (Frun_hook_with_args, Qbuffer_access_fontify_functions,
1672 make_fixnum (start), make_fixnum (end));
1673 }
1674 }
1675
1676 DEFUN ("buffer-substring", Fbuffer_substring, Sbuffer_substring, 2, 2, 0,
1677 doc:
1678
1679
1680
1681
1682
1683
1684 )
1685 (Lisp_Object start, Lisp_Object end)
1686 {
1687 register ptrdiff_t b, e;
1688
1689 validate_region (&start, &end);
1690 b = XFIXNUM (start);
1691 e = XFIXNUM (end);
1692
1693 return make_buffer_string (b, e, 1);
1694 }
1695
1696 DEFUN ("buffer-substring-no-properties", Fbuffer_substring_no_properties,
1697 Sbuffer_substring_no_properties, 2, 2, 0,
1698 doc:
1699
1700 )
1701 (Lisp_Object start, Lisp_Object end)
1702 {
1703 register ptrdiff_t b, e;
1704
1705 validate_region (&start, &end);
1706 b = XFIXNUM (start);
1707 e = XFIXNUM (end);
1708
1709 return make_buffer_string (b, e, 0);
1710 }
1711
1712 DEFUN ("buffer-string", Fbuffer_string, Sbuffer_string, 0, 0, 0,
1713 doc:
1714
1715
1716
1717
1718
1719 )
1720 (void)
1721 {
1722 return make_buffer_string_both (BEGV, BEGV_BYTE, ZV, ZV_BYTE, 1);
1723 }
1724
1725 DEFUN ("insert-buffer-substring", Finsert_buffer_substring, Sinsert_buffer_substring,
1726 1, 3, 0,
1727 doc:
1728
1729
1730
1731
1732
1733
1734
1735
1736
1737
1738 )
1739 (Lisp_Object buffer, Lisp_Object start, Lisp_Object end)
1740 {
1741 register EMACS_INT b, e, temp;
1742 register struct buffer *bp, *obuf;
1743 Lisp_Object buf;
1744
1745 buf = Fget_buffer (buffer);
1746 if (NILP (buf))
1747 nsberror (buffer);
1748 bp = XBUFFER (buf);
1749 if (!BUFFER_LIVE_P (bp))
1750 error ("Selecting deleted buffer");
1751
1752 b = !NILP (start) ? fix_position (start) : BUF_BEGV (bp);
1753 e = !NILP (end) ? fix_position (end) : BUF_ZV (bp);
1754 if (b > e)
1755 temp = b, b = e, e = temp;
1756
1757 if (!(BUF_BEGV (bp) <= b && e <= BUF_ZV (bp)))
1758 args_out_of_range (start, end);
1759
1760 obuf = current_buffer;
1761 set_buffer_internal_1 (bp);
1762 update_buffer_properties (b, e);
1763 set_buffer_internal_1 (obuf);
1764
1765 insert_from_buffer (bp, b, e - b, 0);
1766 return Qnil;
1767 }
1768
1769 DEFUN ("compare-buffer-substrings", Fcompare_buffer_substrings, Scompare_buffer_substrings,
1770 6, 6, 0,
1771 doc:
1772
1773
1774
1775
1776
1777
1778
1779
1780
1781 )
1782 (Lisp_Object buffer1, Lisp_Object start1, Lisp_Object end1, Lisp_Object buffer2, Lisp_Object start2, Lisp_Object end2)
1783 {
1784 register EMACS_INT begp1, endp1, begp2, endp2, temp;
1785 register struct buffer *bp1, *bp2;
1786 register Lisp_Object trt
1787 = (!NILP (BVAR (current_buffer, case_fold_search))
1788 ? BVAR (current_buffer, case_canon_table) : Qnil);
1789 ptrdiff_t chars = 0;
1790 ptrdiff_t i1, i2, i1_byte, i2_byte;
1791
1792
1793
1794 if (NILP (buffer1))
1795 bp1 = current_buffer;
1796 else
1797 {
1798 Lisp_Object buf1;
1799 buf1 = Fget_buffer (buffer1);
1800 if (NILP (buf1))
1801 nsberror (buffer1);
1802 bp1 = XBUFFER (buf1);
1803 if (!BUFFER_LIVE_P (bp1))
1804 error ("Selecting deleted buffer");
1805 }
1806
1807 begp1 = !NILP (start1) ? fix_position (start1) : BUF_BEGV (bp1);
1808 endp1 = !NILP (end1) ? fix_position (end1) : BUF_ZV (bp1);
1809 if (begp1 > endp1)
1810 temp = begp1, begp1 = endp1, endp1 = temp;
1811
1812 if (!(BUF_BEGV (bp1) <= begp1
1813 && begp1 <= endp1
1814 && endp1 <= BUF_ZV (bp1)))
1815 args_out_of_range (start1, end1);
1816
1817
1818
1819 if (NILP (buffer2))
1820 bp2 = current_buffer;
1821 else
1822 {
1823 Lisp_Object buf2;
1824 buf2 = Fget_buffer (buffer2);
1825 if (NILP (buf2))
1826 nsberror (buffer2);
1827 bp2 = XBUFFER (buf2);
1828 if (!BUFFER_LIVE_P (bp2))
1829 error ("Selecting deleted buffer");
1830 }
1831
1832 begp2 = !NILP (start2) ? fix_position (start2) : BUF_BEGV (bp2);
1833 endp2 = !NILP (end2) ? fix_position (end2) : BUF_ZV (bp2);
1834 if (begp2 > endp2)
1835 temp = begp2, begp2 = endp2, endp2 = temp;
1836
1837 if (!(BUF_BEGV (bp2) <= begp2
1838 && begp2 <= endp2
1839 && endp2 <= BUF_ZV (bp2)))
1840 args_out_of_range (start2, end2);
1841
1842 i1 = begp1;
1843 i2 = begp2;
1844 i1_byte = buf_charpos_to_bytepos (bp1, i1);
1845 i2_byte = buf_charpos_to_bytepos (bp2, i2);
1846
1847 while (i1 < endp1 && i2 < endp2)
1848 {
1849
1850
1851 int c1, c2;
1852
1853 if (! NILP (BVAR (bp1, enable_multibyte_characters)))
1854 {
1855 c1 = BUF_FETCH_MULTIBYTE_CHAR (bp1, i1_byte);
1856 i1_byte += buf_next_char_len (bp1, i1_byte);
1857 i1++;
1858 }
1859 else
1860 {
1861 c1 = make_char_multibyte (BUF_FETCH_BYTE (bp1, i1));
1862 i1++;
1863 }
1864
1865 if (! NILP (BVAR (bp2, enable_multibyte_characters)))
1866 {
1867 c2 = BUF_FETCH_MULTIBYTE_CHAR (bp2, i2_byte);
1868 i2_byte += buf_next_char_len (bp2, i2_byte);
1869 i2++;
1870 }
1871 else
1872 {
1873 c2 = make_char_multibyte (BUF_FETCH_BYTE (bp2, i2));
1874 i2++;
1875 }
1876
1877 if (!NILP (trt))
1878 {
1879 c1 = char_table_translate (trt, c1);
1880 c2 = char_table_translate (trt, c2);
1881 }
1882
1883 if (c1 != c2)
1884 return make_fixnum (c1 < c2 ? -1 - chars : chars + 1);
1885
1886 chars++;
1887 rarely_quit (chars);
1888 }
1889
1890
1891
1892 if (chars < endp1 - begp1)
1893 return make_fixnum (chars + 1);
1894 else if (chars < endp2 - begp2)
1895 return make_fixnum (- chars - 1);
1896
1897
1898 return make_fixnum (0);
1899 }
1900
1901
1902
1903
1904
1905 #undef ELEMENT
1906 #undef EQUAL
1907 #define USE_HEURISTIC
1908
1909 #define XVECREF_YVECREF_EQUAL(ctx, xoff, yoff) \
1910 buffer_chars_equal ((ctx), (xoff), (yoff))
1911
1912 #define OFFSET ptrdiff_t
1913
1914 #define EXTRA_CONTEXT_FIELDS \
1915 \
1916 struct buffer *buffer_a; \
1917 struct buffer *buffer_b; \
1918 \
1919 ptrdiff_t beg_a; \
1920 ptrdiff_t beg_b; \
1921 \
1922 bool a_unibyte; \
1923 bool b_unibyte; \
1924
1925 \
1926 unsigned char *deletions; \
1927 unsigned char *insertions; \
1928 struct timespec time_limit; \
1929 sys_jmp_buf jmp; \
1930 unsigned short quitcounter;
1931
1932 #define NOTE_DELETE(ctx, xoff) set_bit ((ctx)->deletions, xoff)
1933 #define NOTE_INSERT(ctx, yoff) set_bit ((ctx)->insertions, yoff)
1934 #define EARLY_ABORT(ctx) compareseq_early_abort (ctx)
1935
1936 struct context;
1937 static void set_bit (unsigned char *, OFFSET);
1938 static bool bit_is_set (const unsigned char *, OFFSET);
1939 static bool buffer_chars_equal (struct context *, OFFSET, OFFSET);
1940 static bool compareseq_early_abort (struct context *);
1941
1942 #include "minmax.h"
1943 #include "diffseq.h"
1944
1945 DEFUN ("replace-buffer-contents", Freplace_buffer_contents,
1946 Sreplace_buffer_contents, 1, 3, "bSource buffer: ",
1947 doc:
1948
1949
1950
1951
1952
1953
1954
1955
1956
1957
1958
1959
1960
1961
1962
1963
1964
1965
1966
1967
1968
1969
1970
1971
1972
1973 )
1974 (Lisp_Object source, Lisp_Object max_secs, Lisp_Object max_costs)
1975 {
1976 struct buffer *a = current_buffer;
1977 Lisp_Object source_buffer = Fget_buffer (source);
1978 if (NILP (source_buffer))
1979 nsberror (source);
1980 struct buffer *b = XBUFFER (source_buffer);
1981 if (! BUFFER_LIVE_P (b))
1982 error ("Selecting deleted buffer");
1983 if (a == b)
1984 error ("Cannot replace a buffer with itself");
1985
1986 ptrdiff_t too_expensive;
1987 if (NILP (max_costs))
1988 too_expensive = 1000000;
1989 else if (FIXNUMP (max_costs))
1990 too_expensive = clip_to_bounds (0, XFIXNUM (max_costs), PTRDIFF_MAX);
1991 else
1992 {
1993 CHECK_INTEGER (max_costs);
1994 too_expensive = NILP (Fnatnump (max_costs)) ? 0 : PTRDIFF_MAX;
1995 }
1996
1997 struct timespec time_limit = make_timespec (0, -1);
1998 if (!NILP (max_secs))
1999 {
2000 struct timespec
2001 tlim = timespec_add (current_timespec (),
2002 lisp_time_argument (max_secs)),
2003 tmax = make_timespec (TYPE_MAXIMUM (time_t), TIMESPEC_HZ - 1);
2004 if (timespec_cmp (tlim, tmax) < 0)
2005 time_limit = tlim;
2006 }
2007
2008 ptrdiff_t min_a = BEGV;
2009 ptrdiff_t min_b = BUF_BEGV (b);
2010 ptrdiff_t size_a = ZV - min_a;
2011 ptrdiff_t size_b = BUF_ZV (b) - min_b;
2012 eassume (size_a >= 0);
2013 eassume (size_b >= 0);
2014 bool a_empty = size_a == 0;
2015 bool b_empty = size_b == 0;
2016
2017
2018
2019
2020 if (a_empty && b_empty)
2021 return Qt;
2022
2023 if (a_empty)
2024 {
2025 Finsert_buffer_substring (source, Qnil, Qnil);
2026 return Qt;
2027 }
2028
2029 if (b_empty)
2030 {
2031 del_range_both (BEGV, BEGV_BYTE, ZV, ZV_BYTE, true);
2032 return Qt;
2033 }
2034
2035 specpdl_ref count = SPECPDL_INDEX ();
2036
2037
2038 ptrdiff_t diags = size_a + size_b + 3;
2039 ptrdiff_t del_bytes = size_a / CHAR_BIT + 1;
2040 ptrdiff_t ins_bytes = size_b / CHAR_BIT + 1;
2041 ptrdiff_t *buffer;
2042 ptrdiff_t bytes_needed;
2043 if (ckd_mul (&bytes_needed, diags, 2 * sizeof *buffer)
2044 || ckd_add (&bytes_needed, bytes_needed, del_bytes + ins_bytes))
2045 memory_full (SIZE_MAX);
2046 USE_SAFE_ALLOCA;
2047 buffer = SAFE_ALLOCA (bytes_needed);
2048 unsigned char *deletions_insertions = memset (buffer + 2 * diags, 0,
2049 del_bytes + ins_bytes);
2050
2051
2052
2053
2054
2055 struct context ctx = {
2056 .buffer_a = a,
2057 .buffer_b = b,
2058 .beg_a = min_a,
2059 .beg_b = min_b,
2060 .a_unibyte = BUF_ZV (a) == BUF_ZV_BYTE (a),
2061 .b_unibyte = BUF_ZV (b) == BUF_ZV_BYTE (b),
2062 .deletions = deletions_insertions,
2063 .insertions = deletions_insertions + del_bytes,
2064 .fdiag = buffer + size_b + 1,
2065 .bdiag = buffer + diags + size_b + 1,
2066 .heuristic = true,
2067 .too_expensive = too_expensive,
2068 .time_limit = time_limit,
2069 };
2070
2071
2072
2073 bool early_abort;
2074 if (! sys_setjmp (ctx.jmp))
2075 early_abort = compareseq (0, size_a, 0, size_b, false, &ctx);
2076 else
2077 early_abort = true;
2078
2079 if (early_abort)
2080 {
2081 del_range (min_a, ZV);
2082 Finsert_buffer_substring (source, Qnil,Qnil);
2083 SAFE_FREE_UNBIND_TO (count, Qnil);
2084 return Qnil;
2085 }
2086
2087 Fundo_boundary ();
2088 bool modification_hooks_inhibited = false;
2089 record_unwind_protect_excursion ();
2090
2091
2092
2093
2094
2095
2096 if (!inhibit_modification_hooks)
2097 {
2098 prepare_to_modify_buffer (BEGV, ZV, NULL);
2099 specbind (Qinhibit_modification_hooks, Qt);
2100 modification_hooks_inhibited = true;
2101 }
2102
2103 ptrdiff_t i = size_a;
2104 ptrdiff_t j = size_b;
2105
2106
2107
2108 while (i >= 0 || j >= 0)
2109 {
2110 rarely_quit (++ctx.quitcounter);
2111
2112
2113
2114 if ((i > 0 && bit_is_set (ctx.deletions, i - 1))
2115 || (j > 0 && bit_is_set (ctx.insertions, j - 1)))
2116 {
2117 ptrdiff_t end_a = min_a + i;
2118 ptrdiff_t end_b = min_b + j;
2119
2120 while (i > 0 && bit_is_set (ctx.deletions, i - 1))
2121 --i;
2122 while (j > 0 && bit_is_set (ctx.insertions, j - 1))
2123 --j;
2124
2125 ptrdiff_t beg_a = min_a + i;
2126 ptrdiff_t beg_b = min_b + j;
2127 eassert (beg_a <= end_a);
2128 eassert (beg_b <= end_b);
2129 eassert (beg_a < end_a || beg_b < end_b);
2130 if (beg_a < end_a)
2131 del_range (beg_a, end_a);
2132 if (beg_b < end_b)
2133 {
2134 SET_PT (beg_a);
2135 Finsert_buffer_substring (source, make_fixed_natnum (beg_b),
2136 make_fixed_natnum (end_b));
2137 }
2138 }
2139 --i;
2140 --j;
2141 }
2142
2143 SAFE_FREE_UNBIND_TO (count, Qnil);
2144
2145 if (modification_hooks_inhibited)
2146 {
2147 signal_after_change (BEGV, size_a, ZV - BEGV);
2148 update_compositions (BEGV, ZV, CHECK_INSIDE);
2149
2150
2151
2152
2153 if (SAVE_MODIFF == MODIFF
2154 && STRINGP (BVAR (a, file_truename)))
2155 Funlock_file (BVAR (a, file_truename));
2156 }
2157
2158 return Qt;
2159 }
2160
2161 static void
2162 set_bit (unsigned char *a, ptrdiff_t i)
2163 {
2164 eassume (0 <= i);
2165 a[i / CHAR_BIT] |= (1 << (i % CHAR_BIT));
2166 }
2167
2168 static bool
2169 bit_is_set (const unsigned char *a, ptrdiff_t i)
2170 {
2171 eassume (0 <= i);
2172 return a[i / CHAR_BIT] & (1 << (i % CHAR_BIT));
2173 }
2174
2175
2176
2177
2178
2179
2180
2181
2182
2183
2184
2185
2186
2187 static bool
2188 buffer_chars_equal (struct context *ctx,
2189 ptrdiff_t pos_a, ptrdiff_t pos_b)
2190 {
2191 if (!++ctx->quitcounter)
2192 {
2193 maybe_quit ();
2194 if (compareseq_early_abort (ctx))
2195 sys_longjmp (ctx->jmp, 1);
2196 }
2197
2198 pos_a += ctx->beg_a;
2199 pos_b += ctx->beg_b;
2200
2201 ptrdiff_t bpos_a =
2202 ctx->a_unibyte ? pos_a : buf_charpos_to_bytepos (ctx->buffer_a, pos_a);
2203 ptrdiff_t bpos_b =
2204 ctx->b_unibyte ? pos_b : buf_charpos_to_bytepos (ctx->buffer_b, pos_b);
2205
2206
2207
2208
2209
2210 if (ctx->a_unibyte && ctx->b_unibyte)
2211 return BUF_FETCH_BYTE (ctx->buffer_a, bpos_a)
2212 == BUF_FETCH_BYTE (ctx->buffer_b, bpos_b);
2213 if (ctx->a_unibyte && !ctx->b_unibyte)
2214 return UNIBYTE_TO_CHAR (BUF_FETCH_BYTE (ctx->buffer_a, bpos_a))
2215 == BUF_FETCH_MULTIBYTE_CHAR (ctx->buffer_b, bpos_b);
2216 if (!ctx->a_unibyte && ctx->b_unibyte)
2217 return BUF_FETCH_MULTIBYTE_CHAR (ctx->buffer_a, bpos_a)
2218 == UNIBYTE_TO_CHAR (BUF_FETCH_BYTE (ctx->buffer_b, bpos_b));
2219 return BUF_FETCH_MULTIBYTE_CHAR (ctx->buffer_a, bpos_a)
2220 == BUF_FETCH_MULTIBYTE_CHAR (ctx->buffer_b, bpos_b);
2221 }
2222
2223 static bool
2224 compareseq_early_abort (struct context *ctx)
2225 {
2226 if (ctx->time_limit.tv_nsec < 0)
2227 return false;
2228 return timespec_cmp (ctx->time_limit, current_timespec ()) < 0;
2229 }
2230
2231
2232 static void
2233 subst_char_in_region_unwind (Lisp_Object arg)
2234 {
2235 bset_undo_list (current_buffer, arg);
2236 }
2237
2238 static void
2239 subst_char_in_region_unwind_1 (Lisp_Object arg)
2240 {
2241 bset_filename (current_buffer, arg);
2242 }
2243
2244 DEFUN ("subst-char-in-region", Fsubst_char_in_region,
2245 Ssubst_char_in_region, 4, 5, 0,
2246 doc:
2247
2248
2249 )
2250 (Lisp_Object start, Lisp_Object end, Lisp_Object fromchar, Lisp_Object tochar, Lisp_Object noundo)
2251 {
2252 register ptrdiff_t pos, pos_byte, stop, i, len, end_byte;
2253
2254
2255
2256
2257 ptrdiff_t changed = 0;
2258 unsigned char fromstr[MAX_MULTIBYTE_LENGTH], tostr[MAX_MULTIBYTE_LENGTH];
2259 unsigned char *p;
2260 specpdl_ref count = SPECPDL_INDEX ();
2261 #define COMBINING_NO 0
2262 #define COMBINING_BEFORE 1
2263 #define COMBINING_AFTER 2
2264 #define COMBINING_BOTH (COMBINING_BEFORE | COMBINING_AFTER)
2265 int maybe_byte_combining = COMBINING_NO;
2266 ptrdiff_t last_changed = 0;
2267 bool multibyte_p
2268 = !NILP (BVAR (current_buffer, enable_multibyte_characters));
2269 int fromc, toc;
2270
2271 restart:
2272
2273 validate_region (&start, &end);
2274 CHECK_CHARACTER (fromchar);
2275 CHECK_CHARACTER (tochar);
2276 fromc = XFIXNAT (fromchar);
2277 toc = XFIXNAT (tochar);
2278
2279 if (multibyte_p)
2280 {
2281 len = CHAR_STRING (fromc, fromstr);
2282 if (CHAR_STRING (toc, tostr) != len)
2283 error ("Characters in `subst-char-in-region' have different byte-lengths");
2284 if (!ASCII_CHAR_P (*tostr))
2285 {
2286
2287
2288
2289
2290 if (!CHAR_HEAD_P (*tostr))
2291 maybe_byte_combining = COMBINING_BOTH;
2292 else if (BYTES_BY_CHAR_HEAD (*tostr) > len)
2293 maybe_byte_combining = COMBINING_AFTER;
2294 }
2295 }
2296 else
2297 {
2298 len = 1;
2299 fromstr[0] = fromc;
2300 tostr[0] = toc;
2301 }
2302
2303 pos = XFIXNUM (start);
2304 pos_byte = CHAR_TO_BYTE (pos);
2305 stop = CHAR_TO_BYTE (XFIXNUM (end));
2306 end_byte = stop;
2307
2308
2309
2310
2311
2312 if (!changed && !NILP (noundo))
2313 {
2314 record_unwind_protect (subst_char_in_region_unwind,
2315 BVAR (current_buffer, undo_list));
2316 bset_undo_list (current_buffer, Qt);
2317
2318 record_unwind_protect (subst_char_in_region_unwind_1,
2319 BVAR (current_buffer, filename));
2320 bset_filename (current_buffer, Qnil);
2321 }
2322
2323 if (pos_byte < GPT_BYTE)
2324 stop = min (stop, GPT_BYTE);
2325 while (1)
2326 {
2327 ptrdiff_t pos_byte_next = pos_byte;
2328
2329 if (pos_byte >= stop)
2330 {
2331 if (pos_byte >= end_byte) break;
2332 stop = end_byte;
2333 }
2334 p = BYTE_POS_ADDR (pos_byte);
2335 if (multibyte_p)
2336 pos_byte_next += next_char_len (pos_byte_next);
2337 else
2338 ++pos_byte_next;
2339 if (pos_byte_next - pos_byte == len
2340 && p[0] == fromstr[0]
2341 && (len == 1
2342 || (p[1] == fromstr[1]
2343 && (len == 2 || (p[2] == fromstr[2]
2344 && (len == 3 || p[3] == fromstr[3]))))))
2345 {
2346 if (changed < 0)
2347
2348
2349 changed = pos;
2350 else if (!changed)
2351 {
2352 changed = -1;
2353 modify_text (pos, XFIXNUM (end));
2354
2355 if (! NILP (noundo))
2356 {
2357 modiff_count m = MODIFF;
2358 if (SAVE_MODIFF == m - 1)
2359 SAVE_MODIFF = m;
2360 if (BUF_AUTOSAVE_MODIFF (current_buffer) == m - 1)
2361 BUF_AUTOSAVE_MODIFF (current_buffer) = m;
2362 }
2363
2364
2365
2366 goto restart;
2367 }
2368
2369
2370
2371 if (maybe_byte_combining
2372 && (maybe_byte_combining == COMBINING_AFTER
2373 ? (pos_byte_next < Z_BYTE
2374 && ! CHAR_HEAD_P (FETCH_BYTE (pos_byte_next)))
2375 : ((pos_byte_next < Z_BYTE
2376 && ! CHAR_HEAD_P (FETCH_BYTE (pos_byte_next)))
2377 || (pos_byte > BEG_BYTE
2378 && ! ASCII_CHAR_P (FETCH_BYTE (pos_byte - 1))))))
2379 {
2380 Lisp_Object tem, string;
2381
2382 tem = BVAR (current_buffer, undo_list);
2383
2384
2385 string = make_multibyte_string ((char *) tostr, 1, len);
2386
2387
2388 replace_range (pos, pos + 1, string,
2389 false, false, true, false, false);
2390 pos_byte_next = CHAR_TO_BYTE (pos);
2391 if (pos_byte_next > pos_byte)
2392
2393
2394
2395 pos--;
2396 else
2397 pos_byte_next += next_char_len (pos_byte_next);
2398
2399 if (! NILP (noundo))
2400 bset_undo_list (current_buffer, tem);
2401 }
2402 else
2403 {
2404 if (NILP (noundo))
2405 record_change (pos, 1);
2406 for (i = 0; i < len; i++) *p++ = tostr[i];
2407
2408 #ifdef HAVE_TREE_SITTER
2409
2410
2411
2412
2413 treesit_record_change (pos_byte, pos_byte + len, pos_byte + len);
2414 #endif
2415 }
2416 last_changed = pos + 1;
2417 }
2418 pos_byte = pos_byte_next;
2419 pos++;
2420 }
2421
2422 if (changed > 0)
2423 {
2424 signal_after_change (changed,
2425 last_changed - changed, last_changed - changed);
2426 update_compositions (changed, last_changed, CHECK_ALL);
2427 }
2428
2429 return unbind_to (count, Qnil);
2430 }
2431
2432
2433 static Lisp_Object check_translation (ptrdiff_t, ptrdiff_t, ptrdiff_t,
2434 Lisp_Object);
2435
2436
2437
2438
2439
2440
2441
2442 static Lisp_Object
2443 check_translation (ptrdiff_t pos, ptrdiff_t pos_byte, ptrdiff_t end,
2444 Lisp_Object val)
2445 {
2446 int initial_buf[16];
2447 int *buf = initial_buf;
2448 ptrdiff_t buf_size = ARRAYELTS (initial_buf);
2449 int *bufalloc = 0;
2450 ptrdiff_t buf_used = 0;
2451 Lisp_Object result = Qnil;
2452
2453 for (; CONSP (val); val = XCDR (val))
2454 {
2455 Lisp_Object elt;
2456 ptrdiff_t len, i;
2457
2458 elt = XCAR (val);
2459 if (! CONSP (elt))
2460 continue;
2461 elt = XCAR (elt);
2462 if (! VECTORP (elt))
2463 continue;
2464 len = ASIZE (elt);
2465 if (len <= end - pos)
2466 {
2467 for (i = 0; i < len; i++)
2468 {
2469 if (buf_used <= i)
2470 {
2471 unsigned char *p = BYTE_POS_ADDR (pos_byte);
2472 int len1;
2473
2474 if (buf_used == buf_size)
2475 {
2476 bufalloc = xpalloc (bufalloc, &buf_size, 1, -1,
2477 sizeof *bufalloc);
2478 if (buf == initial_buf)
2479 memcpy (bufalloc, buf, sizeof initial_buf);
2480 buf = bufalloc;
2481 }
2482 buf[buf_used++] = string_char_and_length (p, &len1);
2483 pos_byte += len1;
2484 }
2485 if (XFIXNUM (AREF (elt, i)) != buf[i])
2486 break;
2487 }
2488 if (i == len)
2489 {
2490 result = XCAR (val);
2491 break;
2492 }
2493 }
2494 }
2495
2496 xfree (bufalloc);
2497 return result;
2498 }
2499
2500
2501 DEFUN ("translate-region-internal", Ftranslate_region_internal,
2502 Stranslate_region_internal, 3, 3, 0,
2503 doc:
2504
2505
2506
2507 )
2508 (Lisp_Object start, Lisp_Object end, Lisp_Object table)
2509 {
2510 int translatable_chars = MAX_CHAR + 1;
2511 bool multibyte = !NILP (BVAR (current_buffer, enable_multibyte_characters));
2512 bool string_multibyte UNINIT;
2513
2514 validate_region (&start, &end);
2515 if (STRINGP (table))
2516 {
2517 if (! multibyte)
2518 table = string_make_unibyte (table);
2519 translatable_chars = min (translatable_chars, SBYTES (table));
2520 string_multibyte = STRING_MULTIBYTE (table);
2521 }
2522 else if (! (CHAR_TABLE_P (table)
2523 && EQ (XCHAR_TABLE (table)->purpose, Qtranslation_table)))
2524 error ("Not a translation table");
2525
2526 ptrdiff_t pos = XFIXNUM (start);
2527 ptrdiff_t pos_byte = CHAR_TO_BYTE (pos);
2528 ptrdiff_t end_pos = XFIXNUM (end);
2529 modify_text (pos, end_pos);
2530
2531 ptrdiff_t characters_changed = 0;
2532
2533 while (pos < end_pos)
2534 {
2535 unsigned char *p = BYTE_POS_ADDR (pos_byte);
2536 unsigned char *str UNINIT;
2537 unsigned char buf[MAX_MULTIBYTE_LENGTH];
2538 int len, oc;
2539
2540 if (multibyte)
2541 oc = string_char_and_length (p, &len);
2542 else
2543 oc = *p, len = 1;
2544 if (oc < translatable_chars)
2545 {
2546 int nc;
2547 int str_len UNINIT;
2548 Lisp_Object val;
2549
2550 if (STRINGP (table))
2551 {
2552
2553 unsigned char *tt = SDATA (table);
2554
2555 if (string_multibyte)
2556 {
2557 str = tt + string_char_to_byte (table, oc);
2558 nc = string_char_and_length (str, &str_len);
2559 }
2560 else
2561 {
2562 nc = tt[oc];
2563 if (! ASCII_CHAR_P (nc) && multibyte)
2564 {
2565 str_len = BYTE8_STRING (nc, buf);
2566 str = buf;
2567 }
2568 else
2569 {
2570 str_len = 1;
2571 str = tt + oc;
2572 }
2573 }
2574 }
2575 else
2576 {
2577 nc = oc;
2578 val = CHAR_TABLE_REF (table, oc);
2579 if (CHARACTERP (val))
2580 {
2581 nc = XFIXNAT (val);
2582 str_len = CHAR_STRING (nc, buf);
2583 str = buf;
2584 }
2585 else if (VECTORP (val) || (CONSP (val)))
2586 {
2587
2588
2589 nc = -1;
2590 }
2591 }
2592
2593 if (nc != oc && nc >= 0)
2594 {
2595
2596 if (len != str_len)
2597 {
2598 Lisp_Object string;
2599
2600
2601
2602 string = make_multibyte_string ((char *) str, 1, str_len);
2603 replace_range (pos, pos + 1, string,
2604 true, false, true, false, false);
2605 len = str_len;
2606 }
2607 else
2608 {
2609 record_change (pos, 1);
2610 while (str_len-- > 0)
2611 *p++ = *str++;
2612 signal_after_change (pos, 1, 1);
2613 update_compositions (pos, pos + 1, CHECK_BORDER);
2614
2615 #ifdef HAVE_TREE_SITTER
2616
2617
2618
2619
2620 treesit_record_change (pos_byte, pos_byte + len,
2621 pos_byte + len);
2622 #endif
2623 }
2624 characters_changed++;
2625 }
2626 else if (nc < 0)
2627 {
2628 if (CONSP (val))
2629 {
2630 val = check_translation (pos, pos_byte, end_pos, val);
2631 if (NILP (val))
2632 {
2633 pos_byte += len;
2634 pos++;
2635 continue;
2636 }
2637
2638 len = ASIZE (XCAR (val));
2639 val = XCDR (val);
2640 }
2641 else
2642 len = 1;
2643
2644 Lisp_Object string
2645 = (VECTORP (val)
2646 ? Fconcat (1, &val)
2647 : Fmake_string (make_fixnum (1), val, Qnil));
2648 replace_range (pos, pos + len, string, true, false, true, false,
2649 false);
2650 pos_byte += SBYTES (string);
2651 pos += SCHARS (string);
2652 characters_changed += SCHARS (string);
2653 end_pos += SCHARS (string) - len;
2654 continue;
2655 }
2656 }
2657 pos_byte += len;
2658 pos++;
2659 }
2660
2661 return make_fixnum (characters_changed);
2662 }
2663
2664 DEFUN ("delete-region", Fdelete_region, Sdelete_region, 2, 2, "r",
2665 doc:
2666
2667 )
2668 (Lisp_Object start, Lisp_Object end)
2669 {
2670 validate_region (&start, &end);
2671 del_range (XFIXNUM (start), XFIXNUM (end));
2672 return Qnil;
2673 }
2674
2675 DEFUN ("delete-and-extract-region", Fdelete_and_extract_region,
2676 Sdelete_and_extract_region, 2, 2, 0,
2677 doc: )
2678 (Lisp_Object start, Lisp_Object end)
2679 {
2680 validate_region (&start, &end);
2681 if (XFIXNUM (start) == XFIXNUM (end))
2682 return empty_unibyte_string;
2683 return del_range_1 (XFIXNUM (start), XFIXNUM (end), 1, 1);
2684 }
2685
2686
2687
2688
2689
2690
2691
2692
2693
2694
2695
2696
2697
2698
2699 static Lisp_Object labeled_restrictions;
2700
2701
2702
2703 static void
2704 labeled_restrictions_add (Lisp_Object buf, Lisp_Object restrictions)
2705 {
2706 labeled_restrictions = nconc2 (list1 (list2 (buf, restrictions)),
2707 labeled_restrictions);
2708 }
2709
2710
2711
2712
2713 static void
2714 labeled_restrictions_remove (Lisp_Object buf)
2715 {
2716 labeled_restrictions = Fdelq (Fassoc (buf, labeled_restrictions, Qnil),
2717 labeled_restrictions);
2718 }
2719
2720
2721
2722
2723
2724
2725
2726 static Lisp_Object
2727 labeled_restrictions_get_bound (Lisp_Object buf, bool begv, bool outermost)
2728 {
2729 if (NILP (Fbuffer_live_p (buf)))
2730 return Qnil;
2731 Lisp_Object restrictions = assq_no_quit (buf, labeled_restrictions);
2732 if (NILP (restrictions))
2733 return Qnil;
2734 restrictions = XCAR (XCDR (restrictions));
2735 Lisp_Object bounds
2736 = outermost
2737 ? XCDR (assq_no_quit (Qoutermost_restriction, restrictions))
2738 : XCDR (XCAR (restrictions));
2739 eassert (! NILP (bounds));
2740 Lisp_Object marker = begv ? XCAR (bounds) : XCAR (XCDR (bounds));
2741 eassert (EQ (Fmarker_buffer (marker), buf));
2742 return marker;
2743 }
2744
2745
2746
2747
2748 static Lisp_Object
2749 labeled_restrictions_peek_label (Lisp_Object buf)
2750 {
2751 if (NILP (Fbuffer_live_p (buf)))
2752 return Qnil;
2753 Lisp_Object restrictions = assq_no_quit (buf, labeled_restrictions);
2754 if (NILP (restrictions))
2755 return Qnil;
2756 Lisp_Object label = XCAR (XCAR (XCAR (XCDR (restrictions))));
2757 eassert (! NILP (label));
2758 return label;
2759 }
2760
2761
2762
2763 static void
2764 labeled_restrictions_push (Lisp_Object buf, Lisp_Object restriction)
2765 {
2766 Lisp_Object restrictions = assq_no_quit (buf, labeled_restrictions);
2767 if (NILP (restrictions))
2768 labeled_restrictions_add (buf, list1 (restriction));
2769 else
2770 XSETCDR (restrictions, list1 (nconc2 (list1 (restriction),
2771 XCAR (XCDR (restrictions)))));
2772 }
2773
2774
2775
2776
2777 static void
2778 labeled_restrictions_pop (Lisp_Object buf)
2779 {
2780 Lisp_Object restrictions = assq_no_quit (buf, labeled_restrictions);
2781 if (NILP (restrictions))
2782 return;
2783 if (EQ (labeled_restrictions_peek_label (buf), Qoutermost_restriction))
2784 labeled_restrictions_remove (buf);
2785 else
2786 XSETCDR (restrictions, list1 (XCDR (XCAR (XCDR (restrictions)))));
2787 }
2788
2789
2790 void
2791 labeled_restrictions_remove_in_current_buffer (void)
2792 {
2793 labeled_restrictions_remove (Fcurrent_buffer ());
2794 }
2795
2796 static void
2797 unwind_reset_outermost_restriction (Lisp_Object buf)
2798 {
2799 Lisp_Object begv = labeled_restrictions_get_bound (buf, true, false);
2800 Lisp_Object zv = labeled_restrictions_get_bound (buf, false, false);
2801 if (! NILP (begv) && ! NILP (zv))
2802 {
2803 SET_BUF_BEGV_BOTH (XBUFFER (buf),
2804 marker_position (begv), marker_byte_position (begv));
2805 SET_BUF_ZV_BOTH (XBUFFER (buf),
2806 marker_position (zv), marker_byte_position (zv));
2807 }
2808 else
2809 labeled_restrictions_remove (buf);
2810 }
2811
2812
2813
2814
2815
2816
2817
2818
2819
2820
2821
2822 void
2823 reset_outermost_restrictions (void)
2824 {
2825 Lisp_Object val, buf;
2826 for (val = labeled_restrictions; CONSP (val); val = XCDR (val))
2827 {
2828 buf = XCAR (XCAR (val));
2829 eassert (BUFFERP (buf));
2830 Lisp_Object begv = labeled_restrictions_get_bound (buf, true, true);
2831 Lisp_Object zv = labeled_restrictions_get_bound (buf, false, true);
2832 if (! NILP (begv) && ! NILP (zv))
2833 {
2834 SET_BUF_BEGV_BOTH (XBUFFER (buf),
2835 marker_position (begv), marker_byte_position (begv));
2836 SET_BUF_ZV_BOTH (XBUFFER (buf),
2837 marker_position (zv), marker_byte_position (zv));
2838 record_unwind_protect (unwind_reset_outermost_restriction, buf);
2839 }
2840 else
2841 labeled_restrictions_remove (buf);
2842 }
2843 }
2844
2845
2846
2847 static Lisp_Object
2848 labeled_restrictions_save (void)
2849 {
2850 Lisp_Object buf = Fcurrent_buffer ();
2851 Lisp_Object restrictions = assq_no_quit (buf, labeled_restrictions);
2852 if (! NILP (restrictions))
2853 restrictions = XCAR (XCDR (restrictions));
2854 return Fcons (buf, Fcopy_sequence (restrictions));
2855 }
2856
2857 static void
2858 labeled_restrictions_restore (Lisp_Object buf_and_restrictions)
2859 {
2860 Lisp_Object buf = XCAR (buf_and_restrictions);
2861 Lisp_Object restrictions = XCDR (buf_and_restrictions);
2862 labeled_restrictions_remove (buf);
2863 if (! NILP (restrictions))
2864 labeled_restrictions_add (buf, restrictions);
2865 }
2866
2867 static void
2868 unwind_labeled_narrow_to_region (Lisp_Object label)
2869 {
2870 Finternal__unlabel_restriction (label);
2871 Fwiden ();
2872 }
2873
2874
2875
2876 void
2877 labeled_narrow_to_region (Lisp_Object begv, Lisp_Object zv,
2878 Lisp_Object label)
2879 {
2880 Finternal__labeled_narrow_to_region (begv, zv, label);
2881 record_unwind_protect (restore_point_unwind, Fpoint_marker ());
2882 record_unwind_protect (unwind_labeled_narrow_to_region, label);
2883 }
2884
2885 DEFUN ("widen", Fwiden, Swiden, 0, 0, "",
2886 doc:
2887
2888
2889
2890
2891
2892
2893 )
2894 (void)
2895 {
2896 Lisp_Object buf = Fcurrent_buffer ();
2897 Lisp_Object label = labeled_restrictions_peek_label (buf);
2898
2899 if (NILP (label))
2900 {
2901 if (BEG != BEGV || Z != ZV)
2902 current_buffer->clip_changed = 1;
2903 BEGV = BEG;
2904 BEGV_BYTE = BEG_BYTE;
2905 SET_BUF_ZV_BOTH (current_buffer, Z, Z_BYTE);
2906 }
2907 else
2908 {
2909 Lisp_Object begv = labeled_restrictions_get_bound (buf, true, false);
2910 Lisp_Object zv = labeled_restrictions_get_bound (buf, false, false);
2911 eassert (! NILP (begv) && ! NILP (zv));
2912 ptrdiff_t begv_charpos = marker_position (begv);
2913 ptrdiff_t zv_charpos = marker_position (zv);
2914 if (begv_charpos != BEGV || zv_charpos != ZV)
2915 current_buffer->clip_changed = 1;
2916 SET_BUF_BEGV_BOTH (current_buffer,
2917 begv_charpos, marker_byte_position (begv));
2918 SET_BUF_ZV_BOTH (current_buffer,
2919 zv_charpos, marker_byte_position (zv));
2920
2921
2922
2923
2924 if (EQ (label, Qoutermost_restriction))
2925 labeled_restrictions_pop (buf);
2926 }
2927
2928 invalidate_current_column ();
2929 return Qnil;
2930 }
2931
2932 DEFUN ("narrow-to-region", Fnarrow_to_region, Snarrow_to_region, 2, 2, "r",
2933 doc:
2934
2935
2936
2937
2938
2939
2940
2941
2942
2943
2944
2945
2946
2947
2948 )
2949 (Lisp_Object start, Lisp_Object end)
2950 {
2951 EMACS_INT s = fix_position (start), e = fix_position (end);
2952
2953 if (e < s)
2954 {
2955 EMACS_INT tem = s; s = e; e = tem;
2956 }
2957
2958 if (!(BEG <= s && s <= e && e <= Z))
2959 args_out_of_range (start, end);
2960
2961 Lisp_Object buf = Fcurrent_buffer ();
2962 if (! NILP (labeled_restrictions_peek_label (buf)))
2963 {
2964
2965
2966 Lisp_Object begv = labeled_restrictions_get_bound (buf, true, false);
2967 Lisp_Object zv = labeled_restrictions_get_bound (buf, false, false);
2968 eassert (! NILP (begv) && ! NILP (zv));
2969 ptrdiff_t begv_charpos = marker_position (begv);
2970 ptrdiff_t zv_charpos = marker_position (zv);
2971 if (s < begv_charpos) s = begv_charpos;
2972 if (s > zv_charpos) s = zv_charpos;
2973 if (e < begv_charpos) e = begv_charpos;
2974 if (e > zv_charpos) e = zv_charpos;
2975 }
2976
2977 if (BEGV != s || ZV != e)
2978 current_buffer->clip_changed = 1;
2979
2980 SET_BUF_BEGV (current_buffer, s);
2981 SET_BUF_ZV (current_buffer, e);
2982
2983 if (PT < s)
2984 SET_PT (s);
2985 if (e < PT)
2986 SET_PT (e);
2987
2988 invalidate_current_column ();
2989 return Qnil;
2990 }
2991
2992 DEFUN ("internal--labeled-narrow-to-region", Finternal__labeled_narrow_to_region,
2993 Sinternal__labeled_narrow_to_region, 3, 3, 0,
2994 doc:
2995
2996 )
2997 (Lisp_Object start, Lisp_Object end, Lisp_Object label)
2998 {
2999 Lisp_Object buf = Fcurrent_buffer ();
3000 Lisp_Object outermost_restriction = list3 (Qoutermost_restriction,
3001 Fpoint_min_marker (),
3002 Fpoint_max_marker ());
3003 Fnarrow_to_region (start, end);
3004 if (NILP (labeled_restrictions_peek_label (buf)))
3005 labeled_restrictions_push (buf, outermost_restriction);
3006 labeled_restrictions_push (buf, list3 (label,
3007 Fpoint_min_marker (),
3008 Fpoint_max_marker ()));
3009 return Qnil;
3010 }
3011
3012 DEFUN ("internal--unlabel-restriction", Finternal__unlabel_restriction,
3013 Sinternal__unlabel_restriction, 1, 1, 0,
3014 doc:
3015
3016 )
3017 (Lisp_Object label)
3018 {
3019 Lisp_Object buf = Fcurrent_buffer ();
3020 if (EQ (labeled_restrictions_peek_label (buf), label))
3021 labeled_restrictions_pop (buf);
3022 return Qnil;
3023 }
3024
3025 static Lisp_Object
3026 save_restriction_save_1 (void)
3027 {
3028 if (BEGV == BEG && ZV == Z)
3029
3030
3031
3032 return Fcurrent_buffer ();
3033 else
3034
3035
3036 {
3037 Lisp_Object beg, end;
3038
3039 beg = build_marker (current_buffer, BEGV, BEGV_BYTE);
3040 end = build_marker (current_buffer, ZV, ZV_BYTE);
3041
3042
3043 XMARKER (end)->insertion_type = 1;
3044
3045 return Fcons (beg, end);
3046 }
3047 }
3048
3049 static void
3050 save_restriction_restore_1 (Lisp_Object data)
3051 {
3052 struct buffer *cur = NULL;
3053 struct buffer *buf = (CONSP (data)
3054 ? XMARKER (XCAR (data))->buffer
3055 : XBUFFER (data));
3056
3057 if (buf && buf != current_buffer && !NILP (BVAR (buf, pt_marker)))
3058 {
3059
3060
3061
3062 cur = current_buffer;
3063 set_buffer_internal (buf);
3064 }
3065
3066 if (CONSP (data))
3067
3068 {
3069 struct Lisp_Marker *beg = XMARKER (XCAR (data));
3070 struct Lisp_Marker *end = XMARKER (XCDR (data));
3071 eassert (buf == end->buffer);
3072
3073 if (buf
3074 && (beg->charpos != BUF_BEGV (buf) || end->charpos != BUF_ZV (buf)))
3075
3076
3077 {
3078 ptrdiff_t pt = BUF_PT (buf);
3079
3080 SET_BUF_BEGV_BOTH (buf, beg->charpos, beg->bytepos);
3081 SET_BUF_ZV_BOTH (buf, end->charpos, end->bytepos);
3082
3083 if (pt < beg->charpos || pt > end->charpos)
3084
3085 SET_BUF_PT_BOTH (buf,
3086 clip_to_bounds (beg->charpos, pt, end->charpos),
3087 clip_to_bounds (beg->bytepos, BUF_PT_BYTE (buf),
3088 end->bytepos));
3089
3090 buf->clip_changed = 1;
3091 }
3092
3093 detach_marker (XCAR (data));
3094 detach_marker (XCDR (data));
3095 free_cons (XCONS (data));
3096 }
3097 else
3098
3099 {
3100 if (buf
3101 && (BUF_BEGV (buf) != BUF_BEG (buf) || BUF_ZV (buf) != BUF_Z (buf)))
3102
3103 {
3104 SET_BUF_BEGV_BOTH (buf, BUF_BEG (buf), BUF_BEG_BYTE (buf));
3105 SET_BUF_ZV_BOTH (buf, BUF_Z (buf), BUF_Z_BYTE (buf));
3106
3107 buf->clip_changed = 1;
3108 }
3109 }
3110
3111
3112 invalidate_current_column ();
3113
3114 if (cur)
3115 set_buffer_internal (cur);
3116 }
3117
3118 Lisp_Object
3119 save_restriction_save (void)
3120 {
3121 Lisp_Object restriction = save_restriction_save_1 ();
3122 Lisp_Object labeled_restrictions = labeled_restrictions_save ();
3123 return Fcons (restriction, labeled_restrictions);
3124 }
3125
3126 void
3127 save_restriction_restore (Lisp_Object data)
3128 {
3129 labeled_restrictions_restore (XCDR (data));
3130 save_restriction_restore_1 (XCAR (data));
3131 }
3132
3133 DEFUN ("save-restriction", Fsave_restriction, Ssave_restriction, 0, UNEVALLED, 0,
3134 doc:
3135
3136
3137
3138
3139
3140
3141
3142
3143
3144
3145
3146
3147
3148
3149
3150 )
3151 (Lisp_Object body)
3152 {
3153 register Lisp_Object val;
3154 specpdl_ref count = SPECPDL_INDEX ();
3155
3156 record_unwind_protect (save_restriction_restore, save_restriction_save ());
3157 val = Fprogn (body);
3158 return unbind_to (count, val);
3159 }
3160
3161
3162
3163 DEFUN ("ngettext", Fngettext, Sngettext, 3, 3, 0,
3164 doc:
3165
3166
3167
3168
3169 )
3170 (Lisp_Object msgid, Lisp_Object msgid_plural, Lisp_Object n)
3171 {
3172 CHECK_STRING (msgid);
3173 CHECK_STRING (msgid_plural);
3174 CHECK_INTEGER (n);
3175
3176
3177 return BASE_EQ (n, make_fixnum (1)) ? msgid : msgid_plural;
3178 }
3179
3180 DEFUN ("message", Fmessage, Smessage, 1, MANY, 0,
3181 doc:
3182
3183
3184
3185
3186
3187
3188
3189
3190
3191
3192
3193
3194
3195
3196
3197
3198
3199 )
3200 (ptrdiff_t nargs, Lisp_Object *args)
3201 {
3202 if (NILP (args[0])
3203 || (STRINGP (args[0])
3204 && SBYTES (args[0]) == 0))
3205 {
3206 message1 (0);
3207 return args[0];
3208 }
3209 else
3210 {
3211 Lisp_Object val = Fformat_message (nargs, args);
3212 message3 (val);
3213 return val;
3214 }
3215 }
3216
3217 DEFUN ("message-box", Fmessage_box, Smessage_box, 1, MANY, 0,
3218 doc:
3219
3220
3221
3222
3223
3224
3225
3226
3227 )
3228 (ptrdiff_t nargs, Lisp_Object *args)
3229 {
3230 if (NILP (args[0]))
3231 {
3232 message1 (0);
3233 return Qnil;
3234 }
3235 else
3236 {
3237 Lisp_Object val = Fformat_message (nargs, args);
3238 Lisp_Object pane, menu;
3239
3240 pane = list1 (Fcons (build_string ("OK"), Qt));
3241 menu = Fcons (val, pane);
3242 Fx_popup_dialog (Qt, menu, Qt);
3243 return val;
3244 }
3245 }
3246
3247 DEFUN ("message-or-box", Fmessage_or_box, Smessage_or_box, 1, MANY, 0,
3248 doc:
3249
3250
3251
3252
3253
3254
3255
3256
3257
3258
3259 )
3260 (ptrdiff_t nargs, Lisp_Object *args)
3261 {
3262 if ((NILP (last_nonmenu_event) || CONSP (last_nonmenu_event))
3263 && use_dialog_box)
3264 return Fmessage_box (nargs, args);
3265 return Fmessage (nargs, args);
3266 }
3267
3268 DEFUN ("current-message", Fcurrent_message, Scurrent_message, 0, 0, 0,
3269 doc: )
3270 (void)
3271 {
3272 return current_message ();
3273 }
3274
3275
3276 DEFUN ("propertize", Fpropertize, Spropertize, 1, MANY, 0,
3277 doc:
3278
3279
3280
3281
3282
3283 )
3284 (ptrdiff_t nargs, Lisp_Object *args)
3285 {
3286 Lisp_Object properties, string;
3287 ptrdiff_t i;
3288
3289
3290 if ((nargs & 1) == 0)
3291 xsignal2 (Qwrong_number_of_arguments, Qpropertize, make_fixnum (nargs));
3292
3293 properties = string = Qnil;
3294
3295
3296 CHECK_STRING (args[0]);
3297 string = Fcopy_sequence (args[0]);
3298
3299 for (i = 1; i < nargs; i += 2)
3300 properties = Fcons (args[i], Fcons (args[i + 1], properties));
3301
3302 Fadd_text_properties (make_fixnum (0),
3303 make_fixnum (SCHARS (string)),
3304 properties, string);
3305 return string;
3306 }
3307
3308
3309
3310
3311
3312
3313
3314 static ptrdiff_t
3315 str2num (char *str, char **str_end)
3316 {
3317 ptrdiff_t n = 0;
3318 for (; c_isdigit (*str); str++)
3319 if (ckd_mul (&n, n, 10) || ckd_add (&n, n, *str - '0'))
3320 n = PTRDIFF_MAX;
3321 *str_end = str;
3322 return n;
3323 }
3324
3325 DEFUN ("format", Fformat, Sformat, 1, MANY, 0,
3326 doc:
3327
3328
3329
3330
3331
3332
3333
3334
3335
3336
3337
3338
3339
3340
3341
3342
3343
3344
3345
3346
3347
3348
3349
3350
3351
3352
3353
3354
3355
3356
3357
3358
3359
3360
3361
3362
3363
3364
3365
3366
3367
3368
3369
3370
3371
3372
3373
3374
3375
3376
3377
3378
3379
3380
3381
3382
3383
3384
3385
3386
3387
3388
3389
3390
3391
3392
3393
3394
3395 )
3396 (ptrdiff_t nargs, Lisp_Object *args)
3397 {
3398 return styled_format (nargs, args, false);
3399 }
3400
3401 DEFUN ("format-message", Fformat_message, Sformat_message, 1, MANY, 0,
3402 doc:
3403
3404
3405
3406
3407
3408
3409
3410
3411 )
3412 (ptrdiff_t nargs, Lisp_Object *args)
3413 {
3414 return styled_format (nargs, args, true);
3415 }
3416
3417
3418
3419 static Lisp_Object
3420 styled_format (ptrdiff_t nargs, Lisp_Object *args, bool message)
3421 {
3422 enum
3423 {
3424
3425
3426
3427 USEFUL_PRECISION_MAX = ((1 - LDBL_MIN_EXP)
3428 * (FLT_RADIX == 2 || FLT_RADIX == 10 ? 1
3429 : FLT_RADIX == 16 ? 4
3430 : -1)),
3431
3432
3433
3434
3435 SPRINTF_BUFSIZE = (sizeof "-." + (LDBL_MAX_10_EXP + 1)
3436 + USEFUL_PRECISION_MAX)
3437 };
3438 verify (USEFUL_PRECISION_MAX > 0);
3439
3440 ptrdiff_t n;
3441 char initial_buffer[1000 + SPRINTF_BUFSIZE];
3442 char *buf = initial_buffer;
3443 ptrdiff_t bufsize = sizeof initial_buffer;
3444 ptrdiff_t max_bufsize = STRING_BYTES_BOUND + 1;
3445 char *p;
3446 specpdl_ref buf_save_value_index UNINIT;
3447 char *format, *end;
3448 ptrdiff_t nchars;
3449
3450
3451
3452
3453 bool maybe_combine_byte;
3454 Lisp_Object val;
3455 bool arg_intervals = false;
3456 USE_SAFE_ALLOCA;
3457 sa_avail -= sizeof initial_buffer;
3458
3459
3460 struct info
3461 {
3462
3463
3464 Lisp_Object argument;
3465
3466
3467 ptrdiff_t start, end;
3468
3469
3470 ptrdiff_t fbeg;
3471
3472
3473 bool_bf intervals : 1;
3474 } *info;
3475
3476 CHECK_STRING (args[0]);
3477 char *format_start = SSDATA (args[0]);
3478 bool multibyte_format = STRING_MULTIBYTE (args[0]);
3479 ptrdiff_t formatlen = SBYTES (args[0]);
3480 bool fmt_props = !!string_intervals (args[0]);
3481
3482
3483 ptrdiff_t nspec_bound = SCHARS (args[0]) >> 1;
3484
3485
3486 ptrdiff_t info_size, alloca_size;
3487 if (ckd_mul (&info_size, nspec_bound, sizeof *info)
3488 || ckd_add (&alloca_size, formatlen, info_size)
3489 || SIZE_MAX < alloca_size)
3490 memory_full (SIZE_MAX);
3491 info = SAFE_ALLOCA (alloca_size);
3492
3493
3494
3495 char *discarded = (char *) &info[nspec_bound];
3496 memset (discarded, 0, formatlen);
3497
3498
3499
3500
3501
3502
3503
3504
3505
3506 bool multibyte = multibyte_format;
3507 for (ptrdiff_t i = 1; !multibyte && i < nargs; i++)
3508 if (STRINGP (args[i]) && STRING_MULTIBYTE (args[i]))
3509 multibyte = true;
3510
3511 Lisp_Object quoting_style = message ? Ftext_quoting_style () : Qnil;
3512
3513 ptrdiff_t ispec;
3514 ptrdiff_t nspec = 0;
3515
3516
3517 bool new_result = false;
3518
3519
3520
3521 retry:
3522
3523 p = buf;
3524 nchars = 0;
3525
3526
3527 n = 0;
3528 ispec = 0;
3529
3530
3531 format = format_start;
3532 end = format + formatlen;
3533 maybe_combine_byte = false;
3534
3535 while (format != end)
3536 {
3537
3538
3539 ptrdiff_t n0 = n;
3540 ptrdiff_t ispec0 = ispec;
3541 char *format0 = format;
3542 char const *convsrc = format;
3543 unsigned char format_char = *format++;
3544
3545
3546
3547
3548
3549
3550 ptrdiff_t convbytes = 1;
3551 enum { CONVBYTES_ROOM = SPRINTF_BUFSIZE - 1 };
3552 eassert (p <= buf + bufsize - SPRINTF_BUFSIZE);
3553
3554 if (format_char == '%')
3555 {
3556
3557
3558
3559
3560
3561
3562
3563
3564
3565
3566
3567
3568
3569
3570
3571
3572
3573
3574
3575
3576
3577
3578 ptrdiff_t num;
3579 char *num_end;
3580 if (c_isdigit (*format))
3581 {
3582 num = str2num (format, &num_end);
3583 if (*num_end == '$')
3584 {
3585 n = num - 1;
3586 format = num_end + 1;
3587 }
3588 }
3589
3590 bool minus_flag = false;
3591 bool plus_flag = false;
3592 bool space_flag = false;
3593 bool sharp_flag = false;
3594 bool zero_flag = false;
3595
3596 for (; ; format++)
3597 {
3598 switch (*format)
3599 {
3600 case '-': minus_flag = true; continue;
3601 case '+': plus_flag = true; continue;
3602 case ' ': space_flag = true; continue;
3603 case '#': sharp_flag = true; continue;
3604 case '0': zero_flag = true; continue;
3605 }
3606 break;
3607 }
3608
3609
3610 space_flag &= ! plus_flag;
3611 zero_flag &= ! minus_flag;
3612
3613 num = str2num (format, &num_end);
3614 if (max_bufsize <= num)
3615 string_overflow ();
3616 ptrdiff_t field_width = num;
3617
3618 bool precision_given = *num_end == '.';
3619 ptrdiff_t precision = (precision_given
3620 ? str2num (num_end + 1, &num_end)
3621 : PTRDIFF_MAX);
3622 format = num_end;
3623
3624 if (format == end)
3625 error ("Format string ends in middle of format specifier");
3626
3627 char conversion = *format++;
3628 memset (&discarded[format0 - format_start], 1,
3629 format - format0 - (conversion == '%'));
3630 info[ispec].fbeg = format0 - format_start;
3631 if (conversion == '%')
3632 {
3633 new_result = true;
3634 goto copy_char;
3635 }
3636
3637 ++n;
3638 if (! (n < nargs))
3639 error ("Not enough arguments for format string");
3640
3641 struct info *spec = &info[ispec++];
3642 if (nspec < ispec)
3643 {
3644 spec->argument = args[n];
3645 spec->intervals = false;
3646 nspec = ispec;
3647 }
3648 Lisp_Object arg = spec->argument;
3649
3650
3651
3652
3653
3654 if ((conversion == 'S'
3655 || (conversion == 's'
3656 && ! STRINGP (arg) && ! SYMBOLP (arg))))
3657 {
3658 if (EQ (arg, args[n]))
3659 {
3660 Lisp_Object noescape = conversion == 'S' ? Qnil : Qt;
3661 spec->argument = arg = Fprin1_to_string (arg, noescape, Qnil);
3662 if (STRING_MULTIBYTE (arg) && ! multibyte)
3663 {
3664 multibyte = true;
3665 goto retry;
3666 }
3667 }
3668 conversion = 's';
3669 }
3670 else if (conversion == 'c')
3671 {
3672 if (FIXNUMP (arg) && ! ASCII_CHAR_P (XFIXNUM (arg)))
3673 {
3674 if (!multibyte)
3675 {
3676 multibyte = true;
3677 goto retry;
3678 }
3679 spec->argument = arg = Fchar_to_string (arg);
3680 }
3681
3682 if (!EQ (arg, args[n]))
3683 conversion = 's';
3684 zero_flag = false;
3685 }
3686
3687 if (SYMBOLP (arg))
3688 {
3689 spec->argument = arg = SYMBOL_NAME (arg);
3690 if (STRING_MULTIBYTE (arg) && ! multibyte)
3691 {
3692 multibyte = true;
3693 goto retry;
3694 }
3695 }
3696
3697 bool float_conversion
3698 = conversion == 'e' || conversion == 'f' || conversion == 'g';
3699
3700 if (conversion == 's')
3701 {
3702 if (format == end && format - format_start == 2
3703 && ! string_intervals (args[0]))
3704 {
3705 val = arg;
3706 goto return_val;
3707 }
3708
3709
3710
3711 ptrdiff_t prec = -1;
3712 if (precision_given)
3713 prec = precision;
3714
3715
3716
3717
3718
3719
3720
3721 ptrdiff_t width, nbytes;
3722 ptrdiff_t nchars_string;
3723 if (prec == 0)
3724 width = nchars_string = nbytes = 0;
3725 else
3726 {
3727 ptrdiff_t nch, nby;
3728 nchars_string = SCHARS (arg);
3729 width = lisp_string_width (arg, 0, nchars_string, prec,
3730 &nch, &nby, false);
3731 if (prec < 0)
3732 nbytes = SBYTES (arg);
3733 else
3734 {
3735 nchars_string = nch;
3736 nbytes = nby;
3737 }
3738 }
3739
3740 convbytes = nbytes;
3741 if (convbytes && multibyte && ! STRING_MULTIBYTE (arg))
3742 convbytes = count_size_as_multibyte (SDATA (arg), nbytes);
3743
3744 ptrdiff_t padding
3745 = width < field_width ? field_width - width : 0;
3746
3747 if (max_bufsize - padding <= convbytes)
3748 string_overflow ();
3749 convbytes += padding;
3750 if (convbytes <= buf + bufsize - p)
3751 {
3752
3753
3754 if (fmt_props)
3755 spec->start = nchars;
3756 if (! minus_flag)
3757 {
3758 memset (p, ' ', padding);
3759 p += padding;
3760 nchars += padding;
3761 }
3762
3763
3764 if (!fmt_props)
3765 spec->start = nchars;
3766
3767 if (p > buf
3768 && multibyte
3769 && !ASCII_CHAR_P (*((unsigned char *) p - 1))
3770 && STRING_MULTIBYTE (arg)
3771 && !CHAR_HEAD_P (SREF (arg, 0)))
3772 maybe_combine_byte = true;
3773
3774 p += copy_text (SDATA (arg), (unsigned char *) p,
3775 nbytes,
3776 STRING_MULTIBYTE (arg), multibyte);
3777
3778 nchars += nchars_string;
3779
3780 if (minus_flag)
3781 {
3782 memset (p, ' ', padding);
3783 p += padding;
3784 nchars += padding;
3785 }
3786 spec->end = nchars;
3787
3788
3789
3790 if (string_intervals (arg))
3791 spec->intervals = arg_intervals = true;
3792
3793 new_result = true;
3794 convbytes = CONVBYTES_ROOM;
3795 }
3796 }
3797 else if (! (conversion == 'c' || conversion == 'd'
3798 || float_conversion || conversion == 'i'
3799 || conversion == 'o' || conversion == 'x'
3800 || conversion == 'X'))
3801 {
3802 unsigned char *p = (unsigned char *) format - 1;
3803 if (multibyte_format)
3804 error ("Invalid format operation %%%c", STRING_CHAR (p));
3805 else
3806 error (*p <= 127 ? "Invalid format operation %%%c"
3807 : "Invalid format operation char #o%03o",
3808 *p);
3809 }
3810 else if (! (FIXNUMP (arg) || ((BIGNUMP (arg) || FLOATP (arg))
3811 && conversion != 'c')))
3812 error ("Format specifier doesn't match argument type");
3813 else
3814 {
3815
3816 enum { pMlen = sizeof PRIdMAX - 2 };
3817
3818
3819 if (conversion == 'd' || conversion == 'i')
3820 sharp_flag = false;
3821
3822
3823
3824
3825
3826
3827 char convspec[sizeof "%FF.*d" + max (sizeof "L" - 1, pMlen)];
3828 char *f = convspec;
3829 *f++ = '%';
3830
3831 *f = '+'; f += plus_flag;
3832 *f = ' '; f += space_flag;
3833 *f = '#'; f += sharp_flag;
3834 *f++ = '.';
3835 *f++ = '*';
3836 if (! (float_conversion || conversion == 'c'))
3837 {
3838 memcpy (f, PRIdMAX, pMlen);
3839 f += pMlen;
3840 zero_flag &= ! precision_given;
3841 }
3842 *f++ = conversion;
3843 *f = '\0';
3844
3845 int prec = -1;
3846 if (precision_given)
3847 prec = min (precision, USEFUL_PRECISION_MAX);
3848
3849
3850
3851
3852 char prefix[sizeof "-0x" - 1];
3853 int prefixlen = 0;
3854
3855
3856
3857
3858
3859
3860
3861
3862
3863
3864
3865
3866
3867 ptrdiff_t sprintf_bytes;
3868 if (float_conversion)
3869 {
3870
3871
3872
3873
3874
3875 bool format_as_long_double = false;
3876 double darg;
3877 long double ldarg UNINIT;
3878
3879 if (FLOATP (arg))
3880 darg = XFLOAT_DATA (arg);
3881 else
3882 {
3883 bool format_bignum_as_double = false;
3884 if (LDBL_MANT_DIG <= DBL_MANT_DIG)
3885 {
3886 if (FIXNUMP (arg))
3887 darg = XFIXNUM (arg);
3888 else
3889 format_bignum_as_double = true;
3890 }
3891 else
3892 {
3893 if (INTEGERP (arg))
3894 {
3895 intmax_t iarg;
3896 uintmax_t uarg;
3897 if (integer_to_intmax (arg, &iarg))
3898 ldarg = iarg;
3899 else if (integer_to_uintmax (arg, &uarg))
3900 ldarg = uarg;
3901 else
3902 format_bignum_as_double = true;
3903 }
3904 if (!format_bignum_as_double)
3905 {
3906 darg = ldarg;
3907 format_as_long_double = darg != ldarg;
3908 }
3909 }
3910 if (format_bignum_as_double)
3911 darg = bignum_to_double (arg);
3912 }
3913
3914 if (format_as_long_double)
3915 {
3916 f[-1] = 'L';
3917 *f++ = conversion;
3918 *f = '\0';
3919 sprintf_bytes = sprintf (p, convspec, prec, ldarg);
3920 }
3921 else
3922 sprintf_bytes = sprintf (p, convspec, prec, darg);
3923 }
3924 else if (conversion == 'c')
3925 {
3926
3927 p[0] = XFIXNUM (arg);
3928 p[1] = '\0';
3929 sprintf_bytes = prec != 0;
3930 }
3931 else if (BIGNUMP (arg))
3932 bignum_arg:
3933 {
3934 int base = ((conversion == 'd' || conversion == 'i') ? 10
3935 : conversion == 'o' ? 8 : 16);
3936 sprintf_bytes = bignum_bufsize (arg, base);
3937 if (sprintf_bytes <= buf + bufsize - p)
3938 {
3939 int signedbase = conversion == 'X' ? -base : base;
3940 sprintf_bytes = bignum_to_c_string (p, sprintf_bytes,
3941 arg, signedbase);
3942 bool negative = p[0] == '-';
3943 prec = min (precision, sprintf_bytes - prefixlen);
3944 prefix[prefixlen] = plus_flag ? '+' : ' ';
3945 prefixlen += (plus_flag | space_flag) & !negative;
3946 prefix[prefixlen] = '0';
3947 prefix[prefixlen + 1] = conversion;
3948 prefixlen += sharp_flag && base == 16 ? 2 : 0;
3949 }
3950 }
3951 else if (conversion == 'd' || conversion == 'i')
3952 {
3953 if (FIXNUMP (arg))
3954 {
3955 intmax_t x = XFIXNUM (arg);
3956 sprintf_bytes = sprintf (p, convspec, prec, x);
3957 }
3958 else
3959 {
3960 strcpy (f - pMlen - 1, "f");
3961 double x = XFLOAT_DATA (arg);
3962
3963
3964
3965 x = trunc (x);
3966 x = x ? x : 0;
3967
3968 sprintf_bytes = sprintf (p, convspec, 0, x);
3969 bool signedp = ! c_isdigit (p[0]);
3970 prec = min (precision, sprintf_bytes - signedp);
3971 }
3972 }
3973 else
3974 {
3975 uintmax_t x;
3976 bool negative;
3977 if (FIXNUMP (arg))
3978 {
3979 if (binary_as_unsigned)
3980 {
3981 x = XUFIXNUM (arg);
3982 negative = false;
3983 }
3984 else
3985 {
3986 EMACS_INT i = XFIXNUM (arg);
3987 negative = i < 0;
3988 x = negative ? -i : i;
3989 }
3990 }
3991 else
3992 {
3993 double d = XFLOAT_DATA (arg);
3994 double abs_d = fabs (d);
3995 if (abs_d < UINTMAX_MAX + 1.0)
3996 {
3997 negative = d <= -1;
3998 x = abs_d;
3999 }
4000 else
4001 {
4002 arg = double_to_integer (d);
4003 goto bignum_arg;
4004 }
4005 }
4006 p[0] = negative ? '-' : plus_flag ? '+' : ' ';
4007 bool signedp = negative | plus_flag | space_flag;
4008 sprintf_bytes = sprintf (p + signedp, convspec, prec, x);
4009 sprintf_bytes += signedp;
4010 }
4011
4012
4013
4014
4015
4016
4017
4018
4019 ptrdiff_t excess_precision
4020 = precision_given ? precision - prec : 0;
4021 ptrdiff_t trailing_zeros = 0;
4022 if (excess_precision != 0 && float_conversion)
4023 {
4024 if (! c_isdigit (p[sprintf_bytes - 1])
4025 || (conversion == 'g'
4026 && ! (sharp_flag && strchr (p, '.'))))
4027 excess_precision = 0;
4028 trailing_zeros = excess_precision;
4029 }
4030 ptrdiff_t leading_zeros = excess_precision - trailing_zeros;
4031
4032
4033
4034 ptrdiff_t numwidth;
4035 if (ckd_add (&numwidth, prefixlen + sprintf_bytes,
4036 excess_precision))
4037 numwidth = PTRDIFF_MAX;
4038 ptrdiff_t padding
4039 = numwidth < field_width ? field_width - numwidth : 0;
4040 if (max_bufsize - (prefixlen + sprintf_bytes) <= excess_precision
4041 || max_bufsize - padding <= numwidth)
4042 string_overflow ();
4043 convbytes = numwidth + padding;
4044
4045 if (convbytes <= buf + bufsize - p)
4046 {
4047 bool signedp = p[0] == '-' || p[0] == '+' || p[0] == ' ';
4048 int beglen = (signedp
4049 + ((p[signedp] == '0'
4050 && (p[signedp + 1] == 'x'
4051 || p[signedp + 1] == 'X'))
4052 ? 2 : 0));
4053 eassert (prefixlen == 0 || beglen == 0
4054 || (beglen == 1 && p[0] == '-'
4055 && ! (prefix[0] == '-' || prefix[0] == '+'
4056 || prefix[0] == ' ')));
4057 if (zero_flag && 0 <= char_hexdigit (p[beglen]))
4058 {
4059 leading_zeros += padding;
4060 padding = 0;
4061 }
4062 if (leading_zeros == 0 && sharp_flag && conversion == 'o'
4063 && p[beglen] != '0')
4064 {
4065 leading_zeros++;
4066 padding -= padding != 0;
4067 }
4068
4069 int endlen = 0;
4070 if (trailing_zeros
4071 && (conversion == 'e' || conversion == 'g'))
4072 {
4073 char *e = strchr (p, 'e');
4074 if (e)
4075 endlen = p + sprintf_bytes - e;
4076 }
4077
4078 ptrdiff_t midlen = sprintf_bytes - beglen - endlen;
4079 ptrdiff_t leading_padding = minus_flag ? 0 : padding;
4080 ptrdiff_t trailing_padding = padding - leading_padding;
4081
4082
4083
4084
4085
4086
4087
4088
4089
4090
4091
4092
4093
4094
4095
4096
4097 ptrdiff_t incr
4098 = (padding + leading_zeros + prefixlen
4099 + sprintf_bytes + trailing_zeros);
4100
4101
4102 if (incr != sprintf_bytes)
4103 {
4104
4105
4106
4107
4108 char *src = p + sprintf_bytes;
4109 char *dst = p + incr;
4110 dst -= trailing_padding;
4111 memset (dst, ' ', trailing_padding);
4112 src -= endlen;
4113 dst -= endlen;
4114 memmove (dst, src, endlen);
4115 dst -= trailing_zeros;
4116 memset (dst, '0', trailing_zeros);
4117 src -= midlen;
4118 dst -= midlen;
4119 memmove (dst, src, midlen);
4120 dst -= leading_zeros;
4121 memset (dst, '0', leading_zeros);
4122 dst -= prefixlen;
4123 memcpy (dst, prefix, prefixlen);
4124 src -= beglen;
4125 dst -= beglen;
4126 memmove (dst, src, beglen);
4127 dst -= leading_padding;
4128 memset (dst, ' ', leading_padding);
4129 }
4130
4131 p += incr;
4132 spec->start = nchars;
4133 spec->end = nchars += incr;
4134 new_result = true;
4135 convbytes = CONVBYTES_ROOM;
4136 }
4137 }
4138 }
4139 else
4140 {
4141 unsigned char str[MAX_MULTIBYTE_LENGTH];
4142
4143 if ((format_char == '`' || format_char == '\'')
4144 && EQ (quoting_style, Qcurve))
4145 {
4146 if (! multibyte)
4147 {
4148 multibyte = true;
4149 goto retry;
4150 }
4151 convsrc = format_char == '`' ? uLSQM : uRSQM;
4152 convbytes = 3;
4153 new_result = true;
4154 }
4155 else if (format_char == '`' && EQ (quoting_style, Qstraight))
4156 {
4157 convsrc = "'";
4158 new_result = true;
4159 }
4160 else
4161 {
4162
4163 if (multibyte_format)
4164 {
4165
4166 if (p > buf
4167 && !ASCII_CHAR_P (*((unsigned char *) p - 1))
4168 && !CHAR_HEAD_P (format_char))
4169 maybe_combine_byte = true;
4170
4171 while (! CHAR_HEAD_P (*format))
4172 format++;
4173
4174 convbytes = format - format0;
4175 memset (&discarded[format0 + 1 - format_start], 2,
4176 convbytes - 1);
4177 }
4178 else if (multibyte && !ASCII_CHAR_P (format_char))
4179 {
4180 int c = BYTE8_TO_CHAR (format_char);
4181 convbytes = CHAR_STRING (c, str);
4182 convsrc = (char *) str;
4183 new_result = true;
4184 }
4185 }
4186
4187 copy_char:
4188 memcpy (p, convsrc, convbytes);
4189 p += convbytes;
4190 nchars++;
4191 convbytes = CONVBYTES_ROOM;
4192 }
4193
4194 ptrdiff_t used = p - buf;
4195 ptrdiff_t buflen_needed;
4196 if (ckd_add (&buflen_needed, used, convbytes))
4197 string_overflow ();
4198 if (bufsize <= buflen_needed)
4199 {
4200 if (max_bufsize <= buflen_needed)
4201 string_overflow ();
4202
4203
4204
4205
4206
4207 bufsize = (buflen_needed <= max_bufsize / 2
4208 ? buflen_needed * 2 : max_bufsize);
4209
4210 if (buf == initial_buffer)
4211 {
4212 buf = xmalloc (bufsize);
4213 buf_save_value_index = SPECPDL_INDEX ();
4214 record_unwind_protect_ptr (xfree, buf);
4215 memcpy (buf, initial_buffer, used);
4216 }
4217 else
4218 {
4219 buf = xrealloc (buf, bufsize);
4220 set_unwind_protect_ptr (buf_save_value_index, xfree, buf);
4221 }
4222
4223 p = buf + used;
4224 if (convbytes != CONVBYTES_ROOM)
4225 {
4226
4227 eassert (CONVBYTES_ROOM < convbytes);
4228 format = format0;
4229 n = n0;
4230 ispec = ispec0;
4231 }
4232 }
4233 }
4234
4235 if (bufsize < p - buf)
4236 emacs_abort ();
4237
4238 if (! new_result)
4239 {
4240 val = args[0];
4241 goto return_val;
4242 }
4243
4244 if (maybe_combine_byte)
4245 nchars = multibyte_chars_in_text ((unsigned char *) buf, p - buf);
4246 val = make_specified_string (buf, nchars, p - buf, multibyte);
4247
4248
4249
4250
4251
4252 if (string_intervals (args[0]) || arg_intervals)
4253 {
4254
4255 Lisp_Object len = make_fixnum (SCHARS (args[0]));
4256 Lisp_Object props = text_property_list (args[0], make_fixnum (0),
4257 len, Qnil);
4258 if (CONSP (props))
4259 {
4260 ptrdiff_t bytepos = 0, position = 0, translated = 0;
4261 ptrdiff_t fieldn = 0;
4262
4263
4264
4265
4266
4267
4268
4269 props = Fnreverse (props);
4270
4271
4272
4273
4274
4275 for (Lisp_Object list = props; CONSP (list); list = XCDR (list))
4276 {
4277 Lisp_Object item = XCAR (list);
4278
4279
4280 ptrdiff_t pos = XFIXNUM (XCAR (item));
4281
4282
4283
4284 for (; position < pos; bytepos++)
4285 {
4286 if (! discarded[bytepos])
4287 position++, translated++;
4288 else if (discarded[bytepos] == 1)
4289 {
4290 position++;
4291 if (fieldn < nspec
4292 && bytepos >= info[fieldn].fbeg
4293 && translated == info[fieldn].start)
4294 {
4295 translated += info[fieldn].end - info[fieldn].start;
4296 fieldn++;
4297 }
4298 }
4299 }
4300
4301 XSETCAR (item, make_fixnum (translated));
4302
4303
4304 pos = XFIXNUM (XCAR (XCDR (item)));
4305
4306 for (; position < pos; bytepos++)
4307 {
4308 if (! discarded[bytepos])
4309 position++, translated++;
4310 else if (discarded[bytepos] == 1)
4311 {
4312 position++;
4313 if (fieldn < nspec
4314 && bytepos >= info[fieldn].fbeg
4315 && translated == info[fieldn].start)
4316 {
4317 translated += info[fieldn].end - info[fieldn].start;
4318 fieldn++;
4319 }
4320 }
4321 }
4322
4323 XSETCAR (XCDR (item), make_fixnum (translated));
4324 }
4325
4326 add_text_properties_from_list (val, props, make_fixnum (0));
4327 }
4328
4329
4330 if (arg_intervals)
4331 for (ptrdiff_t i = 0; i < nspec; i++)
4332 if (info[i].intervals)
4333 {
4334 len = make_fixnum (SCHARS (info[i].argument));
4335 Lisp_Object new_len = make_fixnum (info[i].end - info[i].start);
4336 props = text_property_list (info[i].argument,
4337 make_fixnum (0), len, Qnil);
4338 props = extend_property_ranges (props, len, new_len);
4339
4340
4341 if (1 < i && info[i - 1].end)
4342 make_composition_value_copy (props);
4343 add_text_properties_from_list (val, props,
4344 make_fixnum (info[i].start));
4345 }
4346 }
4347
4348 return_val:
4349
4350 SAFE_FREE ();
4351
4352 return val;
4353 }
4354
4355 DEFUN ("char-equal", Fchar_equal, Schar_equal, 2, 2, 0,
4356 doc:
4357
4358 )
4359 (register Lisp_Object c1, Lisp_Object c2)
4360 {
4361 int i1, i2;
4362
4363
4364 CHECK_CHARACTER (c1);
4365 CHECK_CHARACTER (c2);
4366
4367 if (XFIXNUM (c1) == XFIXNUM (c2))
4368 return Qt;
4369 if (NILP (BVAR (current_buffer, case_fold_search)))
4370 return Qnil;
4371
4372 i1 = XFIXNAT (c1);
4373 i2 = XFIXNAT (c2);
4374
4375
4376
4377
4378
4379
4380
4381 if (NILP (BVAR (current_buffer, enable_multibyte_characters)))
4382 {
4383 if (SINGLE_BYTE_CHAR_P (i1))
4384 i1 = UNIBYTE_TO_CHAR (i1);
4385 if (SINGLE_BYTE_CHAR_P (i2))
4386 i2 = UNIBYTE_TO_CHAR (i2);
4387 }
4388
4389 return (downcase (i1) == downcase (i2) ? Qt : Qnil);
4390 }
4391
4392
4393
4394
4395
4396
4397
4398
4399
4400
4401
4402
4403
4404
4405
4406
4407 static void
4408 transpose_markers (ptrdiff_t start1, ptrdiff_t end1,
4409 ptrdiff_t start2, ptrdiff_t end2,
4410 ptrdiff_t start1_byte, ptrdiff_t end1_byte,
4411 ptrdiff_t start2_byte, ptrdiff_t end2_byte)
4412 {
4413 register ptrdiff_t amt1, amt1_byte, amt2, amt2_byte, diff, diff_byte, mpos;
4414 register struct Lisp_Marker *marker;
4415
4416
4417 if (PT < start1)
4418 ;
4419 else if (PT < end1)
4420 TEMP_SET_PT_BOTH (PT + (end2 - end1),
4421 PT_BYTE + (end2_byte - end1_byte));
4422 else if (PT < start2)
4423 TEMP_SET_PT_BOTH (PT + (end2 - start2) - (end1 - start1),
4424 (PT_BYTE + (end2_byte - start2_byte)
4425 - (end1_byte - start1_byte)));
4426 else if (PT < end2)
4427 TEMP_SET_PT_BOTH (PT - (start2 - start1),
4428 PT_BYTE - (start2_byte - start1_byte));
4429
4430
4431
4432
4433
4434
4435
4436
4437
4438
4439 diff = (end2 - start2) - (end1 - start1);
4440 diff_byte = (end2_byte - start2_byte) - (end1_byte - start1_byte);
4441
4442
4443
4444 amt1 = (end2 - start2) + (start2 - end1);
4445 amt2 = (end1 - start1) + (start2 - end1);
4446 amt1_byte = (end2_byte - start2_byte) + (start2_byte - end1_byte);
4447 amt2_byte = (end1_byte - start1_byte) + (start2_byte - end1_byte);
4448
4449 for (marker = BUF_MARKERS (current_buffer); marker; marker = marker->next)
4450 {
4451 mpos = marker->bytepos;
4452 if (mpos >= start1_byte && mpos < end2_byte)
4453 {
4454 if (mpos < end1_byte)
4455 mpos += amt1_byte;
4456 else if (mpos < start2_byte)
4457 mpos += diff_byte;
4458 else
4459 mpos -= amt2_byte;
4460 marker->bytepos = mpos;
4461 }
4462 mpos = marker->charpos;
4463 if (mpos >= start1 && mpos < end2)
4464 {
4465 if (mpos < end1)
4466 mpos += amt1;
4467 else if (mpos < start2)
4468 mpos += diff;
4469 else
4470 mpos -= amt2;
4471 }
4472 marker->charpos = mpos;
4473 }
4474 }
4475
4476 DEFUN ("transpose-regions", Ftranspose_regions, Stranspose_regions, 4, 5,
4477 "(if (< (length mark-ring) 2)\
4478 (error \"Other region must be marked before transposing two regions\")\
4479 (let* ((num (if current-prefix-arg\
4480 (prefix-numeric-value current-prefix-arg)\
4481 0))\
4482 (ring-length (length mark-ring))\
4483 (eltnum (mod num ring-length))\
4484 (eltnum2 (mod (1+ num) ring-length)))\
4485 (list (point) (mark) (elt mark-ring eltnum) (elt mark-ring eltnum2))))",
4486 doc:
4487
4488
4489
4490
4491
4492
4493
4494
4495
4496
4497
4498
4499
4500 )
4501 (Lisp_Object startr1, Lisp_Object endr1, Lisp_Object startr2, Lisp_Object endr2, Lisp_Object leave_markers)
4502 {
4503 register ptrdiff_t start1, end1, start2, end2;
4504 ptrdiff_t start1_byte, start2_byte, len1_byte, len2_byte, end2_byte;
4505 ptrdiff_t gap, len1, len_mid, len2;
4506 unsigned char *start1_addr, *start2_addr, *temp;
4507
4508 INTERVAL cur_intv, tmp_interval1, tmp_interval_mid, tmp_interval2, tmp_interval3;
4509 Lisp_Object buf;
4510
4511 XSETBUFFER (buf, current_buffer);
4512 cur_intv = buffer_intervals (current_buffer);
4513
4514 validate_region (&startr1, &endr1);
4515 validate_region (&startr2, &endr2);
4516
4517 start1 = XFIXNAT (startr1);
4518 end1 = XFIXNAT (endr1);
4519 start2 = XFIXNAT (startr2);
4520 end2 = XFIXNAT (endr2);
4521 gap = GPT;
4522
4523
4524 if (start2 < end1)
4525 {
4526 register ptrdiff_t glumph = start1;
4527 start1 = start2;
4528 start2 = glumph;
4529 glumph = end1;
4530 end1 = end2;
4531 end2 = glumph;
4532 }
4533
4534 len1 = end1 - start1;
4535 len2 = end2 - start2;
4536
4537 if (start2 < end1)
4538 error ("Transposed regions overlap");
4539
4540 else if ((start1 == end1 || start2 == end2) && end1 == start2)
4541 return Qnil;
4542
4543
4544
4545
4546
4547
4548
4549
4550
4551
4552
4553
4554
4555
4556
4557
4558
4559
4560
4561
4562
4563
4564
4565 start1_byte = CHAR_TO_BYTE (start1);
4566 end2_byte = CHAR_TO_BYTE (end2);
4567
4568
4569
4570 if (start1 < gap && gap < end2)
4571 {
4572 if (gap - start1 < end2 - gap)
4573 move_gap_both (start1, start1_byte);
4574 else
4575 move_gap_both (end2, end2_byte);
4576 }
4577
4578 start2_byte = CHAR_TO_BYTE (start2);
4579 len1_byte = CHAR_TO_BYTE (end1) - start1_byte;
4580 len2_byte = end2_byte - start2_byte;
4581
4582 #ifdef BYTE_COMBINING_DEBUG
4583 if (end1 == start2)
4584 {
4585 if (count_combining_before (BYTE_POS_ADDR (start2_byte),
4586 len2_byte, start1, start1_byte)
4587 || count_combining_before (BYTE_POS_ADDR (start1_byte),
4588 len1_byte, end2, start2_byte + len2_byte)
4589 || count_combining_after (BYTE_POS_ADDR (start1_byte),
4590 len1_byte, end2, start2_byte + len2_byte))
4591 emacs_abort ();
4592 }
4593 else
4594 {
4595 if (count_combining_before (BYTE_POS_ADDR (start2_byte),
4596 len2_byte, start1, start1_byte)
4597 || count_combining_before (BYTE_POS_ADDR (start1_byte),
4598 len1_byte, start2, start2_byte)
4599 || count_combining_after (BYTE_POS_ADDR (start2_byte),
4600 len2_byte, end1, start1_byte + len1_byte)
4601 || count_combining_after (BYTE_POS_ADDR (start1_byte),
4602 len1_byte, end2, start2_byte + len2_byte))
4603 emacs_abort ();
4604 }
4605 #endif
4606
4607
4608
4609
4610
4611 if (end1 == start2)
4612 {
4613 modify_text (start1, end2);
4614 record_change (start1, len1 + len2);
4615
4616 tmp_interval1 = copy_intervals (cur_intv, start1, len1);
4617 tmp_interval2 = copy_intervals (cur_intv, start2, len2);
4618
4619
4620 tmp_interval3 = validate_interval_range (buf, &startr1, &endr2, 0);
4621 if (tmp_interval3)
4622 set_text_properties_1 (startr1, endr2, Qnil, buf, tmp_interval3);
4623
4624 USE_SAFE_ALLOCA;
4625
4626
4627 if (len1_byte < len2_byte)
4628 {
4629 temp = SAFE_ALLOCA (len2_byte);
4630
4631
4632
4633
4634 start1_addr = BYTE_POS_ADDR (start1_byte);
4635 start2_addr = BYTE_POS_ADDR (start2_byte);
4636
4637 memcpy (temp, start2_addr, len2_byte);
4638 memcpy (start1_addr + len2_byte, start1_addr, len1_byte);
4639 memcpy (start1_addr, temp, len2_byte);
4640 }
4641 else
4642
4643 {
4644 temp = SAFE_ALLOCA (len1_byte);
4645 start1_addr = BYTE_POS_ADDR (start1_byte);
4646 start2_addr = BYTE_POS_ADDR (start2_byte);
4647 memcpy (temp, start1_addr, len1_byte);
4648 memcpy (start1_addr, start2_addr, len2_byte);
4649 memcpy (start1_addr + len2_byte, temp, len1_byte);
4650 }
4651
4652 SAFE_FREE ();
4653 graft_intervals_into_buffer (tmp_interval1, start1 + len2,
4654 len1, current_buffer, 0);
4655 graft_intervals_into_buffer (tmp_interval2, start1,
4656 len2, current_buffer, 0);
4657 update_compositions (start1, start1 + len2, CHECK_BORDER);
4658 update_compositions (start1 + len2, end2, CHECK_TAIL);
4659 }
4660
4661 else
4662 {
4663 len_mid = start2_byte - (start1_byte + len1_byte);
4664
4665 if (len1_byte == len2_byte)
4666
4667 {
4668 USE_SAFE_ALLOCA;
4669
4670 modify_text (start1, end2);
4671 record_change (start1, len1);
4672 record_change (start2, len2);
4673 tmp_interval1 = copy_intervals (cur_intv, start1, len1);
4674 tmp_interval2 = copy_intervals (cur_intv, start2, len2);
4675
4676 tmp_interval3 = validate_interval_range (buf, &startr1, &endr1, 0);
4677 if (tmp_interval3)
4678 set_text_properties_1 (startr1, endr1, Qnil, buf, tmp_interval3);
4679
4680 tmp_interval3 = validate_interval_range (buf, &startr2, &endr2, 0);
4681 if (tmp_interval3)
4682 set_text_properties_1 (startr2, endr2, Qnil, buf, tmp_interval3);
4683
4684 temp = SAFE_ALLOCA (len1_byte);
4685 start1_addr = BYTE_POS_ADDR (start1_byte);
4686 start2_addr = BYTE_POS_ADDR (start2_byte);
4687 memcpy (temp, start1_addr, len1_byte);
4688 memcpy (start1_addr, start2_addr, len2_byte);
4689 memcpy (start2_addr, temp, len1_byte);
4690 SAFE_FREE ();
4691
4692 graft_intervals_into_buffer (tmp_interval1, start2,
4693 len1, current_buffer, 0);
4694 graft_intervals_into_buffer (tmp_interval2, start1,
4695 len2, current_buffer, 0);
4696 }
4697
4698 else if (len1_byte < len2_byte)
4699
4700 {
4701 USE_SAFE_ALLOCA;
4702
4703 modify_text (start1, end2);
4704 record_change (start1, (end2 - start1));
4705 tmp_interval1 = copy_intervals (cur_intv, start1, len1);
4706 tmp_interval_mid = copy_intervals (cur_intv, end1, len_mid);
4707 tmp_interval2 = copy_intervals (cur_intv, start2, len2);
4708
4709 tmp_interval3 = validate_interval_range (buf, &startr1, &endr2, 0);
4710 if (tmp_interval3)
4711 set_text_properties_1 (startr1, endr2, Qnil, buf, tmp_interval3);
4712
4713
4714 temp = SAFE_ALLOCA (len2_byte);
4715 start1_addr = BYTE_POS_ADDR (start1_byte);
4716 start2_addr = BYTE_POS_ADDR (start2_byte);
4717 memcpy (temp, start2_addr, len2_byte);
4718 memcpy (start1_addr + len_mid + len2_byte, start1_addr, len1_byte);
4719 memmove (start1_addr + len2_byte, start1_addr + len1_byte, len_mid);
4720 memcpy (start1_addr, temp, len2_byte);
4721 SAFE_FREE ();
4722
4723 graft_intervals_into_buffer (tmp_interval1, end2 - len1,
4724 len1, current_buffer, 0);
4725 graft_intervals_into_buffer (tmp_interval_mid, start1 + len2,
4726 len_mid, current_buffer, 0);
4727 graft_intervals_into_buffer (tmp_interval2, start1,
4728 len2, current_buffer, 0);
4729 }
4730 else
4731
4732 {
4733 USE_SAFE_ALLOCA;
4734
4735 record_change (start1, (end2 - start1));
4736 modify_text (start1, end2);
4737
4738 tmp_interval1 = copy_intervals (cur_intv, start1, len1);
4739 tmp_interval_mid = copy_intervals (cur_intv, end1, len_mid);
4740 tmp_interval2 = copy_intervals (cur_intv, start2, len2);
4741
4742 tmp_interval3 = validate_interval_range (buf, &startr1, &endr2, 0);
4743 if (tmp_interval3)
4744 set_text_properties_1 (startr1, endr2, Qnil, buf, tmp_interval3);
4745
4746
4747 temp = SAFE_ALLOCA (len1_byte);
4748 start1_addr = BYTE_POS_ADDR (start1_byte);
4749 start2_addr = BYTE_POS_ADDR (start2_byte);
4750 memcpy (temp, start1_addr, len1_byte);
4751 memcpy (start1_addr, start2_addr, len2_byte);
4752 memmove (start1_addr + len2_byte, start1_addr + len1_byte, len_mid);
4753 memcpy (start1_addr + len2_byte + len_mid, temp, len1_byte);
4754 SAFE_FREE ();
4755
4756 graft_intervals_into_buffer (tmp_interval1, end2 - len1,
4757 len1, current_buffer, 0);
4758 graft_intervals_into_buffer (tmp_interval_mid, start1 + len2,
4759 len_mid, current_buffer, 0);
4760 graft_intervals_into_buffer (tmp_interval2, start1,
4761 len2, current_buffer, 0);
4762 }
4763
4764 update_compositions (start1, start1 + len2, CHECK_BORDER);
4765 update_compositions (end2 - len1, end2, CHECK_BORDER);
4766 }
4767
4768
4769
4770
4771 if (NILP (leave_markers))
4772 {
4773 transpose_markers (start1, end1, start2, end2,
4774 start1_byte, start1_byte + len1_byte,
4775 start2_byte, start2_byte + len2_byte);
4776 }
4777 else
4778 {
4779
4780
4781
4782
4783 adjust_markers_bytepos (start1, start1_byte, end2, end2_byte, 0);
4784 }
4785
4786 #ifdef HAVE_TREE_SITTER
4787
4788
4789
4790 treesit_record_change (start1_byte, end2_byte, end2_byte);
4791 #endif
4792
4793 signal_after_change (start1, end2 - start1, end2 - start1);
4794 return Qnil;
4795 }
4796
4797
4798 void
4799 syms_of_editfns (void)
4800 {
4801 DEFSYM (Qbuffer_access_fontify_functions, "buffer-access-fontify-functions");
4802 DEFSYM (Qwall, "wall");
4803 DEFSYM (Qpropertize, "propertize");
4804
4805 staticpro (&labeled_restrictions);
4806
4807 DEFVAR_LISP ("inhibit-field-text-motion", Vinhibit_field_text_motion,
4808 doc: );
4809 Vinhibit_field_text_motion = Qnil;
4810
4811 DEFVAR_LISP ("buffer-access-fontify-functions",
4812 Vbuffer_access_fontify_functions,
4813 doc:
4814
4815 );
4816 Vbuffer_access_fontify_functions = Qnil;
4817
4818 {
4819 Lisp_Object obuf;
4820 obuf = Fcurrent_buffer ();
4821
4822 Fset_buffer (Vprin1_to_string_buffer);
4823
4824 Fset (Fmake_local_variable (Qbuffer_access_fontify_functions), Qnil);
4825 Fset_buffer (obuf);
4826 }
4827
4828 DEFVAR_LISP ("buffer-access-fontified-property",
4829 Vbuffer_access_fontified_property,
4830 doc:
4831
4832 );
4833 Vbuffer_access_fontified_property = Qnil;
4834
4835 DEFVAR_LISP ("system-name", Vsystem_name,
4836 doc: );
4837 Vsystem_name = cached_system_name = Qnil;
4838
4839 DEFVAR_LISP ("user-full-name", Vuser_full_name,
4840 doc: );
4841
4842 DEFVAR_LISP ("user-login-name", Vuser_login_name,
4843 doc: );
4844 Vuser_login_name = Qnil;
4845
4846 DEFVAR_LISP ("user-real-login-name", Vuser_real_login_name,
4847 doc: );
4848
4849 DEFVAR_LISP ("operating-system-release", Voperating_system_release,
4850 doc:
4851
4852 );
4853
4854 DEFVAR_BOOL ("binary-as-unsigned",
4855 binary_as_unsigned,
4856 doc:
4857
4858
4859
4860
4861
4862
4863 );
4864 binary_as_unsigned = false;
4865
4866 DEFSYM (Qoutermost_restriction, "outermost-restriction");
4867 Funintern (Qoutermost_restriction, Qnil);
4868
4869 defsubr (&Spropertize);
4870 defsubr (&Schar_equal);
4871 defsubr (&Sgoto_char);
4872 defsubr (&Sstring_to_char);
4873 defsubr (&Schar_to_string);
4874 defsubr (&Sbyte_to_string);
4875 defsubr (&Sbuffer_substring);
4876 defsubr (&Sbuffer_substring_no_properties);
4877 defsubr (&Sbuffer_string);
4878 defsubr (&Sget_pos_property);
4879
4880 defsubr (&Spoint_marker);
4881 defsubr (&Smark_marker);
4882 defsubr (&Spoint);
4883 defsubr (&Sregion_beginning);
4884 defsubr (&Sregion_end);
4885
4886
4887 DEFSYM (Qfield, "field");
4888
4889
4890 DEFSYM (Qboundary, "boundary");
4891
4892 defsubr (&Sfield_beginning);
4893 defsubr (&Sfield_end);
4894 defsubr (&Sfield_string);
4895 defsubr (&Sfield_string_no_properties);
4896 defsubr (&Sdelete_field);
4897 defsubr (&Sconstrain_to_field);
4898
4899 defsubr (&Sline_beginning_position);
4900 defsubr (&Sline_end_position);
4901 defsubr (&Spos_bol);
4902 defsubr (&Spos_eol);
4903
4904 defsubr (&Ssave_excursion);
4905 defsubr (&Ssave_current_buffer);
4906
4907 defsubr (&Sbuffer_size);
4908 defsubr (&Spoint_max);
4909 defsubr (&Spoint_min);
4910 defsubr (&Spoint_min_marker);
4911 defsubr (&Spoint_max_marker);
4912 defsubr (&Sgap_position);
4913 defsubr (&Sgap_size);
4914 defsubr (&Sposition_bytes);
4915 defsubr (&Sbyte_to_position);
4916
4917 defsubr (&Sbobp);
4918 defsubr (&Seobp);
4919 defsubr (&Sbolp);
4920 defsubr (&Seolp);
4921 defsubr (&Sfollowing_char);
4922 defsubr (&Sprevious_char);
4923 defsubr (&Schar_after);
4924 defsubr (&Schar_before);
4925 defsubr (&Sinsert);
4926 defsubr (&Sinsert_before_markers);
4927 defsubr (&Sinsert_and_inherit);
4928 defsubr (&Sinsert_and_inherit_before_markers);
4929 defsubr (&Sinsert_char);
4930 defsubr (&Sinsert_byte);
4931
4932 defsubr (&Sngettext);
4933
4934 defsubr (&Suser_login_name);
4935 defsubr (&Sgroup_name);
4936 defsubr (&Suser_real_login_name);
4937 defsubr (&Suser_uid);
4938 defsubr (&Suser_real_uid);
4939 defsubr (&Sgroup_gid);
4940 defsubr (&Sgroup_real_gid);
4941 defsubr (&Suser_full_name);
4942 defsubr (&Semacs_pid);
4943 defsubr (&Ssystem_name);
4944 defsubr (&Smessage);
4945 defsubr (&Smessage_box);
4946 defsubr (&Smessage_or_box);
4947 defsubr (&Scurrent_message);
4948 defsubr (&Sformat);
4949 defsubr (&Sformat_message);
4950
4951 defsubr (&Sinsert_buffer_substring);
4952 defsubr (&Scompare_buffer_substrings);
4953 defsubr (&Sreplace_buffer_contents);
4954 defsubr (&Ssubst_char_in_region);
4955 defsubr (&Stranslate_region_internal);
4956 defsubr (&Sdelete_region);
4957 defsubr (&Sdelete_and_extract_region);
4958 defsubr (&Swiden);
4959 defsubr (&Snarrow_to_region);
4960 defsubr (&Sinternal__labeled_narrow_to_region);
4961 defsubr (&Sinternal__unlabel_restriction);
4962 defsubr (&Ssave_restriction);
4963 defsubr (&Stranspose_regions);
4964 }