blob: b5f9b32740f52bed4c01aa7ae1d40a82034ed8aa [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
15#include <map>
16#include <string>
17
18#include "clang/AST/ASTConsumer.h"
19#include "clang/AST/ASTContext.h"
20#include "clang/AST/RecursiveASTVisitor.h"
21#include "clang/Tooling/Tooling.h"
22#include "gtest/gtest.h"
23
24#include "clang/AST/ASTConsumer.h"
25
26using namespace clang::tooling;
27
28namespace {
29// For each variable name encountered, whether its initializer was a
30// constant.
31typedef std::map<std::string, bool> VarInfoMap;
32
33/// \brief Records information on variable initializers to a map.
34class EvaluateConstantInitializersVisitor
35 : public clang::RecursiveASTVisitor<EvaluateConstantInitializersVisitor> {
36 public:
37 explicit EvaluateConstantInitializersVisitor(VarInfoMap &VarInfo)
38 : VarInfo(VarInfo) {}
39
40 /// \brief Checks that isConstantInitializer and EvaluateAsRValue agree
41 /// and don't crash.
42 ///
43 /// For each VarDecl with an initializer this also records in VarInfo
44 /// whether the initializer could be evaluated as a constant.
45 bool VisitVarDecl(const clang::VarDecl *VD) {
46 if (const clang::Expr *Init = VD->getInit()) {
47 clang::Expr::EvalResult Result;
48 bool WasEvaluated = Init->EvaluateAsRValue(Result, VD->getASTContext());
49 VarInfo[VD->getNameAsString()] = WasEvaluated;
50 EXPECT_EQ(WasEvaluated, Init->isConstantInitializer(VD->getASTContext(),
51 false /*ForRef*/));
52 }
53 return true;
54 }
55
56 private:
57 VarInfoMap &VarInfo;
58};
59
60class EvaluateConstantInitializersAction : public clang::ASTFrontendAction {
61 public:
David Blaikie6beb6aa2014-08-10 19:56:51 +000062 std::unique_ptr<clang::ASTConsumer>
63 CreateASTConsumer(clang::CompilerInstance &Compiler,
64 llvm::StringRef FilePath) override {
65 return llvm::make_unique<Consumer>();
James Dennett0492ef02014-03-14 17:44:10 +000066 }
67
68 private:
69 class Consumer : public clang::ASTConsumer {
70 public:
71 ~Consumer() override {}
72
73 void HandleTranslationUnit(clang::ASTContext &Ctx) override {
74 VarInfoMap VarInfo;
75 EvaluateConstantInitializersVisitor Evaluator(VarInfo);
76 Evaluator.TraverseDecl(Ctx.getTranslationUnitDecl());
77 EXPECT_EQ(2u, VarInfo.size());
78 EXPECT_FALSE(VarInfo["Dependent"]);
79 EXPECT_TRUE(VarInfo["Constant"]);
80 EXPECT_EQ(2u, VarInfo.size());
81 }
82 };
83};
84}
85
86TEST(EvaluateAsRValue, FailsGracefullyForUnknownTypes) {
87 // This is a regression test; the AST library used to trigger assertion
88 // failures because it assumed that the type of initializers was always
89 // known (which is true only after template instantiation).
90 std::string ModesToTest[] = {"-std=c++03", "-std=c++11", "-std=c++1y"};
91 for (std::string const &Mode : ModesToTest) {
92 std::vector<std::string> Args(1, Mode);
NAKAMURA Takumi7ebbda82014-03-15 00:36:51 +000093 Args.push_back("-fno-delayed-template-parsing");
James Dennett0492ef02014-03-14 17:44:10 +000094 ASSERT_TRUE(runToolOnCodeWithArgs(
95 new EvaluateConstantInitializersAction(),
James Dennett622831b2014-03-14 20:08:11 +000096 "template <typename T>"
97 "struct vector {"
98 " explicit vector(int size);"
99 "};"
100 "template <typename R>"
101 "struct S {"
102 " vector<R> intervals() const {"
103 " vector<R> Dependent(2);"
104 " return Dependent;"
105 " }"
106 "};"
107 "void doSomething() {"
108 " int Constant = 2 + 2;"
109 " (void) Constant;"
110 "}",
James Dennett0492ef02014-03-14 17:44:10 +0000111 Args));
112 }
113}