aboutsummaryrefslogtreecommitdiffstats
path: root/src/buffer.c
diff options
context:
space:
mode:
authorPaul Eggert2011-04-25 00:14:46 -0700
committerPaul Eggert2011-04-25 00:14:46 -0700
commiteab3844f965646b62e242aa622754b86d1fd3444 (patch)
tree10246105e5facc5d61ccf797dfa05debdb1877c1 /src/buffer.c
parent0df1eac54fdf82a80a7611fe421d94a23ebd4a0a (diff)
downloademacs-eab3844f965646b62e242aa622754b86d1fd3444.tar.gz
emacs-eab3844f965646b62e242aa622754b86d1fd3444.zip
lisp.h: Fix a problem with aliasing and vector headers.
GCC 4.6.0 optimizes based on type-based alias analysis. For example, if b is of type struct buffer * and v of type struct Lisp_Vector *, then gcc -O2 was incorrectly assuming that &b->size != &v->size, and therefore "v->size = 1; b->size = 2; return v->size;" must therefore return 1. This assumption is incorrect for Emacs, since it type-puns struct Lisp_Vector * with many other types. To fix this problem, this patch adds a new type struct vector_header that documents the constraints on layout of vectors and pseudovectors, and helps optimizing compilers not get fooled by Emacs's type punning. It also adds the macros XSETTYPED_PVECTYPE XSETTYPED_PSEUDOVECTOR, TYPED_PSEUDOVECTORP, for similar reasons. * lisp.h (XVECTOR_SIZE): New convenience macro. All previous uses of XVECTOR (foo)->size replaced to use this macro, to avoid the hassle of writing XVECTOR (foo)->header.size. (XVECTOR_HEADER_SIZE): New macro, for use in XSETPSEUDOVECTOR. (XSETTYPED_PVECTYPE): New macro, specifying the name of the size member. (XSETPVECTYPE): Rewrite in terms of new macro. (XSETPVECTYPESIZE): New macro, specifying both type and size. This is a bit clearer, and further avoids the possibility of undesirable aliasing. (XSETTYPED_PSEUDOVECTOR): New macro, specifying the size. (XSETPSEUDOVECTOR): Rewrite in terms of XSETTYPED_PSEUDOVECTOR and XVECTOR_HEADER_SIZE. (XSETSUBR): Rewrite in terms of XSETTYPED_PSEUDOVECTOR and XSIZE, since Lisp_Subr is a special case (no "next" field). (ASIZE): Rewrite in terms of XVECTOR_SIZE. (struct vector_header): New type. (TYPED_PSEUDOVECTORP): New macro, also specifying the C type of the object, to help avoid aliasing. (PSEUDOVECTORP): Rewrite in terms of TYPED_PSEUDOVECTORP. (SUBRP): Likewise, since Lisp_Subr is a special case. * lisp.h (struct Lisp_Vector, struct Lisp_Char_Table): (struct Lisp_Sub_Char_Table, struct Lisp_Bool_Vector): (struct Lisp_Hash_Table): Combine first two members into a single struct vector_header member. All uses of "size" and "next" members changed to be "header.size" and "header.next". * buffer.h (struct buffer): Likewise. * font.h (struct font_spec, struct font_entity, struct font): Likewise. * frame.h (struct frame): Likewise. * process.h (struct Lisp_Process): Likewise. * termhooks.h (struct terminal): Likewise. * window.c (struct save_window_data, struct saved_window): Likewise. * window.h (struct window): Likewise. * alloc.c (allocate_buffer, Fmake_bool_vector, allocate_pseudovector): Use XSETPVECTYPESIZE, not XSETPVECTYPE, to avoid aliasing problems. * buffer.c (init_buffer_once): Likewise. * lread.c (defsubr): Use XSETTYPED_PVECTYPE, since Lisp_Subr is a special case. * process.c (Fformat_network_address): Use local var for size, for brevity.
Diffstat (limited to 'src/buffer.c')
-rw-r--r--src/buffer.c18
1 files changed, 9 insertions, 9 deletions
diff --git a/src/buffer.c b/src/buffer.c
index c649836adb8..6ced4fbae87 100644
--- a/src/buffer.c
+++ b/src/buffer.c
@@ -43,7 +43,7 @@ along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>. */
43struct buffer *current_buffer; /* the current buffer */ 43struct buffer *current_buffer; /* the current buffer */
44 44
45/* First buffer in chain of all buffers (in reverse order of creation). 45/* First buffer in chain of all buffers (in reverse order of creation).
46 Threaded through ->next. */ 46 Threaded through ->header.next. */
47 47
48struct buffer *all_buffers; 48struct buffer *all_buffers;
49 49
@@ -359,7 +359,7 @@ even if it is dead. The return value is never nil. */)
359 b->prevent_redisplay_optimizations_p = 1; 359 b->prevent_redisplay_optimizations_p = 1;
360 360
361 /* Put this on the chain of all buffers including killed ones. */ 361 /* Put this on the chain of all buffers including killed ones. */
362 b->next = all_buffers; 362 b->header.next.buffer = all_buffers;
363 all_buffers = b; 363 all_buffers = b;
364 364
365 /* An ordinary buffer normally doesn't need markers 365 /* An ordinary buffer normally doesn't need markers
@@ -588,7 +588,7 @@ CLONE nil means the indirect buffer's state is reset to default values. */)
588 BVAR (b, width_table) = Qnil; 588 BVAR (b, width_table) = Qnil;
589 589
590 /* Put this on the chain of all buffers including killed ones. */ 590 /* Put this on the chain of all buffers including killed ones. */
591 b->next = all_buffers; 591 b->header.next.buffer = all_buffers;
592 all_buffers = b; 592 all_buffers = b;
593 593
594 name = Fcopy_sequence (name); 594 name = Fcopy_sequence (name);
@@ -1458,7 +1458,7 @@ with SIGHUP. */)
1458 1458
1459 GCPRO1 (buffer); 1459 GCPRO1 (buffer);
1460 1460
1461 for (other = all_buffers; other; other = other->next) 1461 for (other = all_buffers; other; other = other->header.next.buffer)
1462 /* all_buffers contains dead buffers too; 1462 /* all_buffers contains dead buffers too;
1463 don't re-kill them. */ 1463 don't re-kill them. */
1464 if (other->base_buffer == b && !NILP (BVAR (other, name))) 1464 if (other->base_buffer == b && !NILP (BVAR (other, name)))
@@ -2099,7 +2099,7 @@ DEFUN ("buffer-swap-text", Fbuffer_swap_text, Sbuffer_swap_text,
2099 2099
2100 { /* This is probably harder to make work. */ 2100 { /* This is probably harder to make work. */
2101 struct buffer *other; 2101 struct buffer *other;
2102 for (other = all_buffers; other; other = other->next) 2102 for (other = all_buffers; other; other = other->header.next.buffer)
2103 if (other->base_buffer == other_buffer 2103 if (other->base_buffer == other_buffer
2104 || other->base_buffer == current_buffer) 2104 || other->base_buffer == current_buffer)
2105 error ("One of the buffers to swap has indirect buffers"); 2105 error ("One of the buffers to swap has indirect buffers");
@@ -2476,7 +2476,7 @@ current buffer is cleared. */)
2476 2476
2477 /* Copy this buffer's new multibyte status 2477 /* Copy this buffer's new multibyte status
2478 into all of its indirect buffers. */ 2478 into all of its indirect buffers. */
2479 for (other = all_buffers; other; other = other->next) 2479 for (other = all_buffers; other; other = other->header.next.buffer)
2480 if (other->base_buffer == current_buffer && !NILP (BVAR (other, name))) 2480 if (other->base_buffer == current_buffer && !NILP (BVAR (other, name)))
2481 { 2481 {
2482 BVAR (other, enable_multibyte_characters) 2482 BVAR (other, enable_multibyte_characters)
@@ -4178,7 +4178,7 @@ static int last_overlay_modification_hooks_used;
4178static void 4178static void
4179add_overlay_mod_hooklist (Lisp_Object functionlist, Lisp_Object overlay) 4179add_overlay_mod_hooklist (Lisp_Object functionlist, Lisp_Object overlay)
4180{ 4180{
4181 int oldsize = XVECTOR (last_overlay_modification_hooks)->size; 4181 int oldsize = XVECTOR_SIZE (last_overlay_modification_hooks);
4182 4182
4183 if (last_overlay_modification_hooks_used == oldsize) 4183 if (last_overlay_modification_hooks_used == oldsize)
4184 last_overlay_modification_hooks = larger_vector 4184 last_overlay_modification_hooks = larger_vector
@@ -4973,9 +4973,9 @@ init_buffer_once (void)
4973 buffer_local_symbols.text = &buffer_local_symbols.own_text; 4973 buffer_local_symbols.text = &buffer_local_symbols.own_text;
4974 BUF_INTERVALS (&buffer_defaults) = 0; 4974 BUF_INTERVALS (&buffer_defaults) = 0;
4975 BUF_INTERVALS (&buffer_local_symbols) = 0; 4975 BUF_INTERVALS (&buffer_local_symbols) = 0;
4976 XSETPVECTYPE (&buffer_defaults, PVEC_BUFFER); 4976 XSETPVECTYPESIZE (&buffer_defaults, PVEC_BUFFER, 0);
4977 XSETBUFFER (Vbuffer_defaults, &buffer_defaults); 4977 XSETBUFFER (Vbuffer_defaults, &buffer_defaults);
4978 XSETPVECTYPE (&buffer_local_symbols, PVEC_BUFFER); 4978 XSETPVECTYPESIZE (&buffer_local_symbols, PVEC_BUFFER, 0);
4979 XSETBUFFER (Vbuffer_local_symbols, &buffer_local_symbols); 4979 XSETBUFFER (Vbuffer_local_symbols, &buffer_local_symbols);
4980 4980
4981 /* Set up the default values of various buffer slots. */ 4981 /* Set up the default values of various buffer slots. */