blob: d6341977a571e9e20532c174e0b2fbbcbaf706d9 [file] [log] [blame]
Googlerb0019ca2021-11-26 14:20:10 +00001// 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 "lifetime_annotations/lifetime_annotations.h"
6
7#include <string>
8#include <utility>
9
Luca Versaric21d92f2022-05-25 00:56:30 -070010#include "gmock/gmock.h"
11#include "gtest/gtest.h"
12#include "absl/status/status.h"
13#include "common/status_test_matchers.h"
Marcel Hlopko97a68a12022-03-09 11:38:51 +000014#include "lifetime_annotations/test/named_func_lifetimes.h"
15#include "lifetime_annotations/test/run_on_code.h"
Martin Brænne2af40a02022-06-27 01:28:16 -070016#include "lifetime_annotations/type_lifetimes.h"
Lukasz Anforowiczcec7a8a2022-04-27 10:24:51 -070017#include "clang/ASTMatchers/ASTMatchFinder.h"
18#include "clang/ASTMatchers/ASTMatchers.h"
19#include "llvm/Support/FormatVariadic.h"
Googlerb0019ca2021-11-26 14:20:10 +000020
Martin Brænne9fb786f2022-06-23 01:18:19 -070021// This file contains tests both for the "legacy" lifetime annotations
22// (`[[clang::annotate("lifetimes", ...)]]` placed on a function declaration)
23// and the newer annotations (`[[clang::annotate_type("lifetime", ...")]]`
24// placed on a type). This is because we expect we may continue to use the
25// "legacy" style of annotations in sidecar files.
26//
27// Some tests only test one style of annotation where testing the other style
28// does not make sense for the particular test.
29
Martin Brænne1a207c52022-04-19 00:05:38 -070030namespace clang {
31namespace tidy {
32namespace lifetimes {
Googlerb0019ca2021-11-26 14:20:10 +000033namespace {
34
Luca Versaric21d92f2022-05-25 00:56:30 -070035using crubit::IsOkAndHolds;
36using crubit::StatusIs;
Googlerb0019ca2021-11-26 14:20:10 +000037using testing::StartsWith;
Googlerb0019ca2021-11-26 14:20:10 +000038
Lukasz Anforowicz63f10da2022-01-26 16:59:36 +000039bool IsOverloaded(const clang::FunctionDecl* func) {
40 return !func->getDeclContext()->lookup(func->getDeclName()).isSingleResult();
41}
42
Googlerb0019ca2021-11-26 14:20:10 +000043std::string QualifiedName(const clang::FunctionDecl* func) {
44 std::string str;
45 llvm::raw_string_ostream ostream(str);
46 func->printQualifiedName(ostream);
Lukasz Anforowicz63f10da2022-01-26 16:59:36 +000047 if (IsOverloaded(func)) {
Martin Brænne2af40a02022-06-27 01:28:16 -070048 ostream << "[" << StripAttributes(func->getType()).getAsString() << "]";
Lukasz Anforowicz63f10da2022-01-26 16:59:36 +000049 }
Googlerb0019ca2021-11-26 14:20:10 +000050 ostream.flush();
51 return str;
52}
53
Martin Brænne9fb786f2022-06-23 01:18:19 -070054// Prepends definitions for lifetime annotation macros to the code.
55std::string WithLifetimeMacros(absl::string_view code) {
56 std::string result = R"(
Martin Brænne8bc2de32022-06-24 07:01:52 -070057 // TODO(mboehme): We would prefer `$(...)` to be a variadic macro that
58 // stringizes each of its macro arguments individually. This is possible but
59 // requires some contortions: https://stackoverflow.com/a/5958315
Martin Brænne9fb786f2022-06-23 01:18:19 -070060 #define $(l) [[clang::annotate_type("lifetime", #l)]]
Martin Brænne8bc2de32022-06-24 07:01:52 -070061 #define $2(l1, l2) [[clang::annotate_type("lifetime", #l1, #l2)]]
62 #define $3(l1, l2, l3) [[clang::annotate_type("lifetime", #l1, #l2, #l3)]]
Martin Brænne9fb786f2022-06-23 01:18:19 -070063 )";
64 for (char l = 'a'; l <= 'z'; ++l) {
65 absl::StrAppendFormat(&result, "#define $%c $(%c)\n", l, l);
66 }
Martin Brænneddf2e822022-06-27 00:43:56 -070067 absl::StrAppend(&result, "#define $static $(static)\n");
Martin Brænne9fb786f2022-06-23 01:18:19 -070068 absl::StrAppend(&result, code);
69 return result;
70}
71
Googlerb0019ca2021-11-26 14:20:10 +000072class LifetimeAnnotationsTest : public testing::Test {
73 protected:
74 absl::StatusOr<NamedFuncLifetimes> GetNamedLifetimeAnnotations(
75 absl::string_view code,
76 const clang::tooling::FileContentMappings& file_contents =
77 clang::tooling::FileContentMappings()) {
78 absl::StatusOr<NamedFuncLifetimes> result;
Googler656e5b22022-03-01 09:40:47 +000079 bool success = runOnCodeWithLifetimeHandlers(
Googler1cd66872021-12-10 11:40:03 +000080 llvm::StringRef(code.data(), code.size()),
Googlerb0019ca2021-11-26 14:20:10 +000081 [&result](clang::ASTContext& ast_context,
82 const LifetimeAnnotationContext& lifetime_context) {
83 using clang::ast_matchers::findAll;
84 using clang::ast_matchers::functionDecl;
85 using clang::ast_matchers::match;
86
87 NamedFuncLifetimes named_func_lifetimes;
88 for (const auto& node :
89 match(findAll(functionDecl().bind("func")), ast_context)) {
90 if (const auto* func =
91 node.getNodeAs<clang::FunctionDecl>("func")) {
Martin Brænnebd003e92022-06-27 23:22:27 -070092 // Skip various categories of function:
93 // - Template instantiation don't contain any annotations that
94 // aren't present in the template itself, but they may contain
95 // reference-like types (which will obviously be unannotated),
96 // which will generate nuisance "lifetime elision not enabled"
97 // errors.
98 // - Implicitly defaulted functions obviously cannot contain
99 // lifetime annotations. They will need to be handled through
100 // `AnalyzeDefaultedFunction()` in analyze.cc.
101 if (func->isTemplateInstantiation() ||
102 (func->isDefaulted() && !func->isExplicitlyDefaulted())) {
103 continue;
104 }
105
Googlerbea5dd12021-11-30 11:47:04 +0000106 LifetimeSymbolTable symbol_table;
Googlerb0019ca2021-11-26 14:20:10 +0000107 llvm::Expected<FunctionLifetimes> func_lifetimes =
Googlerbea5dd12021-11-30 11:47:04 +0000108 GetLifetimeAnnotations(func, lifetime_context, &symbol_table);
Googlerb0019ca2021-11-26 14:20:10 +0000109
Martin Brænne2af40a02022-06-27 01:28:16 -0700110 std::string new_entry;
111 if (func_lifetimes) {
112 new_entry = NameLifetimes(*func_lifetimes, symbol_table);
113 } else {
114 new_entry = absl::StrCat(
115 "ERROR: ", llvm::toString(func_lifetimes.takeError()));
Googlerb0019ca2021-11-26 14:20:10 +0000116 }
Martin Brænne2af40a02022-06-27 01:28:16 -0700117
Lukasz Anforowicz63f10da2022-01-26 16:59:36 +0000118 std::string func_name = QualifiedName(func);
Lukasz Anforowicz63f10da2022-01-26 16:59:36 +0000119 std::optional<llvm::StringRef> old_entry =
120 named_func_lifetimes.Get(func_name);
121 if (old_entry.has_value()) {
122 if (new_entry != old_entry.value()) {
123 result = absl::UnknownError(
124 llvm::formatv(
125 "Unexpectedly different lifetimes for function '{0}'."
126 "Old: '{1}'. New: '{2}'.",
127 func_name, old_entry.value(), new_entry)
128 .str());
129 return;
130 }
131 } else {
132 named_func_lifetimes.Add(std::move(func_name),
133 std::move(new_entry));
134 }
Googlerb0019ca2021-11-26 14:20:10 +0000135 }
136 }
137
138 result = std::move(named_func_lifetimes);
139 },
140 {}, file_contents);
141
Googler656e5b22022-03-01 09:40:47 +0000142 if (!success) {
143 return absl::UnknownError(
144 "Error extracting lifetimes. (Compilation error?)");
145 }
146
Googlerb0019ca2021-11-26 14:20:10 +0000147 return result;
148 }
149};
150
151TEST_F(LifetimeAnnotationsTest, NoLifetimes) {
152 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
153 int f(int);
154 )"),
155 IsOkAndHolds(LifetimesAre({{"f", "()"}})));
156}
157
Googler656e5b22022-03-01 09:40:47 +0000158TEST_F(LifetimeAnnotationsTest, Failure_CompileError) {
159 EXPECT_THAT(
160 GetNamedLifetimeAnnotations(R"(
161 undefined f(undefined);
162 )"),
163 StatusIs(absl::StatusCode::kUnknown,
164 StartsWith("Error extracting lifetimes. (Compilation error?)")));
165}
166
Googlerb0019ca2021-11-26 14:20:10 +0000167TEST_F(LifetimeAnnotationsTest, Failure_NoAnnotationsNoLifetimeElision) {
168 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
169 int** f(int*);
170 )"),
Martin Brænne2af40a02022-06-27 01:28:16 -0700171 IsOkAndHolds(LifetimesAre(
172 {{"f", "ERROR: Lifetime elision not enabled for 'f'"}})));
Googlerb0019ca2021-11-26 14:20:10 +0000173}
174
Googler6804a012022-01-05 07:04:36 +0000175TEST_F(LifetimeAnnotationsTest, Failure_NoOutputAnnotationNoLifetimeElision) {
176 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
177 int* f();
178 )"),
Martin Brænne2af40a02022-06-27 01:28:16 -0700179 // We specifically want to see this error message rather than
180 // "Cannot elide output lifetimes".
181 IsOkAndHolds(LifetimesAre(
182 {{"f", "ERROR: Lifetime elision not enabled for 'f'"}})));
Googler6804a012022-01-05 07:04:36 +0000183}
184
Googlerb0019ca2021-11-26 14:20:10 +0000185TEST_F(LifetimeAnnotationsTest, Failure_NoAnnotationsElisionPragmaInWrongFile) {
186 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
Googler33f768d2021-11-29 10:33:49 +0000187 #pragma clang lifetime_elision
Googlerb0019ca2021-11-26 14:20:10 +0000188 #include "header.h"
189 )",
190 {std::make_pair("header.h", R"(
191 int** f(int*);
192 )")}),
Martin Brænne2af40a02022-06-27 01:28:16 -0700193 IsOkAndHolds(LifetimesAre(
194 {{"f", "ERROR: Lifetime elision not enabled for 'f'"}})));
Googlerb0019ca2021-11-26 14:20:10 +0000195}
196
197TEST_F(LifetimeAnnotationsTest, LifetimeElision_OneInputLifetime) {
198 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
Googler33f768d2021-11-29 10:33:49 +0000199 #pragma clang lifetime_elision
Googlerb0019ca2021-11-26 14:20:10 +0000200 int** f(int*);
201 )"),
202 IsOkAndHolds(LifetimesAre({{"f", "a -> (a, a)"}})));
203}
204
205TEST_F(LifetimeAnnotationsTest, LifetimeElision_NoOutputLifetimes) {
206 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
Googler33f768d2021-11-29 10:33:49 +0000207 #pragma clang lifetime_elision
Googlerb0019ca2021-11-26 14:20:10 +0000208 void f(int**, int *);
209 )"),
210 IsOkAndHolds(LifetimesAre({{"f", "(a, b), c"}})));
211}
212
213TEST_F(LifetimeAnnotationsTest, LifetimeElision_Templates) {
214 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
Googler33f768d2021-11-29 10:33:49 +0000215 #pragma clang lifetime_elision
Googlerb0019ca2021-11-26 14:20:10 +0000216 template <class T> class vector {};
217 int* f(vector<int *>);
218 vector<int*> g(int *);
219 )"),
220 IsOkAndHolds(LifetimesAre({{"f", "a -> a"}, {"g", "a -> a"}})));
221}
222
Kinuko Yasuda74d5a3c2022-08-29 14:40:04 -0700223TEST_F(LifetimeAnnotationsTest, LifetimeElision_TemplatesWithConstant) {
224 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
225 #pragma clang lifetime_elision
226 template <class T, bool B> class vector {};
227 int* f(vector<int *, true>);
228 vector<int*, false> g(int *);
229 )"),
230 IsOkAndHolds(LifetimesAre({{"f", "a -> a"}, {"g", "a -> a"}})));
231}
232
Googlere4d4ec42022-03-28 06:52:23 -0700233TEST_F(LifetimeAnnotationsTest, LifetimeElision_NestedTemplates) {
Luca Versari6ee6b342022-04-05 05:40:52 -0700234 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
Googlerbdc55802022-03-24 11:21:37 +0000235 #pragma clang lifetime_elision
236 template <class T>
237 struct Outer {
238 template <class U>
239 struct Inner {
240 };
241 };
242 void f(Outer<int *>::Inner<int *> &);
243 Outer<int *>::Inner<int *> g(int *);
244 )"),
Luca Versari6ee6b342022-04-05 05:40:52 -0700245 IsOkAndHolds(LifetimesAre(
246 {{"f", "(<a>::<b>, c)"}, {"g", "a -> <a>::<a>"}})));
Googlerbdc55802022-03-24 11:21:37 +0000247}
248
Googlerfbb37d22022-03-02 15:13:53 +0000249TEST_F(LifetimeAnnotationsTest, LifetimeElision_LifetimeParameterizedType) {
250 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
251 #pragma clang lifetime_elision
252 struct [[clang::annotate("lifetime_params", "s")]] string_view{};
253 string_view f(string_view);
254 )"),
255 IsOkAndHolds(LifetimesAre({{"f", "a -> a"}})));
256}
257
Googlerb0019ca2021-11-26 14:20:10 +0000258TEST_F(LifetimeAnnotationsTest, LifetimeElision_Method) {
259 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
Googler33f768d2021-11-29 10:33:49 +0000260 #pragma clang lifetime_elision
Googlerb0019ca2021-11-26 14:20:10 +0000261 struct S {
262 int** method(int *, int *);
263 };
264 )"),
Googler6804a012022-01-05 07:04:36 +0000265 IsOkAndHolds(LifetimesAre({{"S::method", "a: b, c -> (a, a)"}})));
Googlerb0019ca2021-11-26 14:20:10 +0000266}
267
Devin Jeanpierre108e9c02022-06-02 07:10:09 -0700268TEST_F(LifetimeAnnotationsTest, LifetimeElision_Destructor) {
269 EXPECT_THAT(GetNamedLifetimeAnnotations(R"cc(
270 // Note: this works even without #pragma clang lifetime_elision
271 struct S {
272 ~S();
273 };
274 )cc"),
275 IsOkAndHolds(LifetimesAre({{"S::~S", "a:"}})));
276}
277
Lukasz Anforowicz63f10da2022-01-26 16:59:36 +0000278TEST_F(LifetimeAnnotationsTest, LifetimeElision_ExplicitlyDefaultedCtor) {
279 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
280 #pragma clang lifetime_elision
281 struct S {
282 S() = default;
283 };)"),
284 IsOkAndHolds(LifetimesAre({{"S::S", "a:"}})));
285}
286
Googler8a8e82e2022-02-28 14:53:56 +0000287TEST_F(LifetimeAnnotationsTest, LifetimeElision_ArrayParamLifetimes) {
288 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
289 #pragma clang lifetime_elision
290 void f(int pair[2]);
291 )"),
292 IsOkAndHolds(LifetimesAre({{"f", "a"}})));
293}
294
295TEST_F(LifetimeAnnotationsTest, LifetimeElision_ArrayParamAsTypedefLifetimes) {
296 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
297 #pragma clang lifetime_elision
Googler656e5b22022-03-01 09:40:47 +0000298 typedef int Arr[2];
Googler8a8e82e2022-02-28 14:53:56 +0000299 void f(Arr);
300 )"),
301 IsOkAndHolds(LifetimesAre({{"f", "a"}})));
302}
303
304TEST_F(LifetimeAnnotationsTest, LifetimeElision_FunctionPointerLifetimes) {
305 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
306 #pragma clang lifetime_elision
307 void f(void (*)());
308 )"),
309 IsOkAndHolds(LifetimesAre({{"f", "a"}})));
310}
311
312TEST_F(LifetimeAnnotationsTest,
313 LifetimeElision_FunctionPointerAsTypedefLifetimes) {
314 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
315 #pragma clang lifetime_elision
316 typedef void (*FunctionPointer)();
317 void f(FunctionPointer hook);
318 )"),
319 IsOkAndHolds(LifetimesAre({{"f", "a"}})));
320}
321
322TEST_F(LifetimeAnnotationsTest, LifetimeElision_FunctionReferenceLifetimes) {
323 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
324 #pragma clang lifetime_elision
325 typedef void (&FunctionReference)();
326 void f(FunctionReference hook);
327 )"),
328 IsOkAndHolds(LifetimesAre({{"f", "a"}})));
329}
330
331TEST_F(LifetimeAnnotationsTest,
332 LifetimeElision_FunctionReferenceAsTypedefLifetimes) {
333 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
334 #pragma clang lifetime_elision
335 void f(void (&)());
336 )"),
337 IsOkAndHolds(LifetimesAre({{"f", "a"}})));
338}
339
Googler2e1f0c12022-03-01 09:26:30 +0000340TEST_F(LifetimeAnnotationsTest,
341 LifetimeElision_PointerToMemberDoesNotGetLifetime) {
342 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
343 #pragma clang lifetime_elision
344 struct S {};
345 void f(int S::*ptr_to_member);
346 )"),
347 IsOkAndHolds(LifetimesAre({{"f", "()"}})));
348}
349
Googler8a8e82e2022-02-28 14:53:56 +0000350TEST_F(LifetimeAnnotationsTest, LifetimeElision_FailureTooFewInputLifetimes) {
351 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
352 #pragma clang lifetime_elision
353 int* f();
354 )"),
Martin Brænne2af40a02022-06-27 01:28:16 -0700355 IsOkAndHolds(LifetimesAre(
356 {{"f",
357 "ERROR: Cannot elide output lifetimes for 'f' because it "
358 "is a non-member function that does not have exactly one "
359 "input lifetime"}})));
Googler8a8e82e2022-02-28 14:53:56 +0000360}
361
362TEST_F(LifetimeAnnotationsTest, LifetimeElision_FailureTooManyInputLifetimes) {
363 EXPECT_THAT(GetNamedLifetimeAnnotations(R"(
364 #pragma clang lifetime_elision
365 int* f(int**);
366 )"),
Martin Brænne2af40a02022-06-27 01:28:16 -0700367 IsOkAndHolds(LifetimesAre(
368 {{"f",
369 "ERROR: Cannot elide output lifetimes for 'f' because it "
370 "is a non-member function that does not have exactly one "
371 "input lifetime"}})));
Googler8a8e82e2022-02-28 14:53:56 +0000372}
373
374TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_NoLifetimes) {
375 EXPECT_THAT(GetNamedLifetimeAnnotations(R"_(
376 [[clang::annotate("lifetimes", "()")]]
377 void f(int);
378 )_"),
379 IsOkAndHolds(LifetimesAre({{"f", "()"}})));
380}
381
Martin Brænne9fb786f2022-06-23 01:18:19 -0700382TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_BadAttributeArgument) {
383 EXPECT_THAT(
384 GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
385 void f(int* [[clang::annotate_type("lifetime", 1)]]);
386 )")),
Martin Brænne2af40a02022-06-27 01:28:16 -0700387 IsOkAndHolds(LifetimesAre(
388 {{"f", "ERROR: cannot evaluate argument as a string literal"}})));
Martin Brænne9fb786f2022-06-23 01:18:19 -0700389}
390
Googler8a8e82e2022-02-28 14:53:56 +0000391TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_Simple) {
Martin Brænne9fb786f2022-06-23 01:18:19 -0700392 EXPECT_THAT(GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
Googler8a8e82e2022-02-28 14:53:56 +0000393 [[clang::annotate("lifetimes", "a -> a")]]
Martin Brænne9fb786f2022-06-23 01:18:19 -0700394 int* f1(int*);
395 int* $a f2(int* $a);
396 )")),
397 IsOkAndHolds(LifetimesAre({{"f1", "a -> a"}, {"f2", "a -> a"}})));
398}
399
400TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_SimpleRef) {
401 EXPECT_THAT(GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
402 [[clang::annotate("lifetimes", "a -> a")]]
403 int& f1(int&);
404 int& $a f2(int& $a);
405 )")),
406 IsOkAndHolds(LifetimesAre({{"f1", "a -> a"}, {"f2", "a -> a"}})));
407}
408
409TEST_F(LifetimeAnnotationsTest,
410 LifetimeAnnotation_Invalid_MultipleLifetimesOnPointer) {
411 EXPECT_THAT(
412 GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
Martin Brænne8bc2de32022-06-24 07:01:52 -0700413 void f(int* $2(a, b));
Martin Brænne9fb786f2022-06-23 01:18:19 -0700414 )")),
Martin Brænne2af40a02022-06-27 01:28:16 -0700415 IsOkAndHolds(LifetimesAre(
416 {{"f", "ERROR: Expected a single lifetime but 2 were given"}})));
Martin Brænne9fb786f2022-06-23 01:18:19 -0700417}
418
419TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_Static) {
420 EXPECT_THAT(GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
421 [[clang::annotate("lifetimes", "static -> static")]]
422 int* f1(int*);
423 int* $static f2(int* $static);
424 )")),
425 IsOkAndHolds(LifetimesAre(
426 {{"f1", "static -> static"}, {"f2", "static -> static"}})));
427}
428
429TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_PartialElision) {
430 EXPECT_THAT(GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
431 #pragma clang lifetime_elision
432 int* $a f(int* $a, int*, int* $a);
433 )")),
434 IsOkAndHolds(LifetimesAre({{"f", "a, b, a -> a"}})));
Googler8a8e82e2022-02-28 14:53:56 +0000435}
436
437TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_MultiplePtr) {
Martin Brænne9fb786f2022-06-23 01:18:19 -0700438 EXPECT_THAT(GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
Googler8a8e82e2022-02-28 14:53:56 +0000439 [[clang::annotate("lifetimes", "(a, b) -> a")]]
Martin Brænne9fb786f2022-06-23 01:18:19 -0700440 int* f1(int**);
441 int* $a f2(int* $a * $b);
442 )")),
443 IsOkAndHolds(LifetimesAre(
444 {{"f1", "(a, b) -> a"}, {"f2", "(a, b) -> a"}})));
Googler8a8e82e2022-02-28 14:53:56 +0000445}
446
447TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_MultipleArguments) {
Martin Brænne9fb786f2022-06-23 01:18:19 -0700448 EXPECT_THAT(
449 GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
Googler8a8e82e2022-02-28 14:53:56 +0000450 [[clang::annotate("lifetimes", "a, b -> a")]]
Martin Brænne9fb786f2022-06-23 01:18:19 -0700451 int* f1(int*, int*);
452 int* $a f2(int* $a, int* $b);
453 )")),
454 IsOkAndHolds(LifetimesAre({{"f1", "a, b -> a"}, {"f2", "a, b -> a"}})));
Googler8a8e82e2022-02-28 14:53:56 +0000455}
456
457TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_NoReturn) {
Martin Brænne9fb786f2022-06-23 01:18:19 -0700458 EXPECT_THAT(GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
Googler8a8e82e2022-02-28 14:53:56 +0000459 [[clang::annotate("lifetimes", "a, b")]]
Martin Brænne9fb786f2022-06-23 01:18:19 -0700460 void f1(int*, int*);
461 void f2(int* $a, int* $b);
462 )")),
463 IsOkAndHolds(LifetimesAre({{"f1", "a, b"}, {"f2", "a, b"}})));
Googler8a8e82e2022-02-28 14:53:56 +0000464}
465
Martin Brænne9fb786f2022-06-23 01:18:19 -0700466TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_ParamWithoutLifetime) {
467 EXPECT_THAT(GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
Googler8a8e82e2022-02-28 14:53:56 +0000468 [[clang::annotate("lifetimes", "a, (), a -> a")]]
Martin Brænne9fb786f2022-06-23 01:18:19 -0700469 int* f1(int*, int, int*);
470 int* $a f2(int* $a, int, int* $a);
471 )")),
472 IsOkAndHolds(LifetimesAre(
473 {{"f1", "a, (), a -> a"}, {"f2", "a, (), a -> a"}})));
474}
475
476TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_LifetimeParameterizedType) {
477 // Use a custom delimiter so that the `")` in the `clang::annotate` attribute
478 // below doesn't prematurely terminate the string.
479 EXPECT_THAT(GetNamedLifetimeAnnotations(WithLifetimeMacros(R"code(
480 struct [[clang::annotate("lifetime_params", "a", "b")]] S_param {};
481
482 [[clang::annotate("lifetimes", "([a, b]) -> ([a, b])")]]
483 S_param f1(S_param s);
484
Martin Brænne8bc2de32022-06-24 07:01:52 -0700485 S_param $2(a, b) f2(S_param $2(a, b) s);
Martin Brænne9fb786f2022-06-23 01:18:19 -0700486 )code")),
487 IsOkAndHolds(LifetimesAre({{"f1", "([a, b]) -> ([a, b])"},
488 {"f2", "([a, b]) -> ([a, b])"}})));
489}
490
Martin Brænne8bc2de32022-06-24 07:01:52 -0700491TEST_F(
492 LifetimeAnnotationsTest,
493 LifetimeAnnotation_LifetimeParameterizedType_Invalid_WrongNumberOfLifetimes) {
Martin Brænne2af40a02022-06-27 01:28:16 -0700494 EXPECT_THAT(
495 GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
Martin Brænne9fb786f2022-06-23 01:18:19 -0700496 struct [[clang::annotate("lifetime_params", "a", "b")]] S_param {};
497
Martin Brænne8bc2de32022-06-24 07:01:52 -0700498 void f(S_param $3(a, b, c) s);
Martin Brænne9fb786f2022-06-23 01:18:19 -0700499 )")),
Martin Brænne2af40a02022-06-27 01:28:16 -0700500 IsOkAndHolds(LifetimesAre({{"f",
501 "ERROR: Type has 2 lifetime parameters but 3 "
502 "lifetime arguments were given"}})));
Martin Brænne9fb786f2022-06-23 01:18:19 -0700503}
504
Martin Brænne8bc2de32022-06-24 07:01:52 -0700505TEST_F(
506 LifetimeAnnotationsTest,
507 LifetimeAnnotation_LifetimeParameterizedType_Invalid_MultipleAnnotateAttributes) {
Martin Brænne2af40a02022-06-27 01:28:16 -0700508 EXPECT_THAT(GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
Martin Brænne8bc2de32022-06-24 07:01:52 -0700509 struct [[clang::annotate("lifetime_params", "a", "b")]] S_param {};
510
511 void f(S_param $a $b s);
512 )")),
Martin Brænne2af40a02022-06-27 01:28:16 -0700513 IsOkAndHolds(LifetimesAre(
514 {{"f",
515 "ERROR: Only one `[[annotate_type(\"lifetime\", ...)]]` "
516 "attribute may be placed on a type"}})));
Martin Brænne8bc2de32022-06-24 07:01:52 -0700517}
518
Martin Brænne9fb786f2022-06-23 01:18:19 -0700519TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_Template) {
520 EXPECT_THAT(GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
521 template <class T> class vector {};
522
523 [[clang::annotate("lifetimes", "(a, b) -> a")]]
524 int* f1(const vector<int *> &);
525 int* $a f2(const vector<int * $a> & $b);
526 )")),
527 IsOkAndHolds(LifetimesAre(
528 {{"f1", "(a, b) -> a"}, {"f2", "(a, b) -> a"}})));
529}
530
531TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_VariadicTemplate) {
532 EXPECT_THAT(GetNamedLifetimeAnnotations(WithLifetimeMacros(R"code(
533 template <class... T> class variadic{};
534
535 [[clang::annotate("lifetimes", "(<a, b>, c)")]]
536 void f1(const variadic<int *, int *> &);
537 void f2(const variadic<int * $a, int * $b> & $c);
538 )code")),
539 IsOkAndHolds(LifetimesAre(
540 {{"f1", "(<a, b>, c)"}, {"f2", "(<a, b>, c)"}})));
Googler8a8e82e2022-02-28 14:53:56 +0000541}
542
Martin Brænne33f4f502022-06-27 01:15:15 -0700543TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_VariadicTemplateWithCtor) {
Martin Brænne33f4f502022-06-27 01:15:15 -0700544 EXPECT_THAT(GetNamedLifetimeAnnotations(WithLifetimeMacros(R"code(
545 template <typename... Args> struct S { S() $a {} };
546 template <typename T, typename... Args>
547 struct S<T, Args...> {
548 S(T t, Args... args) $a {}
549 };
550
Martin Brænnebd003e92022-06-27 23:22:27 -0700551 void target(int* $a a, int* $b b) {
Martin Brænne33f4f502022-06-27 01:15:15 -0700552 S<int*, int*> s = {a, b};
553 }
554 )code")),
Martin Brænnebd003e92022-06-27 23:22:27 -0700555 IsOkAndHolds(LifetimesAre({{"S::S<Args...>", "a:"},
556 {"S<type-parameter-0-0, "
557 "type-parameter-0-1...>::"
558 "S<type-parameter-0-0, "
559 "type-parameter-0-1...>",
560 "a: (), ()"},
561 {"target", "a, b"}})));
Martin Brænne33f4f502022-06-27 01:15:15 -0700562}
563
Googler8a8e82e2022-02-28 14:53:56 +0000564TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_Method) {
Martin Brænne9fb786f2022-06-23 01:18:19 -0700565 EXPECT_THAT(
566 GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
Googler8a8e82e2022-02-28 14:53:56 +0000567 struct S {
568 [[clang::annotate("lifetimes", "a: -> a")]]
Martin Brænne9fb786f2022-06-23 01:18:19 -0700569 int* f1();
570 int* $a f2() $a;
Googler8a8e82e2022-02-28 14:53:56 +0000571 };
Martin Brænne9fb786f2022-06-23 01:18:19 -0700572 )")),
573 IsOkAndHolds(LifetimesAre({{"S::f1", "a: -> a"}, {"S::f2", "a: -> a"}})));
Googler8a8e82e2022-02-28 14:53:56 +0000574}
575
576TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_MethodWithParam) {
Martin Brænne9fb786f2022-06-23 01:18:19 -0700577 EXPECT_THAT(GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
Googler8a8e82e2022-02-28 14:53:56 +0000578 struct S {
579 [[clang::annotate("lifetimes", "a: b -> a")]]
Martin Brænne9fb786f2022-06-23 01:18:19 -0700580 int* f1(int*);
581 int* $a f2(int* $b) $a;
Googler8a8e82e2022-02-28 14:53:56 +0000582 };
Martin Brænne9fb786f2022-06-23 01:18:19 -0700583 )")),
584 IsOkAndHolds(LifetimesAre(
585 {{"S::f1", "a: b -> a"}, {"S::f2", "a: b -> a"}})));
Googler8a8e82e2022-02-28 14:53:56 +0000586}
587
Luca Versari95ce68f2022-03-24 14:44:19 +0000588TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_MethodWithLifetimeParams) {
Martin Brænne9fb786f2022-06-23 01:18:19 -0700589 EXPECT_THAT(GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
Luca Versari95ce68f2022-03-24 14:44:19 +0000590 struct [[clang::annotate("lifetime_params", "x", "y")]] S {
Luca Versari6ee6b342022-04-05 05:40:52 -0700591 [[clang::annotate("lifetimes", "([x, y], a): -> x")]]
Martin Brænne9fb786f2022-06-23 01:18:19 -0700592 int* f1();
593 // It's implied that the lifetime parameters of `this` are $x and $y
594 // because this is a member function on struct with those lifetime
595 // parameters.
596 // TODO(mboehme): This doesn't work yet. We need some special handling
597 // to know that in this context, the type `S` doesn't need lifetimes
598 // put on it.
599 // TODO(mboehme): How do we resolve this difference relative to the
600 // "legacy" lifetime annotations? Does this mean that they should also
601 // not include the lifetimes x and y?
602 // int* $x f2() $a;
Luca Versari95ce68f2022-03-24 14:44:19 +0000603 };
Martin Brænne9fb786f2022-06-23 01:18:19 -0700604 )")),
605 IsOkAndHolds(LifetimesAre({{"S::f1", "([x, y], a): -> x"}})));
Luca Versari95ce68f2022-03-24 14:44:19 +0000606}
607
Googler8a8e82e2022-02-28 14:53:56 +0000608TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_Invalid_MissingThis) {
Martin Brænne2af40a02022-06-27 01:28:16 -0700609 EXPECT_THAT(
610 GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
Googler8a8e82e2022-02-28 14:53:56 +0000611 struct S {
612 [[clang::annotate("lifetimes", "-> a")]]
613 int* f();
614 };
Martin Brænne9fb786f2022-06-23 01:18:19 -0700615 )")),
Martin Brænne2af40a02022-06-27 01:28:16 -0700616 IsOkAndHolds(LifetimesAre(
617 {{"S::f",
618 "ERROR: Invalid lifetime annotation: too few lifetimes"}})));
Martin Brænne9fb786f2022-06-23 01:18:19 -0700619 EXPECT_THAT(GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
620 struct S {
621 int* $a f();
622 };
623 )")),
Martin Brænne2af40a02022-06-27 01:28:16 -0700624 IsOkAndHolds(LifetimesAre(
625 {{"S::f", "ERROR: Lifetime elision not enabled for 'f'"}})));
Googler8a8e82e2022-02-28 14:53:56 +0000626}
627
628TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_Invalid_ThisOnFreeFunction) {
Martin Brænne2af40a02022-06-27 01:28:16 -0700629 EXPECT_THAT(
630 GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
Googler8a8e82e2022-02-28 14:53:56 +0000631 [[clang::annotate("lifetimes", "a: a -> a")]]
632 int* f(int*);
Martin Brænne9fb786f2022-06-23 01:18:19 -0700633 )")),
Martin Brænne2af40a02022-06-27 01:28:16 -0700634 IsOkAndHolds(LifetimesAre(
635 {{"f", "ERROR: Invalid lifetime annotation: too many lifetimes"}})));
636 EXPECT_THAT(
637 GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
Martin Brænne9fb786f2022-06-23 01:18:19 -0700638 int* $a f(int* $a) $a;
639 )")),
Martin Brænne2af40a02022-06-27 01:28:16 -0700640 IsOkAndHolds(LifetimesAre({{"f",
641 "ERROR: Encountered a `this` lifetime on a "
642 "function with no `this` parameter"}})));
Googler8a8e82e2022-02-28 14:53:56 +0000643}
644
645TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_Invalid_WrongNumber) {
Martin Brænne2af40a02022-06-27 01:28:16 -0700646 EXPECT_THAT(
647 GetNamedLifetimeAnnotations(R"(
Googler8a8e82e2022-02-28 14:53:56 +0000648 [[clang::annotate("lifetimes", "a -> a")]]
649 int* f(int**);
650 )"),
Martin Brænne2af40a02022-06-27 01:28:16 -0700651 IsOkAndHolds(LifetimesAre(
652 {{"f", "ERROR: Invalid lifetime annotation: too few lifetimes"}})));
Googler8a8e82e2022-02-28 14:53:56 +0000653}
654
Luca Versaricc7e6cb2022-04-05 08:08:23 -0700655TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_Callback) {
656 EXPECT_THAT(
Martin Brænne9fb786f2022-06-23 01:18:19 -0700657 GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
Luca Versaricc7e6cb2022-04-05 08:08:23 -0700658 [[clang::annotate("lifetimes", "b, ((a -> a), static) -> b")]]
Martin Brænne9fb786f2022-06-23 01:18:19 -0700659 int* f1(int*, int* (*)(int*));
660 int* $b f2(int* $b, int* $a (* $static)(int* $a));
661 )")),
662 IsOkAndHolds(LifetimesAre({{"f1", "b, ((a -> a), static) -> b"},
663 {"f2", "b, ((a -> a), static) -> b"}})));
Luca Versaricc7e6cb2022-04-05 08:08:23 -0700664}
665
666TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_CallbackMultipleParams) {
667 EXPECT_THAT(
Martin Brænne9fb786f2022-06-23 01:18:19 -0700668 GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
Luca Versaricc7e6cb2022-04-05 08:08:23 -0700669 [[clang::annotate("lifetimes", "c, ((a, b -> a), static) -> c")]]
Martin Brænne9fb786f2022-06-23 01:18:19 -0700670 int* f1(int*, int* (*)(int*, int*));
671 int* $c f2(int* $c, int* $a (* $static)(int* $a, int* $b));
672 )")),
673 IsOkAndHolds(LifetimesAre({{"f1", "c, ((a, b -> a), static) -> c"},
674 {"f2", "c, ((a, b -> a), static) -> c"}})));
Luca Versaricc7e6cb2022-04-05 08:08:23 -0700675}
676
677TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_CallbackTmplFunc) {
Martin Brænne9fb786f2022-06-23 01:18:19 -0700678 EXPECT_THAT(GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
Luca Versaricc7e6cb2022-04-05 08:08:23 -0700679 template <typename Func>
680 struct function;
681 [[clang::annotate("lifetimes", "a, ((b -> b)) -> a")]]
Martin Brænne9fb786f2022-06-23 01:18:19 -0700682 int* f1(int*, function<int*(int*)>);
683 int* $a f2(int* $a, function<int* $b(int* $b)>);
684 )")),
685 IsOkAndHolds(LifetimesAre({{"f1", "a, ((b -> b)) -> a"},
686 {"f2", "a, ((b -> b)) -> a"}})));
Luca Versaricc7e6cb2022-04-05 08:08:23 -0700687}
688
689TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_MultipleCallbacks) {
Martin Brænne9fb786f2022-06-23 01:18:19 -0700690 EXPECT_THAT(GetNamedLifetimeAnnotations(WithLifetimeMacros(R"(
Luca Versaricc7e6cb2022-04-05 08:08:23 -0700691 [[clang::annotate("lifetimes", "a, ((b -> b), static), ((c -> c), static) -> a")]]
Martin Brænne9fb786f2022-06-23 01:18:19 -0700692 int* f1(int*, int* (*)(int*), int* (*)(int*));
693 int* $a f2(int* $a, int* $b (* $static)(int* $b), int* $c (* $static)(int* $c));
694 )")),
Luca Versaricc7e6cb2022-04-05 08:08:23 -0700695 IsOkAndHolds(LifetimesAre(
Martin Brænne9fb786f2022-06-23 01:18:19 -0700696 {{"f1", "a, ((b -> b), static), ((c -> c), static) -> a"},
697 {"f2", "a, ((b -> b), static), ((c -> c), static) -> a"}})));
Luca Versaricc7e6cb2022-04-05 08:08:23 -0700698}
699
700TEST_F(LifetimeAnnotationsTest, LifetimeAnnotation_ReturnFunctionPtr) {
Martin Brænne9fb786f2022-06-23 01:18:19 -0700701 EXPECT_THAT(GetNamedLifetimeAnnotations(WithLifetimeMacros(R"_(
Luca Versaricc7e6cb2022-04-05 08:08:23 -0700702 typedef int* (*FP)(int*);
703 [[clang::annotate("lifetimes", "a -> ((b -> b), static)")]]
704 FP f(int*);
Martin Brænne9fb786f2022-06-23 01:18:19 -0700705 // TODO(mboehme): Need to support lifetime parameters on type aliases to
706 // be able to express this in the new syntax.
707 )_")),
Luca Versaricc7e6cb2022-04-05 08:08:23 -0700708 IsOkAndHolds(LifetimesAre({{"f", "a -> ((b -> b), static)"}})));
709}
710
Googlerb0019ca2021-11-26 14:20:10 +0000711} // namespace
Martin Brænne1a207c52022-04-19 00:05:38 -0700712} // namespace lifetimes
713} // namespace tidy
714} // namespace clang