Skip to content

Commit 433b6af

Browse files
committed
[CIR] Add support for C++ conversion operators
This fairly simple addition enables codegen for C++ conversion operators
1 parent fc2850f commit 433b6af

File tree

2 files changed

+131
-0
lines changed

2 files changed

+131
-0
lines changed

clang/lib/CIR/CodeGen/CIRGenModule.cpp

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1235,6 +1235,7 @@ void CIRGenModule::emitTopLevelDecl(Decl *decl) {
12351235
decl->getDeclKindName());
12361236
break;
12371237

1238+
case Decl::CXXConversion:
12381239
case Decl::CXXMethod:
12391240
case Decl::Function: {
12401241
auto *fd = cast<FunctionDecl>(decl);
Lines changed: 130 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,130 @@
1+
// RUN: %clang_cc1 -std=c++11 -triple aarch64-none-linux-android21 -fclangir -emit-cir %s -o %t.cir
2+
// RUN: FileCheck --check-prefix=CIR --input-file=%t.cir %s
3+
// RUN: %clang_cc1 -std=c++11 -triple aarch64-none-linux-android21 -fclangir -emit-llvm %s -o %t-cir.ll
4+
// RUN: FileCheck --check-prefix=LLVM --input-file=%t-cir.ll %s
5+
// RUN: %clang_cc1 -std=c++11 -triple aarch64-none-linux-android21 -emit-llvm %s -o %t.ll
6+
// RUN: FileCheck --check-prefix=OGCG --input-file=%t.ll %s
7+
8+
struct inline_operator {
9+
operator int() const {
10+
return 987;
11+
}
12+
13+
int operator+(inline_operator) {
14+
return 666;
15+
}
16+
};
17+
18+
struct out_of_line_operator {
19+
operator int();
20+
};
21+
22+
out_of_line_operator::operator int() {
23+
return 123;
24+
}
25+
26+
void test() {
27+
int x = 42;
28+
29+
inline_operator i;
30+
x = i;
31+
32+
out_of_line_operator o;
33+
x = o;
34+
}
35+
36+
// CIR: cir.func dso_local @_ZN20out_of_line_operatorcviEv(%[[THIS_ARG:.+]]: !cir.ptr<!rec_out_of_line_operator>{{.*}}) -> !s32i
37+
// CIR: %[[THIS_ALLOCA:.+]] = cir.alloca !cir.ptr<!rec_out_of_line_operator>, !cir.ptr<!cir.ptr<!rec_out_of_line_operator>>, ["this", init]
38+
// CIR: %[[RETVAL:.+]] = cir.alloca !s32i, !cir.ptr<!s32i>, ["__retval"]
39+
// CIR: cir.store %[[THIS_ARG]], %[[THIS_ALLOCA]] : !cir.ptr<!rec_out_of_line_operator>, !cir.ptr<!cir.ptr<!rec_out_of_line_operator>>
40+
// CIR: %[[THIS_LOAD:.+]] = cir.load %[[THIS_ALLOCA]] : !cir.ptr<!cir.ptr<!rec_out_of_line_operator>>, !cir.ptr<!rec_out_of_line_operator>
41+
// CIR: %[[CONST_123:.+]] = cir.const #cir.int<123> : !s32i
42+
// CIR: cir.store %[[CONST_123]], %[[RETVAL]] : !s32i, !cir.ptr<!s32i>
43+
// CIR: %[[RET_LOAD:.+]] = cir.load %[[RETVAL]] : !cir.ptr<!s32i>, !s32i
44+
// CIR: cir.return %[[RET_LOAD]] : !s32i
45+
// CIR: }
46+
47+
// CIR: cir.func comdat linkonce_odr @_ZNK15inline_operatorcviEv(%[[INLINE_THIS_ARG:.+]]: !cir.ptr<!rec_inline_operator>{{.*}}) -> !s32i
48+
// CIR: %[[INLINE_THIS_ALLOCA:.+]] = cir.alloca !cir.ptr<!rec_inline_operator>, !cir.ptr<!cir.ptr<!rec_inline_operator>>, ["this", init]
49+
// CIR: %[[INLINE_RETVAL:.+]] = cir.alloca !s32i, !cir.ptr<!s32i>, ["__retval"]
50+
// CIR: cir.store %[[INLINE_THIS_ARG]], %[[INLINE_THIS_ALLOCA]] : !cir.ptr<!rec_inline_operator>, !cir.ptr<!cir.ptr<!rec_inline_operator>>
51+
// CIR: %[[INLINE_THIS_LOAD:.+]] = cir.load %[[INLINE_THIS_ALLOCA]] : !cir.ptr<!cir.ptr<!rec_inline_operator>>, !cir.ptr<!rec_inline_operator>
52+
// CIR: %[[CONST_987:.+]] = cir.const #cir.int<987> : !s32i
53+
// CIR: cir.store %[[CONST_987]], %[[INLINE_RETVAL]] : !s32i, !cir.ptr<!s32i>
54+
// CIR: %[[INLINE_RET_LOAD:.+]] = cir.load %[[INLINE_RETVAL]] : !cir.ptr<!s32i>, !s32i
55+
// CIR: cir.return %[[INLINE_RET_LOAD]] : !s32i
56+
// CIR: }
57+
58+
// CIR: cir.func dso_local @_Z4testv()
59+
// CIR: %[[X_ALLOCA:.+]] = cir.alloca !s32i, !cir.ptr<!s32i>, ["x", init]
60+
// CIR: %[[I_ALLOCA:.+]] = cir.alloca {{.*}}, {{.*}}, ["i"]
61+
// CIR: %[[O_ALLOCA:.+]] = cir.alloca {{.*}}, {{.*}}, ["o"]
62+
// CIR: %[[CONST_42:.+]] = cir.const #cir.int<42> : !s32i
63+
// CIR: cir.store align(4) %[[CONST_42]], %[[X_ALLOCA]] : !s32i, !cir.ptr<!s32i>
64+
// CIR: %[[INLINE_CALL:.+]] = cir.call @_ZNK15inline_operatorcviEv(%[[I_ALLOCA]]) : ({{.*}}) -> !s32i
65+
// CIR: cir.store align(4) %[[INLINE_CALL]], %[[X_ALLOCA]] : !s32i, !cir.ptr<!s32i>
66+
// CIR: %[[OUTLINE_CALL:.+]] = cir.call @_ZN20out_of_line_operatorcviEv(%[[O_ALLOCA]]) : ({{.*}}) -> !s32i
67+
// CIR: cir.store align(4) %[[OUTLINE_CALL]], %[[X_ALLOCA]] : !s32i, !cir.ptr<!s32i>
68+
// CIR: cir.return
69+
// CIR: }
70+
71+
// LLVM: define dso_local i32 @_ZN20out_of_line_operatorcviEv(ptr %[[PARAM0:.+]])
72+
// LLVM: %[[THIS_ALLOCA:.+]] = alloca ptr, i64 1
73+
// LLVM: %[[RETVAL:.+]] = alloca i32, i64 1
74+
// LLVM: store ptr %[[PARAM0]], ptr %[[THIS_ALLOCA]]
75+
// LLVM: %[[THIS_LOAD:.+]] = load ptr, ptr %[[THIS_ALLOCA]]
76+
// LLVM: store i32 123, ptr %[[RETVAL]]
77+
// LLVM: %[[RET_LOAD:.+]] = load i32, ptr %[[RETVAL]]
78+
// LLVM: ret i32 %[[RET_LOAD]]
79+
// LLVM: }
80+
81+
// LLVM: define linkonce_odr i32 @_ZNK15inline_operatorcviEv(ptr %[[INLINE_PARAM0:.+]])
82+
// LLVM: %[[INLINE_THIS_ALLOCA:.+]] = alloca ptr, i64 1
83+
// LLVM: %[[INLINE_RETVAL:.+]] = alloca i32, i64 1
84+
// LLVM: store ptr %[[INLINE_PARAM0]], ptr %[[INLINE_THIS_ALLOCA]]
85+
// LLVM: %[[INLINE_THIS_LOAD:.+]] = load ptr, ptr %[[INLINE_THIS_ALLOCA]]
86+
// LLVM: store i32 987, ptr %[[INLINE_RETVAL]]
87+
// LLVM: %[[INLINE_RET_LOAD:.+]] = load i32, ptr %[[INLINE_RETVAL]]
88+
// LLVM: ret i32 %[[INLINE_RET_LOAD]]
89+
// LLVM: }
90+
91+
// LLVM: define dso_local void @_Z4testv()
92+
// LLVM: %[[X_ALLOCA:.+]] = alloca i32, i64 1
93+
// LLVM: %[[I_ALLOCA:.+]] = alloca {{.*}}, i64 1
94+
// LLVM: %[[O_ALLOCA:.+]] = alloca {{.*}}, i64 1
95+
// LLVM: store i32 42, ptr %[[X_ALLOCA]]
96+
// LLVM: %[[INLINE_CALL:.+]] = call i32 @_ZNK15inline_operatorcviEv(ptr %[[I_ALLOCA]])
97+
// LLVM: store i32 %[[INLINE_CALL]], ptr %[[X_ALLOCA]]
98+
// LLVM: %[[OUTLINE_CALL:.+]] = call i32 @_ZN20out_of_line_operatorcviEv(ptr %[[O_ALLOCA]])
99+
// LLVM: store i32 %[[OUTLINE_CALL]], ptr %[[X_ALLOCA]]
100+
// LLVM: ret void
101+
// LLVM: }
102+
103+
// OGCG: define dso_local noundef i32 @_ZN20out_of_line_operatorcviEv(ptr {{.*}} %[[THIS_PARAM:.+]])
104+
// OGCG: entry:
105+
// OGCG: %[[THIS_ADDR:.+]] = alloca ptr
106+
// OGCG: store ptr %[[THIS_PARAM]], ptr %[[THIS_ADDR]]
107+
// OGCG: %[[THIS_LOAD:.+]] = load ptr, ptr %[[THIS_ADDR]]
108+
// OGCG: ret i32 123
109+
// OGCG: }
110+
111+
// OGCG: define dso_local void @_Z4testv()
112+
// OGCG: entry:
113+
// OGCG: %[[X_VAR:.+]] = alloca i32
114+
// OGCG: %[[I_VAR:.+]] = alloca {{.*}}
115+
// OGCG: %[[O_VAR:.+]] = alloca {{.*}}
116+
// OGCG: store i32 42, ptr %[[X_VAR]]
117+
// OGCG: %[[INLINE_CALL:.+]] = call noundef i32 @_ZNK15inline_operatorcviEv(ptr {{.*}} %[[I_VAR]])
118+
// OGCG: store i32 %[[INLINE_CALL]], ptr %[[X_VAR]]
119+
// OGCG: %[[OUTLINE_CALL:.+]] = call noundef i32 @_ZN20out_of_line_operatorcviEv(ptr {{.*}} %[[O_VAR]])
120+
// OGCG: store i32 %[[OUTLINE_CALL]], ptr %[[X_VAR]]
121+
// OGCG: ret void
122+
// OGCG: }
123+
124+
// OGCG: define linkonce_odr noundef i32 @_ZNK15inline_operatorcviEv(ptr {{.*}} %[[INLINE_THIS_PARAM:.+]])
125+
// OGCG: entry:
126+
// OGCG: %[[INLINE_THIS_ADDR:.+]] = alloca ptr
127+
// OGCG: store ptr %[[INLINE_THIS_PARAM]], ptr %[[INLINE_THIS_ADDR]]
128+
// OGCG: %[[INLINE_THIS_LOAD:.+]] = load ptr, ptr %[[INLINE_THIS_ADDR]]
129+
// OGCG: ret i32 987
130+
// OGCG: }

0 commit comments

Comments
 (0)