blob: 830c901fe0b15c330b76c7dbf1438b12cb699b8b [file] [log] [blame]
Michael Forster7e4244a2022-04-25 00:39:01 -07001// Part of the Crubit project, under the Apache License v2.0 with LLVM
2// Exceptions. See /LICENSE for license information.
3// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
4
5#include "rs_bindings_from_cc/importers/function.h"
6
Lukasz Anforowiczcec7a8a2022-04-27 10:24:51 -07007#include "absl/strings/substitute.h"
Michael Forster7e4244a2022-04-25 00:39:01 -07008
9namespace crubit {
10
11std::optional<IR::Item> FunctionDeclImporter::Import(
12 clang::FunctionDecl* function_decl) {
13 if (!ictx_.IsFromCurrentTarget(function_decl)) return std::nullopt;
14 if (function_decl->isDeleted()) return std::nullopt;
15
16 clang::tidy::lifetimes::LifetimeSymbolTable lifetime_symbol_table;
17 llvm::Expected<clang::tidy::lifetimes::FunctionLifetimes> lifetimes =
18 clang::tidy::lifetimes::GetLifetimeAnnotations(
19 function_decl, *ictx_.invocation_.lifetime_context_,
20 &lifetime_symbol_table);
21
22 std::vector<FuncParam> params;
23 std::set<std::string> errors;
24 auto add_error = [&errors](std::string msg) {
25 auto result = errors.insert(std::move(msg));
26 CRUBIT_CHECK(result.second && "Duplicated error message");
27 };
28 if (auto* method_decl =
29 clang::dyn_cast<clang::CXXMethodDecl>(function_decl)) {
30 if (!ictx_.type_mapper_.Contains(method_decl->getParent())) {
31 return ictx_.ImportUnsupportedItem(function_decl,
32 "Couldn't import the parent");
33 }
34
35 // non-static member functions receive an implicit `this` parameter.
36 if (method_decl->isInstance()) {
37 std::optional<clang::tidy::lifetimes::ValueLifetimes> this_lifetimes;
38 if (lifetimes) {
39 this_lifetimes = lifetimes->GetThisLifetimes();
40 }
41 auto param_type = ictx_.type_mapper_.ConvertQualType(
42 method_decl->getThisType(), this_lifetimes,
43 /*nullable=*/false);
44 if (!param_type.ok()) {
45 add_error(absl::StrCat("`this` parameter is not supported: ",
46 param_type.status().message()));
47 } else {
48 params.push_back({*std::move(param_type), Identifier("__this")});
49 }
50 }
51 }
52
53 if (lifetimes) {
54 CRUBIT_CHECK(lifetimes->IsValidForDecl(function_decl));
55 }
56
57 for (unsigned i = 0; i < function_decl->getNumParams(); ++i) {
58 const clang::ParmVarDecl* param = function_decl->getParamDecl(i);
59 std::optional<clang::tidy::lifetimes::ValueLifetimes> param_lifetimes;
60 if (lifetimes) {
61 param_lifetimes = lifetimes->GetParamLifetimes(i);
62 }
63 auto param_type =
64 ictx_.type_mapper_.ConvertQualType(param->getType(), param_lifetimes);
65 if (!param_type.ok()) {
66 add_error(absl::Substitute("Parameter #$0 is not supported: $1", i,
67 param_type.status().message()));
68 continue;
69 }
70
71 if (const clang::RecordType* record_type =
72 clang::dyn_cast<clang::RecordType>(param->getType())) {
73 if (clang::RecordDecl* record_decl =
74 clang::dyn_cast<clang::RecordDecl>(record_type->getDecl())) {
75 // TODO(b/200067242): non-trivial_abi structs, when passed by value,
76 // have a different representation which needs special support. We
77 // currently do not support it.
78 if (!record_decl->canPassInRegisters()) {
79 add_error(
80 absl::Substitute("Non-trivial_abi type '$0' is not "
81 "supported by value as parameter #$1",
82 param->getType().getAsString(), i));
83 }
84 }
85 }
86
87 std::optional<Identifier> param_name = ictx_.GetTranslatedIdentifier(param);
88 CRUBIT_CHECK(param_name.has_value()); // No known failure cases.
89 params.push_back({*param_type, *std::move(param_name)});
90 }
91
92 if (const clang::RecordType* record_return_type =
93 clang::dyn_cast<clang::RecordType>(function_decl->getReturnType())) {
94 if (clang::RecordDecl* record_decl =
95 clang::dyn_cast<clang::RecordDecl>(record_return_type->getDecl())) {
96 // TODO(b/200067242): non-trivial_abi structs, when passed by value,
97 // have a different representation which needs special support. We
98 // currently do not support it.
99 if (!record_decl->canPassInRegisters()) {
100 add_error(
101 absl::Substitute("Non-trivial_abi type '$0' is not supported "
102 "by value as a return type",
103 function_decl->getReturnType().getAsString()));
104 }
105 }
106 }
107
108 std::optional<clang::tidy::lifetimes::ValueLifetimes> return_lifetimes;
109 if (lifetimes) {
110 return_lifetimes = lifetimes->GetReturnLifetimes();
111 }
112
113 auto return_type = ictx_.type_mapper_.ConvertQualType(
114 function_decl->getReturnType(), return_lifetimes);
115 if (!return_type.ok()) {
116 add_error(absl::StrCat("Return type is not supported: ",
117 return_type.status().message()));
118 }
119
120 llvm::DenseSet<clang::tidy::lifetimes::Lifetime> all_free_lifetimes;
121 if (lifetimes) {
122 all_free_lifetimes = lifetimes->AllFreeLifetimes();
123 }
124
125 std::vector<LifetimeName> lifetime_params;
126 for (clang::tidy::lifetimes::Lifetime lifetime : all_free_lifetimes) {
127 std::optional<llvm::StringRef> name =
128 lifetime_symbol_table.LookupLifetime(lifetime);
129 CRUBIT_CHECK(name.has_value());
130 lifetime_params.push_back(
131 {.name = name->str(), .id = LifetimeId(lifetime.Id())});
132 }
133 llvm::sort(lifetime_params,
134 [](const LifetimeName& l1, const LifetimeName& l2) {
135 return l1.name < l2.name;
136 });
137
138 llvm::Optional<MemberFuncMetadata> member_func_metadata;
139 if (auto* method_decl =
140 clang::dyn_cast<clang::CXXMethodDecl>(function_decl)) {
141 switch (method_decl->getAccess()) {
142 case clang::AS_public:
143 break;
144 case clang::AS_protected:
145 case clang::AS_private:
146 case clang::AS_none:
147 // No need for IR to include Func representing private methods.
148 // TODO(lukasza): Revisit this for protected methods.
149 return std::nullopt;
150 }
151 llvm::Optional<MemberFuncMetadata::InstanceMethodMetadata>
152 instance_metadata;
153 if (method_decl->isInstance()) {
154 MemberFuncMetadata::ReferenceQualification reference;
155 switch (method_decl->getRefQualifier()) {
156 case clang::RQ_LValue:
157 reference = MemberFuncMetadata::kLValue;
158 break;
159 case clang::RQ_RValue:
160 reference = MemberFuncMetadata::kRValue;
161 break;
162 case clang::RQ_None:
163 reference = MemberFuncMetadata::kUnqualified;
164 break;
165 }
166 instance_metadata = MemberFuncMetadata::InstanceMethodMetadata{
167 .reference = reference,
168 .is_const = method_decl->isConst(),
169 .is_virtual = method_decl->isVirtual(),
170 .is_explicit_ctor = false,
171 };
172 if (auto* ctor_decl =
173 clang::dyn_cast<clang::CXXConstructorDecl>(function_decl)) {
174 instance_metadata->is_explicit_ctor = ctor_decl->isExplicit();
175 }
176 }
177
178 member_func_metadata = MemberFuncMetadata{
179 .record_id = GenerateItemId(method_decl->getParent()),
180 .instance_method_metadata = instance_metadata};
181 }
182
183 if (!errors.empty()) {
184 return ictx_.ImportUnsupportedItem(function_decl, errors);
185 }
186
187 bool has_c_calling_convention =
188 function_decl->getType()->getAs<clang::FunctionType>()->getCallConv() ==
189 clang::CC_C;
190 std::optional<UnqualifiedIdentifier> translated_name =
191 ictx_.GetTranslatedName(function_decl);
192
193 // Silence ClangTidy, checked above: calling `add_error` if
194 // `!return_type.ok()` and returning early if `!errors.empty()`.
195 CRUBIT_CHECK(return_type.ok());
196
197 if (translated_name.has_value()) {
198 return Func{
199 .name = *translated_name,
200 .owning_target = ictx_.GetOwningTarget(function_decl),
201 .doc_comment = ictx_.GetComment(function_decl),
202 .mangled_name = ictx_.GetMangledName(function_decl),
203 .return_type = *return_type,
204 .params = std::move(params),
205 .lifetime_params = std::move(lifetime_params),
206 .is_inline = function_decl->isInlined(),
207 .member_func_metadata = std::move(member_func_metadata),
208 .has_c_calling_convention = has_c_calling_convention,
209 .source_loc = ictx_.ConvertSourceLocation(function_decl->getBeginLoc()),
210 .id = GenerateItemId(function_decl),
211 };
212 }
213 return std::nullopt;
214}
215
216} // namespace crubit