aboutsummaryrefslogtreecommitdiff
path: root/gcc/ada/gcc-interface/decl.c
diff options
context:
space:
mode:
authorEric Botcazou <ebotcazou@adacore.com>2019-01-27 19:14:14 +0000
committerEric Botcazou <ebotcazou@gcc.gnu.org>2019-01-27 19:14:14 +0000
commit683ccd0568569c67b117e87638782c6d870791b2 (patch)
treed3ad36179e7f2d6037dde651ff566d81ef28ab8a /gcc/ada/gcc-interface/decl.c
parent33731c666114349344848ddd4218f98ef486398e (diff)
downloadgcc-683ccd0568569c67b117e87638782c6d870791b2.zip
gcc-683ccd0568569c67b117e87638782c6d870791b2.tar.gz
gcc-683ccd0568569c67b117e87638782c6d870791b2.tar.bz2
repinfo.adb (List_Component_Layout): Remove superfluous space for zero-sized field.
* repinfo.adb (List_Component_Layout): Remove superfluous space for zero-sized field. * gcc-interface/ada-tree.h (TYPE_IS_EXTRA_SUBTYPE_P): New macro. * gcc-interface/gigi.h (create_extra_subtype): Declare. * gcc-interface/decl.c (TYPE_ARRAY_SIZE_LIMIT): Likewise. (update_n_elem): New function. (gnat_to_gnu_entity): Use create_extra_subtype to create extra subtypes instead of doing it manually. <E_Array_Type>: Use update_n_elem to compute the maximum size. Use the index type instead of base type for the bounds. Set TYPE_ARRAY_MAX_SIZE of the array to the maximum size. <E_Array_Subtype>: Create an extra subtype using the index type of the base array type for self-referential bounds. Use update_n_elem to compute the maximum size. Set TYPE_ARRAY_MAX_SIZE of the array to the maximum size. (gnat_to_gnu_field): Clear DECL_NONADDRESSABLE_P on discriminants. * gcc-interface/misc.c (gnat_get_alias_set): Return the alias set of the base type for an extra subtype. (gnat_type_max_size): Remove obsolete code. * gcc-interface/trans.c (Attribute_to_gnu): Minor tweak. (can_be_lower_p): Deal with pathological types. * gcc-interface/utils.c (create_extra_subtype): New function. (create_field_decl): Minor tweak. (max_size) <tcc_reference>: Compute a better value by using the extra subtypes on the self-referential bounds. <tcc_binary>: Rewrite. Deal with "negative value" in unsigned types. <tcc_expression>: Likewise. * gcc-interface/utils2.c (compare_arrays): Retrieve the original bounds of the arrays upfront. Swap only if the second length is not constant. Use comparisons on the original bounds consistently for the null tests. (build_binary_op): Use TYPE_IS_EXTRA_SUBTYPE_P macro. (build_allocator): Minor tweak. From-SVN: r268318
Diffstat (limited to 'gcc/ada/gcc-interface/decl.c')
-rw-r--r--gcc/ada/gcc-interface/decl.c362
1 files changed, 173 insertions, 189 deletions
diff --git a/gcc/ada/gcc-interface/decl.c b/gcc/ada/gcc-interface/decl.c
index a1cd694..ed015ba 100644
--- a/gcc/ada/gcc-interface/decl.c
+++ b/gcc/ada/gcc-interface/decl.c
@@ -85,6 +85,12 @@
#define FOREIGN_FORCE_REALIGN_STACK 0
#endif
+/* The largest TYPE_ARRAY_MAX_SIZE value we set on an array type.
+ It's an artibrary limit (256 MB) above which we consider that
+ the allocation is essentially unbounded. */
+
+#define TYPE_ARRAY_SIZE_LIMIT (1 << 28)
+
struct incomplete
{
struct incomplete *next;
@@ -216,6 +222,7 @@ static bool cannot_be_superflat (Node_Id);
static bool constructor_address_p (tree);
static bool allocatable_size_p (tree, bool);
static bool initial_value_needs_conversion (tree, tree);
+static tree update_n_elem (tree, tree, tree);
static int compare_field_bitpos (const PTR, const PTR);
static bool components_to_record (Node_Id, Entity_Id, tree, tree, int, bool,
bool, bool, bool, bool, bool, bool, tree,
@@ -1760,12 +1767,10 @@ gnat_to_gnu_entity (Entity_Id gnat_entity, tree gnu_expr, bool definition)
if (gnu_high
&& !tree_int_cst_equal (gnu_high, TYPE_MAX_VALUE (gnu_type)))
{
- tree gnu_subtype = make_unsigned_type (esize);
- SET_TYPE_RM_MAX_VALUE (gnu_subtype, gnu_high);
- TREE_TYPE (gnu_subtype) = gnu_type;
- TYPE_EXTRA_SUBTYPE_P (gnu_subtype) = 1;
TYPE_NAME (gnu_type) = create_concat_name (gnat_entity, "UMT");
- gnu_type = gnu_subtype;
+ gnu_type
+ = create_extra_subtype (gnu_type, TYPE_MIN_VALUE (gnu_type),
+ gnu_high);
}
}
goto discrete_type;
@@ -2052,7 +2057,7 @@ gnat_to_gnu_entity (Entity_Id gnat_entity, tree gnu_expr, bool definition)
tree gnu_template_reference, gnu_template_fields, gnu_fat_type;
tree *gnu_index_types = XALLOCAVEC (tree, ndim);
tree *gnu_temp_fields = XALLOCAVEC (tree, ndim);
- tree gnu_max_size = size_one_node, gnu_max_size_unit, tem, t;
+ tree gnu_max_size = size_one_node, tem, t;
Entity_Id gnat_index, gnat_name;
int index;
tree comp_type;
@@ -2165,17 +2170,27 @@ gnat_to_gnu_entity (Entity_Id gnat_entity, tree gnu_expr, bool definition)
{
char field_name[16];
tree gnu_index_type = get_unpadded_type (Etype (gnat_index));
- tree gnu_index_base_type
- = maybe_character_type (get_base_type (gnu_index_type));
- tree gnu_lb_field, gnu_hb_field, gnu_orig_min, gnu_orig_max;
+ tree gnu_orig_min = TYPE_MIN_VALUE (gnu_index_type);
+ tree gnu_orig_max = TYPE_MAX_VALUE (gnu_index_type);
+ tree gnu_index_base_type = get_base_type (gnu_index_type);
+ tree gnu_lb_field, gnu_hb_field;
tree gnu_min, gnu_max, gnu_high;
+ /* Update the maximum size of the array in elements. */
+ if (gnu_max_size)
+ gnu_max_size
+ = update_n_elem (gnu_max_size, gnu_orig_min, gnu_orig_max);
+
+ /* Now build the self-referential bounds of the index type. */
+ gnu_index_type = maybe_character_type (gnu_index_type);
+ gnu_index_base_type = maybe_character_type (gnu_index_base_type);
+
/* Make the FIELD_DECLs for the low and high bounds of this
type and then make extractions of these fields from the
template. */
sprintf (field_name, "LB%d", index);
gnu_lb_field = create_field_decl (get_identifier (field_name),
- gnu_index_base_type,
+ gnu_index_type,
gnu_template_type, NULL_TREE,
NULL_TREE, 0, 0);
Sloc_to_locus (Sloc (gnat_entity),
@@ -2183,7 +2198,7 @@ gnat_to_gnu_entity (Entity_Id gnat_entity, tree gnu_expr, bool definition)
field_name[0] = 'U';
gnu_hb_field = create_field_decl (get_identifier (field_name),
- gnu_index_base_type,
+ gnu_index_type,
gnu_template_type, NULL_TREE,
NULL_TREE, 0, 0);
Sloc_to_locus (Sloc (gnat_entity),
@@ -2193,10 +2208,10 @@ gnat_to_gnu_entity (Entity_Id gnat_entity, tree gnu_expr, bool definition)
/* We can't use build_component_ref here since the template type
isn't complete yet. */
- gnu_orig_min = build3 (COMPONENT_REF, gnu_index_base_type,
+ gnu_orig_min = build3 (COMPONENT_REF, TREE_TYPE (gnu_lb_field),
gnu_template_reference, gnu_lb_field,
NULL_TREE);
- gnu_orig_max = build3 (COMPONENT_REF, gnu_index_base_type,
+ gnu_orig_max = build3 (COMPONENT_REF, TREE_TYPE (gnu_hb_field),
gnu_template_reference, gnu_hb_field,
NULL_TREE);
TREE_READONLY (gnu_orig_min) = TREE_READONLY (gnu_orig_max) = 1;
@@ -2222,25 +2237,6 @@ gnat_to_gnu_entity (Entity_Id gnat_entity, tree gnu_expr, bool definition)
gnu_orig_max),
gnat_entity);
- /* Update the maximum size of the array in elements. */
- if (gnu_max_size)
- {
- tree gnu_min
- = convert (sizetype, TYPE_MIN_VALUE (gnu_index_type));
- tree gnu_max
- = convert (sizetype, TYPE_MAX_VALUE (gnu_index_type));
- tree gnu_this_max
- = size_binop (PLUS_EXPR, size_one_node,
- size_binop (MINUS_EXPR, gnu_max, gnu_min));
-
- if (TREE_CODE (gnu_this_max) == INTEGER_CST
- && TREE_OVERFLOW (gnu_this_max))
- gnu_max_size = NULL_TREE;
- else
- gnu_max_size
- = size_binop (MULT_EXPR, gnu_max_size, gnu_this_max);
- }
-
TYPE_NAME (gnu_index_types[index])
= create_concat_name (gnat_entity, field_name);
}
@@ -2262,17 +2258,10 @@ gnat_to_gnu_entity (Entity_Id gnat_entity, tree gnu_expr, bool definition)
Set_Component_Size (gnat_entity,
annotate_value (TYPE_SIZE (comp_type)));
- /* Compute the maximum size of the array in units and bits. */
+ /* Compute the maximum size of the array in units. */
if (gnu_max_size)
- {
- gnu_max_size_unit = size_binop (MULT_EXPR, gnu_max_size,
- TYPE_SIZE_UNIT (comp_type));
- gnu_max_size = size_binop (MULT_EXPR,
- convert (bitsizetype, gnu_max_size),
- TYPE_SIZE (comp_type));
- }
- else
- gnu_max_size_unit = NULL_TREE;
+ gnu_max_size
+ = size_binop (MULT_EXPR, gnu_max_size, TYPE_SIZE_UNIT (comp_type));
/* Now build the array type. */
tem = comp_type;
@@ -2329,14 +2318,8 @@ gnat_to_gnu_entity (Entity_Id gnat_entity, tree gnu_expr, bool definition)
if (gnu_max_size
&& TREE_CODE (gnu_max_size) == INTEGER_CST
&& !TREE_OVERFLOW (gnu_max_size)
- && TREE_CODE (gnu_max_size_unit) == INTEGER_CST
- && !TREE_OVERFLOW (gnu_max_size_unit))
- {
- TYPE_SIZE (tem) = size_binop (MIN_EXPR, gnu_max_size,
- TYPE_SIZE (tem));
- TYPE_SIZE_UNIT (tem) = size_binop (MIN_EXPR, gnu_max_size_unit,
- TYPE_SIZE_UNIT (tem));
- }
+ && compare_tree_int (gnu_max_size, TYPE_ARRAY_SIZE_LIMIT) <= 0)
+ TYPE_ARRAY_MAX_SIZE (tem) = gnu_max_size;
create_type_decl (create_concat_name (gnat_entity, "XUA"), tem,
artificial_p, debug_info_p, gnat_entity);
@@ -2400,7 +2383,7 @@ gnat_to_gnu_entity (Entity_Id gnat_entity, tree gnu_expr, bool definition)
const int ndim = Number_Dimensions (gnat_entity);
tree gnu_base_type = gnu_type;
tree *gnu_index_types = XALLOCAVEC (tree, ndim);
- tree gnu_max_size = size_one_node, gnu_max_size_unit;
+ tree gnu_max_size = size_one_node;
bool need_index_type_struct = false;
int index;
@@ -2416,27 +2399,83 @@ gnat_to_gnu_entity (Entity_Id gnat_entity, tree gnu_expr, bool definition)
gnat_base_index = Next_Index (gnat_base_index))
{
tree gnu_index_type = get_unpadded_type (Etype (gnat_index));
- tree gnu_index_base_type
- = maybe_character_type (get_base_type (gnu_index_type));
- tree gnu_orig_min
- = convert (gnu_index_base_type,
- TYPE_MIN_VALUE (gnu_index_type));
- tree gnu_orig_max
- = convert (gnu_index_base_type,
- TYPE_MAX_VALUE (gnu_index_type));
- tree gnu_min = convert (sizetype, gnu_orig_min);
- tree gnu_max = convert (sizetype, gnu_orig_max);
+ tree gnu_orig_min = TYPE_MIN_VALUE (gnu_index_type);
+ tree gnu_orig_max = TYPE_MAX_VALUE (gnu_index_type);
+ tree gnu_index_base_type = get_base_type (gnu_index_type);
tree gnu_base_index_type
= get_unpadded_type (Etype (gnat_base_index));
- tree gnu_base_index_base_type
- = maybe_character_type (get_base_type (gnu_base_index_type));
- tree gnu_base_orig_min
- = convert (gnu_base_index_base_type,
- TYPE_MIN_VALUE (gnu_base_index_type));
- tree gnu_base_orig_max
- = convert (gnu_base_index_base_type,
- TYPE_MAX_VALUE (gnu_base_index_type));
- tree gnu_high;
+ tree gnu_base_orig_min = TYPE_MIN_VALUE (gnu_base_index_type);
+ tree gnu_base_orig_max = TYPE_MAX_VALUE (gnu_base_index_type);
+ tree gnu_min, gnu_max, gnu_high;
+
+ /* We try to define subtypes for discriminants used as bounds
+ that are more restrictive than those declared by using the
+ bounds of the index type of the base array type. This will
+ make it possible to calculate the maximum size of the record
+ type more conservatively. This may have already been done by
+ the front-end (Exp_Ch3.Adjust_Discriminants), in which case
+ there will be a conversion that needs to be removed first. */
+ if (CONTAINS_PLACEHOLDER_P (gnu_orig_min)
+ && TYPE_RM_SIZE (gnu_base_index_type)
+ && !tree_int_cst_lt (TYPE_RM_SIZE (gnu_index_type),
+ TYPE_RM_SIZE (gnu_base_index_type)))
+ {
+ gnu_orig_min = remove_conversions (gnu_orig_min, false);
+ TREE_TYPE (gnu_orig_min)
+ = create_extra_subtype (TREE_TYPE (gnu_orig_min),
+ gnu_base_orig_min,
+ gnu_base_orig_max);
+ }
+
+ if (CONTAINS_PLACEHOLDER_P (gnu_orig_max)
+ && TYPE_RM_SIZE (gnu_base_index_type)
+ && !tree_int_cst_lt (TYPE_RM_SIZE (gnu_index_type),
+ TYPE_RM_SIZE (gnu_base_index_type)))
+ {
+ gnu_orig_max = remove_conversions (gnu_orig_max, false);
+ TREE_TYPE (gnu_orig_max)
+ = create_extra_subtype (TREE_TYPE (gnu_orig_max),
+ gnu_base_orig_min,
+ gnu_base_orig_max);
+ }
+
+ /* Update the maximum size of the array in elements. Here we
+ see if any constraint on the index type of the base type
+ can be used in the case of self-referential bounds on the
+ index type of the array type. We look for a non-"infinite"
+ and non-self-referential bound from any type involved and
+ handle each bound separately. */
+ if (gnu_max_size)
+ {
+ if (CONTAINS_PLACEHOLDER_P (gnu_orig_min))
+ gnu_min = gnu_base_orig_min;
+ else
+ gnu_min = gnu_orig_min;
+
+ if (TREE_CODE (gnu_min) != INTEGER_CST
+ || TREE_OVERFLOW (gnu_min))
+ gnu_min = TYPE_MIN_VALUE (TREE_TYPE (gnu_min));
+
+ if (CONTAINS_PLACEHOLDER_P (gnu_orig_max))
+ gnu_max = gnu_base_orig_max;
+ else
+ gnu_max = gnu_orig_max;
+
+ if (TREE_CODE (gnu_max) != INTEGER_CST
+ || TREE_OVERFLOW (gnu_max))
+ gnu_max = TYPE_MAX_VALUE (TREE_TYPE (gnu_max));
+
+ gnu_max_size
+ = update_n_elem (gnu_max_size, gnu_min, gnu_max);
+ }
+
+ /* Convert the bounds to the base type for consistency below. */
+ gnu_index_base_type = maybe_character_type (gnu_index_base_type);
+ gnu_orig_min = convert (gnu_index_base_type, gnu_orig_min);
+ gnu_orig_max = convert (gnu_index_base_type, gnu_orig_max);
+
+ gnu_min = convert (sizetype, gnu_orig_min);
+ gnu_max = convert (sizetype, gnu_orig_max);
/* See if the base array type is already flat. If it is, we
are probably compiling an ACATS test but it will cause the
@@ -2470,7 +2509,8 @@ gnat_to_gnu_entity (Entity_Id gnat_entity, tree gnu_expr, bool definition)
&& TREE_OVERFLOW (gnu_min) && TREE_OVERFLOW (gnu_max)
&& !TREE_OVERFLOW
(convert (sizetype,
- fold_build2 (MINUS_EXPR, gnu_index_type,
+ fold_build2 (MINUS_EXPR,
+ gnu_index_base_type,
gnu_orig_max,
gnu_orig_min))))
{
@@ -2512,12 +2552,9 @@ gnat_to_gnu_entity (Entity_Id gnat_entity, tree gnu_expr, bool definition)
}
/* Finally we use (hb >= lb) ? hb : lb - 1 for the upper bound
- in all the other cases. Note that, here as well as above,
- the condition used in the comparison must be equivalent to
- the condition (length != 0). This is relied upon in order
- to optimize array comparisons in compare_arrays. Moreover
- we use int_const_binop for the shift by 1 if the bound is
- constant to avoid any unwanted overflow. */
+ in all the other cases. Note that we use int_const_binop for
+ the shift by 1 if the bound is constant to avoid any unwanted
+ overflow. */
else
gnu_high
= build_cond_expr (sizetype,
@@ -2538,65 +2575,6 @@ gnat_to_gnu_entity (Entity_Id gnat_entity, tree gnu_expr, bool definition)
= create_index_type (gnu_min, gnu_high, gnu_index_type,
gnat_entity);
- /* Update the maximum size of the array in elements. Here we
- see if any constraint on the index type of the base type
- can be used in the case of self-referential bound on the
- index type of the subtype. We look for a non-"infinite"
- and non-self-referential bound from any type involved and
- handle each bound separately. */
- if (gnu_max_size)
- {
- tree gnu_base_min = convert (sizetype, gnu_base_orig_min);
- tree gnu_base_max = convert (sizetype, gnu_base_orig_max);
- tree gnu_base_base_min
- = convert (sizetype,
- TYPE_MIN_VALUE (gnu_base_index_base_type));
- tree gnu_base_base_max
- = convert (sizetype,
- TYPE_MAX_VALUE (gnu_base_index_base_type));
-
- if (!CONTAINS_PLACEHOLDER_P (gnu_min)
- || !(TREE_CODE (gnu_base_min) == INTEGER_CST
- && !TREE_OVERFLOW (gnu_base_min)))
- gnu_base_min = gnu_min;
-
- if (!CONTAINS_PLACEHOLDER_P (gnu_max)
- || !(TREE_CODE (gnu_base_max) == INTEGER_CST
- && !TREE_OVERFLOW (gnu_base_max)))
- gnu_base_max = gnu_max;
-
- if ((TREE_CODE (gnu_base_min) == INTEGER_CST
- && TREE_OVERFLOW (gnu_base_min))
- || operand_equal_p (gnu_base_min, gnu_base_base_min, 0)
- || (TREE_CODE (gnu_base_max) == INTEGER_CST
- && TREE_OVERFLOW (gnu_base_max))
- || operand_equal_p (gnu_base_max, gnu_base_base_max, 0))
- gnu_max_size = NULL_TREE;
- else
- {
- tree gnu_this_max;
-
- /* Use int_const_binop if the bounds are constant to
- avoid any unwanted overflow. */
- if (TREE_CODE (gnu_base_min) == INTEGER_CST
- && TREE_CODE (gnu_base_max) == INTEGER_CST)
- gnu_this_max
- = int_const_binop (PLUS_EXPR, size_one_node,
- int_const_binop (MINUS_EXPR,
- gnu_base_max,
- gnu_base_min));
- else
- gnu_this_max
- = size_binop (PLUS_EXPR, size_one_node,
- size_binop (MINUS_EXPR,
- gnu_base_max,
- gnu_base_min));
-
- gnu_max_size
- = size_binop (MULT_EXPR, gnu_max_size, gnu_this_max);
- }
- }
-
/* We need special types for debugging information to point to
the index types if they have variable bounds, are not integer
types, are biased or are wider than sizetype. These are GNAT
@@ -2646,17 +2624,10 @@ gnat_to_gnu_entity (Entity_Id gnat_entity, tree gnu_expr, bool definition)
}
}
- /* Compute the maximum size of the array in units and bits. */
+ /* Compute the maximum size of the array in units. */
if (gnu_max_size)
- {
- gnu_max_size_unit = size_binop (MULT_EXPR, gnu_max_size,
- TYPE_SIZE_UNIT (gnu_type));
- gnu_max_size = size_binop (MULT_EXPR,
- convert (bitsizetype, gnu_max_size),
- TYPE_SIZE (gnu_type));
- }
- else
- gnu_max_size_unit = NULL_TREE;
+ gnu_max_size
+ = size_binop (MULT_EXPR, gnu_max_size, TYPE_SIZE_UNIT (gnu_type));
/* Now build the array type. */
for (index = ndim - 1; index >= 0; index --)
@@ -2776,21 +2747,12 @@ gnat_to_gnu_entity (Entity_Id gnat_entity, tree gnu_expr, bool definition)
= (Is_Packed (gnat_entity)
|| Is_Packed_Array_Impl_Type (gnat_entity));
- /* If the size is self-referential and the maximum size doesn't
- overflow, use it. */
- if (CONTAINS_PLACEHOLDER_P (TYPE_SIZE (gnu_type))
- && gnu_max_size
- && !(TREE_CODE (gnu_max_size) == INTEGER_CST
- && TREE_OVERFLOW (gnu_max_size))
- && !(TREE_CODE (gnu_max_size_unit) == INTEGER_CST
- && TREE_OVERFLOW (gnu_max_size_unit)))
- {
- TYPE_SIZE (gnu_type) = size_binop (MIN_EXPR, gnu_max_size,
- TYPE_SIZE (gnu_type));
- TYPE_SIZE_UNIT (gnu_type)
- = size_binop (MIN_EXPR, gnu_max_size_unit,
- TYPE_SIZE_UNIT (gnu_type));
- }
+ /* If the maximum size doesn't overflow, use it. */
+ if (gnu_max_size
+ && TREE_CODE (gnu_max_size) == INTEGER_CST
+ && !TREE_OVERFLOW (gnu_max_size)
+ && compare_tree_int (gnu_max_size, TYPE_ARRAY_SIZE_LIMIT) <= 0)
+ TYPE_ARRAY_MAX_SIZE (gnu_type) = gnu_max_size;
/* Set our alias set to that of our base type. This gives all
array subtypes the same alias set. */
@@ -2850,17 +2812,10 @@ gnat_to_gnu_entity (Entity_Id gnat_entity, tree gnu_expr, bool definition)
TYPE_MODULUS for modular types so we make an extra
subtype if necessary. */
if (TYPE_MODULAR_P (gnu_inner))
- {
- tree gnu_subtype
- = make_unsigned_type (TYPE_PRECISION (gnu_inner));
- TREE_TYPE (gnu_subtype) = gnu_inner;
- TYPE_EXTRA_SUBTYPE_P (gnu_subtype) = 1;
- SET_TYPE_RM_MIN_VALUE (gnu_subtype,
- TYPE_MIN_VALUE (gnu_inner));
- SET_TYPE_RM_MAX_VALUE (gnu_subtype,
- TYPE_MAX_VALUE (gnu_inner));
- gnu_inner = gnu_subtype;
- }
+ gnu_inner
+ = create_extra_subtype (gnu_inner,
+ TYPE_MIN_VALUE (gnu_inner),
+ TYPE_MAX_VALUE (gnu_inner));
TYPE_HAS_ACTUAL_BOUNDS_P (gnu_inner) = 1;
@@ -3259,7 +3214,7 @@ gnat_to_gnu_entity (Entity_Id gnat_entity, tree gnu_expr, bool definition)
&& !Is_Access_Type (Etype (Node (gnat_constr)))
&& Ekind (Entity (Node (gnat_constr))) == E_Discriminant)
{
- Entity_Id gnat_discr = Entity (Node (gnat_constr));
+ const Entity_Id gnat_discr = Entity (Node (gnat_constr));
tree gnu_discr_type = gnat_to_gnu_type (Etype (gnat_discr));
tree gnu_ref
= gnat_to_gnu_entity (Original_Record_Component (gnat_discr),
@@ -3270,20 +3225,10 @@ gnat_to_gnu_entity (Entity_Id gnat_entity, tree gnu_expr, bool definition)
gcc_assert (TREE_TYPE (TREE_OPERAND (gnu_ref, 0)) == gnu_type);
if (gnu_discr_type != TREE_TYPE (gnu_ref))
- {
- const unsigned prec = TYPE_PRECISION (TREE_TYPE (gnu_ref));
- tree gnu_subtype
- = TYPE_UNSIGNED (TREE_TYPE (gnu_ref))
- ? make_unsigned_type (prec) : make_signed_type (prec);
- TREE_TYPE (gnu_subtype) = TREE_TYPE (gnu_ref);
- TYPE_EXTRA_SUBTYPE_P (gnu_subtype) = 1;
- SET_TYPE_RM_MIN_VALUE (gnu_subtype,
- TYPE_MIN_VALUE (gnu_discr_type));
- SET_TYPE_RM_MAX_VALUE (gnu_subtype,
- TYPE_MAX_VALUE (gnu_discr_type));
- TREE_TYPE (gnu_ref)
- = TREE_TYPE (TREE_OPERAND (gnu_ref, 1)) = gnu_subtype;
- }
+ TREE_TYPE (gnu_ref)
+ = create_extra_subtype (TREE_TYPE (gnu_ref),
+ TYPE_MIN_VALUE (gnu_discr_type),
+ TYPE_MAX_VALUE (gnu_discr_type));
}
/* If this is a derived type with discriminants and these discriminants
@@ -6399,6 +6344,37 @@ initial_value_needs_conversion (tree gnu_type, tree gnu_expr)
/* In all the other cases, convert the expression to the object's type. */
return true;
}
+
+/* Add the contribution of [MIN, MAX] to the current number of elements N_ELEM
+ of an array type and return the result, or NULL_TREE if it overflowed. */
+
+static tree
+update_n_elem (tree n_elem, tree min, tree max)
+{
+ /* First deal with the empty case. */
+ if (TREE_CODE (min) == INTEGER_CST
+ && TREE_CODE (max) == INTEGER_CST
+ && tree_int_cst_lt (max, min))
+ return size_zero_node;
+
+ min = convert (sizetype, min);
+ max = convert (sizetype, max);
+
+ /* Compute the number of elements in this dimension. */
+ tree this_n_elem
+ = size_binop (PLUS_EXPR, size_one_node, size_binop (MINUS_EXPR, max, min));
+
+ if (TREE_CODE (this_n_elem) == INTEGER_CST && TREE_OVERFLOW (this_n_elem))
+ return NULL_TREE;
+
+ /* Multiply the current number of elements by the result. */
+ n_elem = size_binop (MULT_EXPR, n_elem, this_n_elem);
+
+ if (TREE_CODE (n_elem) == INTEGER_CST && TREE_OVERFLOW (n_elem))
+ return NULL_TREE;
+
+ return n_elem;
+}
/* Given GNAT_ENTITY, elaborate all expressions that are required to
be elaborated at the point of its definition, but do nothing else. */
@@ -7222,12 +7198,20 @@ gnat_to_gnu_field (Entity_Id gnat_field, tree gnu_record_type, int packed,
DECL_ALIASED_P (gnu_field) = is_aliased;
TREE_SIDE_EFFECTS (gnu_field) = TREE_THIS_VOLATILE (gnu_field) = is_volatile;
+ /* If this is a discriminant, then we treat it specially: first, we set its
+ index number for the back-annotation; second, we record whether it cannot
+ be changed once it has been set for the computation of loop invariants;
+ third, we make it addressable in order for the optimizer to more easily
+ see that it cannot be modified by assignments to the other fields of the
+ record (see create_field_decl for a more detailed explanation), which is
+ crucial to hoist the offset and size computations of dynamic fields. */
if (Ekind (gnat_field) == E_Discriminant)
{
- DECL_INVARIANT_P (gnu_field)
- = No (Discriminant_Default_Value (gnat_field));
DECL_DISCRIMINANT_NUMBER (gnu_field)
= UI_To_gnu (Discriminant_Number (gnat_field), sizetype);
+ DECL_INVARIANT_P (gnu_field)
+ = No (Discriminant_Default_Value (gnat_field));
+ DECL_NONADDRESSABLE_P (gnu_field) = 0;
}
return gnu_field;