199 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			199 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			C++
		
	
	
	
// REQUIRES: x86-registered-target
 | 
						|
// RUN: %clang_cc1 -x c++ %s -triple i386-apple-darwin10 -fasm-blocks -emit-llvm -o - -std=c++11 | FileCheck %s
 | 
						|
 | 
						|
// rdar://13645930
 | 
						|
 | 
						|
struct Foo {
 | 
						|
  static int *ptr;
 | 
						|
  static int a, b;
 | 
						|
  int arr[4];
 | 
						|
  struct Bar {
 | 
						|
    static int *ptr;
 | 
						|
    char arr[2];
 | 
						|
  };
 | 
						|
};
 | 
						|
 | 
						|
void t1() {
 | 
						|
// CHECK-LABEL: define void @_Z2t1v()
 | 
						|
  Foo::ptr = (int *)0xDEADBEEF;
 | 
						|
  Foo::Bar::ptr = (int *)0xDEADBEEF;
 | 
						|
// 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)
 | 
						|
  __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]
 | 
						|
}
 | 
						|
 | 
						|
int gvar = 10;
 | 
						|
void t2() {
 | 
						|
  int lvar = 10;
 | 
						|
  __asm mov eax, offset Foo::ptr
 | 
						|
  __asm mov eax, offset Foo::Bar::ptr
 | 
						|
// CHECK-LABEL: define void @_Z2t2v()
 | 
						|
// CHECK: call void asm sideeffect inteldialect
 | 
						|
// CHECK-SAME: mov eax, $0
 | 
						|
// CHECK-SAME: mov eax, $1
 | 
						|
// CHECK-SAME: "r,r,~{eax},~{dirflag},~{fpsr},~{flags}"(i32** @_ZN3Foo3ptrE, i32** @_ZN3Foo3Bar3ptrE)
 | 
						|
}
 | 
						|
 | 
						|
// CHECK-LABEL: define void @_Z2t3v()
 | 
						|
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
 | 
						|
// 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}"()
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
struct T4 {
 | 
						|
  int x;
 | 
						|
  static int y;
 | 
						|
  void test();
 | 
						|
};
 | 
						|
 | 
						|
// CHECK-LABEL: define void @_ZN2T44testEv(
 | 
						|
void T4::test() {
 | 
						|
// CHECK: [[T0:%.*]] = alloca [[T4:%.*]]*,
 | 
						|
// CHECK: [[THIS:%.*]] = load [[T4]]*, [[T4]]** [[T0]]
 | 
						|
// CHECK: [[X:%.*]] = getelementptr inbounds [[T4]], [[T4]]* [[THIS]], i32 0, i32 0
 | 
						|
  __asm mov eax, x;
 | 
						|
  __asm mov y, eax;
 | 
						|
// 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* {{.*}})
 | 
						|
}
 | 
						|
 | 
						|
template <class T> struct T5 {
 | 
						|
  template <class U> static T create(U);
 | 
						|
  void run();
 | 
						|
};
 | 
						|
// CHECK-LABEL: define void @_Z5test5v()
 | 
						|
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
 | 
						|
  // CHECK: call void asm sideeffect inteldialect
 | 
						|
  // CHECK-SAME: push $0
 | 
						|
  // CHECK-SAME: call dword ptr $2
 | 
						|
  // CHECK-SAME: mov $1, eax
 | 
						|
  // CHECK-SAME: "=*m,=*m,*m,~{esp},~{dirflag},~{fpsr},~{flags}"(i32* %y, i32* %x, i32 (float)* @_ZN2T5IiE6createIfEEiT_)
 | 
						|
}
 | 
						|
 | 
						|
// Just verify this doesn't emit an error.
 | 
						|
void test6() {
 | 
						|
  __asm {
 | 
						|
   a:
 | 
						|
   jmp a
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void t7_struct() {
 | 
						|
  struct A {
 | 
						|
    int a;
 | 
						|
    int b;
 | 
						|
  };
 | 
						|
  __asm mov eax, [eax].A.b
 | 
						|
  // CHECK-LABEL: define void @_Z9t7_structv
 | 
						|
  // CHECK: call void asm sideeffect inteldialect
 | 
						|
  // CHECK-SAME: mov eax, [eax + $$4]
 | 
						|
  // CHECK-SAME: "~{eax},~{dirflag},~{fpsr},~{flags}"()
 | 
						|
}
 | 
						|
 | 
						|
void t7_typedef() {
 | 
						|
  typedef struct {
 | 
						|
    int a;
 | 
						|
    int b;
 | 
						|
  } A;
 | 
						|
  __asm mov eax, [eax].A.b
 | 
						|
  // CHECK-LABEL: define void @_Z10t7_typedefv
 | 
						|
  // CHECK: call void asm sideeffect inteldialect
 | 
						|
  // CHECK-SAME: mov eax, [eax + $$4]
 | 
						|
  // CHECK-SAME: "~{eax},~{dirflag},~{fpsr},~{flags}"()
 | 
						|
}
 | 
						|
 | 
						|
void t7_using() {
 | 
						|
  using A = struct {
 | 
						|
    int a;
 | 
						|
    int b;
 | 
						|
  };
 | 
						|
  __asm mov eax, [eax].A.b
 | 
						|
  // CHECK-LABEL: define void @_Z8t7_usingv
 | 
						|
  // CHECK: call void asm sideeffect inteldialect
 | 
						|
  // CHECK-SAME: mov eax, [eax + $$4]
 | 
						|
  // CHECK-SAME: "~{eax},~{dirflag},~{fpsr},~{flags}"()
 | 
						|
}
 | 
						|
 | 
						|
void t8() {
 | 
						|
  __asm some_label:
 | 
						|
  // CHECK-LABEL: define void @_Z2t8v()
 | 
						|
  // CHECK: call void asm sideeffect inteldialect
 | 
						|
  // CHECK-SAME: L__MSASMLABEL_.${:uid}__some_label:
 | 
						|
  // CHECK-SAME: "~{dirflag},~{fpsr},~{flags}"()
 | 
						|
  struct A {
 | 
						|
    static void g() {
 | 
						|
      __asm jmp some_label ; This should jump forwards
 | 
						|
      __asm some_label:
 | 
						|
      __asm nop
 | 
						|
      // CHECK-LABEL: define internal void @_ZZ2t8vEN1A1gEv()
 | 
						|
      // 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}"()
 | 
						|
    }
 | 
						|
  };
 | 
						|
  A::g();
 | 
						|
}
 | 
						|
 | 
						|
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
 | 
						|
      // CHECK-SAME: mov eax, dword ptr [eax + $$4]
 | 
						|
      // CHECK-SAME: "~{eax},~{dirflag},~{fpsr},~{flags}"()
 | 
						|
    }
 | 
						|
  };
 | 
						|
  A AA;
 | 
						|
  AA.g();
 | 
						|
}
 | 
						|
 |