blob: 4530bc95fb80e145e39299e04944734b9f209002 [file] [log] [blame]
James Dennett0492ef02014-03-14 17:44:10 +00001//===- unittests/AST/EvaluateAsRValueTest.cpp -----------------------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// \file
11// \brief Unit tests for evaluation of constant initializers.
12//
13//===----------------------------------------------------------------------===//
14
Chandler Carruth575bc3ba2015-01-14 11:23:58 +000015#include "clang/AST/ASTConsumer.h"
James Dennett0492ef02014-03-14 17:44:10 +000016#include "clang/AST/ASTConsumer.h"
17#include "clang/AST/ASTContext.h"
18#include "clang/AST/RecursiveASTVisitor.h"
19#include "clang/Tooling/Tooling.h"
20#include "gtest/gtest.h"
Chandler Carruth575bc3ba2015-01-14 11:23:58 +000021#include <map>
22#include <string>
James Dennett0492ef02014-03-14 17:44:10 +000023
24using namespace clang::tooling;
25
26namespace {
27// For each variable name encountered, whether its initializer was a
28// constant.
29typedef std::map<std::string, bool> VarInfoMap;
30
31/// \brief Records information on variable initializers to a map.
32class EvaluateConstantInitializersVisitor
33 : public clang::RecursiveASTVisitor<EvaluateConstantInitializersVisitor> {
34 public:
35 explicit EvaluateConstantInitializersVisitor(VarInfoMap &VarInfo)
36 : VarInfo(VarInfo) {}
37
38 /// \brief Checks that isConstantInitializer and EvaluateAsRValue agree
39 /// and don't crash.
40 ///
41 /// For each VarDecl with an initializer this also records in VarInfo
42 /// whether the initializer could be evaluated as a constant.
43 bool VisitVarDecl(const clang::VarDecl *VD) {
44 if (const clang::Expr *Init = VD->getInit()) {
45 clang::Expr::EvalResult Result;
46 bool WasEvaluated = Init->EvaluateAsRValue(Result, VD->getASTContext());
47 VarInfo[VD->getNameAsString()] = WasEvaluated;
48 EXPECT_EQ(WasEvaluated, Init->isConstantInitializer(VD->getASTContext(),
49 false /*ForRef*/));
50 }
51 return true;
52 }
53
54 private:
55 VarInfoMap &VarInfo;
56};
57
58class EvaluateConstantInitializersAction : public clang::ASTFrontendAction {
59 public:
David Blaikie6beb6aa2014-08-10 19:56:51 +000060 std::unique_ptr<clang::ASTConsumer>
61 CreateASTConsumer(clang::CompilerInstance &Compiler,
62 llvm::StringRef FilePath) override {
63 return llvm::make_unique<Consumer>();
James Dennett0492ef02014-03-14 17:44:10 +000064 }
65
66 private:
67 class Consumer : public clang::ASTConsumer {
68 public:
Angel Garcia Gomezb5250d32015-10-20 12:52:55 +000069 ~Consumer() override = default;
James Dennett0492ef02014-03-14 17:44:10 +000070
71 void HandleTranslationUnit(clang::ASTContext &Ctx) override {
72 VarInfoMap VarInfo;
73 EvaluateConstantInitializersVisitor Evaluator(VarInfo);
74 Evaluator.TraverseDecl(Ctx.getTranslationUnitDecl());
75 EXPECT_EQ(2u, VarInfo.size());
76 EXPECT_FALSE(VarInfo["Dependent"]);
77 EXPECT_TRUE(VarInfo["Constant"]);
78 EXPECT_EQ(2u, VarInfo.size());
79 }
80 };
81};
Alexander Kornienkoab9db512015-06-22 23:07:51 +000082}
James Dennett0492ef02014-03-14 17:44:10 +000083
84TEST(EvaluateAsRValue, FailsGracefullyForUnknownTypes) {
85 // This is a regression test; the AST library used to trigger assertion
86 // failures because it assumed that the type of initializers was always
87 // known (which is true only after template instantiation).
88 std::string ModesToTest[] = {"-std=c++03", "-std=c++11", "-std=c++1y"};
89 for (std::string const &Mode : ModesToTest) {
90 std::vector<std::string> Args(1, Mode);
NAKAMURA Takumi7ebbda82014-03-15 00:36:51 +000091 Args.push_back("-fno-delayed-template-parsing");
James Dennett0492ef02014-03-14 17:44:10 +000092 ASSERT_TRUE(runToolOnCodeWithArgs(
93 new EvaluateConstantInitializersAction(),
James Dennett622831b2014-03-14 20:08:11 +000094 "template <typename T>"
95 "struct vector {"
96 " explicit vector(int size);"
97 "};"
98 "template <typename R>"
99 "struct S {"
100 " vector<R> intervals() const {"
101 " vector<R> Dependent(2);"
102 " return Dependent;"
103 " }"
104 "};"
105 "void doSomething() {"
106 " int Constant = 2 + 2;"
107 " (void) Constant;"
108 "}",
James Dennett0492ef02014-03-14 17:44:10 +0000109 Args));
110 }
111}