| /* |
| * 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/trace_to_text/trace_to_systrace.h" |
| |
| #include <inttypes.h> |
| #include <stdio.h> |
| |
| #include <algorithm> |
| #include <functional> |
| #include <map> |
| #include <memory> |
| #include <unordered_map> |
| #include <utility> |
| #include <vector> |
| |
| #include "perfetto/base/build_config.h" |
| #include "perfetto/base/logging.h" |
| #include "perfetto/base/paged_memory.h" |
| #include "perfetto/base/string_writer.h" |
| #include "perfetto/trace_processor/trace_processor.h" |
| #include "perfetto/traced/sys_stats_counters.h" |
| #include "tools/trace_to_text/ftrace_event_formatter.h" |
| #include "tools/trace_to_text/process_formatter.h" |
| #include "tools/trace_to_text/utils.h" |
| |
| #include "perfetto/trace/trace.pb.h" |
| #include "perfetto/trace/trace_packet.pb.h" |
| #include "perfetto/trace_processor/raw_query.pb.h" |
| |
| // When running in Web Assembly, fflush() is a no-op and the stdio buffering |
| // sends progress updates to JS only when a write ends with \n. |
| #if PERFETTO_BUILDFLAG(PERFETTO_OS_WASM) |
| #define PROGRESS_CHAR "\n" |
| #else |
| #define PROGRESS_CHAR "\r" |
| #endif |
| |
| namespace perfetto { |
| namespace trace_to_text { |
| |
| namespace { |
| using protos::FtraceEvent; |
| using protos::FtraceEventBundle; |
| using protos::ProcessTree; |
| using protos::Trace; |
| using protos::TracePacket; |
| using protos::SysStats; |
| |
| // Having an empty traceEvents object is necessary for trace viewer to |
| // load the json properly. |
| const char kTraceHeader[] = R"({ |
| "traceEvents": [], |
| )"; |
| |
| const char kTraceFooter[] = R"(\n", |
| "controllerTraceDataKey": "systraceController" |
| })"; |
| |
| const char kProcessDumpHeader[] = |
| "" |
| "\"androidProcessDump\": " |
| "\"PROCESS DUMP\\nUSER PID PPID VSZ RSS WCHAN " |
| "PC S NAME COMM \\n"; |
| |
| const char kThreadHeader[] = "USER PID TID CMD \\n"; |
| |
| const char kSystemTraceEvents[] = |
| "" |
| " \"systemTraceEvents\": \""; |
| |
| const char kFtraceHeader[] = |
| "# tracer: nop\n" |
| "#\n" |
| "# entries-in-buffer/entries-written: 30624/30624 #P:4\n" |
| "#\n" |
| "# _-----=> irqs-off\n" |
| "# / _----=> need-resched\n" |
| "# | / _---=> hardirq/softirq\n" |
| "# || / _--=> preempt-depth\n" |
| "# ||| / delay\n" |
| "# TASK-PID TGID CPU# |||| TIMESTAMP FUNCTION\n" |
| "# | | | | |||| | |\n"; |
| |
| const char kFtraceJsonHeader[] = |
| "# tracer: nop\\n" |
| "#\\n" |
| "# entries-in-buffer/entries-written: 30624/30624 #P:4\\n" |
| "#\\n" |
| "# _-----=> irqs-off\\n" |
| "# / _----=> need-resched\\n" |
| "# | / _---=> hardirq/softirq\\n" |
| "# || / _--=> preempt-depth\\n" |
| "# ||| / delay\\n" |
| "# TASK-PID TGID CPU# |||| TIMESTAMP FUNCTION\\n" |
| "# | | | | |||| | |\\n"; |
| |
| class QueryWriter { |
| public: |
| QueryWriter(trace_processor::TraceProcessor* tp, std::ostream* output) |
| : tp_(tp), |
| buffer_(base::PagedMemory::Allocate(kBufferSize)), |
| global_writer_(static_cast<char*>(buffer_.Get()), kBufferSize), |
| output_(output) {} |
| |
| template <typename Callback> |
| bool RunQuery(base::StringView sql, Callback callback) { |
| auto iterator = tp_->ExecuteQuery(sql); |
| if (!iterator.IsValid()) { |
| PERFETTO_ELOG("Error creating SQL iterator"); |
| return false; |
| } |
| |
| char buffer[2048]; |
| for (uint32_t rows = 0;; rows++) { |
| using Result = trace_processor::TraceProcessor::Iterator::NextResult; |
| |
| auto result = iterator.Next(); |
| if (PERFETTO_UNLIKELY(result == Result::kError)) { |
| PERFETTO_ELOG("Error while writing systrace %s", |
| iterator.GetLastError().value().c_str()); |
| return false; |
| } else if (result == Result::kEOF) { |
| break; |
| } |
| |
| base::StringWriter line_writer(buffer, base::ArraySize(buffer)); |
| callback(&iterator, &line_writer); |
| |
| if (global_writer_.pos() + line_writer.pos() >= global_writer_.size()) { |
| fprintf(stderr, "Writing row %" PRIu32 PROGRESS_CHAR, rows); |
| auto str = global_writer_.GetStringView(); |
| output_->write(str.data(), static_cast<std::streamsize>(str.size())); |
| global_writer_.reset(); |
| } |
| global_writer_.AppendStringView(line_writer.GetStringView()); |
| } |
| |
| // Flush any dangling pieces in the global writer. |
| auto str = global_writer_.GetStringView(); |
| output_->write(str.data(), static_cast<std::streamsize>(str.size())); |
| global_writer_.reset(); |
| return true; |
| } |
| |
| private: |
| static constexpr uint32_t kBufferSize = 1024u * 1024u * 16u; |
| |
| trace_processor::TraceProcessor* tp_ = nullptr; |
| base::PagedMemory buffer_; |
| base::StringWriter global_writer_; |
| std::ostream* output_ = nullptr; |
| }; |
| |
| } // namespace |
| |
| int TraceToExperimentalSystrace(std::istream* input, |
| std::ostream* output, |
| bool wrap_in_json) { |
| trace_processor::Config config; |
| std::unique_ptr<trace_processor::TraceProcessor> tp = |
| trace_processor::TraceProcessor::CreateInstance(config); |
| |
| // 1MB chunk size seems the best tradeoff on a MacBook Pro 2013 - i7 2.8 GHz. |
| constexpr size_t kChunkSize = 1024 * 1024; |
| |
| // Printing the status update on stderr can be a perf bottleneck. On WASM print |
| // status updates more frequently because it can be slower to parse each chunk. |
| #if PERFETTO_BUILDFLAG(PERFETTO_OS_WASM) |
| constexpr int kStderrRate = 1; |
| #else |
| constexpr int kStderrRate = 128; |
| #endif |
| uint64_t file_size = 0; |
| for (int i = 0;; i++) { |
| if (i % kStderrRate == 0) { |
| fprintf(stderr, "Loading trace %.2f MB" PROGRESS_CHAR, file_size / 1.0e6); |
| fflush(stderr); |
| } |
| |
| std::unique_ptr<uint8_t[]> buf(new uint8_t[kChunkSize]); |
| input->read(reinterpret_cast<char*>(buf.get()), kChunkSize); |
| if (input->bad()) { |
| PERFETTO_ELOG("Failed when reading trace"); |
| return 1; |
| } |
| |
| auto rsize = input->gcount(); |
| if (rsize <= 0) |
| break; |
| file_size += static_cast<uint64_t>(rsize); |
| tp->Parse(std::move(buf), static_cast<size_t>(rsize)); |
| } |
| tp->NotifyEndOfFile(); |
| |
| fprintf(stderr, "Loaded trace" PROGRESS_CHAR); |
| fflush(stderr); |
| |
| using Iterator = trace_processor::TraceProcessor::Iterator; |
| |
| QueryWriter q_writer(tp.get(), output); |
| if (wrap_in_json) { |
| *output << kTraceHeader; |
| |
| *output << kProcessDumpHeader; |
| |
| // Write out all the processes in the trace. |
| // TODO(lalitm): change this query to actually use ppid when it is exposed |
| // by the process table. |
| static const char kPSql[] = "select pid, 0 as ppid, name from process"; |
| auto p_callback = [](Iterator* it, base::StringWriter* writer) { |
| uint32_t pid = static_cast<uint32_t>(it->Get(0 /* col */).long_value); |
| uint32_t ppid = static_cast<uint32_t>(it->Get(1 /* col */).long_value); |
| const char* name = it->Get(2 /* col */).string_value; |
| FormatProcess(pid, ppid, name, writer); |
| }; |
| if (!q_writer.RunQuery(kPSql, p_callback)) |
| return 1; |
| |
| *output << kThreadHeader; |
| |
| // Write out all the threads in the trace. |
| static const char kTSql[] = |
| "select tid, COALESCE(upid, 0), thread.name " |
| "from thread inner join process using (upid)"; |
| auto t_callback = [](Iterator* it, base::StringWriter* writer) { |
| uint32_t tid = static_cast<uint32_t>(it->Get(0 /* col */).long_value); |
| uint32_t tgid = static_cast<uint32_t>(it->Get(1 /* col */).long_value); |
| const char* name = it->Get(2 /* col */).string_value; |
| FormatThread(tid, tgid, name, writer); |
| }; |
| if (!q_writer.RunQuery(kTSql, t_callback)) |
| return 1; |
| |
| *output << "\","; |
| *output << kSystemTraceEvents; |
| *output << kFtraceJsonHeader; |
| } else { |
| *output << "TRACE:\n"; |
| *output << kFtraceHeader; |
| } |
| |
| fprintf(stderr, "Converting trace events" PROGRESS_CHAR); |
| fflush(stderr); |
| |
| static const char kRawSql[] = "select to_ftrace(id) from raw"; |
| auto raw_callback = [wrap_in_json](Iterator* it, base::StringWriter* writer) { |
| const char* line = it->Get(0 /* col */).string_value; |
| if (wrap_in_json) { |
| for (uint32_t i = 0; line[i] != '\0'; i++) { |
| char c = line[i]; |
| if (c == '\n') { |
| writer->AppendLiteral("\\n"); |
| continue; |
| } |
| |
| if (c == '\\' || c == '"') |
| writer->AppendChar('\\'); |
| writer->AppendChar(c); |
| } |
| writer->AppendChar('\\'); |
| writer->AppendChar('n'); |
| } else { |
| writer->AppendString(line); |
| writer->AppendChar('\n'); |
| } |
| }; |
| if (!q_writer.RunQuery(kRawSql, raw_callback)) |
| return 1; |
| |
| if (wrap_in_json) |
| *output << kTraceFooter; |
| |
| return 0; |
| } |
| |
| int TraceToSystrace(std::istream* input, |
| std::ostream* output, |
| bool wrap_in_json) { |
| std::multimap<uint64_t, std::string> ftrace_sorted; |
| std::vector<std::string> proc_dump; |
| std::vector<std::string> thread_dump; |
| std::unordered_map<uint32_t /*tid*/, uint32_t /*tgid*/> thread_map; |
| std::unordered_map<uint32_t /*tid*/, std::string> thread_names; |
| |
| std::vector<const char*> meminfo_strs = BuildMeminfoCounterNames(); |
| std::vector<const char*> vmstat_strs = BuildVmstatCounterNames(); |
| |
| std::vector<protos::TracePacket> packets_to_process; |
| |
| ForEachPacketInTrace( |
| input, [&thread_map, &packets_to_process, &proc_dump, &thread_names, |
| &thread_dump](const protos::TracePacket& packet) { |
| if (!packet.has_process_tree()) { |
| packets_to_process.emplace_back(std::move(packet)); |
| } |
| if (packet.has_ftrace_events()) { |
| const FtraceEventBundle& bundle = packet.ftrace_events(); |
| for (const FtraceEvent& event : bundle.event()) { |
| if (!event.has_sched_switch()) |
| continue; |
| const auto& sched_switch = event.sched_switch(); |
| thread_names[static_cast<uint32_t>(sched_switch.prev_pid())] = |
| sched_switch.prev_comm(); |
| thread_names[static_cast<uint32_t>(sched_switch.next_pid())] = |
| sched_switch.next_comm(); |
| } |
| return; |
| } |
| const ProcessTree& process_tree = packet.process_tree(); |
| for (const auto& process : process_tree.processes()) { |
| // Main threads will have the same pid as tgid. |
| thread_map[static_cast<uint32_t>(process.pid())] = |
| static_cast<uint32_t>(process.pid()); |
| std::string p = FormatProcess(process); |
| proc_dump.emplace_back(p); |
| } |
| for (const auto& thread : process_tree.threads()) { |
| // Populate thread map for matching tids to tgids. |
| thread_map[static_cast<uint32_t>(thread.tid())] = |
| static_cast<uint32_t>(thread.tgid()); |
| if (thread.has_name()) { |
| thread_names[static_cast<uint32_t>(thread.tid())] = thread.name(); |
| } |
| std::string t = FormatThread(thread); |
| thread_dump.emplace_back(t); |
| } |
| }); |
| |
| for (const auto& packet : packets_to_process) { |
| if (packet.has_ftrace_events()) { |
| const FtraceEventBundle& bundle = packet.ftrace_events(); |
| for (const FtraceEvent& event : bundle.event()) { |
| std::string line = FormatFtraceEvent(event.timestamp(), bundle.cpu(), |
| event, thread_map, thread_names); |
| if (line == "") |
| continue; |
| ftrace_sorted.emplace(event.timestamp(), line); |
| } |
| } // packet.has_ftrace_events |
| |
| if (packet.has_sys_stats()) { |
| const SysStats& sys_stats = packet.sys_stats(); |
| for (const auto& meminfo : sys_stats.meminfo()) { |
| FtraceEvent event; |
| uint64_t ts = static_cast<uint64_t>(packet.timestamp()); |
| char str[256]; |
| event.set_timestamp(ts); |
| event.set_pid(1); |
| sprintf(str, "C|1|%s|%" PRIu64, meminfo_strs[meminfo.key()], |
| static_cast<uint64_t>(meminfo.value())); |
| event.mutable_print()->set_buf(str); |
| ftrace_sorted.emplace( |
| ts, FormatFtraceEvent(ts, 0, event, thread_map, thread_names)); |
| } |
| for (const auto& vmstat : sys_stats.vmstat()) { |
| FtraceEvent event; |
| uint64_t ts = static_cast<uint64_t>(packet.timestamp()); |
| char str[256]; |
| event.set_timestamp(ts); |
| event.set_pid(1); |
| sprintf(str, "C|1|%s|%" PRIu64, vmstat_strs[vmstat.key()], |
| static_cast<uint64_t>(vmstat.value())); |
| event.mutable_print()->set_buf(str); |
| ftrace_sorted.emplace( |
| ts, FormatFtraceEvent(ts, 0, event, thread_map, thread_names)); |
| } |
| } |
| } |
| |
| if (wrap_in_json) { |
| *output << kTraceHeader; |
| *output << kProcessDumpHeader; |
| for (const auto& process : proc_dump) { |
| *output << process << "\\n"; |
| } |
| *output << kThreadHeader; |
| for (const auto& thread : thread_dump) { |
| *output << thread << "\\n"; |
| } |
| *output << "\","; |
| *output << kSystemTraceEvents; |
| *output << kFtraceJsonHeader; |
| } else { |
| *output << "TRACE:\n"; |
| *output << kFtraceHeader; |
| } |
| |
| fprintf(stderr, "\n"); |
| size_t total_events = ftrace_sorted.size(); |
| size_t written_events = 0; |
| std::vector<char> escaped_str; |
| for (auto it = ftrace_sorted.begin(); it != ftrace_sorted.end(); it++) { |
| if (wrap_in_json) { |
| escaped_str.clear(); |
| escaped_str.reserve(it->second.size() * 101 / 100); |
| for (char c : it->second) { |
| if (c == '\\' || c == '"') |
| escaped_str.push_back('\\'); |
| escaped_str.push_back(c); |
| } |
| escaped_str.push_back('\\'); |
| escaped_str.push_back('n'); |
| escaped_str.push_back('\0'); |
| *output << escaped_str.data(); |
| } else { |
| *output << it->second; |
| *output << "\n"; |
| } |
| if (!StdoutIsTty() && (written_events++ % 1000 == 0 || |
| written_events == ftrace_sorted.size())) { |
| fprintf(stderr, "Writing trace: %.2f %%" PROGRESS_CHAR, |
| written_events * 100.0 / total_events); |
| fflush(stderr); |
| output->flush(); |
| } |
| } |
| |
| if (wrap_in_json) |
| *output << kTraceFooter; |
| |
| return 0; |
| } |
| |
| } // namespace trace_to_text |
| } // namespace perfetto |