diff options
Diffstat (limited to 'gcc/c-common.c')
-rw-r--r-- | gcc/c-common.c | 184 |
1 files changed, 30 insertions, 154 deletions
diff --git a/gcc/c-common.c b/gcc/c-common.c index 6989d40..5cc34c9 100644 --- a/gcc/c-common.c +++ b/gcc/c-common.c @@ -156,7 +156,6 @@ static void init_attributes PARAMS ((void)); static void record_function_format PARAMS ((tree, tree, enum format_type, int, int)); static void record_international_format PARAMS ((tree, tree, int)); -static tree c_find_base_decl PARAMS ((tree)); static int default_valid_lang_attribute PARAMS ((tree, tree, tree, tree)); /* Keep a stack of if statements. We record the number of compound @@ -3245,10 +3244,10 @@ c_apply_type_quals_to_decl (type_quals, decl) alias set for the type pointed to by the type of the decl. */ - int pointed_to_alias_set + HOST_WIDE_INT pointed_to_alias_set = get_alias_set (TREE_TYPE (TREE_TYPE (decl))); - if (!pointed_to_alias_set) + if (pointed_to_alias_set == 0) /* It's not legal to make a subset of alias set zero. */ ; else @@ -3261,91 +3260,16 @@ c_apply_type_quals_to_decl (type_quals, decl) } } -/* T is an expression with pointer type. Find the DECL on which this - expression is based. (For example, in `a[i]' this would be `a'.) - If there is no such DECL, or a unique decl cannot be determined, - NULL_TREE is retured. */ - -static tree -c_find_base_decl (t) - tree t; -{ - int i; - tree decl; - - if (t == NULL_TREE || t == error_mark_node) - return NULL_TREE; - - if (!POINTER_TYPE_P (TREE_TYPE (t))) - return NULL_TREE; - - decl = NULL_TREE; - - if (TREE_CODE (t) == FIELD_DECL - || TREE_CODE (t) == PARM_DECL - || TREE_CODE (t) == VAR_DECL) - /* Aha, we found a pointer-typed declaration. */ - return t; - - /* It would be nice to deal with COMPONENT_REFs here. If we could - tell that `a' and `b' were the same, then `a->f' and `b->f' are - also the same. */ - - /* Handle general expressions. */ - switch (TREE_CODE_CLASS (TREE_CODE (t))) - { - case '1': - case '2': - case '3': - for (i = TREE_CODE_LENGTH (TREE_CODE (t)); --i >= 0;) - { - tree d = c_find_base_decl (TREE_OPERAND (t, i)); - if (d) - { - if (!decl) - decl = d; - else if (d && d != decl) - /* Two different declarations. That's confusing; let's - just assume we don't know what's going on. */ - decl = NULL_TREE; - } - } - break; - - default: - break; - } - - return decl; -} /* Return the typed-based alias set for T, which may be an expression - or a type. */ + or a type. Return -1 if we don't do anything special. */ -int +HOST_WIDE_INT c_get_alias_set (t) tree t; { - tree type; tree u; - if (t == error_mark_node) - return 0; - - /* For a bit field reference that's not to a specific field, - all we can say is the aliasing information for the underlying object. */ - if (TREE_CODE (t) == BIT_FIELD_REF) - t = TREE_OPERAND (t, 0); - - /* If this is a type, use it, otherwise get the type of the expression. - If the type is an error type, say this may alias anything. */ - type = TYPE_P (t) ? t : TREE_TYPE (t); - if (type == error_mark_node) - return 0; - - /* Deal with special cases first; for certain kinds of references - we're interested in more than just the type. */ - /* Permit type-punning when accessing a union, provided the access is directly through the union. For example, this code does not permit taking the address of a union member and then storing @@ -3359,81 +3283,37 @@ c_get_alias_set (t) && TREE_CODE (TREE_TYPE (TREE_OPERAND (u, 0))) == UNION_TYPE) return 0; - if (TREE_CODE (t) == INDIRECT_REF) - { - /* Check for accesses through restrict-qualified pointers. */ - tree op = TREE_OPERAND (t, 0); - tree decl = c_find_base_decl (op); - - if (decl && DECL_POINTER_ALIAS_SET_KNOWN_P (decl)) - /* We use the alias set indicated in the declaration. */ - return DECL_POINTER_ALIAS_SET (decl); - - /* If this is a char *, the ANSI C standard says it can alias - anything. */ - if (TREE_CODE (TREE_TYPE (op)) == INTEGER_TYPE - && (TYPE_PRECISION (TREE_TYPE (op)) - == TYPE_PRECISION (char_type_node))) - return 0; - } + /* If this is a char *, the ANSI C standard says it can alias + anything. */ + else if (TREE_CODE (t) == INDIRECT_REF + && TREE_CODE (TREE_TYPE (TREE_OPERAND (t, 0))) == INTEGER_TYPE + && (TYPE_PRECISION (TREE_TYPE (TREE_OPERAND (t, 0))) + == TYPE_PRECISION (char_type_node))) + return 0; - /* From here on, only the type matters. */ - - if (TREE_CODE (t) == COMPONENT_REF - && DECL_BIT_FIELD_TYPE (TREE_OPERAND (t, 1))) - /* Since build_modify_expr calls get_unwidened for stores to - component references, the type of a bit field can be changed - from (say) `unsigned int : 16' to `unsigned short' or from - `enum E : 16' to `short'. We want the real type of the - bit-field in this case, not some the integral equivalent. */ - type = DECL_BIT_FIELD_TYPE (TREE_OPERAND (t, 1)); - - if (TYPE_ALIAS_SET_KNOWN_P (type)) - /* If we've already calculated the value, just return it. */ - return TYPE_ALIAS_SET (type); - else if (TYPE_MAIN_VARIANT (type) != type) - /* The C standard specifically allows aliasing between - cv-qualified variants of types. */ - TYPE_ALIAS_SET (type) = c_get_alias_set (TYPE_MAIN_VARIANT (type)); - else if (TREE_CODE (type) == INTEGER_TYPE) - { - tree signed_variant; + /* That's all the expressions we handle specially. */ + if (! TYPE_P (t)) + return -1; + if (TREE_CODE (t) == INTEGER_TYPE) + { /* The C standard specifically allows aliasing between signed and unsigned variants of the same type. We treat the signed variant as canonical. */ - signed_variant = signed_type (type); + tree signed_variant = signed_type (t); - if (signed_variant != type) - TYPE_ALIAS_SET (type) = c_get_alias_set (signed_variant); - else if (signed_variant == signed_char_type_node) + if (signed_variant == signed_char_type_node) /* The C standard guarantess that any object may be accessed via an lvalue that has character type. We don't have to check for unsigned_char_type_node or char_type_node because we are specifically looking at the signed variant. */ - TYPE_ALIAS_SET (type) = 0; + return 0; + else if (signed_variant != t) + return get_alias_set (signed_variant); } - else if (TREE_CODE (type) == ARRAY_TYPE) - /* Anything that can alias one of the array elements can alias - the entire array as well. */ - TYPE_ALIAS_SET (type) = c_get_alias_set (TREE_TYPE (type)); - else if (TREE_CODE (type) == FUNCTION_TYPE) - /* There are no objects of FUNCTION_TYPE, so there's no point in - using up an alias set for them. (There are, of course, - pointers and references to functions, but that's - different.) */ - TYPE_ALIAS_SET (type) = 0; - else if (TREE_CODE (type) == RECORD_TYPE - || TREE_CODE (type) == UNION_TYPE) - /* If TYPE is a struct or union type then we're reading or - writing an entire struct. Thus, we don't know anything about - aliasing. (In theory, such an access can only alias objects - whose type is the same as one of the fields, recursively, but - we don't yet make any use of that information.) */ - TYPE_ALIAS_SET (type) = 0; - else if (POINTER_TYPE_P (type)) + else if (POINTER_TYPE_P (t)) { - tree t; + tree t1; /* Unfortunately, there is no canonical form of a pointer type. In particular, if we have `typedef int I', then `int *', and @@ -3458,19 +3338,14 @@ c_get_alias_set (t) can dereference IPP and CIPP. So, we ignore cv-qualifiers on the pointed-to types. This issue has been reported to the C++ committee. */ - t = TYPE_MAIN_VARIANT (TREE_TYPE (type)); - t = ((TREE_CODE (type) == POINTER_TYPE) - ? build_pointer_type (t) : build_reference_type (t)); - if (t != type) - TYPE_ALIAS_SET (type) = c_get_alias_set (t); + t1 = TYPE_MAIN_VARIANT (TREE_TYPE (t)); + t1 = ((TREE_CODE (t) == POINTER_TYPE) + ? build_pointer_type (t1) : build_reference_type (t1)); + if (t1 != t) + return get_alias_set (t1); } - if (! TYPE_ALIAS_SET_KNOWN_P (type)) - /* TYPE is something we haven't seen before. Put it in a new - alias set. */ - TYPE_ALIAS_SET (type) = new_alias_set (); - - return TYPE_ALIAS_SET (type); + return -1; } /* Build tree nodes and builtin functions common to both C and C++ language @@ -3480,6 +3355,7 @@ c_get_alias_set (t) NO_BUILTINS and NO_NONANSI_BUILTINS contain the respective values of the language frontend flags flag_no_builtin and flag_no_nonansi_builtin. */ + void c_common_nodes_and_builtins (cplus_mode, no_builtins, no_nonansi_builtins) int cplus_mode, no_builtins, no_nonansi_builtins; |