2013-12-04 11:41:33 +08:00
|
|
|
// REQUIRES: x86-registered-target
|
2014-05-06 11:13:27 +08:00
|
|
|
// RUN: %clang_cc1 -x c++ %s -triple i386-apple-darwin10 -fasm-blocks -emit-llvm -o - -std=c++11 | FileCheck %s
|
2013-04-03 04:03:29 +08:00
|
|
|
|
2013-05-03 08:10:13 +08:00
|
|
|
// rdar://13645930
|
|
|
|
|
2013-04-03 04:03:29 +08:00
|
|
|
struct Foo {
|
|
|
|
static int *ptr;
|
|
|
|
static int a, b;
|
2013-04-12 07:57:29 +08:00
|
|
|
int arr[4];
|
2013-04-03 04:03:29 +08:00
|
|
|
struct Bar {
|
|
|
|
static int *ptr;
|
2013-04-12 07:57:29 +08:00
|
|
|
char arr[2];
|
2013-04-03 04:03:29 +08:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
void t1() {
|
2017-05-05 02:19:52 +08:00
|
|
|
// CHECK-LABEL: define void @_Z2t1v()
|
2013-04-03 04:03:29 +08:00
|
|
|
Foo::ptr = (int *)0xDEADBEEF;
|
|
|
|
Foo::Bar::ptr = (int *)0xDEADBEEF;
|
2017-05-05 02:19:52 +08:00
|
|
|
// CHECK: call void asm sideeffect inteldialect
|
|
|
|
// CHECK-SAME: mov eax, $0
|
|
|
|
// CHECK-SAME: mov eax, $1
|
|
|
|
// CHECK-SAME: mov eax, $2
|
|
|
|
// CHECK-SAME: mov eax, dword ptr $3
|
|
|
|
// CHECK-SAME: mov eax, dword ptr $4
|
|
|
|
// CHECK-SAME: "*m,*m,*m,*m,*m,~{eax},~{dirflag},~{fpsr},~{flags}"(i32** @_ZN3Foo3ptrE, i32** @_ZN3Foo3Bar3ptrE, i32** @_ZN3Foo3ptrE, i32** @_ZN3Foo3ptrE, i32** @_ZN3Foo3ptrE)
|
2013-05-03 08:10:13 +08:00
|
|
|
__asm mov eax, Foo ::ptr
|
|
|
|
__asm mov eax, Foo :: Bar :: ptr
|
|
|
|
__asm mov eax, [Foo:: ptr]
|
|
|
|
__asm mov eax, dword ptr [Foo :: ptr]
|
|
|
|
__asm mov eax, dword ptr [Foo :: ptr]
|
2013-04-03 04:03:29 +08:00
|
|
|
}
|
2013-04-12 07:37:53 +08:00
|
|
|
|
|
|
|
int gvar = 10;
|
|
|
|
void t2() {
|
|
|
|
int lvar = 10;
|
|
|
|
__asm mov eax, offset Foo::ptr
|
|
|
|
__asm mov eax, offset Foo::Bar::ptr
|
2017-05-05 02:19:52 +08:00
|
|
|
// CHECK-LABEL: define void @_Z2t2v()
|
|
|
|
// CHECK: call void asm sideeffect inteldialect
|
|
|
|
// CHECK-SAME: mov eax, $0
|
|
|
|
// CHECK-SAME: mov eax, $1
|
2019-12-31 03:33:56 +08:00
|
|
|
// CHECK-SAME: "i,i,~{eax},~{dirflag},~{fpsr},~{flags}"(i32** @_ZN3Foo3ptrE, i32** @_ZN3Foo3Bar3ptrE)
|
2013-04-12 07:37:53 +08:00
|
|
|
}
|
2013-04-12 07:57:29 +08:00
|
|
|
|
2013-08-15 14:47:53 +08:00
|
|
|
// CHECK-LABEL: define void @_Z2t3v()
|
2013-04-12 07:57:29 +08:00
|
|
|
void t3() {
|
|
|
|
__asm mov eax, LENGTH Foo::ptr
|
|
|
|
__asm mov eax, LENGTH Foo::Bar::ptr
|
|
|
|
__asm mov eax, LENGTH Foo::arr
|
|
|
|
__asm mov eax, LENGTH Foo::Bar::arr
|
|
|
|
|
|
|
|
__asm mov eax, TYPE Foo::ptr
|
|
|
|
__asm mov eax, TYPE Foo::Bar::ptr
|
|
|
|
__asm mov eax, TYPE Foo::arr
|
|
|
|
__asm mov eax, TYPE Foo::Bar::arr
|
|
|
|
|
|
|
|
__asm mov eax, SIZE Foo::ptr
|
|
|
|
__asm mov eax, SIZE Foo::Bar::ptr
|
|
|
|
__asm mov eax, SIZE Foo::arr
|
|
|
|
__asm mov eax, SIZE Foo::Bar::arr
|
2017-05-05 02:19:52 +08:00
|
|
|
// CHECK: call void asm sideeffect inteldialect
|
|
|
|
// CHECK-SAME: mov eax, $$1
|
|
|
|
// CHECK-SAME: mov eax, $$1
|
|
|
|
// CHECK-SAME: mov eax, $$4
|
|
|
|
// CHECK-SAME: mov eax, $$2
|
|
|
|
// CHECK-SAME: mov eax, $$4
|
|
|
|
// CHECK-SAME: mov eax, $$4
|
|
|
|
// CHECK-SAME: mov eax, $$4
|
|
|
|
// CHECK-SAME: mov eax, $$1
|
|
|
|
// CHECK-SAME: mov eax, $$4
|
|
|
|
// CHECK-SAME: mov eax, $$4
|
|
|
|
// CHECK-SAME: mov eax, $$16
|
|
|
|
// CHECK-SAME: mov eax, $$2
|
|
|
|
// CHECK-SAME: "~{eax},~{dirflag},~{fpsr},~{flags}"()
|
2013-05-03 08:10:13 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
struct T4 {
|
|
|
|
int x;
|
|
|
|
static int y;
|
|
|
|
void test();
|
|
|
|
};
|
|
|
|
|
2013-08-15 14:47:53 +08:00
|
|
|
// CHECK-LABEL: define void @_ZN2T44testEv(
|
2013-05-03 08:10:13 +08:00
|
|
|
void T4::test() {
|
|
|
|
// CHECK: [[T0:%.*]] = alloca [[T4:%.*]]*,
|
2015-02-28 05:19:58 +08:00
|
|
|
// CHECK: [[THIS:%.*]] = load [[T4]]*, [[T4]]** [[T0]]
|
2015-02-28 03:18:17 +08:00
|
|
|
// CHECK: [[X:%.*]] = getelementptr inbounds [[T4]], [[T4]]* [[THIS]], i32 0, i32 0
|
2013-05-03 08:10:13 +08:00
|
|
|
__asm mov eax, x;
|
|
|
|
__asm mov y, eax;
|
2017-05-05 02:19:52 +08:00
|
|
|
// CHECK: call void asm sideeffect inteldialect
|
|
|
|
// CHECK-SAME: mov eax, $1
|
|
|
|
// CHECK-SAME: mov $0, eax
|
|
|
|
// CHECK-SAME: "=*m,*m,~{eax},~{dirflag},~{fpsr},~{flags}"(i32* @_ZN2T41yE, i32* {{.*}})
|
2013-05-03 08:10:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
template <class T> struct T5 {
|
|
|
|
template <class U> static T create(U);
|
|
|
|
void run();
|
|
|
|
};
|
2013-08-15 14:47:53 +08:00
|
|
|
// CHECK-LABEL: define void @_Z5test5v()
|
2013-05-03 08:10:13 +08:00
|
|
|
void test5() {
|
|
|
|
// CHECK: [[X:%.*]] = alloca i32
|
|
|
|
// CHECK: [[Y:%.*]] = alloca i32
|
|
|
|
int x, y;
|
|
|
|
__asm push y
|
|
|
|
__asm call T5<int>::create<float>
|
|
|
|
__asm mov x, eax
|
2017-05-05 02:19:52 +08:00
|
|
|
// CHECK: call void asm sideeffect inteldialect
|
|
|
|
// CHECK-SAME: push $0
|
2019-12-22 13:09:37 +08:00
|
|
|
// CHECK-SAME: call dword ptr ${2:P}
|
2017-05-05 02:19:52 +08:00
|
|
|
// CHECK-SAME: mov $1, eax
|
|
|
|
// CHECK-SAME: "=*m,=*m,*m,~{esp},~{dirflag},~{fpsr},~{flags}"(i32* %y, i32* %x, i32 (float)* @_ZN2T5IiE6createIfEEiT_)
|
2013-04-12 07:57:29 +08:00
|
|
|
}
|
2013-05-25 02:32:55 +08:00
|
|
|
|
|
|
|
// Just verify this doesn't emit an error.
|
|
|
|
void test6() {
|
|
|
|
__asm {
|
|
|
|
a:
|
|
|
|
jmp a
|
|
|
|
}
|
|
|
|
}
|
2014-05-06 11:13:27 +08:00
|
|
|
|
|
|
|
void t7_struct() {
|
|
|
|
struct A {
|
|
|
|
int a;
|
|
|
|
int b;
|
|
|
|
};
|
|
|
|
__asm mov eax, [eax].A.b
|
|
|
|
// CHECK-LABEL: define void @_Z9t7_structv
|
2017-05-05 02:19:52 +08:00
|
|
|
// CHECK: call void asm sideeffect inteldialect
|
2017-08-24 16:47:26 +08:00
|
|
|
// CHECK-SAME: mov eax, [eax + $$4]
|
2017-05-05 02:19:52 +08:00
|
|
|
// CHECK-SAME: "~{eax},~{dirflag},~{fpsr},~{flags}"()
|
2014-05-06 11:13:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void t7_typedef() {
|
|
|
|
typedef struct {
|
|
|
|
int a;
|
|
|
|
int b;
|
|
|
|
} A;
|
|
|
|
__asm mov eax, [eax].A.b
|
|
|
|
// CHECK-LABEL: define void @_Z10t7_typedefv
|
2017-05-05 02:19:52 +08:00
|
|
|
// CHECK: call void asm sideeffect inteldialect
|
2017-08-24 16:47:26 +08:00
|
|
|
// CHECK-SAME: mov eax, [eax + $$4]
|
2017-05-05 02:19:52 +08:00
|
|
|
// CHECK-SAME: "~{eax},~{dirflag},~{fpsr},~{flags}"()
|
2014-05-06 11:13:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void t7_using() {
|
|
|
|
using A = struct {
|
|
|
|
int a;
|
|
|
|
int b;
|
|
|
|
};
|
|
|
|
__asm mov eax, [eax].A.b
|
|
|
|
// CHECK-LABEL: define void @_Z8t7_usingv
|
2017-05-05 02:19:52 +08:00
|
|
|
// CHECK: call void asm sideeffect inteldialect
|
2017-08-24 16:47:26 +08:00
|
|
|
// CHECK-SAME: mov eax, [eax + $$4]
|
2017-05-05 02:19:52 +08:00
|
|
|
// CHECK-SAME: "~{eax},~{dirflag},~{fpsr},~{flags}"()
|
2014-05-06 11:13:27 +08:00
|
|
|
}
|
2014-09-22 10:21:54 +08:00
|
|
|
|
|
|
|
void t8() {
|
|
|
|
__asm some_label:
|
|
|
|
// CHECK-LABEL: define void @_Z2t8v()
|
2017-05-05 02:19:52 +08:00
|
|
|
// CHECK: call void asm sideeffect inteldialect
|
|
|
|
// CHECK-SAME: L__MSASMLABEL_.${:uid}__some_label:
|
|
|
|
// CHECK-SAME: "~{dirflag},~{fpsr},~{flags}"()
|
2014-09-22 10:21:54 +08:00
|
|
|
struct A {
|
|
|
|
static void g() {
|
|
|
|
__asm jmp some_label ; This should jump forwards
|
|
|
|
__asm some_label:
|
|
|
|
__asm nop
|
|
|
|
// CHECK-LABEL: define internal void @_ZZ2t8vEN1A1gEv()
|
2017-05-05 02:19:52 +08:00
|
|
|
// CHECK: call void asm sideeffect inteldialect
|
|
|
|
// CHECK-SAME: jmp L__MSASMLABEL_.${:uid}__some_label
|
|
|
|
// CHECK-SAME: L__MSASMLABEL_.${:uid}__some_label:
|
|
|
|
// CHECK-SAME: nop
|
|
|
|
// CHECK-SAME: "~{dirflag},~{fpsr},~{flags}"()
|
2014-09-22 10:21:54 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
A::g();
|
|
|
|
}
|
|
|
|
|
2017-08-09 21:31:41 +08:00
|
|
|
void t9() {
|
|
|
|
// CHECK-LABEL: define void @_Z2t9v()
|
|
|
|
struct A {
|
|
|
|
int a;
|
|
|
|
int b;
|
|
|
|
void g() {
|
|
|
|
__asm mov eax, dword ptr [eax]this.b
|
|
|
|
// CHECK: call void asm sideeffect inteldialect
|
2017-08-24 16:47:26 +08:00
|
|
|
// CHECK-SAME: mov eax, dword ptr [eax + $$4]
|
2017-08-09 21:31:41 +08:00
|
|
|
// CHECK-SAME: "~{eax},~{dirflag},~{fpsr},~{flags}"()
|
|
|
|
}
|
|
|
|
};
|
|
|
|
A AA;
|
|
|
|
AA.g();
|
|
|
|
}
|
|
|
|
|