diff options
| author | Eli Zaretskii | 2011-05-09 05:59:23 -0400 |
|---|---|---|
| committer | Eli Zaretskii | 2011-05-09 05:59:23 -0400 |
| commit | 14fe7b530dc927a88169a841afc0cd806593dea8 (patch) | |
| tree | 510192ce4c22c74ffec5b97327ea5e4a03a5a66c /src/window.c | |
| parent | 6eea50c73be34e865dabf14cbd2d0e7c4f64e6a0 (diff) | |
| download | emacs-14fe7b530dc927a88169a841afc0cd806593dea8.tar.gz emacs-14fe7b530dc927a88169a841afc0cd806593dea8.zip | |
Backport revisions 2011-04-24T05:30:24Z!eggert@cs.ucla.edu..2011-04-25T19:40:22Z!eggert@cs.ucla.edu (inclusive) from trunk (bug#8623)
The next log entry shows the actual changes by Paul Eggert.
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.
src/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.
src/lisp.h: Say "vectorlike header" rather than "vector header.
(struct vectorlike_header): Rename from struct vector_header.
(XVECTORLIKE_HEADER_SIZE): Renamed from XVECTOR_HEADER_SIZE.
All uses changed.
(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.
src/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".
src/buffer.h (struct buffer): Likewise.
src/font.h (struct font_spec, struct font_entity, struct font): Likewise.
src/frame.h (struct frame): Likewise.
src/process.h (struct Lisp_Process): Likewise.
src/termhooks.h (struct terminal): Likewise.
src/window.c (struct save_window_data, struct saved_window): Likewise.
src/window.h (struct window): Likewise.
src/alloc.c (allocate_buffer, Fmake_bool_vector, allocate_pseudovector):
Use XSETPVECTYPESIZE, not XSETPVECTYPE, to avoid aliasing problems.
src/buffer.c (init_buffer_once): Likewise.
src/lread.c (defsubr): Use XSETTYPED_PVECTYPE, since Lisp_Subr is a
special case.
src/process.c (Fformat_network_address): Use local var for size,
for brevity.
src/fns.c (vector): Remove; this old hack is no longer needed.
src/bytecode.c (exec_byte_code): Don't use XVECTOR before CHECK_VECTOR.
Diffstat (limited to 'src/window.c')
| -rw-r--r-- | src/window.c | 14 |
1 files changed, 5 insertions, 9 deletions
diff --git a/src/window.c b/src/window.c index 3e6062a7153..894ff65b600 100644 --- a/src/window.c +++ b/src/window.c | |||
| @@ -5931,8 +5931,7 @@ zero means top of window, negative means relative to bottom of window. */) | |||
| 5931 | 5931 | ||
| 5932 | struct save_window_data | 5932 | struct save_window_data |
| 5933 | { | 5933 | { |
| 5934 | EMACS_UINT size; | 5934 | struct vectorlike_header header; |
| 5935 | struct Lisp_Vector *next_from_Lisp_Vector_struct; | ||
| 5936 | Lisp_Object selected_frame; | 5935 | Lisp_Object selected_frame; |
| 5937 | Lisp_Object current_window; | 5936 | Lisp_Object current_window; |
| 5938 | Lisp_Object current_buffer; | 5937 | Lisp_Object current_buffer; |
| @@ -5954,10 +5953,7 @@ struct save_window_data | |||
| 5954 | /* This is saved as a Lisp_Vector */ | 5953 | /* This is saved as a Lisp_Vector */ |
| 5955 | struct saved_window | 5954 | struct saved_window |
| 5956 | { | 5955 | { |
| 5957 | /* these first two must agree with struct Lisp_Vector in lisp.h */ | 5956 | struct vectorlike_header header; |
| 5958 | EMACS_UINT size; | ||
| 5959 | struct Lisp_Vector *next_from_Lisp_Vector_struct; | ||
| 5960 | |||
| 5961 | Lisp_Object window; | 5957 | Lisp_Object window; |
| 5962 | Lisp_Object buffer, start, pointm, mark; | 5958 | Lisp_Object buffer, start, pointm, mark; |
| 5963 | Lisp_Object left_col, top_line, total_cols, total_lines; | 5959 | Lisp_Object left_col, top_line, total_cols, total_lines; |
| @@ -6141,7 +6137,7 @@ the return value is nil. Otherwise the value is t. */) | |||
| 6141 | dead. */ | 6137 | dead. */ |
| 6142 | delete_all_subwindows (XWINDOW (FRAME_ROOT_WINDOW (f))); | 6138 | delete_all_subwindows (XWINDOW (FRAME_ROOT_WINDOW (f))); |
| 6143 | 6139 | ||
| 6144 | for (k = 0; k < saved_windows->size; k++) | 6140 | for (k = 0; k < saved_windows->header.size; k++) |
| 6145 | { | 6141 | { |
| 6146 | p = SAVED_WINDOW_N (saved_windows, k); | 6142 | p = SAVED_WINDOW_N (saved_windows, k); |
| 6147 | w = XWINDOW (p->window); | 6143 | w = XWINDOW (p->window); |
| @@ -7078,10 +7074,10 @@ compare_window_configurations (c1, c2, ignore_positions) | |||
| 7078 | return 0; | 7074 | return 0; |
| 7079 | 7075 | ||
| 7080 | /* Verify that the two confis have the same number of windows. */ | 7076 | /* Verify that the two confis have the same number of windows. */ |
| 7081 | if (sw1->size != sw2->size) | 7077 | if (sw1->header.size != sw2->header.size) |
| 7082 | return 0; | 7078 | return 0; |
| 7083 | 7079 | ||
| 7084 | for (i = 0; i < sw1->size; i++) | 7080 | for (i = 0; i < sw1->header.size; i++) |
| 7085 | { | 7081 | { |
| 7086 | struct saved_window *p1, *p2; | 7082 | struct saved_window *p1, *p2; |
| 7087 | int w1_is_current, w2_is_current; | 7083 | int w1_is_current, w2_is_current; |