blob: e1353a72b396cfb721a1faedd5ef8de123a5364a [file] [log] [blame]
Daniel Dunbara5728872009-12-15 20:14:24 +00001// RUN: %clang_cc1 %s -emit-llvm -o - -triple=x86_64-apple-darwin9 | FileCheck %s
Anders Carlsson7b699562009-09-29 03:38:56 +00002
Anders Carlssonf57b4e42009-10-03 15:02:02 +00003struct A { int a; void f(); virtual void vf(); };
4struct B { int b; virtual void g(); };
Anders Carlsson7b699562009-09-29 03:38:56 +00005struct C : B, A { };
6
7void (A::*pa)();
8void (A::*volatile vpa)();
9void (B::*pb)();
10void (C::*pc)();
11
Anders Carlssonf57b4e42009-10-03 15:02:02 +000012// CHECK: @pa2 = global %0 { i64 ptrtoint (void ()* @_ZN1A1fEv to i64), i64 0 }, align 8
13void (A::*pa2)() = &A::f;
14
15// CHECK: @pa3 = global %0 { i64 1, i64 0 }, align 8
16void (A::*pa3)() = &A::vf;
17
Anders Carlsson2c51f092009-10-03 15:13:22 +000018// CHECK: @pc2 = global %0 { i64 ptrtoint (void ()* @_ZN1A1fEv to i64), i64 16 }, align 8
19void (C::*pc2)() = &C::f;
20
21// CHECK: @pc3 = global %0 { i64 1, i64 0 }, align 8
22void (A::*pc3)() = &A::vf;
23
Anders Carlsson7b699562009-09-29 03:38:56 +000024void f() {
Anders Carlssonee383162009-10-03 14:39:13 +000025 // CHECK: store i64 0, i64* getelementptr inbounds (%0* @pa, i32 0, i32 0)
26 // CHECK: store i64 0, i64* getelementptr inbounds (%0* @pa, i32 0, i32 1)
Anders Carlsson7b699562009-09-29 03:38:56 +000027 pa = 0;
Anders Carlssonee383162009-10-03 14:39:13 +000028
29 // CHECK: volatile store i64 0, i64* getelementptr inbounds (%0* @vpa, i32 0, i32 0)
30 // CHECK: volatile store i64 0, i64* getelementptr inbounds (%0* @vpa, i32 0, i32 1)
Anders Carlsson7b699562009-09-29 03:38:56 +000031 vpa = 0;
Anders Carlssonee383162009-10-03 14:39:13 +000032
Daniel Dunbar47d1e822009-11-11 03:48:26 +000033 // CHECK: store i64 {{.*}}, i64* getelementptr inbounds (%0* @pc, i32 0, i32 0)
34 // CHECK: [[ADJ:%[a-zA-Z0-9\.]+]] = add i64 {{.*}}, 16
Anders Carlssonee383162009-10-03 14:39:13 +000035 // CHECK: store i64 [[ADJ]], i64* getelementptr inbounds (%0* @pc, i32 0, i32 1)
Anders Carlssonf57b4e42009-10-03 15:02:02 +000036 pc = pa;
Eli Friedman9b372742009-11-27 04:56:40 +000037
38 // CHECK: store i64 {{.*}}, i64* getelementptr inbounds (%0* @pa, i32 0, i32 0)
39 // CHECK: [[ADJ:%[a-zA-Z0-9\.]+]] = sub i64 {{.*}}, 16
40 // CHECK: store i64 [[ADJ]], i64* getelementptr inbounds (%0* @pa, i32 0, i32 1)
41 pa = static_cast<void (A::*)()>(pc);
Anders Carlsson7b699562009-09-29 03:38:56 +000042}
Anders Carlssona024d172009-10-03 15:43:24 +000043
44void f2() {
45 // CHECK: [[pa2ptr:%[a-zA-Z0-9\.]+]] = getelementptr inbounds %0* %pa2, i32 0, i32 0
46 // CHECK: store i64 ptrtoint (void ()* @_ZN1A1fEv to i64), i64* [[pa2ptr]]
47 // CHECK: [[pa2adj:%[a-zA-Z0-9\.]+]] = getelementptr inbounds %0* %pa2, i32 0, i32 1
48 // CHECK: store i64 0, i64* [[pa2adj]]
49 void (A::*pa2)() = &A::f;
50
51 // CHECK: [[pa3ptr:%[a-zA-Z0-9\.]+]] = getelementptr inbounds %0* %pa3, i32 0, i32 0
52 // CHECK: store i64 1, i64* [[pa3ptr]]
53 // CHECK: [[pa3adj:%[a-zA-Z0-9\.]+]] = getelementptr inbounds %0* %pa3, i32 0, i32 1
Daniel Dunbar47d1e822009-11-11 03:48:26 +000054 // CHECK: store i64 0, i64* [[pa3adj]]
Anders Carlssona024d172009-10-03 15:43:24 +000055 void (A::*pa3)() = &A::vf;
56}
Anders Carlsson375c31c2009-10-03 19:43:08 +000057
58void f3(A *a, A &ar) {
59 (a->*pa)();
60 (ar.*pa)();
61}
Anders Carlsson3eea6352009-10-13 17:41:28 +000062
Anders Carlssonbc0e0782009-11-23 20:04:44 +000063bool f4() {
64 return pa;
65}
66
Anders Carlsson3eea6352009-10-13 17:41:28 +000067// PR5177
68namespace PR5177 {
69 struct A {
70 bool foo(int*) const;
71 } a;
72
73 struct B1 {
74 bool (A::*pmf)(int*) const;
75 const A* pa;
76
77 B1() : pmf(&A::foo), pa(&a) {}
78 bool operator()() const { return (pa->*pmf)(new int); }
79 };
80
81 void bar(B1 b2) { while (b2()) ; }
82}
Anders Carlssonbb378cb2009-10-18 20:31:03 +000083
84// PR5138
85namespace PR5138 {
86 struct foo {
87 virtual void bar(foo *);
88 };
89
90 extern "C" {
91 void baz(foo *);
92 }
93
94 void (foo::*ptr1)(void *) = (void (foo::*)(void *))&foo::bar;
95 void (*ptr2)(void *) = (void (*)(void *))&baz;
96
97 void (foo::*ptr3)(void) = (void (foo::*)(void))&foo::bar;
98}
Anders Carlssona4c98cd2009-11-23 21:47:44 +000099
100// PR5593
101namespace PR5593 {
102 struct A { };
103
104 bool f(void (A::*f)()) {
105 return f && f;
106 }
107}
Eli Friedmanb81c7862009-12-11 07:36:43 +0000108
109namespace PR5718 {
110 struct A { };
111
112 bool f(void (A::*f)(), void (A::*g)()) {
113 return f == g;
114 }
115}
Eli Friedman3a173702009-12-11 09:26:29 +0000116
117namespace BoolMemberPointer {
118 struct A { };
119
120 bool f(void (A::*f)()) {
121 return !f;
122 }
123
124 bool g(void (A::*f)()) {
125 if (!!f)
126 return true;
127 return false;
128 }
129}
130
Anders Carlsson7af4ec72010-01-05 05:04:05 +0000131// PR5940
132namespace PR5940 {
133 class foo {
134 public:
135 virtual void baz(void);
136 };
137
138 void foo::baz(void) {
139 void (foo::*ptr)(void) = &foo::baz;
140 }
141}
Eli Friedman3005efe2010-01-16 00:00:48 +0000142
143namespace MemberPointerImpCast {
144 struct A {
145 int x;
146 };
147 struct B : public A {
148 };
149 void f(B* obj, void (A::*method)()) {
150 (obj->*method)();
151 }
152}