aboutsummaryrefslogtreecommitdiffstats
path: root/src/process.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/process.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/process.c')
-rw-r--r--src/process.c15
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}