| ; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py |
| ; RUN: llc -o - %s -mtriple=x86_64--unknown-linux-gnu | FileCheck %s |
| |
| @base = external dso_local local_unnamed_addr global i16, align 2 |
| |
| ; We should be able to merge the two loads here |
| define i16 @unify_through_trivial_asm() { |
| ; CHECK-LABEL: unify_through_trivial_asm: |
| ; CHECK: # %bb.0: |
| ; CHECK-NEXT: movzwl {{.*}}(%rip), %eax |
| ; CHECK-NEXT: #APP |
| ; CHECK-NEXT: nop |
| ; CHECK-NEXT: #NO_APP |
| ; CHECK-NEXT: movzwl {{.*}}(%rip), %eax |
| ; CHECK-NEXT: incl %eax |
| ; CHECK-NEXT: # kill: def $ax killed $ax killed $eax |
| ; CHECK-NEXT: retq |
| %x = load i16, i16* @base, align 2 |
| tail call void asm sideeffect "nop", "r,~{dirflag},~{fpsr},~{flags}"(i16 %x) |
| %x2 = load i16, i16* @base, align 2 |
| %v = add i16 %x2, 1 |
| ret i16 %v |
| } |
| |
| ; The asm call prevents the merging the loads here. |
| define i16 @unify_through_trival_asm_w_memory_clobber() { |
| ; CHECK-LABEL: unify_through_trival_asm_w_memory_clobber: |
| ; CHECK: # %bb.0: |
| ; CHECK-NEXT: movzwl {{.*}}(%rip), %eax |
| ; CHECK-NEXT: #APP |
| ; CHECK-NEXT: nop |
| ; CHECK-NEXT: #NO_APP |
| ; CHECK-NEXT: movzwl {{.*}}(%rip), %eax |
| ; CHECK-NEXT: incl %eax |
| ; CHECK-NEXT: # kill: def $ax killed $ax killed $eax |
| ; CHECK-NEXT: retq |
| %x = load i16, i16* @base, align 2 |
| tail call void asm sideeffect "nop", "+r,~{dirflag},~{fpsr},~{flags},~{base},~{memory}"(i16 %x) |
| %x2 = load i16, i16* @base, align 2 |
| %v = add i16 %x2, 1 |
| ret i16 %v |
| } |
| |
| define dso_local void @fulltest() local_unnamed_addr { |
| ; CHECK-LABEL: fulltest: |
| ; CHECK: # %bb.0: # %entry |
| ; CHECK-NEXT: movzwl {{.*}}(%rip), %edx |
| ; CHECK-NEXT: addl $16, %edx |
| ; CHECK-NEXT: xorl %eax, %eax |
| ; CHECK-NEXT: # kill: def $dx killed $dx killed $edx |
| ; CHECK-NEXT: #APP |
| ; CHECK-NEXT: outb %al, %dx |
| ; CHECK-NEXT: #NO_APP |
| ; CHECK-NEXT: movzwl {{.*}}(%rip), %edx |
| ; CHECK-NEXT: addl $16, %edx |
| ; CHECK-NEXT: movb $1, %al |
| ; CHECK-NEXT: # kill: def $dx killed $dx killed $edx |
| ; CHECK-NEXT: #APP |
| ; CHECK-NEXT: outb %al, %dx |
| ; CHECK-NEXT: #NO_APP |
| ; CHECK-NEXT: movzwl {{.*}}(%rip), %edx |
| ; CHECK-NEXT: addl $16, %edx |
| ; CHECK-NEXT: movb $2, %al |
| ; CHECK-NEXT: # kill: def $dx killed $dx killed $edx |
| ; CHECK-NEXT: #APP |
| ; CHECK-NEXT: outb %al, %dx |
| ; CHECK-NEXT: #NO_APP |
| ; CHECK-NEXT: movzwl {{.*}}(%rip), %edx |
| ; CHECK-NEXT: addl $16, %edx |
| ; CHECK-NEXT: movb $3, %al |
| ; CHECK-NEXT: # kill: def $dx killed $dx killed $edx |
| ; CHECK-NEXT: #APP |
| ; CHECK-NEXT: outb %al, %dx |
| ; CHECK-NEXT: #NO_APP |
| ; CHECK-NEXT: movzwl {{.*}}(%rip), %edx |
| ; CHECK-NEXT: addl $16, %edx |
| ; CHECK-NEXT: movb $4, %al |
| ; CHECK-NEXT: # kill: def $dx killed $dx killed $edx |
| ; CHECK-NEXT: #APP |
| ; CHECK-NEXT: outb %al, %dx |
| ; CHECK-NEXT: #NO_APP |
| ; CHECK-NEXT: movzwl {{.*}}(%rip), %edx |
| ; CHECK-NEXT: addl $16, %edx |
| ; CHECK-NEXT: movb $5, %al |
| ; CHECK-NEXT: # kill: def $dx killed $dx killed $edx |
| ; CHECK-NEXT: #APP |
| ; CHECK-NEXT: outb %al, %dx |
| ; CHECK-NEXT: #NO_APP |
| ; CHECK-NEXT: movzwl {{.*}}(%rip), %edx |
| ; CHECK-NEXT: addl $16, %edx |
| ; CHECK-NEXT: movb $6, %al |
| ; CHECK-NEXT: # kill: def $dx killed $dx killed $edx |
| ; CHECK-NEXT: #APP |
| ; CHECK-NEXT: outb %al, %dx |
| ; CHECK-NEXT: #NO_APP |
| ; CHECK-NEXT: movzwl {{.*}}(%rip), %edx |
| ; CHECK-NEXT: addl $16, %edx |
| ; CHECK-NEXT: movb $7, %al |
| ; CHECK-NEXT: # kill: def $dx killed $dx killed $edx |
| ; CHECK-NEXT: #APP |
| ; CHECK-NEXT: outb %al, %dx |
| ; CHECK-NEXT: #NO_APP |
| ; CHECK-NEXT: movzwl {{.*}}(%rip), %edx |
| ; CHECK-NEXT: addl $16, %edx |
| ; CHECK-NEXT: movb $8, %al |
| ; CHECK-NEXT: # kill: def $dx killed $dx killed $edx |
| ; CHECK-NEXT: #APP |
| ; CHECK-NEXT: outb %al, %dx |
| ; CHECK-NEXT: #NO_APP |
| ; CHECK-NEXT: retq |
| entry: |
| %0 = load i16, i16* @base, align 2 |
| %add = add i16 %0, 16 |
| tail call void asm sideeffect "outb %al,${1:w}", "{ax},{dx},~{dirflag},~{fpsr},~{flags}"(i8 0, i16 %add) |
| %1 = load i16, i16* @base, align 2 |
| %add3 = add i16 %1, 16 |
| tail call void asm sideeffect "outb %al,${1:w}", "{ax},{dx},~{dirflag},~{fpsr},~{flags}"(i8 1, i16 %add3) |
| %2 = load i16, i16* @base, align 2 |
| %add6 = add i16 %2, 16 |
| tail call void asm sideeffect "outb %al,${1:w}", "{ax},{dx},~{dirflag},~{fpsr},~{flags}"(i8 2, i16 %add6) |
| %3 = load i16, i16* @base, align 2 |
| %add9 = add i16 %3, 16 |
| tail call void asm sideeffect "outb %al,${1:w}", "{ax},{dx},~{dirflag},~{fpsr},~{flags}"(i8 3, i16 %add9) |
| %4 = load i16, i16* @base, align 2 |
| %add12 = add i16 %4, 16 |
| tail call void asm sideeffect "outb %al,${1:w}", "{ax},{dx},~{dirflag},~{fpsr},~{flags}"(i8 4, i16 %add12) |
| %5 = load i16, i16* @base, align 2 |
| %add15 = add i16 %5, 16 |
| tail call void asm sideeffect "outb %al,${1:w}", "{ax},{dx},~{dirflag},~{fpsr},~{flags}"(i8 5, i16 %add15) |
| %6 = load i16, i16* @base, align 2 |
| %add18 = add i16 %6, 16 |
| tail call void asm sideeffect "outb %al,${1:w}", "{ax},{dx},~{dirflag},~{fpsr},~{flags}"(i8 6, i16 %add18) |
| %7 = load i16, i16* @base, align 2 |
| %add21 = add i16 %7, 16 |
| tail call void asm sideeffect "outb %al,${1:w}", "{ax},{dx},~{dirflag},~{fpsr},~{flags}"(i8 7, i16 %add21) |
| %8 = load i16, i16* @base, align 2 |
| %add24 = add i16 %8, 16 |
| tail call void asm sideeffect "outb %al,${1:w}", "{ax},{dx},~{dirflag},~{fpsr},~{flags}"(i8 8, i16 %add24) |
| ret void |
| } |