aboutsummaryrefslogtreecommitdiff
path: root/gcc
diff options
context:
space:
mode:
authorMartin Jambor <mjambor@suse.cz>2012-03-13 15:29:42 +0100
committerMartin Jambor <jamborm@gcc.gnu.org>2012-03-13 15:29:42 +0100
commit43332529edd116f18441629f484c36a645ccb763 (patch)
treeaf2735f2d09cbd822cac8b47433a66135561bf9f /gcc
parent3985be599e422e875867c8f64dd0eb8b39d77f63 (diff)
downloadgcc-43332529edd116f18441629f484c36a645ccb763.zip
gcc-43332529edd116f18441629f484c36a645ccb763.tar.gz
gcc-43332529edd116f18441629f484c36a645ccb763.tar.bz2
expr.c (expand_assignment): Handle misaligned scalar writes to memory through top-level MEM_REFs by calling...
2012-03-13 Martin Jambor <mjambor@suse.cz> * expr.c (expand_assignment): Handle misaligned scalar writes to memory through top-level MEM_REFs by calling store_bit_field. * testsuite/gcc.dg/misaligned-expand-2.c: New test. From-SVN: r185336
Diffstat (limited to 'gcc')
-rw-r--r--gcc/ChangeLog5
-rw-r--r--gcc/expr.c22
-rw-r--r--gcc/testsuite/ChangeLog4
-rw-r--r--gcc/testsuite/gcc.dg/misaligned-expand-2.c42
4 files changed, 66 insertions, 7 deletions
diff --git a/gcc/ChangeLog b/gcc/ChangeLog
index d5c679c..19bd4ed0 100644
--- a/gcc/ChangeLog
+++ b/gcc/ChangeLog
@@ -1,3 +1,8 @@
+2012-03-13 Martin Jambor <mjambor@suse.cz>
+
+ * expr.c (expand_assignment): Handle misaligned scalar writes to
+ memory through top-level MEM_REFs by calling store_bit_field.
+
2012-03-13 Richard Guenther <rguenther@suse.de>
PR middle-end/52134
diff --git a/gcc/expr.c b/gcc/expr.c
index e6fc100..59b76a4 100644
--- a/gcc/expr.c
+++ b/gcc/expr.c
@@ -4593,10 +4593,12 @@ expand_assignment (tree to, tree from, bool nontemporal)
if ((TREE_CODE (to) == MEM_REF
|| TREE_CODE (to) == TARGET_MEM_REF)
&& mode != BLKmode
+ && !mem_ref_refers_to_non_mem_p (to)
&& ((align = get_object_or_type_alignment (to))
< GET_MODE_ALIGNMENT (mode))
- && ((icode = optab_handler (movmisalign_optab, mode))
- != CODE_FOR_nothing))
+ && (((icode = optab_handler (movmisalign_optab, mode))
+ != CODE_FOR_nothing)
+ || SLOW_UNALIGNED_ACCESS (mode, align)))
{
addr_space_t as
= TYPE_ADDR_SPACE (TREE_TYPE (TREE_TYPE (TREE_OPERAND (to, 0))));
@@ -4639,11 +4641,17 @@ expand_assignment (tree to, tree from, bool nontemporal)
if (TREE_THIS_VOLATILE (to))
MEM_VOLATILE_P (mem) = 1;
- create_fixed_operand (&ops[0], mem);
- create_input_operand (&ops[1], reg, mode);
- /* The movmisalign<mode> pattern cannot fail, else the assignment would
- silently be omitted. */
- expand_insn (icode, 2, ops);
+ if (icode != CODE_FOR_nothing)
+ {
+ create_fixed_operand (&ops[0], mem);
+ create_input_operand (&ops[1], reg, mode);
+ /* The movmisalign<mode> pattern cannot fail, else the assignment
+ would silently be omitted. */
+ expand_insn (icode, 2, ops);
+ }
+ else
+ store_bit_field (mem, GET_MODE_BITSIZE (mode),
+ 0, 0, 0, mode, reg);
return;
}
diff --git a/gcc/testsuite/ChangeLog b/gcc/testsuite/ChangeLog
index 1e6c43f..96b775e 100644
--- a/gcc/testsuite/ChangeLog
+++ b/gcc/testsuite/ChangeLog
@@ -1,3 +1,7 @@
+2012-03-13 Martin Jambor <mjambor@suse.cz>
+
+ * gcc.dg/misaligned-expand-2.c: New test.
+
2012-03-13 Richard Guenther <rguenther@suse.de>
PR middle-end/52134
diff --git a/gcc/testsuite/gcc.dg/misaligned-expand-2.c b/gcc/testsuite/gcc.dg/misaligned-expand-2.c
new file mode 100644
index 0000000..f3ae97f
--- /dev/null
+++ b/gcc/testsuite/gcc.dg/misaligned-expand-2.c
@@ -0,0 +1,42 @@
+/* Test that expand can generate correct stores to misaligned data even on
+ strict alignment platforms. */
+
+/* { dg-do run } */
+/* { dg-options "-O0" } */
+
+extern void abort ();
+
+typedef unsigned int myint __attribute__((aligned(1)));
+
+void
+foo (myint *p, unsigned int i)
+{
+ *p = i;
+}
+
+#define cst 0xdeadbeef
+#define NUM 8
+
+struct blah
+{
+ char c;
+ myint i[NUM];
+};
+
+struct blah g;
+
+#define cst 0xdeadbeef
+
+int
+main (int argc, char **argv)
+{
+ int k;
+
+ for (k = 0; k < NUM; k++)
+ {
+ foo (&g.i[k], cst);
+ if (g.i[k] != cst)
+ abort ();
+ }
+ return 0;
+}