summaryrefslogtreecommitdiff
path: root/gcc/stor-layout.c
diff options
context:
space:
mode:
authorrth <rth@138bc75d-0d04-0410-961f-82ee72b054a4>2004-07-10 01:52:53 +0000
committerrth <rth@138bc75d-0d04-0410-961f-82ee72b054a4>2004-07-10 01:52:53 +0000
commit52cc2eeb6b62beb597d35602cb76637d4167cbb4 (patch)
tree236f16934f66ba6982429b03b99f0240e991c0d1 /gcc/stor-layout.c
parent4e5bf7c3c23d701362af6f10314df08d853542ae (diff)
downloadgcc-52cc2eeb6b62beb597d35602cb76637d4167cbb4.tar.gz
* builtins.c (std_gimplify_va_arg_expr): Deny ARGS_GROW_DOWNWARD.
Always align upward to arg boundary. Use size_in_bytes/round_up. Maintain type-correctness of constants. * stor-layout.c (round_up, round_down): Special-case powers of 2. git-svn-id: svn+ssh://gcc.gnu.org/svn/gcc/trunk@84430 138bc75d-0d04-0410-961f-82ee72b054a4
Diffstat (limited to 'gcc/stor-layout.c')
-rw-r--r--gcc/stor-layout.c36
1 files changed, 32 insertions, 4 deletions
diff --git a/gcc/stor-layout.c b/gcc/stor-layout.c
index cf97159b5c5..689768536af 100644
--- a/gcc/stor-layout.c
+++ b/gcc/stor-layout.c
@@ -274,9 +274,24 @@ get_mode_alignment (enum machine_mode mode)
tree
round_up (tree value, int divisor)
{
- tree arg = size_int_type (divisor, TREE_TYPE (value));
+ tree t;
+
+ /* If divisor is a power of two, simplify this to bit manipulation. */
+ if (divisor == (divisor & -divisor))
+ {
+ t = size_int_type (divisor - 1, TREE_TYPE (value));
+ value = size_binop (PLUS_EXPR, value, t);
+ t = size_int_type (-divisor, TREE_TYPE (value));
+ value = size_binop (BIT_AND_EXPR, value, t);
+ }
+ else
+ {
+ t = size_int_type (divisor, TREE_TYPE (value));
+ value = size_binop (CEIL_DIV_EXPR, value, t);
+ value = size_binop (MULT_EXPR, value, t);
+ }
- return size_binop (MULT_EXPR, size_binop (CEIL_DIV_EXPR, value, arg), arg);
+ return value;
}
/* Likewise, but round down. */
@@ -284,9 +299,22 @@ round_up (tree value, int divisor)
tree
round_down (tree value, int divisor)
{
- tree arg = size_int_type (divisor, TREE_TYPE (value));
+ tree t;
+
+ /* If divisor is a power of two, simplify this to bit manipulation. */
+ if (divisor == (divisor & -divisor))
+ {
+ t = size_int_type (-divisor, TREE_TYPE (value));
+ value = size_binop (BIT_AND_EXPR, value, t);
+ }
+ else
+ {
+ t = size_int_type (divisor, TREE_TYPE (value));
+ value = size_binop (FLOOR_DIV_EXPR, value, t);
+ value = size_binop (MULT_EXPR, value, t);
+ }
- return size_binop (MULT_EXPR, size_binop (FLOOR_DIV_EXPR, value, arg), arg);
+ return value;
}
/* Subroutine of layout_decl: Force alignment required for the data type.