diff options
| author | Paul Eggert | 2011-04-25 00:14:46 -0700 |
|---|---|---|
| committer | Paul Eggert | 2011-04-25 00:14:46 -0700 |
| commit | eab3844f965646b62e242aa622754b86d1fd3444 (patch) | |
| tree | 10246105e5facc5d61ccf797dfa05debdb1877c1 /src/process.c | |
| parent | 0df1eac54fdf82a80a7611fe421d94a23ebd4a0a (diff) | |
| download | emacs-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/process.c')
| -rw-r--r-- | src/process.c | 15 |
1 files changed, 8 insertions, 7 deletions
diff --git a/src/process.c b/src/process.c index f25d10dbd67..89a5f3e0386 100644 --- a/src/process.c +++ b/src/process.c | |||
| @@ -1188,25 +1188,26 @@ Returns nil if format of ADDRESS is invalid. */) | |||
| 1188 | if (VECTORP (address)) /* AF_INET or AF_INET6 */ | 1188 | if (VECTORP (address)) /* AF_INET or AF_INET6 */ |
| 1189 | { | 1189 | { |
| 1190 | register struct Lisp_Vector *p = XVECTOR (address); | 1190 | register struct Lisp_Vector *p = XVECTOR (address); |
| 1191 | EMACS_UINT size = p->header.size; | ||
| 1191 | Lisp_Object args[10]; | 1192 | Lisp_Object args[10]; |
| 1192 | int nargs, i; | 1193 | int nargs, i; |
| 1193 | 1194 | ||
| 1194 | if (p->size == 4 || (p->size == 5 && !NILP (omit_port))) | 1195 | if (size == 4 || (size == 5 && !NILP (omit_port))) |
| 1195 | { | 1196 | { |
| 1196 | args[0] = build_string ("%d.%d.%d.%d"); | 1197 | args[0] = build_string ("%d.%d.%d.%d"); |
| 1197 | nargs = 4; | 1198 | nargs = 4; |
| 1198 | } | 1199 | } |
| 1199 | else if (p->size == 5) | 1200 | else if (size == 5) |
| 1200 | { | 1201 | { |
| 1201 | args[0] = build_string ("%d.%d.%d.%d:%d"); | 1202 | args[0] = build_string ("%d.%d.%d.%d:%d"); |
| 1202 | nargs = 5; | 1203 | nargs = 5; |
| 1203 | } | 1204 | } |
| 1204 | else if (p->size == 8 || (p->size == 9 && !NILP (omit_port))) | 1205 | else if (size == 8 || (size == 9 && !NILP (omit_port))) |
| 1205 | { | 1206 | { |
| 1206 | args[0] = build_string ("%x:%x:%x:%x:%x:%x:%x:%x"); | 1207 | args[0] = build_string ("%x:%x:%x:%x:%x:%x:%x:%x"); |
| 1207 | nargs = 8; | 1208 | nargs = 8; |
| 1208 | } | 1209 | } |
| 1209 | else if (p->size == 9) | 1210 | else if (size == 9) |
| 1210 | { | 1211 | { |
| 1211 | args[0] = build_string ("[%x:%x:%x:%x:%x:%x:%x:%x]:%d"); | 1212 | args[0] = build_string ("[%x:%x:%x:%x:%x:%x:%x:%x]:%d"); |
| 1212 | nargs = 9; | 1213 | nargs = 9; |
| @@ -2064,13 +2065,13 @@ get_lisp_to_sockaddr_size (Lisp_Object address, int *familyp) | |||
| 2064 | if (VECTORP (address)) | 2065 | if (VECTORP (address)) |
| 2065 | { | 2066 | { |
| 2066 | p = XVECTOR (address); | 2067 | p = XVECTOR (address); |
| 2067 | if (p->size == 5) | 2068 | if (p->header.size == 5) |
| 2068 | { | 2069 | { |
| 2069 | *familyp = AF_INET; | 2070 | *familyp = AF_INET; |
| 2070 | return sizeof (struct sockaddr_in); | 2071 | return sizeof (struct sockaddr_in); |
| 2071 | } | 2072 | } |
| 2072 | #ifdef AF_INET6 | 2073 | #ifdef AF_INET6 |
| 2073 | else if (p->size == 9) | 2074 | else if (p->header.size == 9) |
| 2074 | { | 2075 | { |
| 2075 | *familyp = AF_INET6; | 2076 | *familyp = AF_INET6; |
| 2076 | return sizeof (struct sockaddr_in6); | 2077 | return sizeof (struct sockaddr_in6); |
| @@ -2089,7 +2090,7 @@ get_lisp_to_sockaddr_size (Lisp_Object address, int *familyp) | |||
| 2089 | struct sockaddr *sa; | 2090 | struct sockaddr *sa; |
| 2090 | *familyp = XINT (XCAR (address)); | 2091 | *familyp = XINT (XCAR (address)); |
| 2091 | p = XVECTOR (XCDR (address)); | 2092 | p = XVECTOR (XCDR (address)); |
| 2092 | return p->size + sizeof (sa->sa_family); | 2093 | return p->header.size + sizeof (sa->sa_family); |
| 2093 | } | 2094 | } |
| 2094 | return 0; | 2095 | return 0; |
| 2095 | } | 2096 | } |