ms-inline-asm.cpp 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  1. // REQUIRES: x86-registered-target
  2. // RUN: %clang_cc1 -x c++ %s -triple i386-apple-darwin10 -fasm-blocks -emit-llvm -o - -std=c++11 | FileCheck %s
  3. // rdar://13645930
  4. struct Foo {
  5. static int *ptr;
  6. static int a, b;
  7. int arr[4];
  8. struct Bar {
  9. static int *ptr;
  10. char arr[2];
  11. };
  12. };
  13. void t1() {
  14. Foo::ptr = (int *)0xDEADBEEF;
  15. Foo::Bar::ptr = (int *)0xDEADBEEF;
  16. __asm mov eax, Foo ::ptr
  17. __asm mov eax, Foo :: Bar :: ptr
  18. __asm mov eax, [Foo:: ptr]
  19. __asm mov eax, dword ptr [Foo :: ptr]
  20. __asm mov eax, dword ptr [Foo :: ptr]
  21. // CHECK: @_Z2t1v
  22. // CHECK: call void asm sideeffect inteldialect "mov eax, dword ptr $0\0A\09mov eax, dword ptr $1\0A\09mov eax, dword ptr $2\0A\09mov eax, dword ptr $3\0A\09mov eax, dword ptr $4", "*m,*m,*m,*m,*m,~{eax},~{dirflag},~{fpsr},~{flags}"(i32** @_ZN3Foo3ptrE, i32** @_ZN3Foo3Bar3ptrE, i32** @_ZN3Foo3ptrE, i32** @_ZN3Foo3ptrE, i32** @_ZN3Foo3ptrE)
  23. }
  24. int gvar = 10;
  25. void t2() {
  26. int lvar = 10;
  27. __asm mov eax, offset Foo::ptr
  28. __asm mov eax, offset Foo::Bar::ptr
  29. // CHECK: t2
  30. // CHECK: call void asm sideeffect inteldialect "mov eax, $0\0A\09mov eax, $1", "r,r,~{eax},~{dirflag},~{fpsr},~{flags}"(i32** @_ZN3Foo3ptrE, i32** @_ZN3Foo3Bar3ptrE)
  31. }
  32. // CHECK-LABEL: define void @_Z2t3v()
  33. void t3() {
  34. __asm mov eax, LENGTH Foo::ptr
  35. __asm mov eax, LENGTH Foo::Bar::ptr
  36. __asm mov eax, LENGTH Foo::arr
  37. __asm mov eax, LENGTH Foo::Bar::arr
  38. __asm mov eax, TYPE Foo::ptr
  39. __asm mov eax, TYPE Foo::Bar::ptr
  40. __asm mov eax, TYPE Foo::arr
  41. __asm mov eax, TYPE Foo::Bar::arr
  42. __asm mov eax, SIZE Foo::ptr
  43. __asm mov eax, SIZE Foo::Bar::ptr
  44. __asm mov eax, SIZE Foo::arr
  45. __asm mov eax, SIZE Foo::Bar::arr
  46. // CHECK: call void asm sideeffect inteldialect "mov eax, $$1\0A\09mov eax, $$1\0A\09mov eax, $$4\0A\09mov eax, $$2\0A\09mov eax, $$4\0A\09mov eax, $$4\0A\09mov eax, $$4\0A\09mov eax, $$1\0A\09mov eax, $$4\0A\09mov eax, $$4\0A\09mov eax, $$16\0A\09mov eax, $$2", "~{eax},~{dirflag},~{fpsr},~{flags}"()
  47. }
  48. struct T4 {
  49. int x;
  50. static int y;
  51. void test();
  52. };
  53. // CHECK-LABEL: define void @_ZN2T44testEv(
  54. void T4::test() {
  55. // CHECK: [[T0:%.*]] = alloca [[T4:%.*]]*,
  56. // CHECK: [[THIS:%.*]] = load [[T4]]** [[T0]]
  57. // CHECK: [[X:%.*]] = getelementptr inbounds [[T4]], [[T4]]* [[THIS]], i32 0, i32 0
  58. __asm mov eax, x;
  59. __asm mov y, eax;
  60. // CHECK: call void asm sideeffect inteldialect "mov eax, dword ptr $1\0A\09mov dword ptr $0, eax", "=*m,*m,~{eax},~{dirflag},~{fpsr},~{flags}"(i32* @_ZN2T41yE, i32* {{.*}})
  61. }
  62. template <class T> struct T5 {
  63. template <class U> static T create(U);
  64. void run();
  65. };
  66. // CHECK-LABEL: define void @_Z5test5v()
  67. void test5() {
  68. // CHECK: [[X:%.*]] = alloca i32
  69. // CHECK: [[Y:%.*]] = alloca i32
  70. int x, y;
  71. __asm push y
  72. __asm call T5<int>::create<float>
  73. __asm mov x, eax
  74. // CHECK: call void asm sideeffect inteldialect "push dword ptr $0\0A\09call dword ptr $2\0A\09mov dword ptr $1, eax", "=*m,=*m,*m,~{esp},~{dirflag},~{fpsr},~{flags}"(i32* %y, i32* %x, i32 (float)* @_ZN2T5IiE6createIfEEiT_)
  75. }
  76. // Just verify this doesn't emit an error.
  77. void test6() {
  78. __asm {
  79. a:
  80. jmp a
  81. }
  82. }
  83. void t7_struct() {
  84. struct A {
  85. int a;
  86. int b;
  87. };
  88. __asm mov eax, [eax].A.b
  89. // CHECK-LABEL: define void @_Z9t7_structv
  90. // CHECK: call void asm sideeffect inteldialect "mov eax, [eax].4", "~{eax},~{dirflag},~{fpsr},~{flags}"()
  91. }
  92. void t7_typedef() {
  93. typedef struct {
  94. int a;
  95. int b;
  96. } A;
  97. __asm mov eax, [eax].A.b
  98. // CHECK-LABEL: define void @_Z10t7_typedefv
  99. // CHECK: call void asm sideeffect inteldialect "mov eax, [eax].4", "~{eax},~{dirflag},~{fpsr},~{flags}"()
  100. }
  101. void t7_using() {
  102. using A = struct {
  103. int a;
  104. int b;
  105. };
  106. __asm mov eax, [eax].A.b
  107. // CHECK-LABEL: define void @_Z8t7_usingv
  108. // CHECK: call void asm sideeffect inteldialect "mov eax, [eax].4", "~{eax},~{dirflag},~{fpsr},~{flags}"()
  109. }
  110. void t8() {
  111. __asm some_label:
  112. // CHECK-LABEL: define void @_Z2t8v()
  113. // CHECK: call void asm sideeffect inteldialect "L__MSASMLABEL_.1__some_label:", "~{dirflag},~{fpsr},~{flags}"()
  114. struct A {
  115. static void g() {
  116. __asm jmp some_label ; This should jump forwards
  117. __asm some_label:
  118. __asm nop
  119. // CHECK-LABEL: define internal void @_ZZ2t8vEN1A1gEv()
  120. // CHECK: call void asm sideeffect inteldialect "jmp L__MSASMLABEL_.2__some_label\0A\09L__MSASMLABEL_.2__some_label:\0A\09nop", "~{dirflag},~{fpsr},~{flags}"()
  121. }
  122. };
  123. A::g();
  124. }