/*
* Copyright (C) 2018 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 "src/perfetto_cmd/pbtxt_to_pb.h"
#include <memory>
#include <string>
#include "gmock/gmock.h"
#include "gtest/gtest.h"
#include "google/protobuf/io/zero_copy_stream_impl_lite.h"
#include "perfetto/config/trace_config.pb.h"
namespace perfetto {
namespace {
using ::testing::StrictMock;
using ::testing::Contains;
using ::testing::ElementsAre;
using ::google::protobuf::io::ZeroCopyInputStream;
using ::google::protobuf::io::ArrayInputStream;
class MockErrorReporter : public ErrorReporter {
public:
MockErrorReporter() {}
~MockErrorReporter() = default;
MOCK_METHOD4(AddError,
void(size_t line,
size_t column_start,
size_t column_end,
const std::string& message));
};
protos::TraceConfig ToProto(const std::string& input) {
StrictMock<MockErrorReporter> reporter;
std::vector<uint8_t> output = PbtxtToPb(input, &reporter);
EXPECT_FALSE(output.empty());
protos::TraceConfig config;
config.ParseFromArray(output.data(), static_cast<int>(output.size()));
return config;
}
void ToErrors(const std::string& input, MockErrorReporter* reporter) {
std::vector<uint8_t> output = PbtxtToPb(input, reporter);
}
TEST(PbtxtToPb, OneField) {
protos::TraceConfig config = ToProto(R"(
duration_ms: 1234
)");
EXPECT_EQ(config.duration_ms(), 1234);
}
TEST(PbtxtToPb, TwoFields) {
protos::TraceConfig config = ToProto(R"(
duration_ms: 1234
file_write_period_ms: 5678
)");
EXPECT_EQ(config.duration_ms(), 1234);
EXPECT_EQ(config.file_write_period_ms(), 5678);
}
TEST(PbtxtToPb, Semicolons) {
protos::TraceConfig config = ToProto(R"(
duration_ms: 1234;
file_write_period_ms: 5678;
)");
EXPECT_EQ(config.duration_ms(), 1234);
EXPECT_EQ(config.file_write_period_ms(), 5678);
}
TEST(PbtxtToPb, NestedMessage) {
protos::TraceConfig config = ToProto(R"(
buffers: {
size_kb: 123
}
)");
ASSERT_EQ(config.buffers().size(), 1);
EXPECT_EQ(config.buffers().Get(0).size_kb(), 123);
}
TEST(PbtxtToPb, SplitNested) {
protos::TraceConfig config = ToProto(R"(
buffers: {
size_kb: 1
}
duration_ms: 1000;
buffers: {
size_kb: 2
}
)");
ASSERT_EQ(config.buffers().size(), 2);
EXPECT_EQ(config.buffers().Get(0).size_kb(), 1);
EXPECT_EQ(config.buffers().Get(1).size_kb(), 2);
EXPECT_EQ(config.duration_ms(), 1000);
}
TEST(PbtxtToPb, MultipleNestedMessage) {
protos::TraceConfig config = ToProto(R"(
buffers: {
size_kb: 1
}
buffers: {
size_kb: 2
}
)");
ASSERT_EQ(config.buffers().size(), 2);
EXPECT_EQ(config.buffers().Get(0).size_kb(), 1);
EXPECT_EQ(config.buffers().Get(1).size_kb(), 2);
}
TEST(PbtxtToPb, NestedMessageCrossFile) {
protos::TraceConfig config = ToProto(R"(
data_sources {
config {
ftrace_config {
drain_period_ms: 42
}
}
}
)");
ASSERT_EQ(
config.data_sources().Get(0).config().ftrace_config().drain_period_ms(),
42);
}
TEST(PbtxtToPb, Booleans) {
protos::TraceConfig config = ToProto(R"(
write_into_file: false; deferred_start: true;
)");
EXPECT_EQ(config.write_into_file(), false);
EXPECT_EQ(config.deferred_start(), true);
}
TEST(PbtxtToPb, Comments) {
protos::TraceConfig config = ToProto(R"(
write_into_file: false # deferred_start: true;
buffers# 1
# 2
:# 3
# 4
{# 5
# 6
fill_policy# 7
# 8
:# 9
# 10
RING_BUFFER# 11
# 12
;# 13
# 14
} # 15
# 16
)");
EXPECT_EQ(config.write_into_file(), false);
EXPECT_EQ(config.deferred_start(), false);
}
TEST(PbtxtToPb, Enums) {
protos::TraceConfig config = ToProto(R"(
buffers: {
fill_policy: RING_BUFFER
}
)");
EXPECT_EQ(config.buffers().Get(0).fill_policy(),
protos::TraceConfig::BufferConfig::RING_BUFFER);
}
TEST(PbtxtToPb, AllFieldTypes) {
protos::TraceConfig config = ToProto(R"(
data_sources {
config {
for_testing {
dummy_fields {
field_uint32: 1;
field_uint64: 2;
field_int32: 3;
field_int64: 4;
field_fixed64: 5;
field_sfixed64: 6;
field_fixed32: 7;
field_sfixed32: 8;
field_double: 9;
field_float: 10;
field_sint64: 11;
field_sint32: 12;
field_string: "13";
field_bytes: "14";
}
}
}
}
)");
const auto& fields =
config.data_sources().Get(0).config().for_testing().dummy_fields();
ASSERT_EQ(fields.field_uint32(), 1);
ASSERT_EQ(fields.field_uint64(), 2);
ASSERT_EQ(fields.field_int32(), 3);
ASSERT_EQ(fields.field_int64(), 4);
ASSERT_EQ(fields.field_fixed64(), 5);
ASSERT_EQ(fields.field_sfixed64(), 6);
ASSERT_EQ(fields.field_fixed32(), 7);
ASSERT_EQ(fields.field_sfixed32(), 8);
ASSERT_EQ(fields.field_double(), 9);
ASSERT_EQ(fields.field_float(), 10);
ASSERT_EQ(fields.field_sint64(), 11);
ASSERT_EQ(fields.field_sint32(), 12);
ASSERT_EQ(fields.field_string(), "13");
ASSERT_EQ(fields.field_bytes(), "14");
}
TEST(PbtxtToPb, NegativeNumbers) {
protos::TraceConfig config = ToProto(R"(
data_sources {
config {
for_testing {
dummy_fields {
field_int32: -1;
field_int64: -2;
field_fixed64: -3;
field_sfixed64: -4;
field_fixed32: -5;
field_sfixed32: -6;
field_double: -7;
field_float: -8;
field_sint64: -9;
field_sint32: -10;
}
}
}
}
)");
const auto& fields =
config.data_sources().Get(0).config().for_testing().dummy_fields();
ASSERT_EQ(fields.field_int32(), -1);
ASSERT_EQ(fields.field_int64(), -2);
ASSERT_EQ(fields.field_fixed64(), -3);
ASSERT_EQ(fields.field_sfixed64(), -4);
ASSERT_EQ(fields.field_fixed32(), -5);
ASSERT_EQ(fields.field_sfixed32(), -6);
ASSERT_EQ(fields.field_double(), -7);
ASSERT_EQ(fields.field_float(), -8);
ASSERT_EQ(fields.field_sint64(), -9);
ASSERT_EQ(fields.field_sint32(), -10);
}
TEST(PbtxtToPb, EofEndsNumeric) {
protos::TraceConfig config = ToProto(R"(duration_ms: 1234)");
EXPECT_EQ(config.duration_ms(), 1234);
}
TEST(PbtxtToPb, EofEndsIdentifier) {
protos::TraceConfig config = ToProto(R"(enable_extra_guardrails: true)");
EXPECT_EQ(config.enable_extra_guardrails(), true);
}
TEST(PbtxtToPb, ExampleConfig) {
protos::TraceConfig config = ToProto(R"(
buffers {
size_kb: 100024
fill_policy: RING_BUFFER
}
data_sources {
config {
name: "linux.ftrace"
target_buffer: 0
ftrace_config {
buffer_size_kb: 512 # 4 (page size) * 128
drain_period_ms: 200
ftrace_events: "binder_lock"
ftrace_events: "binder_locked"
atrace_categories: "gfx"
}
}
}
data_sources {
config {
name: "linux.process_stats"
target_buffer: 0
}
}
data_sources {
config {
name: "linux.inode_file_map"
target_buffer: 0
inode_file_config {
scan_delay_ms: 1000
scan_interval_ms: 1000
scan_batch_size: 500
mount_point_mapping: {
mountpoint: "/data"
scan_roots: "/data/app"
}
}
}
}
producers {
producer_name: "perfetto.traced_probes"
shm_size_kb: 4096
page_size_kb: 4
}
duration_ms: 10000
)");
EXPECT_EQ(config.duration_ms(), 10000);
EXPECT_EQ(config.buffers().Get(0).size_kb(), 100024);
EXPECT_EQ(config.data_sources().Get(0).config().name(), "linux.ftrace");
EXPECT_EQ(config.data_sources().Get(0).config().target_buffer(), 0);
EXPECT_EQ(config.producers().Get(0).producer_name(),
"perfetto.traced_probes");
}
TEST(PbtxtToPb, Strings) {
protos::TraceConfig config = ToProto(R"(
data_sources {
config {
ftrace_config {
ftrace_events: "binder_lock"
ftrace_events: "foo/bar"
ftrace_events: "foo\\bar"
ftrace_events: "newline\nnewline"
ftrace_events: "\"quoted\""
ftrace_events: "\a\b\f\n\r\t\v\\\'\"\?"
}
}
}
)");
auto events =
config.data_sources().Get(0).config().ftrace_config().ftrace_events();
EXPECT_THAT(events, Contains("binder_lock"));
EXPECT_THAT(events, Contains("foo/bar"));
EXPECT_THAT(events, Contains("foo\\bar"));
EXPECT_THAT(events, Contains("newline\nnewline"));
EXPECT_THAT(events, Contains("\"quoted\""));
EXPECT_THAT(events, Contains("\a\b\f\n\r\t\v\\\'\"\?"));
}
TEST(PbtxtToPb, UnknownField) {
MockErrorReporter reporter;
EXPECT_CALL(reporter,
AddError(2, 5, 11,
"No field named \"not_a_label\" in proto TraceConfig"));
ToErrors(R"(
not_a_label: false
)",
&reporter);
}
TEST(PbtxtToPb, UnknownNestedField) {
MockErrorReporter reporter;
EXPECT_CALL(
reporter,
AddError(
4, 5, 16,
"No field named \"not_a_field_name\" in proto DataSourceConfig"));
ToErrors(R"(
data_sources {
config {
not_a_field_name {
}
}
}
)",
&reporter);
}
TEST(PbtxtToPb, BadBoolean) {
MockErrorReporter reporter;
EXPECT_CALL(reporter, AddError(2, 22, 3,
"Expected 'true' or 'false' for boolean field "
"write_into_file in proto TraceConfig instead "
"saw 'foo'"));
ToErrors(R"(
write_into_file: foo;
)",
&reporter);
}
TEST(PbtxtToPb, MissingBoolean) {
MockErrorReporter reporter;
EXPECT_CALL(reporter, AddError(3, 3, 0, "Unexpected end of input"));
ToErrors(R"(
write_into_file:
)",
&reporter);
}
TEST(PbtxtToPb, RootProtoMustNotEndWithBrace) {
MockErrorReporter reporter;
EXPECT_CALL(reporter, AddError(2, 5, 0, "Unmatched closing brace"));
ToErrors(R"(
}
)",
&reporter);
}
TEST(PbtxtToPb, SawNonRepeatedFieldTwice) {
MockErrorReporter reporter;
EXPECT_CALL(
reporter,
AddError(3, 5, 15,
"Saw non-repeating field 'write_into_file' more than once"));
ToErrors(R"(
write_into_file: true;
write_into_file: true;
)",
&reporter);
}
TEST(PbtxtToPb, WrongTypeBoolean) {
MockErrorReporter reporter;
EXPECT_CALL(reporter,
AddError(2, 18, 4,
"Expected value of type uint32 for field duration_ms in "
"proto TraceConfig instead saw 'true'"));
ToErrors(R"(
duration_ms: true;
)",
&reporter);
}
TEST(PbtxtToPb, WrongTypeNumber) {
MockErrorReporter reporter;
EXPECT_CALL(reporter,
AddError(2, 14, 3,
"Expected value of type message for field buffers in "
"proto TraceConfig instead saw '100'"));
ToErrors(R"(
buffers: 100;
)",
&reporter);
}
TEST(PbtxtToPb, NestedMessageDidNotTerminate) {
MockErrorReporter reporter;
EXPECT_CALL(reporter, AddError(2, 15, 0, "Nested message not closed"));
ToErrors(R"(
buffers: {)",
&reporter);
}
TEST(PbtxtToPb, BadEscape) {
MockErrorReporter reporter;
EXPECT_CALL(reporter, AddError(5, 23, 2,
"Unknown string escape in ftrace_events in "
"proto FtraceConfig: '\\p'"));
ToErrors(R"(
data_sources {
config {
ftrace_config {
ftrace_events: "\p"
}
}
})",
&reporter);
}
// TODO(hjd): Add these tests.
// TEST(PbtxtToPb, WrongTypeString)
// TEST(PbtxtToPb, OverflowOnIntegers)
// TEST(PbtxtToPb, NegativeNumbersForUnsignedInt)
// TEST(PbtxtToPb, UnterminatedString) {
// TEST(PbtxtToPb, NumberIsEof)
// TEST(PbtxtToPb, OneOf)
} // namespace
} // namespace perfetto