0
|
1 /* Copyright (c) 1994, 1995 Free Software Foundation.
|
|
2 Copyright (c) 1995 Ben Wing.
|
|
3
|
|
4 This file is part of XEmacs.
|
|
5
|
|
6 XEmacs is free software; you can redistribute it and/or modify it
|
|
7 under the terms of the GNU General Public License as published by the
|
|
8 Free Software Foundation; either version 2, or (at your option) any
|
|
9 later version.
|
|
10
|
|
11 XEmacs is distributed in the hope that it will be useful, but WITHOUT
|
|
12 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
13 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
14 for more details.
|
|
15
|
|
16 You should have received a copy of the GNU General Public License
|
|
17 along with XEmacs; see the file COPYING. If not, write to
|
|
18 the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
19 Boston, MA 02111-1307, USA. */
|
|
20
|
|
21 /* Synched up with: Not in FSF. */
|
|
22
|
|
23 #ifndef _XEMACS_EXTENTS_H_
|
|
24 #define _XEMACS_EXTENTS_H_
|
|
25
|
|
26 DECLARE_LRECORD (extent, struct extent);
|
|
27 #define XEXTENT(x) XRECORD (x, extent, struct extent)
|
|
28 #define XSETEXTENT(x, p) XSETRECORD (x, p, extent)
|
|
29 #define EXTENTP(x) RECORDP (x, extent)
|
|
30 #define GC_EXTENTP(x) GC_RECORDP (x, extent)
|
|
31 #define CHECK_EXTENT(x) CHECK_RECORD (x, extent)
|
|
32 #define CONCHECK_EXTENT(x) CONCHECK_RECORD (x, extent)
|
|
33
|
|
34 struct extent
|
|
35 {
|
|
36 struct lrecord_header lheader;
|
|
37
|
|
38 Memind start;
|
|
39 Memind end;
|
|
40 Lisp_Object object; /* A buffer, string, Qnil (extent detached from no
|
|
41 buffer), Qt (destroyed extent) */
|
|
42
|
|
43 /* Extent properties are conceptually a plist, but the most common
|
|
44 props are implemented as bits instead of conses.
|
|
45 */
|
|
46 struct
|
|
47 {
|
|
48 Lisp_Object face;
|
|
49
|
|
50 /* These flags are simply an optimization for common boolean properties
|
|
51 which go onto the extent's property list. Any of them would work if
|
|
52 done in the normal way, but the space savings of doing these in this
|
|
53 way is significant. Note that if you add a flag, there are numerous
|
|
54 places in extents.c that need to know about it.
|
|
55
|
|
56 Another consideration is that some of these properties are accessed
|
|
57 during redisplay, so it's good for access to them to be fast (a bit
|
|
58 reference instead of a search down a plist).
|
|
59
|
|
60 `begin_glyph_layout' and `end_glyph_layout' are unusual in
|
|
61 that they have 4 states instead of 2.
|
|
62
|
|
63 Other special extent properties are stored in an auxiliary
|
|
64 structure that sits at the beginning of the plist. The has_aux
|
|
65 flag indicates whether this structure exists. The has_parent
|
|
66 flag is an optimization indicating whether the extent has a parent
|
|
67 (this could also be determined by looking in the aux structure). */
|
|
68
|
|
69 unsigned int begin_glyph_layout :2; /* 2 text, margins, or whitespace */
|
|
70 unsigned int end_glyph_layout :2; /* 4 text, margins, or whitespace */
|
|
71 unsigned int has_parent : 1; /* 5 extent has a parent */
|
|
72 unsigned int has_aux : 1; /* 6 extent has an aux. structure */
|
|
73 unsigned int start_open : 1; /* 7 insertion behavior at start */
|
|
74 unsigned int end_open : 1; /* 8 insertion behavior at end */
|
|
75 unsigned int unused9 : 1; /* 9 unused */
|
|
76 unsigned int unique : 1; /* 10 there may be only one attached */
|
|
77 unsigned int duplicable : 1; /* 11 copied to strings by kill/undo */
|
96
|
78 unsigned int REPLICATING : 1; /* 12 invoke old extent-replica behav.*/
|
|
79 /* Not used any more */
|
0
|
80 unsigned int detachable : 1; /* 13 extent detaches if text deleted */
|
|
81 unsigned int internal : 1; /* 14 used by map-extents etc. */
|
207
|
82 unsigned int in_red_event : 1; /* 15 An event has been spawned for
|
|
83 initial redisplay. Not exported to
|
|
84 the lisp level */
|
0
|
85 unsigned int unused16 : 1; /* 16 unused */
|
|
86 /* --- Adding more flags will cause the extent struct grow by another
|
|
87 word. It's not clear that this would make a difference, however,
|
|
88 because on 32-bit machines things tend to get allocated in chunks
|
|
89 of 4 bytes. */
|
|
90 } flags;
|
|
91 /* The plist may have an auxiliary structure as its first element */
|
|
92 Lisp_Object plist;
|
|
93 };
|
|
94
|
|
95 /* Basic properties of an extent (not affected by the extent's parent) */
|
|
96 #define extent_object(e) ((e)->object)
|
|
97 #define extent_start(e) ((e)->start + 0)
|
|
98 #define extent_end(e) ((e)->end + 0)
|
|
99 #define set_extent_start(e, val) ((e)->start = (val))
|
|
100 #define set_extent_end(e, val) ((e)->end = (val))
|
|
101 #define extent_endpoint(e, endp) ((endp) ? extent_end (e) : extent_start (e))
|
|
102 #define set_extent_endpoint(e, val, endp) \
|
|
103 ((endp) ? set_extent_end (e, val) : set_extent_start (e, val))
|
|
104 #define extent_detached_p(e) (extent_start (e) < 0)
|
|
105
|
|
106 /* the layouts for glyphs (extent->flags.glyph_layout). Must fit in 2 bits. */
|
|
107 #define GL_TEXT 0
|
|
108 #define GL_OUTSIDE_MARGIN 1
|
|
109 #define GL_INSIDE_MARGIN 2
|
|
110 #define GL_WHITESPACE 3
|
|
111
|
|
112 /* Additional information that may be present in an extent. The idea is
|
|
113 that fast access is provided to this information, but since (hopefully)
|
|
114 most extents won't have this set on them, we usually don't need to
|
|
115 have this structure around and thus the size of an extent is smaller. */
|
|
116
|
185
|
117 typedef struct extent_auxiliary extent_auxiliary;
|
0
|
118 struct extent_auxiliary
|
|
119 {
|
|
120 struct lcrecord_header header;
|
|
121
|
|
122 Lisp_Object begin_glyph;
|
|
123 Lisp_Object end_glyph;
|
|
124 Lisp_Object parent;
|
|
125 /* We use a weak list here. Originally I didn't do this and
|
|
126 depended on having the extent's finalization method remove
|
|
127 itself from its parent's children list. This runs into
|
|
128 lots and lots of problems though because everything is in
|
|
129 a really really bizarre state when an extent's finalization
|
|
130 method is called (it happens in sweep_extents() by way of
|
|
131 ADDITIONAL_FREE_extent()) and it's extremely difficult to
|
|
132 avoid getting hosed by just-freed objects. */
|
|
133 Lisp_Object children;
|
|
134 Lisp_Object invisible;
|
|
135 Lisp_Object read_only;
|
|
136 Lisp_Object mouse_face;
|
207
|
137 Lisp_Object initial_redisplay_function;
|
0
|
138 int priority;
|
|
139 };
|
|
140
|
|
141 extern struct extent_auxiliary extent_auxiliary_defaults;
|
|
142
|
|
143 DECLARE_LRECORD (extent_auxiliary, struct extent_auxiliary);
|
|
144 #define XEXTENT_AUXILIARY(x) \
|
|
145 XRECORD (x, extent_auxiliary, struct extent_auxiliary)
|
|
146 #define XSETEXTENT_AUXILIARY(x, p) XSETRECORD (x, p, extent_auxiliary)
|
|
147 #define EXTENT_AUXILIARYP(x) RECORDP (x, extent_auxiliary)
|
|
148 #define GC_EXTENT_AUXILIARYP(x) GC_RECORDP (x, extent_auxiliary)
|
|
149 #define CHECK_EXTENT_AUXILIARY(x) CHECK_RECORD (x, extent_auxiliary)
|
|
150 #define CONCHECK_EXTENT_AUXILIARY(x) CONCHECK_RECORD (x, extent_auxiliary)
|
|
151
|
|
152 struct extent_info
|
|
153 {
|
|
154 struct lcrecord_header header;
|
|
155
|
|
156 struct extent_list *extents;
|
|
157 struct stack_of_extents *soe;
|
|
158 };
|
|
159
|
|
160 DECLARE_LRECORD (extent_info, struct extent_info);
|
|
161 #define XEXTENT_INFO(x) \
|
|
162 XRECORD (x, extent_info, struct extent_info)
|
|
163 #define XSETEXTENT_INFO(x, p) XSETRECORD (x, p, extent_info)
|
|
164 #define EXTENT_INFOP(x) RECORDP (x, extent_info)
|
|
165 #define GC_EXTENT_INFOP(x) GC_RECORDP (x, extent_info)
|
|
166 #define CHECK_EXTENT_INFO(x) CHECK_RECORD (x, extent_info)
|
|
167 #define CONCHECK_EXTENT_INFO(x) CONCHECK_RECORD (x, extent_info)
|
|
168
|
|
169 void flush_cached_extent_info (Lisp_Object extent_info);
|
|
170
|
|
171 /* Note that we take pains in all the macros below never to evaluate
|
|
172 the extent argument more than once. This may not be necessary
|
|
173 but is much less likely to introduce subtle bugs. */
|
|
174
|
|
175 MAC_DECLARE_EXTERN (EXTENT, MTancestor_extent)
|
|
176 MAC_DECLARE_EXTERN (EXTENT, MTaux_extent)
|
|
177 MAC_DECLARE_EXTERN (EXTENT, MTplist_extent)
|
|
178 MAC_DECLARE_EXTERN (EXTENT, MTensure_extent)
|
|
179 MAC_DECLARE_EXTERN (EXTENT, MTset_extent)
|
|
180
|
|
181 /* extent_ancestor() chases all the parent links until there aren't any
|
|
182 more. extent_ancestor_1() does the same thing but it a function;
|
|
183 the following macro optimizes the most common case. */
|
|
184
|
|
185 #define extent_ancestor(e) \
|
|
186 MAC_BEGIN \
|
|
187 MAC_DECLARE (EXTENT, MTancestor_extent, e) \
|
|
188 (MTancestor_extent->flags.has_parent ? \
|
|
189 extent_ancestor_1 (MTancestor_extent) : \
|
|
190 MTancestor_extent) \
|
|
191 MAC_END
|
|
192
|
|
193 /* a "normal" field is one that is stored in the `struct flags' structure
|
|
194 in an extent. an "aux" field is one that is stored in the extent's
|
|
195 auxiliary structure.
|
|
196
|
|
197 The functions below that have `extent_no_chase' in their name operate
|
|
198 on an extent directly (ignoring its parent), and should normally
|
|
199 only be used on extents known not to have a parent. The other
|
|
200 versions chase down any parent links. */
|
|
201
|
|
202 #define extent_no_chase_normal_field(e, field) ((e)->flags.field)
|
|
203
|
|
204 #define extent_no_chase_aux_field(e, field) \
|
|
205 MAC_BEGIN \
|
|
206 MAC_DECLARE (EXTENT, MTaux_extent, e) \
|
|
207 (MTaux_extent->flags.has_aux ? \
|
|
208 XEXTENT_AUXILIARY (XCONS (MTaux_extent->plist)->car)->field \
|
|
209 : extent_auxiliary_defaults.field) \
|
|
210 MAC_END
|
|
211
|
|
212 #define extent_normal_field(e, field) \
|
|
213 extent_no_chase_normal_field (extent_ancestor (e), field)
|
|
214
|
|
215 #define extent_aux_field(e, field) \
|
|
216 extent_no_chase_aux_field (extent_ancestor (e), field)
|
|
217
|
|
218 #define ensure_extent_has_auxiliary(e) \
|
|
219 MAC_BEGIN \
|
|
220 MAC_DECLARE (EXTENT, MTensure_extent, e) \
|
|
221 (MTensure_extent->flags.has_aux ? (void) 0 : \
|
|
222 allocate_extent_auxiliary (MTensure_extent)) \
|
185
|
223 MAC_END
|
0
|
224
|
|
225 #define set_extent_no_chase_aux_field(e, field, value) \
|
|
226 MAC_BEGIN \
|
|
227 MAC_DECLARE (EXTENT, MTset_extent, e) \
|
|
228 ensure_extent_has_auxiliary (MTset_extent) \
|
|
229 MAC_SEP \
|
|
230 XEXTENT_AUXILIARY (XCONS (MTset_extent->plist)->car)->field = \
|
|
231 (value) \
|
|
232 MAC_END
|
|
233
|
|
234 #define set_extent_no_chase_normal_field(e, field, value) \
|
|
235 extent_no_chase_normal_field (e, field) = (value)
|
|
236
|
|
237 #define set_extent_aux_field(e, field, value) \
|
|
238 set_extent_no_chase_aux_field (extent_ancestor (e), field, value)
|
|
239
|
|
240 #define set_extent_normal_field(e, field, value) \
|
|
241 set_extent_ancestor_normal_field (extent_no_chase (e), field, value)
|
|
242
|
|
243 /* The `parent' and `children' fields are not affected by any
|
|
244 parent links. We don't provide any settors for these fields
|
|
245 because they need special handling and it's cleaner just to
|
|
246 do this in the particular functions that need to do this. */
|
|
247
|
|
248 #define extent_parent(e) extent_no_chase_aux_field (e, parent)
|
|
249 #define extent_children(e) extent_no_chase_aux_field (e, children)
|
|
250
|
|
251 #define extent_begin_glyph(e) extent_aux_field (e, begin_glyph)
|
|
252 #define extent_end_glyph(e) extent_aux_field (e, end_glyph)
|
|
253 #define extent_priority(e) extent_aux_field (e, priority)
|
|
254 #define extent_invisible(e) extent_aux_field (e, invisible)
|
|
255 #define extent_read_only(e) extent_aux_field (e, read_only)
|
|
256 #define extent_mouse_face(e) extent_aux_field (e, mouse_face)
|
207
|
257 #define extent_initial_redisplay_function(e) extent_aux_field (e, initial_redisplay_function)
|
0
|
258
|
|
259 #define set_extent_begin_glyph(e, value) \
|
|
260 set_extent_aux_field (e, begin_glyph, value)
|
|
261 #define set_extent_end_glyph(e, value) \
|
|
262 set_extent_aux_field (e, end_glyph, value)
|
|
263 #define set_extent_priority(e, value) \
|
|
264 set_extent_aux_field (e, priority, value)
|
|
265 #define set_extent_invisible_1(e, value) \
|
|
266 set_extent_aux_field (e, invisible, value)
|
|
267 #define set_extent_read_only(e, value) \
|
|
268 set_extent_aux_field (e, read_only, value)
|
|
269 #define set_extent_mouse_face(e, value) \
|
|
270 set_extent_aux_field (e, mouse_face, value)
|
207
|
271 /* Use Fset_extent_initial_redisplay_function unless you know what you are ding */
|
|
272 #define set_extent_initial_redisplay_function(e, value) \
|
|
273 set_extent_aux_field (e, initial_redisplay_function, value)
|
0
|
274
|
|
275 #define extent_face(e) extent_normal_field (e, face)
|
|
276 #define extent_begin_glyph_layout(e) \
|
|
277 extent_normal_field (e, begin_glyph_layout)
|
|
278 #define extent_end_glyph_layout(e) extent_normal_field (e, end_glyph_layout)
|
|
279 #define extent_start_open_p(e) extent_normal_field (e, start_open)
|
|
280 #define extent_end_open_p(e) extent_normal_field (e, end_open)
|
|
281 #define extent_unique_p(e) extent_normal_field (e, unique)
|
|
282 #define extent_duplicable_p(e) extent_normal_field (e, duplicable)
|
|
283 #define extent_detachable_p(e) extent_normal_field (e, detachable)
|
|
284 #define extent_internal_p(e) extent_normal_field (e, internal)
|
207
|
285 #define extent_in_red_event_p(e) extent_normal_field (e, in_red_event)
|
0
|
286
|
|
287 #define extent_no_chase_plist_addr(e) \
|
|
288 MAC_BEGIN \
|
|
289 MAC_DECLARE (EXTENT, MTplist_extent, e) \
|
|
290 (MTplist_extent->flags.has_aux ? \
|
|
291 &XCONS (MTplist_extent->plist)->cdr : \
|
|
292 &MTplist_extent->plist) \
|
185
|
293 MAC_END
|
0
|
294 #define extent_no_chase_plist(e) (*extent_no_chase_plist_addr (e))
|
|
295
|
|
296 #define extent_plist_addr(e) extent_no_chase_plist_addr (extent_ancestor (e))
|
|
297 #define extent_plist_slot(e) extent_no_chase_plist (extent_ancestor (e))
|
|
298
|
|
299 /* flags for map_extents() and friends */
|
|
300 #define ME_END_CLOSED (1 << 0)
|
|
301 #define ME_START_OPEN (1 << 1)
|
|
302 #define ME_ALL_EXTENTS_CLOSED (1 << 2)
|
|
303 #define ME_ALL_EXTENTS_OPEN (2 << 2)
|
|
304 #define ME_ALL_EXTENTS_CLOSED_OPEN (3 << 2)
|
|
305 #define ME_ALL_EXTENTS_OPEN_CLOSED (4 << 2)
|
|
306 #define ME_ALL_EXTENTS_MASK (7 << 2)
|
|
307 #define ME_START_IN_REGION (1 << 5)
|
|
308 #define ME_END_IN_REGION (2 << 5)
|
|
309 #define ME_START_AND_END_IN_REGION (3 << 5)
|
|
310 #define ME_START_OR_END_IN_REGION (4 << 5)
|
|
311 #define ME_IN_REGION_MASK (7 << 5)
|
|
312 #define ME_NEGATE_IN_REGION (1 << 8)
|
|
313 /* the following flags are internal-only */
|
|
314 #define ME_INCLUDE_INTERNAL (1 << 9)
|
|
315 #define ME_MIGHT_THROW (1 << 10)
|
|
316 #define ME_MIGHT_MODIFY_TEXT (1 << 11)
|
|
317 #define ME_MIGHT_MODIFY_EXTENTS (1 << 12)
|
|
318 #define ME_MIGHT_MOVE_SOE (1 << 13)
|
|
319 #define ME_MIGHT_CALL_ELISP (ME_MIGHT_THROW | ME_MIGHT_MODIFY_TEXT | \
|
|
320 ME_MIGHT_MODIFY_EXTENTS | ME_MIGHT_MOVE_SOE)
|
|
321
|
|
322
|
|
323 #define EXTENT_LIVE_P(e) (!EQ (extent_object (e), Qt))
|
|
324
|
|
325 #define CHECK_LIVE_EXTENT(x) \
|
|
326 do { CHECK_EXTENT (x); \
|
|
327 if (!EXTENT_LIVE_P (XEXTENT (x))) \
|
|
328 dead_wrong_type_argument (Qextent_live_p, (x)); } while (0)
|
|
329 #define CONCHECK_LIVE_EXTENT(x) \
|
|
330 do { CONCHECK_EXTENT (x); \
|
|
331 if (!EXTENT_LIVE_P (XEXTENT (x))) \
|
|
332 x = wrong_type_argument (Qextent_live_p, (x)); } while (0)
|
|
333
|
|
334 extern Lisp_Object Qextent_live_p;
|
|
335
|
|
336 extern int inside_undo;
|
|
337
|
|
338 struct extent_fragment *extent_fragment_new (Lisp_Object buffer_or_string,
|
|
339 struct frame *frm);
|
|
340 face_index extent_fragment_update (struct window *w,
|
|
341 struct extent_fragment *ef,
|
|
342 /* Note this is in Bytinds */
|
|
343 Bytind pos);
|
|
344 void extent_fragment_delete (struct extent_fragment *ef);
|
|
345
|
|
346 extern Lisp_Object Vlast_highlighted_extent;
|
|
347
|
|
348
|
|
349 #ifdef emacs /* things other than emacs want the structs */
|
|
350
|
|
351 /* from alloc.c */
|
|
352 struct extent *allocate_extent (void);
|
|
353
|
|
354 /* from extents.c */
|
|
355 EXTENT extent_ancestor_1 (EXTENT e);
|
|
356 void allocate_extent_auxiliary (EXTENT ext);
|
|
357 void init_buffer_extents (struct buffer *b);
|
|
358 void uninit_buffer_extents (struct buffer *b);
|
|
359 void map_extents (Bufpos from, Bufpos to, int (*fn) (EXTENT extent,
|
|
360 void * arg),
|
|
361 void *arg, Lisp_Object obj, EXTENT after,
|
|
362 unsigned int flags);
|
|
363
|
|
364 /* Note the following five functions are NOT in Bufpos's */
|
|
365 void adjust_extents (Lisp_Object object, Memind from,
|
|
366 Memind to, int amount);
|
|
367 void adjust_extents_for_deletion (Lisp_Object object, Bytind from,
|
|
368 Bytind to, int gapsize,
|
98
|
369 int numdel, int movegapsize);
|
0
|
370 void verify_extent_modification (Lisp_Object object, Bytind from,
|
|
371 Bytind to,
|
|
372 Lisp_Object inhibit_read_only_value);
|
|
373 void process_extents_for_insertion (Lisp_Object object,
|
|
374 Bytind opoint, Bytecount length);
|
|
375 void process_extents_for_deletion (Lisp_Object object, Bytind from,
|
|
376 Bytind to, int destroy_them);
|
|
377
|
|
378 void set_extent_glyph (EXTENT extent, Lisp_Object glyph, int endp,
|
|
379 unsigned int layout);
|
|
380
|
|
381 void add_string_extents (Lisp_Object string, struct buffer *buf,
|
|
382 Bytind opoint, Bytecount length);
|
|
383 void splice_in_string_extents (Lisp_Object string, struct buffer *buf,
|
|
384 Bytind opoint, Bytecount length,
|
|
385 Bytecount pos);
|
|
386 void copy_string_extents (Lisp_Object new_string,
|
|
387 Lisp_Object old_string,
|
|
388 Bytecount new_pos, Bytecount old_pos,
|
|
389 Bytecount length);
|
|
390
|
|
391 void detach_all_extents (Lisp_Object object);
|
|
392 void set_extent_endpoints (EXTENT extent, Bytind s, Bytind e,
|
|
393 Lisp_Object object);
|
|
394
|
|
395 #ifdef ERROR_CHECK_EXTENTS
|
|
396 void sledgehammer_extent_check (Lisp_Object obj);
|
|
397 #endif
|
|
398
|
|
399 #ifdef MEMORY_USAGE_STATS
|
|
400 int compute_buffer_extent_usage (struct buffer *b,
|
|
401 struct overhead_stats *ovstats);
|
|
402 #endif
|
|
403
|
|
404 #endif /* emacs */
|
|
405
|
|
406 #endif /* _XEMACS_EXTENTS_H_ */
|