llvm/clang/test/CodeGen/asm-goto2.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);
}