| ; Check the memd loads are generated by HexagonLoadStoreWidening pass |
| ; Check that memw loads from adjacent memory location are replaced with memd, |
| ; though the load/stores alias with instructions that occur later in the block. |
| ; The order of memory operations remains unchanged. |
| |
| ; RUN: llc -mtriple=hexagon -verify-machineinstrs < %s | FileCheck %s |
| |
| target triple = "hexagon" |
| |
| ; CHECK-LABEL: load_store_interleaved: |
| ; CHECK: r{{[0-9]+}}:{{[0-9]+}} = memd(r{{[0-9]+}}+#0) |
| ; CHECK: memd(r{{[0-9]+}}+#0) = r{{[0-9]+}}:{{[0-9]+}} |
| ; Function Attrs: mustprogress nounwind |
| define linkonce_odr dso_local void @load_store_interleaved(ptr %p, float %a, float %b) local_unnamed_addr { |
| entry: |
| %0 = load float, ptr %p, align 8 |
| %add0 = fadd float %0, %a |
| store float %add0, ptr %p, align 8 |
| %q = getelementptr i8, ptr %p, i32 4 |
| %1 = load float, ptr %q, align 4 |
| %add1 = fadd float %1, %b |
| store float %add1, ptr %q, align 4 |
| ret void |
| } |
| |
| ; Store can be widened here, but this order of instructions is not currently handled |
| ; CHECK-LABEL: loads_between_stores: |
| ; CHECK: r{{[0-9]+}}:{{[0-9]+}} = memd(r{{[0-9]+}}+#0) |
| ; CHECK-NOT: memd(r{{[0-9]+}}+#4) = r{{[0-9]+}}:{{[0-9]+}} |
| ; Function Attrs: mustprogress nounwind |
| define linkonce_odr dso_local void @loads_between_stores(ptr %p, float %a, float %b) local_unnamed_addr { |
| entry: |
| %add0 = fadd float %b, %a |
| %q = getelementptr i8, ptr %p, i32 4 |
| %r = getelementptr i8, ptr %p, i32 8 |
| store float %add0, ptr %r, align 4 |
| %0 = load float, ptr %p, align 8 |
| %1 = load float, ptr %q, align 4 |
| %add1 = fadd float %1, %0 |
| store float %add1, ptr %q, align 8 |
| ret void |
| } |
| |
| ; CHECK-LABEL: loads_before_stores: |
| ; CHECK: r{{[0-9]+}}:{{[0-9]+}} = memd(r{{[0-9]+}}+#0) |
| ; CHECK: memd(r{{[0-9]+}}+#0) = r{{[0-9]+}}:{{[0-9]+}} |
| ; Function Attrs: mustprogress nounwind |
| define linkonce_odr dso_local void @loads_before_stores(ptr %p, float %a, float %b) local_unnamed_addr { |
| entry: |
| %0 = load float, ptr %p, align 8 |
| %q = getelementptr i8, ptr %p, i32 4 |
| %1 = load float, ptr %q, align 4 |
| %add0 = fadd float %0, %a |
| store float %add0, ptr %p, align 8 |
| %add1 = fadd float %1, %b |
| store float %add1, ptr %q, align 4 |
| ret void |
| } |
| |
| ; Store can be widened here, but this order of instructions is not currently handled |
| ; CHECK-LABEL: store_load_interleaved: |
| ; CHECK: r{{[0-9]+}}:{{[0-9]+}} = memd(r{{[0-9]+}}+#0) |
| ; CHECK-NOT: memd(r{{[0-9]+}}+#0) = r{{[0-9]+}}:{{[0-9]+}} |
| ; Function Attrs: mustprogress nounwind |
| define linkonce_odr dso_local void @store_load_interleaved(ptr %p, float %a, float %b, float %f) local_unnamed_addr { |
| entry: |
| %q = getelementptr i8, ptr %p, i32 4 |
| %r = getelementptr i8, ptr %p, i32 8 |
| store float %f, ptr %r, align 4 |
| %0 = load float, ptr %p, align 8 |
| %add0 = fadd float %0, %a |
| store float %add0, ptr %p, align 8 |
| %1 = load float, ptr %q, align 4 |
| %add1 = fadd float %1, %b |
| %add2 = fadd float %add1, %add0 |
| store float %add2, ptr %q, align 8 |
| ret void |
| } |
| |
| ; CHECK-LABEL: stores_between_loads: |
| ; CHECK-NOT: r{{[0-9]+}}:{{[0-9]+}} = memd(r{{[0-9]+}}+#0) |
| ; CHECK: memd(r{{[0-9]+}}+#0) = r{{[0-9]+}}:{{[0-9]+}} |
| ; Function Attrs: mustprogress nounwind |
| define linkonce_odr dso_local void @stores_between_loads(ptr %p, float %a, float %b, float %f) local_unnamed_addr { |
| entry: |
| %0 = load float, ptr %p, align 8 |
| %add0 = fadd float %f, %0 |
| store float %add0, ptr %p, align 8 |
| %q = getelementptr i8, ptr %p, i32 4 |
| %add1 = fadd float %f, %b |
| store float %add1, ptr %q, align 8 |
| %r = getelementptr i8, ptr %p, i32 8 |
| %1 = load float, ptr %r, align 4 |
| %add2 = fadd float %add1, %1 |
| store float %add2, ptr %r, align 4 |
| ret void |
| } |