/* * 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 "tools/ftrace_proto_gen/ftrace_descriptor_gen.h" namespace perfetto { void GenerateFtraceDescriptors( const google::protobuf::DescriptorPool& descriptor_pool, std::ostream* fout) { const google::protobuf::Descriptor* ftrace_event = descriptor_pool.FindMessageTypeByName("perfetto.protos.FtraceEvent"); const google::protobuf::OneofDescriptor* one_of_event = ftrace_event->FindOneofByName("event"); // Find max id for any ftrace event. int max_id = 0; for (int i = 0; i < one_of_event->field_count(); i++) max_id = std::max(max_id, one_of_event->field(i)->number()); *fout << "// Autogenerated by:\n"; *fout << std::string("// ") + __FILE__ + "\n"; *fout << "// Do not edit.\n"; *fout << R"( #include "src/trace_processor/ftrace_descriptors.h" namespace perfetto { namespace trace_processor { namespace { std::array<MessageDescriptor, )"; *fout << std::to_string(max_id + 1) + "> descriptors{{"; for (int i = 0; i <= max_id; i++) { const google::protobuf::FieldDescriptor* event = ftrace_event->FindFieldByNumber(i); // Skip events that don't exist or are not messages. (Proxy for events) if (!event || event->type() != google::protobuf::FieldDescriptor::TYPE_MESSAGE) { *fout << "{nullptr, 0, {}},"; continue; } const auto* event_descriptor = event->message_type(); // Find the max field id in the event. int max_field_id = 0; for (int j = 0; j < event_descriptor->field_count(); j++) max_field_id = std::max(max_field_id, event_descriptor->field(j)->number()); *fout << "{\"" + event->name() + "\", " << max_field_id << ", " << "{"; for (int j = 0; j <= max_field_id; j++) { const auto* field = event_descriptor->FindFieldByNumber(j); // Skip fields that don't exist or are nested messages. if (!field || field->type() == google::protobuf::FieldDescriptor::TYPE_MESSAGE) { *fout << "{},"; continue; } ProtoType type = ProtoType::FromDescriptor(field->type()); *fout << "{\"" + field->name() + "\", ProtoSchemaType::k" + ToCamelCase(type.ToString()) + "},"; } *fout << "},\n},"; } *fout << "}};\n"; *fout << R"( } // namespace MessageDescriptor* GetMessageDescriptorForId(size_t id) { PERFETTO_CHECK(id < descriptors.size()); return &descriptors[id]; } size_t GetDescriptorsSize() { return descriptors.size(); } )"; *fout << "} // namespace trace_processor\n} // namespace perfetto\n"; } } // namespace perfetto