/*
* Copyright (C) 2015, The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <string>
#include <gtest/gtest.h>
#include "ast_cpp.h"
#include "code_writer.h"
using std::string;
using std::vector;
using std::unique_ptr;
namespace android {
namespace aidl {
namespace cpp {
namespace {
const char kExpectedHeaderOutput[] =
R"(#ifndef HEADER_INCLUDE_GUARD_H_
#define HEADER_INCLUDE_GUARD_H_
#include <string>
#include <memory>
namespace android {
namespace test {
class TestClass {
public:
void NormalMethod(int normalarg, float normal2);
virtual void SubMethod(int subarg) const;
}; // class TestClass
class TestSubClass : public TestClass {
public:
virtual void SubMethod(int subarg) const;
}; // class TestSubClass
} // namespace test
} // namespace android
#endif // HEADER_INCLUDE_GUARD_H_
)";
const char kExpectedEnumOutput[] =
R"(enum Foo {
BAR = 42,
BAZ,
};
)";
const char kExpectedSwitchOutput[] =
R"(switch (var) {
case 2:
{
baz;
}
break;
case 1:
{
foo;
bar;
}
break;
}
)";
const char kExpectedMethodImplOutput[] =
R"(return_type ClassName::MethodName(arg 1, arg 2, arg 3) const {
foo;
bar;
}
)";
} // namespace
class AstCppTests : public ::testing::Test {
protected:
void CompareGeneratedCode(const AstNode& node,
const string& expected_output) {
string actual_output;
CodeWriterPtr writer = GetStringWriter(&actual_output);
node.Write(writer.get());
EXPECT_EQ(expected_output, actual_output);
}
}; // class AstCppTests
TEST_F(AstCppTests, GeneratesHeader) {
unique_ptr<MethodDecl> norm{new MethodDecl(
"void", "NormalMethod",
ArgList{vector<string>{"int normalarg", "float normal2"}})};
unique_ptr<MethodDecl> sub{
new MethodDecl("void", "SubMethod",
ArgList{ "int subarg" },
MethodDecl::IS_CONST | MethodDecl::IS_VIRTUAL)};
unique_ptr<MethodDecl> sub2{
new MethodDecl("void", "SubMethod",
ArgList{ "int subarg" },
MethodDecl::IS_CONST | MethodDecl::IS_VIRTUAL)};
vector<unique_ptr<Declaration>> test_methods;
test_methods.push_back(std::move(norm));
test_methods.push_back(std::move(sub));
vector<unique_ptr<Declaration>> test_sub_methods;
test_sub_methods.push_back(std::move(sub2));
unique_ptr<Declaration> test{new ClassDecl { "TestClass", "",
std::move(test_methods), {} }};
unique_ptr<Declaration> test_sub{new ClassDecl { "TestSubClass",
"TestClass", std::move(test_sub_methods), {} }};
vector<unique_ptr<Declaration>> classes;
classes.push_back(std::move(test));
classes.push_back(std::move(test_sub));
unique_ptr<CppNamespace> test_ns{new CppNamespace {"test",
std::move(classes)}};
vector<unique_ptr<Declaration>> test_ns_vec;
test_ns_vec.push_back(std::move(test_ns));
unique_ptr<CppNamespace> android_ns{new CppNamespace {"android",
std::move(test_ns_vec) }};
CppHeader cpp_header{"HEADER_INCLUDE_GUARD_H_", {"string", "memory"},
std::move(android_ns) };
CompareGeneratedCode(cpp_header, kExpectedHeaderOutput);
}
TEST_F(AstCppTests, GeneratesEnum) {
Enum e("Foo");
e.AddValue("BAR", "42");
e.AddValue("BAZ", "");
CompareGeneratedCode(e, kExpectedEnumOutput);
}
TEST_F(AstCppTests, GeneratesArgList) {
ArgList simple("foo");
CompareGeneratedCode(simple, "(foo)");
ArgList compound({"foo", "bar", "baz"});
CompareGeneratedCode(compound, "(foo, bar, baz)");
std::vector<unique_ptr<AstNode>> args;
args.emplace_back(new LiteralExpression("foo()"));
ArgList nested(std::move(args));
CompareGeneratedCode(nested, "(foo())");
}
TEST_F(AstCppTests, GeneratesStatement) {
Statement s(new LiteralExpression("foo"));
CompareGeneratedCode(s, "foo;\n");
}
TEST_F(AstCppTests, GeneratesComparison) {
Comparison c(
new LiteralExpression("lhs"), "&&", new LiteralExpression("rhs"));
CompareGeneratedCode(c, "((lhs) && (rhs))");
}
TEST_F(AstCppTests, GeneratesStatementBlock) {
StatementBlock block;
block.AddStatement(unique_ptr<AstNode>(new Statement("foo")));
block.AddStatement(unique_ptr<AstNode>(new Statement("bar")));
CompareGeneratedCode(block, "{\nfoo;\nbar;\n}\n");
}
TEST_F(AstCppTests, GeneratesConstructorImpl) {
ConstructorImpl c("ClassName", ArgList({"a", "b", "c"}),
{"baz_(foo)", "bar_(blah)"});
string expected = R"(ClassName::ClassName(a, b, c)
: baz_(foo),
bar_(blah){
}
)";
CompareGeneratedCode(c, expected);
}
TEST_F(AstCppTests, GeneratesAssignment) {
Assignment simple("foo", "8");
CompareGeneratedCode(simple, "foo = 8;\n");
Assignment less_simple("foo", new MethodCall("f", "8"));
CompareGeneratedCode(less_simple, "foo = f(8);\n");
}
TEST_F(AstCppTests, GeneratesMethodCall) {
MethodCall single("single", "arg");
CompareGeneratedCode(single, "single(arg)");
MethodCall multi(
"multi",
ArgList({"has", "some", "args"}));
CompareGeneratedCode(multi, "multi(has, some, args)");
}
TEST_F(AstCppTests, GeneratesIfStatement) {
IfStatement s(new LiteralExpression("foo"));
s.OnTrue()->AddLiteral("on true1");
s.OnFalse()->AddLiteral("on false");
CompareGeneratedCode(s, "if (foo) {\non true1;\n}\nelse {\non false;\n}\n");
IfStatement s2(new LiteralExpression("bar"));
s2.OnTrue()->AddLiteral("on true1");
CompareGeneratedCode(s2, "if (bar) {\non true1;\n}\n");
}
TEST_F(AstCppTests, GeneratesSwitchStatement) {
SwitchStatement s("var");
// These are intentionally out of alphanumeric order. We're testing
// that switch respects case addition order.
auto case2 = s.AddCase("2");
case2->AddStatement(unique_ptr<AstNode>{new Statement{"baz"}});
auto case1 = s.AddCase("1");
case1->AddStatement(unique_ptr<AstNode>{new Statement{"foo"}});
case1->AddStatement(unique_ptr<AstNode>{new Statement{"bar"}});
CompareGeneratedCode(s, kExpectedSwitchOutput);
}
TEST_F(AstCppTests, GeneratesMethodImpl) {
MethodImpl m{"return_type", "ClassName", "MethodName",
ArgList{{"arg 1", "arg 2", "arg 3"}},
true};
auto b = m.GetStatementBlock();
b->AddLiteral("foo");
b->AddLiteral("bar");
CompareGeneratedCode(m, kExpectedMethodImplOutput);
}
} // namespace cpp
} // namespace aidl
} // namespace android