2020-06-09 18:24:30 +08:00
|
|
|
// Copyright 2020 Google LLC
|
|
|
|
//
|
|
|
|
// 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
|
|
|
|
//
|
2022-01-28 17:38:27 +08:00
|
|
|
// https://www.apache.org/licenses/LICENSE-2.0
|
2020-06-09 18:24:30 +08:00
|
|
|
//
|
|
|
|
// 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 "sandboxed_api/tools/clang_generator/emitter.h"
|
|
|
|
|
2020-09-25 16:13:50 +08:00
|
|
|
#include <initializer_list>
|
2022-10-12 20:22:51 +08:00
|
|
|
#include <memory>
|
2020-09-25 16:13:50 +08:00
|
|
|
|
2020-06-09 18:24:30 +08:00
|
|
|
#include "gmock/gmock.h"
|
2020-06-10 20:34:35 +08:00
|
|
|
#include "gtest/gtest.h"
|
2020-10-27 00:08:06 +08:00
|
|
|
#include "absl/status/statusor.h"
|
2020-06-10 20:34:35 +08:00
|
|
|
#include "absl/strings/string_view.h"
|
2021-01-14 01:25:25 +08:00
|
|
|
#include "sandboxed_api/testing.h"
|
2020-09-25 16:13:50 +08:00
|
|
|
#include "sandboxed_api/tools/clang_generator/frontend_action_test_util.h"
|
|
|
|
#include "sandboxed_api/tools/clang_generator/generator.h"
|
2020-06-09 18:24:30 +08:00
|
|
|
#include "sandboxed_api/util/status_matchers.h"
|
|
|
|
|
|
|
|
namespace sapi {
|
|
|
|
namespace {
|
|
|
|
|
2022-03-08 23:16:17 +08:00
|
|
|
using ::testing::ElementsAre;
|
|
|
|
using ::testing::IsEmpty;
|
2020-06-09 18:24:30 +08:00
|
|
|
using ::testing::MatchesRegex;
|
2020-09-25 16:13:50 +08:00
|
|
|
using ::testing::SizeIs;
|
2020-06-09 18:24:30 +08:00
|
|
|
using ::testing::StrEq;
|
2020-06-10 20:34:35 +08:00
|
|
|
using ::testing::StrNe;
|
2020-06-09 18:24:30 +08:00
|
|
|
|
2020-09-25 16:13:50 +08:00
|
|
|
class EmitterForTesting : public Emitter {
|
|
|
|
public:
|
|
|
|
using Emitter::functions_;
|
2022-03-08 23:16:17 +08:00
|
|
|
using Emitter::rendered_types_;
|
2020-09-25 16:13:50 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
class EmitterTest : public FrontendActionTest {};
|
|
|
|
|
|
|
|
TEST_F(EmitterTest, BasicFunctionality) {
|
|
|
|
GeneratorOptions options;
|
|
|
|
options.out_file = "input.h";
|
|
|
|
options.set_function_names<std::initializer_list<std::string>>(
|
|
|
|
{"ExposedFunction"});
|
|
|
|
|
|
|
|
EmitterForTesting emitter;
|
|
|
|
RunFrontendAction(R"(extern "C" void ExposedFunction() {})",
|
2022-09-05 22:14:55 +08:00
|
|
|
std::make_unique<GeneratorAction>(emitter, options));
|
2020-09-25 16:13:50 +08:00
|
|
|
|
|
|
|
EXPECT_THAT(emitter.functions_, SizeIs(1));
|
|
|
|
|
|
|
|
absl::StatusOr<std::string> header = emitter.EmitHeader(options);
|
|
|
|
EXPECT_THAT(header, IsOk());
|
|
|
|
}
|
|
|
|
|
2022-03-08 23:16:17 +08:00
|
|
|
TEST_F(EmitterTest, RelatedTypes) {
|
|
|
|
EmitterForTesting emitter;
|
|
|
|
RunFrontendAction(
|
|
|
|
R"(
|
|
|
|
namespace std {
|
|
|
|
using size_t = unsigned long;
|
|
|
|
} // namespace std
|
|
|
|
using std::size_t;
|
|
|
|
typedef enum { kRed, kGreen, kBlue } Color;
|
|
|
|
struct Channel {
|
|
|
|
Color color;
|
|
|
|
size_t width;
|
|
|
|
size_t height;
|
|
|
|
};
|
2022-03-17 18:29:36 +08:00
|
|
|
struct ByValue { int value; };
|
2022-03-08 23:16:17 +08:00
|
|
|
extern "C" void Colorize(Channel* chan, ByValue v) {}
|
|
|
|
|
|
|
|
typedef struct { int member; } MyStruct;
|
|
|
|
extern "C" void Structize(MyStruct* s);
|
2022-03-17 18:29:36 +08:00
|
|
|
)",
|
2022-09-05 22:14:55 +08:00
|
|
|
std::make_unique<GeneratorAction>(emitter, GeneratorOptions()));
|
2022-03-08 23:16:17 +08:00
|
|
|
|
|
|
|
// Types from "std" should be skipped
|
|
|
|
EXPECT_THAT(emitter.rendered_types_["std"], IsEmpty());
|
|
|
|
|
2022-03-17 18:29:36 +08:00
|
|
|
EXPECT_THAT(UglifyAll(emitter.rendered_types_[""]),
|
2022-03-08 23:16:17 +08:00
|
|
|
ElementsAre("typedef enum { kRed, kGreen, kBlue } Color",
|
2022-03-17 18:12:46 +08:00
|
|
|
"struct Channel {"
|
|
|
|
" Color color;"
|
2022-08-03 19:55:15 +08:00
|
|
|
" size_t width;"
|
|
|
|
" size_t height; }",
|
2022-03-17 18:12:46 +08:00
|
|
|
"struct ByValue { int value; }",
|
2022-03-08 23:16:17 +08:00
|
|
|
"typedef struct { int member; } MyStruct"));
|
|
|
|
}
|
|
|
|
|
2022-03-17 18:12:46 +08:00
|
|
|
TEST_F(EmitterTest, NestedStruct) {
|
|
|
|
EmitterForTesting emitter;
|
|
|
|
RunFrontendAction(
|
|
|
|
R"(
|
|
|
|
struct A {
|
2022-03-17 18:29:36 +08:00
|
|
|
struct B { int number; };
|
2022-03-17 18:12:46 +08:00
|
|
|
B b;
|
|
|
|
int data;
|
|
|
|
};
|
|
|
|
extern "C" void Structize(A* s);
|
2022-03-17 18:29:36 +08:00
|
|
|
)",
|
2022-09-05 22:14:55 +08:00
|
|
|
std::make_unique<GeneratorAction>(emitter, GeneratorOptions()));
|
2022-03-17 18:12:46 +08:00
|
|
|
|
2022-03-17 18:29:36 +08:00
|
|
|
EXPECT_THAT(UglifyAll(emitter.rendered_types_[""]),
|
|
|
|
ElementsAre("struct A {"
|
|
|
|
" struct B { int number; };"
|
2022-08-03 19:55:15 +08:00
|
|
|
" B b;"
|
2022-03-17 18:29:36 +08:00
|
|
|
" int data; }"));
|
2022-03-17 18:12:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(EmitterTest, NestedAnonymousStruct) {
|
|
|
|
EmitterForTesting emitter;
|
|
|
|
RunFrontendAction(
|
|
|
|
R"(
|
|
|
|
struct A {
|
2022-03-17 18:29:36 +08:00
|
|
|
struct { int number; } b;
|
2022-03-17 18:12:46 +08:00
|
|
|
int data;
|
|
|
|
};
|
|
|
|
extern "C" void Structize(A* s);
|
2022-03-17 18:29:36 +08:00
|
|
|
)",
|
2022-09-05 22:14:55 +08:00
|
|
|
std::make_unique<GeneratorAction>(emitter, GeneratorOptions()));
|
2022-03-17 18:29:36 +08:00
|
|
|
|
|
|
|
EXPECT_THAT(UglifyAll(emitter.rendered_types_[""]),
|
|
|
|
ElementsAre("struct A {"
|
|
|
|
" struct { int number; } b;"
|
|
|
|
" int data; }"));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(EmitterTest, ParentNotCollected) {
|
|
|
|
EmitterForTesting emitter;
|
|
|
|
RunFrontendAction(
|
|
|
|
R"(
|
|
|
|
struct A {
|
|
|
|
struct B { int number; };
|
|
|
|
B b;
|
|
|
|
int data;
|
|
|
|
};
|
|
|
|
extern "C" void Structize(A::B* s);
|
|
|
|
)",
|
2022-09-05 22:14:55 +08:00
|
|
|
std::make_unique<GeneratorAction>(emitter, GeneratorOptions()));
|
2022-03-17 18:29:36 +08:00
|
|
|
|
|
|
|
EXPECT_THAT(UglifyAll(emitter.rendered_types_[""]),
|
|
|
|
ElementsAre("struct A {"
|
|
|
|
" struct B { int number; };"
|
2022-08-03 19:55:15 +08:00
|
|
|
" B b;"
|
2022-03-17 18:29:36 +08:00
|
|
|
" int data; }"));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(EmitterTest, RemoveQualifiers) {
|
|
|
|
EmitterForTesting emitter;
|
|
|
|
RunFrontendAction(
|
|
|
|
R"(
|
|
|
|
struct A { int data; };
|
|
|
|
extern "C" void Structize(const A* in, A* out);
|
|
|
|
)",
|
2022-09-05 22:14:55 +08:00
|
|
|
std::make_unique<GeneratorAction>(emitter, GeneratorOptions()));
|
2022-03-17 18:12:46 +08:00
|
|
|
|
2022-03-17 18:29:36 +08:00
|
|
|
EXPECT_THAT(UglifyAll(emitter.rendered_types_[""]),
|
|
|
|
ElementsAre("struct A { int data; }"));
|
2022-03-17 18:12:46 +08:00
|
|
|
}
|
|
|
|
|
2020-06-09 18:24:30 +08:00
|
|
|
TEST(IncludeGuard, CreatesRandomizedGuardForEmptyFilename) {
|
|
|
|
// Copybara will transform the string. This is intentional.
|
|
|
|
constexpr absl::string_view kGeneratedHeaderPrefix =
|
|
|
|
"SANDBOXED_API_GENERATED_HEADER_";
|
|
|
|
|
|
|
|
const std::string include_guard = GetIncludeGuard("");
|
|
|
|
EXPECT_THAT(include_guard, MatchesRegex(absl::StrCat(kGeneratedHeaderPrefix,
|
|
|
|
R"([0-9A-F]+_)")));
|
2020-06-10 20:34:35 +08:00
|
|
|
|
|
|
|
EXPECT_THAT(GetIncludeGuard(""), StrNe(include_guard));
|
2020-06-09 18:24:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(IncludeGuard, BasicFunctionality) {
|
|
|
|
EXPECT_THAT(GetIncludeGuard("boost/graph/compressed_sparse_row_graph.hpp"),
|
|
|
|
StrEq("BOOST_GRAPH_COMPRESSED_SPARSE_ROW_GRAPH_HPP_"));
|
|
|
|
|
|
|
|
// "SAPI_" prefix is there to avoid generating guards starting with "_"
|
|
|
|
EXPECT_THAT(GetIncludeGuard("/usr/include/unistd.h"),
|
|
|
|
StrEq("SAPI_USR_INCLUDE_UNISTD_H_"));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(IncludeGuard, AvoidReservedIdentifiers) {
|
|
|
|
EXPECT_THAT(GetIncludeGuard("9p.h"), StrEq("SAPI_9P_H_"));
|
|
|
|
EXPECT_THAT(GetIncludeGuard("double__under.h"), StrEq("DOUBLE_UNDER_H_"));
|
|
|
|
EXPECT_THAT(GetIncludeGuard("_single.h"), StrEq("SAPI_SINGLE_H_"));
|
|
|
|
EXPECT_THAT(GetIncludeGuard("__double.h"), StrEq("SAPI_DOUBLE_H_"));
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
} // namespace sapi
|