mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2024-11-24 03:33:20 +01:00
af0734bc33
Before this instruction supported output values, it fit fairly naturally as a terminator. However, being a terminator while also supporting outputs causes some trouble, as the physreg->vreg COPY operations cannot be in the same block. Modeling it as a non-terminator allows it to be handled the same way as invoke is handled already. Most of the changes here were created by auditing all the existing users of MachineBasicBlock::isEHPad() and MachineBasicBlock::hasEHPadSuccessor(), and adding calls to isInlineAsmBrIndirectTarget or mayHaveInlineAsmBr, as appropriate. Reviewed By: nickdesaulniers, void Differential Revision: https://reviews.llvm.org/D79794
77 lines
2.0 KiB
LLVM
77 lines
2.0 KiB
LLVM
; RUN: not opt -S %s -verify 2>&1 | FileCheck %s
|
|
|
|
; CHECK: Indirect label missing from arglist.
|
|
; CHECK-NEXT: #test1
|
|
define void @test1() {
|
|
; The %4 in the indirect label list is not found in the blockaddresses in the
|
|
; arg list (bad).
|
|
callbr void asm sideeffect "#test1", "X,X"(i8* blockaddress(@test1, %3), i8* blockaddress(@test1, %2))
|
|
to label %1 [label %4, label %2]
|
|
1:
|
|
ret void
|
|
2:
|
|
ret void
|
|
3:
|
|
ret void
|
|
4:
|
|
ret void
|
|
}
|
|
|
|
; CHECK-NOT: Indirect label missing from arglist.
|
|
define void @test2() {
|
|
; %4 and %2 are both in the indirect label list and the arg list (good).
|
|
callbr void asm sideeffect "${0:l} ${1:l}", "X,X"(i8* blockaddress(@test2, %4), i8* blockaddress(@test2, %2))
|
|
to label %1 [label %4, label %2]
|
|
1:
|
|
ret void
|
|
2:
|
|
ret void
|
|
3:
|
|
ret void
|
|
4:
|
|
ret void
|
|
}
|
|
|
|
; CHECK-NOT: Indirect label missing from arglist.
|
|
define void @test3() {
|
|
; note %2 blockaddress. Such a case is possible when passing the address of
|
|
; a label as an input to the inline asm (both address of label and asm goto
|
|
; use blockaddress constants; we're testing that the indirect label list from
|
|
; the asm goto is in the arg list to the asm).
|
|
callbr void asm sideeffect "${0:l} ${1:l} ${2:l}", "X,X,X"(i8* blockaddress(@test3, %4), i8* blockaddress(@test3, %2), i8* blockaddress(@test3, %3))
|
|
to label %1 [label %3, label %4]
|
|
1:
|
|
ret void
|
|
2:
|
|
ret void
|
|
3:
|
|
ret void
|
|
4:
|
|
ret void
|
|
}
|
|
|
|
;; Ensure you cannot use the return value of a callbr in indirect targets.
|
|
; CHECK: Instruction does not dominate all uses!
|
|
; CHECK-NEXT: #test4
|
|
define i32 @test4(i1 %var) {
|
|
entry:
|
|
%ret = callbr i32 asm sideeffect "#test4", "=r,X"(i8* blockaddress(@test4, %abnormal)) to label %normal [label %abnormal]
|
|
|
|
normal:
|
|
ret i32 0
|
|
|
|
abnormal:
|
|
ret i32 %ret
|
|
}
|
|
|
|
;; Ensure you cannot specify the same label as both normal and indirect targets.
|
|
; CHECK: Duplicate callbr destination!
|
|
; CHECK-NEXT: #test5
|
|
define i32 @test5() {
|
|
entry:
|
|
%ret = callbr i32 asm sideeffect "#test5", "=r,X"(i8* blockaddress(@test5, %both)) to label %both [label %both]
|
|
|
|
both:
|
|
ret i32 0
|
|
}
|