Mercurial > hg > xemacs-beta
annotate src/casetab.c @ 5159:cb303ff63e76
merge
| author | Ben Wing <ben@xemacs.org> |
|---|---|
| date | Fri, 19 Mar 2010 17:02:11 -0500 |
| parents | 88bd4f3ef8e4 |
| children | 5ddbab03b0e6 |
| rev | line source |
|---|---|
| 428 | 1 /* XEmacs routines to deal with case tables. |
| 2 Copyright (C) 1987, 1992, 1993, 1994 Free Software Foundation, Inc. | |
| 3 Copyright (C) 1995 Sun Microsystems, Inc. | |
|
5146
88bd4f3ef8e4
make lrecord UID's have a separate UID space for each object, resurrect debug SOE code in extents.c
Ben Wing <ben@xemacs.org>
parents:
5142
diff
changeset
|
4 Copyright (C) 2002, 2010 Ben Wing. |
| 428 | 5 |
| 6 This file is part of XEmacs. | |
| 7 | |
| 8 XEmacs is free software; you can redistribute it and/or modify it | |
| 9 under the terms of the GNU General Public License as published by the | |
| 10 Free Software Foundation; either version 2, or (at your option) any | |
| 11 later version. | |
| 12 | |
| 13 XEmacs is distributed in the hope that it will be useful, but WITHOUT | |
| 14 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | |
| 15 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License | |
| 16 for more details. | |
| 17 | |
| 18 You should have received a copy of the GNU General Public License | |
| 19 along with XEmacs; see the file COPYING. If not, write to | |
| 20 the Free Software Foundation, Inc., 59 Temple Place - Suite 330, | |
| 21 Boston, MA 02111-1307, USA. */ | |
| 22 | |
| 771 | 23 /* Synched up with: FSF 19.28. Between FSF 19.28 and 19.30, casetab.c |
| 428 | 24 was rewritten to use junky FSF char tables. Meanwhile I rewrote it |
| 771 | 25 to use more logical char tables. --ben */ |
| 428 | 26 |
| 826 | 27 /* Written by Howard Gayle. */ |
| 428 | 28 |
| 29 /* Modified for Mule by Ben Wing. */ | |
| 30 | |
| 826 | 31 /* The four tables in a case table are downcase, upcase, canon, and eqv. |
| 32 Each is a char-table. Their workings are rather non-obvious. | |
| 33 | |
| 34 (1) `downcase' is the only obvious table: Map a character to its | |
| 35 lowercase equivalent. | |
| 771 | 36 |
| 826 | 37 (2) `upcase' does *NOT* map a character to its uppercase equivalent, |
| 38 despite its name. Rather, it maps lowercase characters to their | |
| 39 uppercase equivalent, and uppercase characters to *ANYTHING BUT* their | |
| 40 uppercase equivalent (currently, their lowercase equivalent), and | |
| 41 characters without case to themselves. It is used to determine if a | |
| 42 character "has no case" (no uppercase or lowercase mapping). #### This | |
| 43 is way bogus. Just use the obvious implementation of uppercase mapping | |
| 44 and of NOCASE_P. | |
| 446 | 45 |
| 826 | 46 (3) `canon' maps each character to a "canonical" lowercase, such that if |
| 47 two different uppercase characters map to the same lowercase character, | |
| 48 or vice versa, both characters will have the same entry in the canon | |
| 49 table. | |
| 446 | 50 |
|
4407
4ee73bbe4f8e
Always use boyer_moore in ASCII or Latin-1 buffers with ASCII search strings.
Aidan Kehoe <kehoea@parhasard.net>
parents:
3017
diff
changeset
|
51 (4) `eqv' lists the "equivalence classes" defined by `canon'. Imagine |
| 826 | 52 that all characters are divided into groups having the same `canon' |
|
4407
4ee73bbe4f8e
Always use boyer_moore in ASCII or Latin-1 buffers with ASCII search strings.
Aidan Kehoe <kehoea@parhasard.net>
parents:
3017
diff
changeset
|
53 entry; these groups are called "equivalence classes" and `eqv' lists them |
|
4ee73bbe4f8e
Always use boyer_moore in ASCII or Latin-1 buffers with ASCII search strings.
Aidan Kehoe <kehoea@parhasard.net>
parents:
3017
diff
changeset
|
54 by linking the characters in each equivalence class together in a |
|
4ee73bbe4f8e
Always use boyer_moore in ASCII or Latin-1 buffers with ASCII search strings.
Aidan Kehoe <kehoea@parhasard.net>
parents:
3017
diff
changeset
|
55 circular list. That is, to find out all all the members of a given char's |
| 4890 | 56 equivalence class, you need something like the following code: |
| 826 | 57 |
|
4407
4ee73bbe4f8e
Always use boyer_moore in ASCII or Latin-1 buffers with ASCII search strings.
Aidan Kehoe <kehoea@parhasard.net>
parents:
3017
diff
changeset
|
58 (let* ((char ?i) |
|
4ee73bbe4f8e
Always use boyer_moore in ASCII or Latin-1 buffers with ASCII search strings.
Aidan Kehoe <kehoea@parhasard.net>
parents:
3017
diff
changeset
|
59 (original-char char) |
|
4ee73bbe4f8e
Always use boyer_moore in ASCII or Latin-1 buffers with ASCII search strings.
Aidan Kehoe <kehoea@parhasard.net>
parents:
3017
diff
changeset
|
60 (standard-case-eqv (case-table-eqv (standard-case-table)))) |
|
4ee73bbe4f8e
Always use boyer_moore in ASCII or Latin-1 buffers with ASCII search strings.
Aidan Kehoe <kehoea@parhasard.net>
parents:
3017
diff
changeset
|
61 (loop |
|
4ee73bbe4f8e
Always use boyer_moore in ASCII or Latin-1 buffers with ASCII search strings.
Aidan Kehoe <kehoea@parhasard.net>
parents:
3017
diff
changeset
|
62 with res = (list char) |
|
4ee73bbe4f8e
Always use boyer_moore in ASCII or Latin-1 buffers with ASCII search strings.
Aidan Kehoe <kehoea@parhasard.net>
parents:
3017
diff
changeset
|
63 until (eq (setq char (get-char-table char standard-case-eqv)) |
|
4ee73bbe4f8e
Always use boyer_moore in ASCII or Latin-1 buffers with ASCII search strings.
Aidan Kehoe <kehoea@parhasard.net>
parents:
3017
diff
changeset
|
64 original-char) |
|
4ee73bbe4f8e
Always use boyer_moore in ASCII or Latin-1 buffers with ASCII search strings.
Aidan Kehoe <kehoea@parhasard.net>
parents:
3017
diff
changeset
|
65 do (push char res) |
|
4ee73bbe4f8e
Always use boyer_moore in ASCII or Latin-1 buffers with ASCII search strings.
Aidan Kehoe <kehoea@parhasard.net>
parents:
3017
diff
changeset
|
66 finally return res)) |
|
4ee73bbe4f8e
Always use boyer_moore in ASCII or Latin-1 buffers with ASCII search strings.
Aidan Kehoe <kehoea@parhasard.net>
parents:
3017
diff
changeset
|
67 |
|
4ee73bbe4f8e
Always use boyer_moore in ASCII or Latin-1 buffers with ASCII search strings.
Aidan Kehoe <kehoea@parhasard.net>
parents:
3017
diff
changeset
|
68 (Where #'case-table-eqv doesn't yet exist, and probably never will, given |
|
4ee73bbe4f8e
Always use boyer_moore in ASCII or Latin-1 buffers with ASCII search strings.
Aidan Kehoe <kehoea@parhasard.net>
parents:
3017
diff
changeset
|
69 that the C code needs to keep it in a consistent state so Lisp can't mess |
|
4ee73bbe4f8e
Always use boyer_moore in ASCII or Latin-1 buffers with ASCII search strings.
Aidan Kehoe <kehoea@parhasard.net>
parents:
3017
diff
changeset
|
70 around with it.) |
|
4ee73bbe4f8e
Always use boyer_moore in ASCII or Latin-1 buffers with ASCII search strings.
Aidan Kehoe <kehoea@parhasard.net>
parents:
3017
diff
changeset
|
71 |
|
4ee73bbe4f8e
Always use boyer_moore in ASCII or Latin-1 buffers with ASCII search strings.
Aidan Kehoe <kehoea@parhasard.net>
parents:
3017
diff
changeset
|
72 `canon' is used when doing case-insensitive comparisons. `eqv' is |
| 826 | 73 used in the Boyer-Moore search code. |
| 74 */ | |
| 428 | 75 |
| 76 #include <config.h> | |
| 77 #include "lisp.h" | |
| 78 #include "buffer.h" | |
| 79 #include "opaque.h" | |
| 446 | 80 #include "chartab.h" |
| 81 #include "casetab.h" | |
| 428 | 82 |
| 446 | 83 Lisp_Object Qcase_tablep, Qdowncase, Qupcase; |
| 84 Lisp_Object Vstandard_case_table; | |
| 428 | 85 |
| 446 | 86 Lisp_Object case_table_char (Lisp_Object ch, Lisp_Object table); |
| 428 | 87 |
| 826 | 88 #define STRING256_P(obj) ((STRINGP (obj) && string_char_length (obj) == 256)) |
| 446 | 89 |
| 90 static Lisp_Object | |
| 91 mark_case_table (Lisp_Object obj) | |
| 92 { | |
| 93 Lisp_Case_Table *ct = XCASE_TABLE (obj); | |
| 94 | |
| 95 mark_object (CASE_TABLE_DOWNCASE (ct)); | |
| 96 mark_object (CASE_TABLE_UPCASE (ct)); | |
| 97 mark_object (CASE_TABLE_CANON (ct)); | |
| 98 mark_object (CASE_TABLE_EQV (ct)); | |
| 99 return Qnil; | |
| 100 } | |
| 101 | |
| 102 static void | |
| 2286 | 103 print_case_table (Lisp_Object obj, Lisp_Object printcharfun, |
| 104 int UNUSED (escapeflag)) | |
| 446 | 105 { |
| 106 Lisp_Case_Table *ct = XCASE_TABLE (obj); | |
| 107 if (print_readably) | |
|
5142
f965e31a35f0
reduce lcrecord headers to 2 words, rename printing_unreadable_object
Ben Wing <ben@xemacs.org>
parents:
5127
diff
changeset
|
108 printing_unreadable_lisp_object (obj, 0); |
| 826 | 109 write_fmt_string_lisp |
| 110 (printcharfun, "#<case-table downcase=%s upcase=%s canon=%s eqv=%s ", 4, | |
| 111 CASE_TABLE_DOWNCASE (ct), CASE_TABLE_UPCASE (ct), | |
| 112 CASE_TABLE_CANON (ct), CASE_TABLE_EQV (ct)); | |
|
5146
88bd4f3ef8e4
make lrecord UID's have a separate UID space for each object, resurrect debug SOE code in extents.c
Ben Wing <ben@xemacs.org>
parents:
5142
diff
changeset
|
113 write_fmt_string (printcharfun, "0x%x>", LISP_OBJECT_UID (obj)); |
| 446 | 114 } |
| 115 | |
| 1204 | 116 static const struct memory_description case_table_description [] = { |
| 446 | 117 { XD_LISP_OBJECT, offsetof (Lisp_Case_Table, downcase_table) }, |
| 118 { XD_LISP_OBJECT, offsetof (Lisp_Case_Table, upcase_table) }, | |
| 119 { XD_LISP_OBJECT, offsetof (Lisp_Case_Table, case_canon_table) }, | |
| 120 { XD_LISP_OBJECT, offsetof (Lisp_Case_Table, case_eqv_table) }, | |
| 121 { XD_END } | |
| 122 }; | |
| 123 | |
| 934 | 124 |
|
5118
e0db3c197671
merge up to latest default branch, doesn't compile yet
Ben Wing <ben@xemacs.org>
diff
changeset
|
125 DEFINE_DUMPABLE_LISP_OBJECT ("case-table", case_table, |
|
e0db3c197671
merge up to latest default branch, doesn't compile yet
Ben Wing <ben@xemacs.org>
diff
changeset
|
126 mark_case_table, print_case_table, 0, |
|
e0db3c197671
merge up to latest default branch, doesn't compile yet
Ben Wing <ben@xemacs.org>
diff
changeset
|
127 0, 0, case_table_description, Lisp_Case_Table); |
| 446 | 128 |
| 129 static Lisp_Object | |
| 826 | 130 allocate_case_table (int init_tables) |
| 446 | 131 { |
|
5127
a9c41067dd88
more cleanups, terminology clarification, lots of doc work
Ben Wing <ben@xemacs.org>
parents:
5125
diff
changeset
|
132 Lisp_Object obj = ALLOC_NORMAL_LISP_OBJECT (case_table); |
|
5117
3742ea8250b5
Checking in final CVS version of workspace 'ben-lisp-object'
Ben Wing <ben@xemacs.org>
parents:
3017
diff
changeset
|
133 Lisp_Case_Table *ct = XCASE_TABLE (obj); |
| 446 | 134 |
| 826 | 135 if (init_tables) |
| 136 { | |
| 137 SET_CASE_TABLE_DOWNCASE (ct, MAKE_TRT_TABLE ()); | |
| 138 SET_CASE_TABLE_UPCASE (ct, MAKE_TRT_TABLE ()); | |
| 139 SET_CASE_TABLE_CANON (ct, MAKE_TRT_TABLE ()); | |
| 140 SET_CASE_TABLE_EQV (ct, MAKE_TRT_TABLE ()); | |
| 141 } | |
| 142 else | |
| 143 { | |
| 144 SET_CASE_TABLE_DOWNCASE (ct, Qnil); | |
| 145 SET_CASE_TABLE_UPCASE (ct, Qnil); | |
| 146 SET_CASE_TABLE_CANON (ct, Qnil); | |
| 147 SET_CASE_TABLE_EQV (ct, Qnil); | |
| 148 } | |
|
5117
3742ea8250b5
Checking in final CVS version of workspace 'ben-lisp-object'
Ben Wing <ben@xemacs.org>
parents:
3017
diff
changeset
|
149 return obj; |
| 826 | 150 } |
| 446 | 151 |
| 826 | 152 DEFUN ("make-case-table", Fmake_case_table, 0, 0, 0, /* |
| 153 Create a new, empty case table. | |
| 154 */ | |
| 155 ()) | |
| 156 { | |
| 157 return allocate_case_table (1); | |
| 446 | 158 } |
| 428 | 159 |
| 160 DEFUN ("case-table-p", Fcase_table_p, 1, 1, 0, /* | |
| 444 | 161 Return t if OBJECT is a case table. |
| 428 | 162 See `set-case-table' for more information on these data structures. |
| 163 */ | |
| 444 | 164 (object)) |
| 428 | 165 { |
| 446 | 166 if (CASE_TABLEP (object)) |
| 167 return Qt; | |
| 168 else | |
| 169 { | |
| 170 Lisp_Object down, up, canon, eqv; | |
| 171 if (!CONSP (object)) | |
| 172 return Qnil; | |
| 173 down = XCAR (object); object = XCDR (object); | |
| 174 if (!CONSP (object)) | |
| 175 return Qnil; | |
| 176 up = XCAR (object); object = XCDR (object); | |
| 177 if (!CONSP (object)) | |
| 178 return Qnil; | |
| 179 canon = XCAR (object); object = XCDR (object); | |
| 180 if (!CONSP (object)) | |
| 181 return Qnil; | |
| 182 eqv = XCAR (object); | |
| 428 | 183 |
| 446 | 184 return ((STRING256_P (down) |
| 185 && (NILP (up) || STRING256_P (up)) | |
| 186 && ((NILP (canon) && NILP (eqv)) | |
| 187 || STRING256_P (canon)) | |
| 188 && (NILP (eqv) || STRING256_P (eqv))) | |
| 189 ? Qt : Qnil); | |
| 190 | |
| 191 } | |
| 428 | 192 } |
| 193 | |
| 194 static Lisp_Object | |
| 444 | 195 check_case_table (Lisp_Object object) |
| 428 | 196 { |
| 446 | 197 /* This function can GC */ |
| 444 | 198 while (NILP (Fcase_table_p (object))) |
| 199 object = wrong_type_argument (Qcase_tablep, object); | |
| 200 return object; | |
| 428 | 201 } |
| 202 | |
| 446 | 203 Lisp_Object |
| 204 case_table_char (Lisp_Object ch, Lisp_Object table) | |
| 205 { | |
| 206 Lisp_Object ct_char; | |
| 826 | 207 ct_char = get_char_table (XCHAR (ch), table); |
| 446 | 208 if (NILP (ct_char)) |
| 209 return ch; | |
| 210 else | |
| 211 return ct_char; | |
| 212 } | |
| 213 | |
| 214 DEFUN ("get-case-table", Fget_case_table, 3, 3, 0, /* | |
| 215 Return CHAR-CASE version of CHARACTER in CASE-TABLE. | |
| 216 | |
| 826 | 217 CHAR-CASE is either `downcase' or `upcase'. |
| 446 | 218 */ |
| 219 (char_case, character, case_table)) | |
| 220 { | |
| 221 CHECK_CHAR (character); | |
| 222 CHECK_CASE_TABLE (case_table); | |
| 223 if (EQ (char_case, Qdowncase)) | |
| 224 return case_table_char (character, XCASE_TABLE_DOWNCASE (case_table)); | |
| 225 else if (EQ (char_case, Qupcase)) | |
| 226 return case_table_char (character, XCASE_TABLE_UPCASE (case_table)); | |
| 227 else | |
| 563 | 228 invalid_constant ("Char case must be downcase or upcase", char_case); |
| 446 | 229 |
| 230 return Qnil; /* Not reached. */ | |
| 231 } | |
| 232 | |
| 233 DEFUN ("put-case-table", Fput_case_table, 4, 4, 0, /* | |
| 234 Set CHAR-CASE version of CHARACTER to be VALUE in CASE-TABLE. | |
| 235 | |
| 826 | 236 CHAR-CASE is either `downcase' or `upcase'. |
| 446 | 237 See also `put-case-table-pair'. |
| 238 */ | |
| 239 (char_case, character, value, case_table)) | |
| 240 { | |
| 241 CHECK_CHAR (character); | |
| 242 CHECK_CHAR (value); | |
| 243 | |
| 244 if (EQ (char_case, Qdowncase)) | |
| 245 { | |
| 246 Fput_char_table (character, value, XCASE_TABLE_DOWNCASE (case_table)); | |
| 826 | 247 /* This one is not at all intuitive. See comment at top of file. */ |
| 446 | 248 Fput_char_table (character, value, XCASE_TABLE_UPCASE (case_table)); |
| 249 } | |
| 250 else if (EQ (char_case, Qupcase)) | |
| 251 { | |
| 252 Fput_char_table (character, value, XCASE_TABLE_UPCASE (case_table)); | |
| 826 | 253 Fput_char_table (character, character, |
| 254 XCASE_TABLE_DOWNCASE (case_table)); | |
| 446 | 255 } |
| 256 else | |
| 826 | 257 invalid_constant ("CHAR-CASE must be downcase or upcase", char_case); |
| 446 | 258 |
| 826 | 259 XCASE_TABLE (case_table)->dirty = 1; |
| 446 | 260 return Qnil; |
| 261 } | |
| 262 | |
| 263 DEFUN ("put-case-table-pair", Fput_case_table_pair, 3, 3, 0, /* | |
| 264 Make UC and LC a pair of inter-case-converting letters in CASE-TABLE. | |
| 265 UC is an uppercase character and LC is a downcase character. | |
| 266 */ | |
| 267 (uc, lc, case_table)) | |
| 268 { | |
| 269 CHECK_CHAR (uc); | |
| 270 CHECK_CHAR (lc); | |
| 271 CHECK_CASE_TABLE (case_table); | |
| 272 | |
| 273 Fput_char_table (lc, lc, XCASE_TABLE_DOWNCASE (case_table)); | |
| 274 Fput_char_table (uc, lc, XCASE_TABLE_UPCASE (case_table)); | |
| 275 Fput_char_table (uc, lc, XCASE_TABLE_DOWNCASE (case_table)); | |
| 276 Fput_char_table (lc, uc, XCASE_TABLE_UPCASE (case_table)); | |
| 277 | |
| 826 | 278 XCASE_TABLE (case_table)->dirty = 1; |
| 446 | 279 return Qnil; |
| 280 } | |
| 281 | |
| 282 DEFUN ("copy-case-table", Fcopy_case_table, 1, 1, 0, /* | |
| 283 Return a new case table which is a copy of CASE-TABLE | |
| 284 */ | |
| 285 (case_table)) | |
| 286 { | |
| 287 Lisp_Object new_obj; | |
| 288 CHECK_CASE_TABLE (case_table); | |
| 289 | |
| 826 | 290 new_obj = allocate_case_table (0); |
| 446 | 291 XSET_CASE_TABLE_DOWNCASE |
| 292 (new_obj, Fcopy_char_table (XCASE_TABLE_DOWNCASE (case_table))); | |
| 293 XSET_CASE_TABLE_UPCASE | |
| 294 (new_obj, Fcopy_char_table (XCASE_TABLE_UPCASE (case_table))); | |
| 295 XSET_CASE_TABLE_CANON | |
| 296 (new_obj, Fcopy_char_table (XCASE_TABLE_CANON (case_table))); | |
| 297 XSET_CASE_TABLE_EQV | |
| 298 (new_obj, Fcopy_char_table (XCASE_TABLE_EQV (case_table))); | |
| 299 return new_obj; | |
| 300 } | |
| 301 | |
| 826 | 302 static int |
| 303 compute_canon_mapper (struct chartab_range *range, | |
| 2286 | 304 Lisp_Object UNUSED (table), Lisp_Object val, void *arg) |
| 826 | 305 { |
| 5013 | 306 Lisp_Object casetab = GET_LISP_FROM_VOID (arg); |
| 826 | 307 if (range->type == CHARTAB_RANGE_CHAR) |
| 308 SET_TRT_TABLE_OF (XCASE_TABLE_CANON (casetab), range->ch, | |
| 309 TRT_TABLE_OF (XCASE_TABLE_DOWNCASE (casetab), | |
| 310 TRT_TABLE_OF (XCASE_TABLE_UPCASE (casetab), | |
| 311 XCHAR (val)))); | |
| 312 | |
| 313 return 0; | |
| 314 } | |
| 315 | |
| 316 static int | |
| 317 initialize_identity_mapper (struct chartab_range *range, | |
| 2286 | 318 Lisp_Object UNUSED (table), |
| 319 Lisp_Object UNUSED (val), void *arg) | |
| 826 | 320 { |
| 5013 | 321 Lisp_Object trt = GET_LISP_FROM_VOID (arg); |
| 826 | 322 if (range->type == CHARTAB_RANGE_CHAR) |
| 323 SET_TRT_TABLE_OF (trt, range->ch, range->ch); | |
| 324 | |
| 325 return 0; | |
| 326 } | |
| 327 | |
| 328 static int | |
| 329 compute_up_or_eqv_mapper (struct chartab_range *range, | |
| 2286 | 330 Lisp_Object UNUSED (table), |
| 331 Lisp_Object val, void *arg) | |
| 826 | 332 { |
| 5013 | 333 Lisp_Object inverse = GET_LISP_FROM_VOID (arg); |
| 867 | 334 Ichar toch = XCHAR (val); |
| 826 | 335 |
| 336 if (range->type == CHARTAB_RANGE_CHAR && range->ch != toch) | |
| 337 { | |
| 867 | 338 Ichar c = TRT_TABLE_OF (inverse, toch); |
| 826 | 339 SET_TRT_TABLE_OF (inverse, toch, range->ch); |
| 340 SET_TRT_TABLE_OF (inverse, range->ch, c); | |
| 341 } | |
| 342 | |
| 343 return 0; | |
| 344 } | |
| 345 | |
| 346 /* Recomputing the canonical and equivalency tables from scratch is a | |
| 347 lengthy process, and doing them incrementally is extremely difficult or | |
| 348 perhaps impossible -- and certainly not worth it. To avoid lots of | |
| 349 excessive recomputation when lots of stuff is incrementally added, we | |
| 350 just store a dirty flag and then recompute when a value from the canon | |
| 351 or eqv tables is actually needed. */ | |
| 352 | |
| 353 void | |
| 354 recompute_case_table (Lisp_Object casetab) | |
| 355 { | |
| 356 struct chartab_range range; | |
| 357 | |
| 358 range.type = CHARTAB_RANGE_ALL; | |
| 359 /* Turn off dirty flag first so we don't get infinite recursion when | |
| 360 retrieving the values below! */ | |
| 361 XCASE_TABLE (casetab)->dirty = 0; | |
| 362 map_char_table (XCASE_TABLE_DOWNCASE (casetab), &range, | |
| 5013 | 363 compute_canon_mapper, STORE_LISP_IN_VOID (casetab)); |
| 826 | 364 map_char_table (XCASE_TABLE_CANON (casetab), &range, |
| 365 initialize_identity_mapper, | |
| 5013 | 366 STORE_LISP_IN_VOID (XCASE_TABLE_EQV (casetab))); |
| 826 | 367 map_char_table (XCASE_TABLE_CANON (casetab), &range, |
| 368 compute_up_or_eqv_mapper, | |
| 5013 | 369 STORE_LISP_IN_VOID (XCASE_TABLE_EQV (casetab))); |
| 826 | 370 } |
| 371 | |
| 428 | 372 DEFUN ("current-case-table", Fcurrent_case_table, 0, 1, 0, /* |
| 373 Return the case table of BUFFER, which defaults to the current buffer. | |
| 374 */ | |
| 375 (buffer)) | |
| 376 { | |
| 377 struct buffer *buf = decode_buffer (buffer, 0); | |
| 378 | |
| 446 | 379 return buf->case_table; |
| 428 | 380 } |
| 381 | |
| 382 DEFUN ("standard-case-table", Fstandard_case_table, 0, 0, 0, /* | |
| 383 Return the standard case table. | |
| 384 This is the one used for new buffers. | |
| 385 */ | |
| 386 ()) | |
| 387 { | |
| 446 | 388 return Vstandard_case_table; |
| 428 | 389 } |
| 390 | |
| 826 | 391 static void |
| 392 convert_old_style_syntax_string (Lisp_Object table, Lisp_Object string) | |
| 393 { | |
| 867 | 394 Ichar i; |
| 826 | 395 |
| 396 for (i = 0; i < 256; i++) | |
| 867 | 397 SET_TRT_TABLE_OF (table, i, string_ichar (string, i)); |
| 826 | 398 } |
| 399 | |
| 400 static Lisp_Object | |
| 401 set_case_table (Lisp_Object table, int standard) | |
| 402 { | |
| 403 /* This function can GC */ | |
| 404 struct buffer *buf = | |
| 405 standard ? XBUFFER (Vbuffer_defaults) : current_buffer; | |
|
4906
6ef8256a020a
implement equalp in C, fix case-folding, add equal() method for keymaps
Ben Wing <ben@xemacs.org>
parents:
4890
diff
changeset
|
406 Lisp_Object casetab; |
| 826 | 407 |
| 408 check_case_table (table); | |
| 409 | |
| 410 if (CASE_TABLEP (table)) | |
|
4906
6ef8256a020a
implement equalp in C, fix case-folding, add equal() method for keymaps
Ben Wing <ben@xemacs.org>
parents:
4890
diff
changeset
|
411 casetab = table; |
| 826 | 412 else |
| 413 { | |
| 414 /* For backward compatibility. */ | |
| 415 Lisp_Object down, up, canon, eqv, tail = table; | |
| 416 struct chartab_range range; | |
| 417 | |
|
4906
6ef8256a020a
implement equalp in C, fix case-folding, add equal() method for keymaps
Ben Wing <ben@xemacs.org>
parents:
4890
diff
changeset
|
418 casetab = Fmake_case_table (); |
|
6ef8256a020a
implement equalp in C, fix case-folding, add equal() method for keymaps
Ben Wing <ben@xemacs.org>
parents:
4890
diff
changeset
|
419 |
| 826 | 420 range.type = CHARTAB_RANGE_ALL; |
| 421 | |
| 422 Freset_char_table (XCASE_TABLE_DOWNCASE (casetab)); | |
| 423 Freset_char_table (XCASE_TABLE_UPCASE (casetab)); | |
| 424 Freset_char_table (XCASE_TABLE_CANON (casetab)); | |
| 425 Freset_char_table (XCASE_TABLE_EQV (casetab)); | |
| 426 | |
| 427 down = XCAR (tail); tail = XCDR (tail); | |
| 428 up = XCAR (tail); tail = XCDR (tail); | |
| 429 canon = XCAR (tail); tail = XCDR (tail); | |
| 430 eqv = XCAR (tail); | |
| 431 | |
| 432 convert_old_style_syntax_string (XCASE_TABLE_DOWNCASE (casetab), down); | |
| 433 | |
| 434 if (NILP (up)) | |
| 435 { | |
| 436 map_char_table (XCASE_TABLE_DOWNCASE (casetab), &range, | |
| 437 initialize_identity_mapper, | |
| 5013 | 438 STORE_LISP_IN_VOID (XCASE_TABLE_UPCASE (casetab))); |
| 826 | 439 map_char_table (XCASE_TABLE_DOWNCASE (casetab), &range, |
| 440 compute_up_or_eqv_mapper, | |
| 5013 | 441 STORE_LISP_IN_VOID (XCASE_TABLE_UPCASE (casetab))); |
| 826 | 442 } |
| 443 else | |
| 444 convert_old_style_syntax_string (XCASE_TABLE_UPCASE (casetab), up); | |
| 445 | |
| 446 if (NILP (canon)) | |
| 447 map_char_table (XCASE_TABLE_DOWNCASE (casetab), &range, | |
| 5013 | 448 compute_canon_mapper, STORE_LISP_IN_VOID (casetab)); |
| 826 | 449 else |
| 450 convert_old_style_syntax_string (XCASE_TABLE_CANON (casetab), canon); | |
| 451 | |
| 452 if (NILP (eqv)) | |
| 453 { | |
| 454 map_char_table (XCASE_TABLE_CANON (casetab), &range, | |
| 455 initialize_identity_mapper, | |
| 5013 | 456 STORE_LISP_IN_VOID (XCASE_TABLE_EQV (casetab))); |
| 826 | 457 map_char_table (XCASE_TABLE_CANON (casetab), &range, |
| 458 compute_up_or_eqv_mapper, | |
| 5013 | 459 STORE_LISP_IN_VOID (XCASE_TABLE_EQV (casetab))); |
| 826 | 460 } |
| 461 else | |
| 462 convert_old_style_syntax_string (XCASE_TABLE_CANON (casetab), eqv); | |
| 463 } | |
| 464 | |
|
4906
6ef8256a020a
implement equalp in C, fix case-folding, add equal() method for keymaps
Ben Wing <ben@xemacs.org>
parents:
4890
diff
changeset
|
465 |
|
6ef8256a020a
implement equalp in C, fix case-folding, add equal() method for keymaps
Ben Wing <ben@xemacs.org>
parents:
4890
diff
changeset
|
466 if (standard) |
|
6ef8256a020a
implement equalp in C, fix case-folding, add equal() method for keymaps
Ben Wing <ben@xemacs.org>
parents:
4890
diff
changeset
|
467 Vstandard_case_table = casetab; |
|
6ef8256a020a
implement equalp in C, fix case-folding, add equal() method for keymaps
Ben Wing <ben@xemacs.org>
parents:
4890
diff
changeset
|
468 |
|
6ef8256a020a
implement equalp in C, fix case-folding, add equal() method for keymaps
Ben Wing <ben@xemacs.org>
parents:
4890
diff
changeset
|
469 buf->case_table = casetab; |
|
6ef8256a020a
implement equalp in C, fix case-folding, add equal() method for keymaps
Ben Wing <ben@xemacs.org>
parents:
4890
diff
changeset
|
470 |
| 826 | 471 return buf->case_table; |
| 472 } | |
| 428 | 473 |
| 474 DEFUN ("set-case-table", Fset_case_table, 1, 1, 0, /* | |
| 444 | 475 Select CASE-TABLE as the new case table for the current buffer. |
| 446 | 476 A case table is a case-table object or list |
| 477 (DOWNCASE UPCASE CANONICALIZE EQUIVALENCES) | |
| 428 | 478 where each element is either nil or a string of length 256. |
| 446 | 479 The latter is provided for backward-compatibility. |
| 428 | 480 DOWNCASE maps each character to its lower-case equivalent. |
| 481 UPCASE maps each character to its upper-case equivalent; | |
| 482 if lower and upper case characters are in 1-1 correspondence, | |
| 483 you may use nil and the upcase table will be deduced from DOWNCASE. | |
| 484 CANONICALIZE maps each character to a canonical equivalent; | |
| 485 any two characters that are related by case-conversion have the same | |
| 486 canonical equivalent character; it may be nil, in which case it is | |
| 487 deduced from DOWNCASE and UPCASE. | |
| 488 EQUIVALENCES is a map that cyclicly permutes each equivalence class | |
| 489 (of characters with the same canonical equivalent); it may be nil, | |
| 490 in which case it is deduced from CANONICALIZE. | |
| 491 | |
| 446 | 492 See also `get-case-table', `put-case-table' and `put-case-table-pair'. |
| 428 | 493 */ |
| 444 | 494 (case_table)) |
| 428 | 495 { |
| 446 | 496 /* This function can GC */ |
| 444 | 497 return set_case_table (case_table, 0); |
| 428 | 498 } |
| 499 | |
| 500 DEFUN ("set-standard-case-table", Fset_standard_case_table, 1, 1, 0, /* | |
| 444 | 501 Select CASE-TABLE as the new standard case table for new buffers. |
| 428 | 502 See `set-case-table' for more info on case tables. |
| 503 */ | |
| 444 | 504 (case_table)) |
| 428 | 505 { |
| 446 | 506 /* This function can GC */ |
| 444 | 507 return set_case_table (case_table, 1); |
| 428 | 508 } |
| 509 | |
| 510 | |
| 511 void | |
| 512 syms_of_casetab (void) | |
| 513 { | |
|
5117
3742ea8250b5
Checking in final CVS version of workspace 'ben-lisp-object'
Ben Wing <ben@xemacs.org>
parents:
3017
diff
changeset
|
514 INIT_LISP_OBJECT (case_table); |
| 446 | 515 |
| 563 | 516 DEFSYMBOL_MULTIWORD_PREDICATE (Qcase_tablep); |
| 517 DEFSYMBOL (Qdowncase); | |
| 518 DEFSYMBOL (Qupcase); | |
| 428 | 519 |
| 826 | 520 DEFSUBR (Fmake_case_table); |
| 428 | 521 DEFSUBR (Fcase_table_p); |
| 446 | 522 DEFSUBR (Fget_case_table); |
| 523 DEFSUBR (Fput_case_table); | |
| 524 DEFSUBR (Fput_case_table_pair); | |
| 428 | 525 DEFSUBR (Fcurrent_case_table); |
| 526 DEFSUBR (Fstandard_case_table); | |
| 446 | 527 DEFSUBR (Fcopy_case_table); |
| 428 | 528 DEFSUBR (Fset_case_table); |
| 529 DEFSUBR (Fset_standard_case_table); | |
| 530 } | |
| 531 | |
| 532 void | |
| 533 complex_vars_of_casetab (void) | |
| 534 { | |
| 867 | 535 REGISTER Ichar i; |
| 428 | 536 |
| 446 | 537 staticpro (&Vstandard_case_table); |
| 428 | 538 |
| 826 | 539 Vstandard_case_table = allocate_case_table (1); |
| 428 | 540 |
| 541 for (i = 0; i < 256; i++) | |
| 542 { | |
| 543 unsigned char lowered = tolower (i); | |
| 544 | |
| 826 | 545 SET_TRT_TABLE_OF (XCASE_TABLE_DOWNCASE (Vstandard_case_table), i, |
| 546 lowered); | |
| 428 | 547 } |
| 548 | |
| 549 for (i = 0; i < 256; i++) | |
| 550 { | |
| 551 unsigned char flipped = (isupper (i) ? tolower (i) | |
| 552 : (islower (i) ? toupper (i) : i)); | |
| 553 | |
| 826 | 554 SET_TRT_TABLE_OF (XCASE_TABLE_UPCASE (Vstandard_case_table), i, |
| 555 flipped); | |
| 428 | 556 } |
| 826 | 557 |
| 558 recompute_case_table (Vstandard_case_table); | |
| 428 | 559 } |
