157 lines
6.4 KiB
C
157 lines
6.4 KiB
C
|
// NOTE: Assertions have been autogenerated by utils/update_cc_test_checks.py
|
||
|
// REQUIRES: x86-registered-target
|
||
|
// RUN: %clang_cc1 -triple x86_64-pc-linux-gnu -O0 -emit-llvm %s -o - | FileCheck %s
|
||
|
|
||
|
// CHECK-LABEL: @test0(
|
||
|
// CHECK-NEXT: entry:
|
||
|
// CHECK-NEXT: [[RET:%.*]] = alloca i32, align 4
|
||
|
// CHECK-NEXT: [[TMP0:%.*]] = callbr i32 asm "", "=r,!i,~{dirflag},~{fpsr},~{flags}"() #[[ATTR1:[0-9]+]]
|
||
|
// CHECK-NEXT: to label [[ASM_FALLTHROUGH:%.*]] [label %z.split], !srcloc !2
|
||
|
// CHECK: asm.fallthrough:
|
||
|
// CHECK-NEXT: store i32 [[TMP0]], ptr [[RET]], align 4
|
||
|
// CHECK-NEXT: store i32 42, ptr [[RET]], align 4
|
||
|
// CHECK-NEXT: br label [[Z:%.*]]
|
||
|
// CHECK: z:
|
||
|
// CHECK-NEXT: [[TMP1:%.*]] = load i32, ptr [[RET]], align 4
|
||
|
// CHECK-NEXT: ret i32 [[TMP1]]
|
||
|
// CHECK: z.split:
|
||
|
// CHECK-NEXT: store i32 [[TMP0]], ptr [[RET]], align 4
|
||
|
// CHECK-NEXT: br label [[Z]]
|
||
|
//
|
||
|
int test0 (void) {
|
||
|
int ret;
|
||
|
asm goto ("" : "=r"(ret):::z);
|
||
|
ret = 42;
|
||
|
z:
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
// CHECK-LABEL: @test1(
|
||
|
// CHECK-NEXT: entry:
|
||
|
// CHECK-NEXT: [[RET:%.*]] = alloca i32, align 4
|
||
|
// CHECK-NEXT: [[B:%.*]] = alloca i32, align 4
|
||
|
// CHECK-NEXT: [[TMP0:%.*]] = callbr { i32, i32 } asm "", "=r,=r,!i,~{dirflag},~{fpsr},~{flags}"() #[[ATTR1]]
|
||
|
// CHECK-NEXT: to label [[ASM_FALLTHROUGH:%.*]] [label %z.split], !srcloc !3
|
||
|
// CHECK: asm.fallthrough:
|
||
|
// CHECK-NEXT: [[ASMRESULT:%.*]] = extractvalue { i32, i32 } [[TMP0]], 0
|
||
|
// CHECK-NEXT: [[ASMRESULT1:%.*]] = extractvalue { i32, i32 } [[TMP0]], 1
|
||
|
// CHECK-NEXT: store i32 [[ASMRESULT]], ptr [[RET]], align 4
|
||
|
// CHECK-NEXT: store i32 [[ASMRESULT1]], ptr [[B]], align 4
|
||
|
// CHECK-NEXT: store i32 42, ptr [[RET]], align 4
|
||
|
// CHECK-NEXT: br label [[Z:%.*]]
|
||
|
// CHECK: z:
|
||
|
// CHECK-NEXT: [[TMP1:%.*]] = load i32, ptr [[RET]], align 4
|
||
|
// CHECK-NEXT: ret i32 [[TMP1]]
|
||
|
// CHECK: z.split:
|
||
|
// CHECK-NEXT: [[ASMRESULT2:%.*]] = extractvalue { i32, i32 } [[TMP0]], 0
|
||
|
// CHECK-NEXT: [[ASMRESULT3:%.*]] = extractvalue { i32, i32 } [[TMP0]], 1
|
||
|
// CHECK-NEXT: store i32 [[ASMRESULT2]], ptr [[RET]], align 4
|
||
|
// CHECK-NEXT: store i32 [[ASMRESULT3]], ptr [[B]], align 4
|
||
|
// CHECK-NEXT: br label [[Z]]
|
||
|
//
|
||
|
int test1 (void) {
|
||
|
int ret, b;
|
||
|
asm goto ("" : "=r"(ret), "=r"(b):::z);
|
||
|
ret = 42;
|
||
|
z:
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
// CHECK-LABEL: @test2(
|
||
|
// CHECK-NEXT: entry:
|
||
|
// CHECK-NEXT: [[RET:%.*]] = alloca i32, align 4
|
||
|
// CHECK-NEXT: [[B:%.*]] = alloca i32, align 4
|
||
|
// CHECK-NEXT: [[TMP0:%.*]] = callbr { i32, i32 } asm "", "=r,=r,!i,~{dirflag},~{fpsr},~{flags}"() #[[ATTR1]]
|
||
|
// CHECK-NEXT: to label [[ASM_FALLTHROUGH:%.*]] [label %z.split], !srcloc !4
|
||
|
// CHECK: asm.fallthrough:
|
||
|
// CHECK-NEXT: [[ASMRESULT:%.*]] = extractvalue { i32, i32 } [[TMP0]], 0
|
||
|
// CHECK-NEXT: [[ASMRESULT1:%.*]] = extractvalue { i32, i32 } [[TMP0]], 1
|
||
|
// CHECK-NEXT: store i32 [[ASMRESULT]], ptr [[RET]], align 4
|
||
|
// CHECK-NEXT: store i32 [[ASMRESULT1]], ptr [[B]], align 4
|
||
|
// CHECK-NEXT: [[TMP1:%.*]] = callbr { i32, i32 } asm "", "=r,=r,!i,~{dirflag},~{fpsr},~{flags}"() #[[ATTR1]]
|
||
|
// CHECK-NEXT: to label [[ASM_FALLTHROUGH4:%.*]] [label %z.split9], !srcloc !5
|
||
|
// CHECK: asm.fallthrough4:
|
||
|
// CHECK-NEXT: [[ASMRESULT5:%.*]] = extractvalue { i32, i32 } [[TMP1]], 0
|
||
|
// CHECK-NEXT: [[ASMRESULT6:%.*]] = extractvalue { i32, i32 } [[TMP1]], 1
|
||
|
// CHECK-NEXT: store i32 [[ASMRESULT5]], ptr [[RET]], align 4
|
||
|
// CHECK-NEXT: store i32 [[ASMRESULT6]], ptr [[B]], align 4
|
||
|
// CHECK-NEXT: br label [[Z:%.*]]
|
||
|
// CHECK: z:
|
||
|
// CHECK-NEXT: [[TMP2:%.*]] = load i32, ptr [[RET]], align 4
|
||
|
// CHECK-NEXT: ret i32 [[TMP2]]
|
||
|
// CHECK: z.split:
|
||
|
// CHECK-NEXT: [[ASMRESULT2:%.*]] = extractvalue { i32, i32 } [[TMP0]], 0
|
||
|
// CHECK-NEXT: [[ASMRESULT3:%.*]] = extractvalue { i32, i32 } [[TMP0]], 1
|
||
|
// CHECK-NEXT: store i32 [[ASMRESULT2]], ptr [[RET]], align 4
|
||
|
// CHECK-NEXT: store i32 [[ASMRESULT3]], ptr [[B]], align 4
|
||
|
// CHECK-NEXT: br label [[Z]]
|
||
|
// CHECK: z.split9:
|
||
|
// CHECK-NEXT: [[ASMRESULT7:%.*]] = extractvalue { i32, i32 } [[TMP1]], 0
|
||
|
// CHECK-NEXT: [[ASMRESULT8:%.*]] = extractvalue { i32, i32 } [[TMP1]], 1
|
||
|
// CHECK-NEXT: store i32 [[ASMRESULT7]], ptr [[RET]], align 4
|
||
|
// CHECK-NEXT: store i32 [[ASMRESULT8]], ptr [[B]], align 4
|
||
|
// CHECK-NEXT: br label [[Z]]
|
||
|
//
|
||
|
int test2 (void) {
|
||
|
int ret, b;
|
||
|
asm goto ("" : "=r"(ret), "=r"(b):::z);
|
||
|
asm goto ("" : "=r"(ret), "=r"(b):::z);
|
||
|
z:
|
||
|
return ret;
|
||
|
}
|
||
|
// CHECK-LABEL: @test3(
|
||
|
// CHECK-NEXT: entry:
|
||
|
// CHECK-NEXT: [[RETVAL:%.*]] = alloca i32, align 4
|
||
|
// CHECK-NEXT: [[OUT1_ADDR:%.*]] = alloca i32, align 4
|
||
|
// CHECK-NEXT: store i32 [[OUT1:%.*]], ptr [[OUT1_ADDR]], align 4
|
||
|
// CHECK-NEXT: [[TMP0:%.*]] = callbr i32 asm "", "=r,!i,!i,~{dirflag},~{fpsr},~{flags}"() #[[ATTR1]]
|
||
|
// CHECK-NEXT: to label [[ASM_FALLTHROUGH:%.*]] [label [[LABEL_TRUE_SPLIT:%.*]], label %loop.split], !srcloc !6
|
||
|
// CHECK: asm.fallthrough:
|
||
|
// CHECK-NEXT: store i32 [[TMP0]], ptr [[OUT1_ADDR]], align 4
|
||
|
// CHECK-NEXT: store i32 0, ptr [[RETVAL]], align 4
|
||
|
// CHECK-NEXT: br label [[RETURN:%.*]]
|
||
|
// CHECK: label_true.split:
|
||
|
// CHECK-NEXT: store i32 [[TMP0]], ptr [[OUT1_ADDR]], align 4
|
||
|
// CHECK-NEXT: br label [[LABEL_TRUE:%.*]]
|
||
|
// CHECK: loop.split:
|
||
|
// CHECK-NEXT: store i32 [[TMP0]], ptr [[OUT1_ADDR]], align 4
|
||
|
// CHECK-NEXT: br label [[LOOP:%.*]]
|
||
|
// CHECK: loop:
|
||
|
// CHECK-NEXT: store i32 0, ptr [[RETVAL]], align 4
|
||
|
// CHECK-NEXT: br label [[RETURN]]
|
||
|
// CHECK: label_true:
|
||
|
// CHECK-NEXT: store i32 1, ptr [[RETVAL]], align 4
|
||
|
// CHECK-NEXT: br label [[RETURN]]
|
||
|
// CHECK: return:
|
||
|
// CHECK-NEXT: [[TMP1:%.*]] = load i32, ptr [[RETVAL]], align 4
|
||
|
// CHECK-NEXT: ret i32 [[TMP1]]
|
||
|
//
|
||
|
int test3 (int out1) {
|
||
|
asm goto("" : "=r"(out1)::: label_true, loop);
|
||
|
return 0;
|
||
|
loop:
|
||
|
return 0;
|
||
|
label_true:
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
// CHECK-LABEL: @test4(
|
||
|
// CHECK-NEXT: entry:
|
||
|
// CHECK-NEXT: [[X:%.*]] = alloca i32, align 4
|
||
|
// CHECK-NEXT: br label [[FOO:%.*]]
|
||
|
// CHECK: foo:
|
||
|
// CHECK-NEXT: [[TMP0:%.*]] = callbr i32 asm "", "=r,!i,~{dirflag},~{fpsr},~{flags}"() #[[ATTR1]]
|
||
|
// CHECK-NEXT: to label [[ASM_FALLTHROUGH:%.*]] [label %foo.split], !srcloc !7
|
||
|
// CHECK: asm.fallthrough:
|
||
|
// CHECK-NEXT: store i32 [[TMP0]], ptr [[X]], align 4
|
||
|
// CHECK-NEXT: ret void
|
||
|
// CHECK: foo.split:
|
||
|
// CHECK-NEXT: store i32 [[TMP0]], ptr [[X]], align 4
|
||
|
// CHECK-NEXT: br label [[FOO]]
|
||
|
//
|
||
|
void test4 (void) {
|
||
|
int x;
|
||
|
foo:
|
||
|
asm goto ("" : "=r"(x):::foo);
|
||
|
}
|