blob: 7b79bffad00cd34c8fc73922f189b37f2b80a0d7 [file] [log] [blame]
/*
* 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/trace_processor_storage_impl.h"
#include "perfetto/base/logging.h"
#include "perfetto/ext/base/uuid.h"
#include "src/trace_processor/forwarding_trace_parser.h"
#include "src/trace_processor/importers/common/args_tracker.h"
#include "src/trace_processor/importers/common/args_translation_table.h"
#include "src/trace_processor/importers/common/async_track_set_tracker.h"
#include "src/trace_processor/importers/common/clock_converter.h"
#include "src/trace_processor/importers/common/clock_tracker.h"
#include "src/trace_processor/importers/common/event_tracker.h"
#include "src/trace_processor/importers/common/flow_tracker.h"
#include "src/trace_processor/importers/common/metadata_tracker.h"
#include "src/trace_processor/importers/common/process_tracker.h"
#include "src/trace_processor/importers/common/slice_tracker.h"
#include "src/trace_processor/importers/common/slice_translation_table.h"
#include "src/trace_processor/importers/common/track_tracker.h"
#include "src/trace_processor/importers/proto/chrome_track_event.descriptor.h"
#include "src/trace_processor/importers/proto/default_modules.h"
#include "src/trace_processor/importers/proto/heap_profile_tracker.h"
#include "src/trace_processor/importers/proto/packet_analyzer.h"
#include "src/trace_processor/importers/proto/perf_sample_tracker.h"
#include "src/trace_processor/importers/proto/proto_importer_module.h"
#include "src/trace_processor/importers/proto/proto_trace_reader.h"
#include "src/trace_processor/importers/proto/stack_profile_tracker.h"
#include "src/trace_processor/importers/proto/track_event.descriptor.h"
#include "src/trace_processor/sorter/trace_sorter.h"
#include "src/trace_processor/util/descriptors.h"
namespace perfetto {
namespace trace_processor {
TraceProcessorStorageImpl::TraceProcessorStorageImpl(const Config& cfg) {
context_.config = cfg;
context_.storage.reset(new TraceStorage(context_.config));
context_.track_tracker.reset(new TrackTracker(&context_));
context_.async_track_set_tracker.reset(new AsyncTrackSetTracker(&context_));
context_.args_tracker.reset(new ArgsTracker(&context_));
context_.args_translation_table.reset(
new ArgsTranslationTable(context_.storage.get()));
context_.slice_tracker.reset(new SliceTracker(&context_));
context_.slice_translation_table.reset(
new SliceTranslationTable(context_.storage.get()));
context_.flow_tracker.reset(new FlowTracker(&context_));
context_.event_tracker.reset(new EventTracker(&context_));
context_.process_tracker.reset(new ProcessTracker(&context_));
context_.clock_tracker.reset(new ClockTracker(&context_));
context_.clock_converter.reset(new ClockConverter(&context_));
context_.heap_profile_tracker.reset(new HeapProfileTracker(&context_));
context_.perf_sample_tracker.reset(new PerfSampleTracker(&context_));
context_.global_stack_profile_tracker.reset(new GlobalStackProfileTracker());
context_.metadata_tracker.reset(new MetadataTracker(context_.storage.get()));
context_.global_args_tracker.reset(
new GlobalArgsTracker(context_.storage.get()));
{
context_.descriptor_pool_.reset(new DescriptorPool());
auto status = context_.descriptor_pool_->AddFromFileDescriptorSet(
kTrackEventDescriptor.data(), kTrackEventDescriptor.size());
PERFETTO_DCHECK(status.ok());
status = context_.descriptor_pool_->AddFromFileDescriptorSet(
kChromeTrackEventDescriptor.data(), kChromeTrackEventDescriptor.size());
PERFETTO_DCHECK(status.ok());
}
context_.slice_tracker->SetOnSliceBeginCallback(
[this](TrackId track_id, SliceId slice_id) {
context_.flow_tracker->ClosePendingEventsOnTrack(track_id, slice_id);
});
RegisterDefaultModules(&context_);
}
TraceProcessorStorageImpl::~TraceProcessorStorageImpl() {}
util::Status TraceProcessorStorageImpl::Parse(TraceBlobView blob) {
if (blob.size() == 0)
return util::OkStatus();
if (unrecoverable_parse_error_)
return util::ErrStatus(
"Failed unrecoverably while parsing in a previous Parse call");
if (!context_.chunk_reader)
context_.chunk_reader.reset(new ForwardingTraceParser(&context_));
auto scoped_trace = context_.storage->TraceExecutionTimeIntoStats(
stats::parse_trace_duration_ns);
if (hash_input_size_remaining_ > 0 && !context_.uuid_found_in_trace) {
const size_t hash_size = std::min(hash_input_size_remaining_, blob.size());
hash_input_size_remaining_ -= hash_size;
trace_hash_.Update(reinterpret_cast<const char*>(blob.data()), hash_size);
base::Uuid uuid(static_cast<int64_t>(trace_hash_.digest()), 0);
const StringId id_for_uuid =
context_.storage->InternString(base::StringView(uuid.ToPrettyString()));
context_.metadata_tracker->SetMetadata(metadata::trace_uuid,
Variadic::String(id_for_uuid));
}
util::Status status = context_.chunk_reader->Parse(std::move(blob));
unrecoverable_parse_error_ |= !status.ok();
return status;
}
void TraceProcessorStorageImpl::Flush() {
if (unrecoverable_parse_error_)
return;
if (context_.sorter)
context_.sorter->ExtractEventsForced();
}
void TraceProcessorStorageImpl::NotifyEndOfFile() {
if (unrecoverable_parse_error_ || !context_.chunk_reader)
return;
Flush();
context_.chunk_reader->NotifyEndOfFile();
for (std::unique_ptr<ProtoImporterModule>& module : context_.modules) {
module->NotifyEndOfFile();
}
if (context_.content_analyzer) {
PacketAnalyzer::Get(&context_)->NotifyEndOfFile();
}
context_.event_tracker->FlushPendingEvents();
context_.slice_tracker->FlushPendingSlices();
context_.heap_profile_tracker->NotifyEndOfFile();
context_.args_tracker->Flush();
context_.process_tracker->NotifyEndOfFile();
}
void TraceProcessorStorageImpl::DestroyContext() {
TraceProcessorContext context;
context.storage = std::move(context_.storage);
context.heap_graph_tracker = std::move(context_.heap_graph_tracker);
context.clock_converter = std::move(context_.clock_converter);
// "to_ftrace" textual converter of the "raw" table requires remembering the
// kernel version (inside system_info_tracker) to know how to textualise
// sched_switch.prev_state bitflags.
context.system_info_tracker = std::move(context_.system_info_tracker);
context_ = std::move(context);
}
} // namespace trace_processor
} // namespace perfetto