diff options
author | rth <rth@138bc75d-0d04-0410-961f-82ee72b054a4> | 2004-07-10 01:52:53 +0000 |
---|---|---|
committer | rth <rth@138bc75d-0d04-0410-961f-82ee72b054a4> | 2004-07-10 01:52:53 +0000 |
commit | 52cc2eeb6b62beb597d35602cb76637d4167cbb4 (patch) | |
tree | 236f16934f66ba6982429b03b99f0240e991c0d1 /gcc/stor-layout.c | |
parent | 4e5bf7c3c23d701362af6f10314df08d853542ae (diff) | |
download | gcc-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.c | 36 |
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. |