/* * Copyright (C) 2015 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 #include #include #include #include #include #include #include "command.h" #include "dso.h" #include "ETMDecoder.h" #include "event_attr.h" #include "event_type.h" #include "perf_regs.h" #include "record.h" #include "record_file.h" #include "utils.h" using namespace PerfFileFormat; using namespace simpleperf; class DumpRecordCommand : public Command { public: DumpRecordCommand() : Command("dump", "dump perf record file", // clang-format off "Usage: simpleperf dumprecord [options] [perf_record_file]\n" " Dump different parts of a perf record file. Default file is perf.data.\n" "--dump-etm type1,type2,... Dump etm data. A type is one of raw, packet and element.\n" "--symdir Look for binaries in a directory recursively.\n" // clang-format on ) {} bool Run(const std::vector& args); private: bool ParseOptions(const std::vector& args); void DumpFileHeader(); void DumpAttrSection(); bool DumpDataSection(); bool DumpAuxData(const AuxRecord& aux, ETMDecoder& etm_decoder); bool DumpFeatureSection(); std::string record_filename_ = "perf.data"; std::unique_ptr record_file_reader_; ETMDumpOption etm_dump_option_; }; bool DumpRecordCommand::Run(const std::vector& args) { if (!ParseOptions(args)) { return false; } record_file_reader_ = RecordFileReader::CreateInstance(record_filename_); if (record_file_reader_ == nullptr) { return false; } DumpFileHeader(); DumpAttrSection(); if (!DumpDataSection()) { return false; } return DumpFeatureSection(); } bool DumpRecordCommand::ParseOptions(const std::vector& args) { size_t i; for (i = 0; i < args.size() && !args[i].empty() && args[i][0] == '-'; ++i) { if (args[i] == "--dump-etm") { if (!NextArgumentOrError(args, &i) || !ParseEtmDumpOption(args[i], &etm_dump_option_)) { return false; } } else if (args[i] == "--symdir") { if (!NextArgumentOrError(args, &i)) { return false; } if (!Dso::AddSymbolDir(args[i])) { return false; } } else { ReportUnknownOption(args, i); return false; } } if (i + 1 < args.size()) { LOG(ERROR) << "too many record files"; return false; } if (i + 1 == args.size()) { record_filename_ = args[i]; } return true; } static const std::string GetFeatureNameOrUnknown(int feature) { std::string name = GetFeatureName(feature); return name.empty() ? android::base::StringPrintf("unknown_feature(%d)", feature) : name; } void DumpRecordCommand::DumpFileHeader() { const FileHeader& header = record_file_reader_->FileHeader(); printf("magic: "); for (size_t i = 0; i < 8; ++i) { printf("%c", header.magic[i]); } printf("\n"); printf("header_size: %" PRId64 "\n", header.header_size); if (header.header_size != sizeof(header)) { PLOG(WARNING) << "record file header size " << header.header_size << "doesn't match expected header size " << sizeof(header); } printf("attr_size: %" PRId64 "\n", header.attr_size); if (header.attr_size != sizeof(FileAttr)) { PLOG(WARNING) << "record file attr size " << header.attr_size << " doesn't match expected attr size " << sizeof(FileAttr); } printf("attrs[file section]: offset %" PRId64 ", size %" PRId64 "\n", header.attrs.offset, header.attrs.size); printf("data[file section]: offset %" PRId64 ", size %" PRId64 "\n", header.data.offset, header.data.size); printf("event_types[file section]: offset %" PRId64 ", size %" PRId64 "\n", header.event_types.offset, header.event_types.size); std::vector features; for (size_t i = 0; i < FEAT_MAX_NUM; ++i) { size_t j = i / 8; size_t k = i % 8; if ((header.features[j] & (1 << k)) != 0) { features.push_back(i); } } for (auto& feature : features) { printf("feature: %s\n", GetFeatureNameOrUnknown(feature).c_str()); } } void DumpRecordCommand::DumpAttrSection() { std::vector attrs = record_file_reader_->AttrSection(); for (size_t i = 0; i < attrs.size(); ++i) { const auto& attr = attrs[i]; printf("attr %zu:\n", i + 1); DumpPerfEventAttr(*attr.attr, 1); if (!attr.ids.empty()) { printf(" ids:"); for (const auto& id : attr.ids) { printf(" %" PRId64, id); } printf("\n"); } } } bool DumpRecordCommand::DumpDataSection() { std::unique_ptr etm_decoder; ThreadTree thread_tree; thread_tree.ShowIpForUnknownSymbol(); record_file_reader_->LoadBuildIdAndFileFeatures(thread_tree); auto get_symbol_function = [&](uint32_t pid, uint32_t tid, uint64_t ip, std::string& dso_name, std::string& symbol_name, uint64_t& vaddr_in_file, bool in_kernel) { ThreadEntry* thread = thread_tree.FindThreadOrNew(pid, tid); const MapEntry* map = thread_tree.FindMap(thread, ip, in_kernel); Dso* dso; const Symbol* symbol = thread_tree.FindSymbol(map, ip, &vaddr_in_file, &dso); dso_name = dso->Path(); symbol_name = symbol->DemangledName(); }; auto record_callback = [&](std::unique_ptr r) { r->Dump(); thread_tree.Update(*r); if (r->type() == PERF_RECORD_SAMPLE) { SampleRecord& sr = *static_cast(r.get()); bool in_kernel = sr.InKernel(); if (sr.sample_type & PERF_SAMPLE_CALLCHAIN) { PrintIndented(1, "callchain:\n"); for (size_t i = 0; i < sr.callchain_data.ip_nr; ++i) { if (sr.callchain_data.ips[i] >= PERF_CONTEXT_MAX) { if (sr.callchain_data.ips[i] == PERF_CONTEXT_USER) { in_kernel = false; } continue; } std::string dso_name; std::string symbol_name; uint64_t vaddr_in_file; get_symbol_function(sr.tid_data.pid, sr.tid_data.tid, sr.callchain_data.ips[i], dso_name, symbol_name, vaddr_in_file, in_kernel); PrintIndented(2, "%s (%s[+%" PRIx64 "])\n", symbol_name.c_str(), dso_name.c_str(), vaddr_in_file); } } } else if (r->type() == SIMPLE_PERF_RECORD_CALLCHAIN) { CallChainRecord& cr = *static_cast(r.get()); PrintIndented(1, "callchain:\n"); for (size_t i = 0; i < cr.ip_nr; ++i) { std::string dso_name; std::string symbol_name; uint64_t vaddr_in_file; get_symbol_function(cr.pid, cr.tid, cr.ips[i], dso_name, symbol_name, vaddr_in_file, false); PrintIndented(2, "%s (%s[+%" PRIx64 "])\n", symbol_name.c_str(), dso_name.c_str(), vaddr_in_file); } } else if (r->type() == PERF_RECORD_AUXTRACE_INFO) { etm_decoder = ETMDecoder::Create(*static_cast(r.get()), thread_tree); if (!etm_decoder) { return false; } etm_decoder->EnableDump(etm_dump_option_); } else if (r->type() == PERF_RECORD_AUX) { CHECK(etm_decoder); return DumpAuxData(*static_cast(r.get()), *etm_decoder); } return true; }; return record_file_reader_->ReadDataSection(record_callback); } bool DumpRecordCommand::DumpAuxData(const AuxRecord& aux, ETMDecoder& etm_decoder) { size_t size = aux.data->aux_size; if (size > 0) { std::unique_ptr data(new uint8_t[size]); if (!record_file_reader_->ReadAuxData(aux.Cpu(), aux.data->aux_offset, data.get(), size)) { return false; } return etm_decoder.ProcessData(data.get(), size); } return true; } bool DumpRecordCommand::DumpFeatureSection() { std::map section_map = record_file_reader_->FeatureSectionDescriptors(); for (const auto& pair : section_map) { int feature = pair.first; const auto& section = pair.second; printf("feature section for %s: offset %" PRId64 ", size %" PRId64 "\n", GetFeatureNameOrUnknown(feature).c_str(), section.offset, section.size); if (feature == FEAT_BUILD_ID) { std::vector records = record_file_reader_->ReadBuildIdFeature(); for (auto& r : records) { r.Dump(1); } } else if (feature == FEAT_OSRELEASE) { std::string s = record_file_reader_->ReadFeatureString(feature); PrintIndented(1, "osrelease: %s\n", s.c_str()); } else if (feature == FEAT_ARCH) { std::string s = record_file_reader_->ReadFeatureString(feature); PrintIndented(1, "arch: %s\n", s.c_str()); } else if (feature == FEAT_CMDLINE) { std::vector cmdline = record_file_reader_->ReadCmdlineFeature(); PrintIndented(1, "cmdline: %s\n", android::base::Join(cmdline, ' ').c_str()); } else if (feature == FEAT_FILE) { std::string file_path; uint32_t file_type; uint64_t min_vaddr; uint64_t file_offset_of_min_vaddr; std::vector symbols; std::vector dex_file_offsets; size_t read_pos = 0; PrintIndented(1, "file:\n"); while (record_file_reader_->ReadFileFeature(read_pos, &file_path, &file_type, &min_vaddr, &file_offset_of_min_vaddr, &symbols, &dex_file_offsets)) { PrintIndented(2, "file_path %s\n", file_path.c_str()); PrintIndented(2, "file_type %s\n", DsoTypeToString(static_cast(file_type))); PrintIndented(2, "min_vaddr 0x%" PRIx64 "\n", min_vaddr); PrintIndented(2, "file_offset_of_min_vaddr 0x%" PRIx64 "\n", file_offset_of_min_vaddr); PrintIndented(2, "symbols:\n"); for (const auto& symbol : symbols) { PrintIndented(3, "%s [0x%" PRIx64 "-0x%" PRIx64 "]\n", symbol.DemangledName(), symbol.addr, symbol.addr + symbol.len); } if (file_type == static_cast(DSO_DEX_FILE)) { PrintIndented(2, "dex_file_offsets:\n"); for (uint64_t offset : dex_file_offsets) { PrintIndented(3, "0x%" PRIx64 "\n", offset); } } } } else if (feature == FEAT_META_INFO) { PrintIndented(1, "meta_info:\n"); for (auto& pair : record_file_reader_->GetMetaInfoFeature()) { PrintIndented(2, "%s = %s\n", pair.first.c_str(), pair.second.c_str()); } } else if (feature == FEAT_AUXTRACE) { PrintIndented(1, "file_offsets_of_auxtrace_records:\n"); for (auto offset : record_file_reader_->ReadAuxTraceFeature()) { PrintIndented(2, "%" PRIu64 "\n", offset); } } } return true; } void RegisterDumpRecordCommand() { RegisterCommand("dump", [] { return std::unique_ptr(new DumpRecordCommand); }); }