| /* |
| * Copyright (C) 2019 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/trace_processor/forwarding_trace_parser.h" |
| |
| #include <memory> |
| #include <optional> |
| #include <utility> |
| |
| #include "perfetto/base/logging.h" |
| #include "perfetto/base/status.h" |
| #include "perfetto/ext/base/status_or.h" |
| #include "perfetto/trace_processor/basic_types.h" |
| #include "src/trace_processor/importers/common/chunked_trace_reader.h" |
| #include "src/trace_processor/importers/common/process_tracker.h" |
| #include "src/trace_processor/importers/common/trace_file_tracker.h" |
| #include "src/trace_processor/importers/proto/proto_trace_reader.h" |
| #include "src/trace_processor/sorter/trace_sorter.h" |
| #include "src/trace_processor/storage/stats.h" |
| #include "src/trace_processor/tables/metadata_tables_py.h" |
| #include "src/trace_processor/trace_reader_registry.h" |
| #include "src/trace_processor/types/trace_processor_context.h" |
| #include "src/trace_processor/util/status_macros.h" |
| #include "src/trace_processor/util/trace_type.h" |
| |
| namespace perfetto::trace_processor { |
| namespace { |
| |
| TraceSorter::SortingMode ConvertSortingMode(SortingMode sorting_mode) { |
| switch (sorting_mode) { |
| case SortingMode::kDefaultHeuristics: |
| return TraceSorter::SortingMode::kDefault; |
| case SortingMode::kForceFullSort: |
| return TraceSorter::SortingMode::kFullSort; |
| } |
| PERFETTO_FATAL("For GCC"); |
| } |
| |
| std::optional<TraceSorter::SortingMode> GetMinimumSortingMode( |
| TraceType trace_type, |
| const TraceProcessorContext& context) { |
| switch (trace_type) { |
| case kNinjaLogTraceType: |
| case kSystraceTraceType: |
| case kGzipTraceType: |
| case kCtraceTraceType: |
| return std::nullopt; |
| |
| case kPerfDataTraceType: |
| case kInstrumentsXmlTraceType: |
| return TraceSorter::SortingMode::kDefault; |
| |
| case kUnknownTraceType: |
| case kJsonTraceType: |
| case kFuchsiaTraceType: |
| case kZipFile: |
| case kTarTraceType: |
| case kAndroidLogcatTraceType: |
| case kGeckoTraceType: |
| case kArtMethodTraceType: |
| case kPerfTextTraceType: |
| return TraceSorter::SortingMode::kFullSort; |
| |
| case kProtoTraceType: |
| case kSymbolsTraceType: |
| return ConvertSortingMode(context.config.sorting_mode); |
| |
| case kAndroidDumpstateTraceType: |
| case kAndroidBugreportTraceType: |
| PERFETTO_FATAL( |
| "This trace type should be handled at the ZipParser level"); |
| } |
| PERFETTO_FATAL("For GCC"); |
| } |
| |
| } // namespace |
| |
| ForwardingTraceParser::ForwardingTraceParser(TraceProcessorContext* context, |
| tables::TraceFileTable::Id id) |
| : context_(context), file_id_(id) {} |
| |
| ForwardingTraceParser::~ForwardingTraceParser() = default; |
| |
| base::Status ForwardingTraceParser::Init(const TraceBlobView& blob) { |
| PERFETTO_CHECK(!reader_); |
| |
| { |
| auto scoped_trace = context_->storage->TraceExecutionTimeIntoStats( |
| stats::guess_trace_type_duration_ns); |
| trace_type_ = GuessTraceType(blob.data(), blob.size()); |
| } |
| if (trace_type_ == kUnknownTraceType) { |
| // If renaming this error message don't remove the "(ERR:fmt)" part. |
| // The UI's error_dialog.ts uses it to make the dialog more graceful. |
| return base::ErrStatus("Unknown trace type provided (ERR:fmt)"); |
| } |
| context_->trace_file_tracker->StartParsing(file_id_, trace_type_); |
| ASSIGN_OR_RETURN(reader_, |
| context_->reader_registry->CreateTraceReader(trace_type_)); |
| |
| PERFETTO_DLOG("%s trace detected", TraceTypeToString(trace_type_)); |
| UpdateSorterForTraceType(trace_type_); |
| |
| // TODO(b/334978369) Make sure kProtoTraceType and kSystraceTraceType are |
| // parsed first so that we do not get issues with |
| // SetPidZeroIsUpidZeroIdleProcess() |
| if (trace_type_ == kProtoTraceType || trace_type_ == kSystraceTraceType) { |
| context_->process_tracker->SetPidZeroIsUpidZeroIdleProcess(); |
| } |
| return base::OkStatus(); |
| } |
| |
| void ForwardingTraceParser::UpdateSorterForTraceType(TraceType trace_type) { |
| std::optional<TraceSorter::SortingMode> minimum_sorting_mode = |
| GetMinimumSortingMode(trace_type, *context_); |
| if (!minimum_sorting_mode.has_value()) { |
| return; |
| } |
| |
| if (!context_->sorter) { |
| TraceSorter::EventHandling event_handling; |
| switch (context_->config.parsing_mode) { |
| case ParsingMode::kDefault: |
| event_handling = TraceSorter::EventHandling::kSortAndPush; |
| break; |
| case ParsingMode::kTokenizeOnly: |
| event_handling = TraceSorter::EventHandling::kDrop; |
| break; |
| case ParsingMode::kTokenizeAndSort: |
| event_handling = TraceSorter::EventHandling::kSortAndDrop; |
| break; |
| } |
| if (context_->config.enable_dev_features) { |
| auto it = context_->config.dev_flags.find("drop-after-sort"); |
| if (it != context_->config.dev_flags.end() && it->second == "true") { |
| event_handling = TraceSorter::EventHandling::kSortAndDrop; |
| } |
| } |
| context_->sorter = std::make_shared<TraceSorter>( |
| context_, *minimum_sorting_mode, event_handling); |
| } |
| |
| switch (context_->sorter->sorting_mode()) { |
| case TraceSorter::SortingMode::kDefault: |
| PERFETTO_CHECK(minimum_sorting_mode == |
| TraceSorter::SortingMode::kDefault); |
| break; |
| case TraceSorter::SortingMode::kFullSort: |
| break; |
| } |
| } |
| |
| base::Status ForwardingTraceParser::Parse(TraceBlobView blob) { |
| // If this is the first Parse() call, guess the trace type and create the |
| // appropriate parser. |
| if (!reader_) { |
| RETURN_IF_ERROR(Init(blob)); |
| } |
| trace_size_ += blob.size(); |
| return reader_->Parse(std::move(blob)); |
| } |
| |
| base::Status ForwardingTraceParser::NotifyEndOfFile() { |
| if (reader_) { |
| RETURN_IF_ERROR(reader_->NotifyEndOfFile()); |
| } |
| if (trace_type_ != kUnknownTraceType) { |
| context_->trace_file_tracker->DoneParsing(file_id_, trace_size_); |
| } |
| return base::OkStatus(); |
| } |
| |
| } // namespace perfetto::trace_processor |