//===-- PerfReader.cpp - perfscript reader ---------------------*- C++ -*-===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "PerfReader.h" static cl::opt ShowMmapEvents("show-mmap-events", cl::ReallyHidden, cl::init(false), cl::ZeroOrMore, cl::desc("Print binary load events.")); namespace llvm { namespace sampleprof { PerfReader::PerfReader(cl::list &BinaryFilenames) { // Load the binaries. for (auto Filename : BinaryFilenames) loadBinary(Filename, /*AllowNameConflict*/ false); } ProfiledBinary &PerfReader::loadBinary(const StringRef BinaryPath, bool AllowNameConflict) { // The binary table is currently indexed by the binary name not the full // binary path. This is because the user-given path may not match the one // that was actually executed. StringRef BinaryName = llvm::sys::path::filename(BinaryPath); // Call to load the binary in the ctor of ProfiledBinary. auto Ret = BinaryTable.insert({BinaryName, ProfiledBinary(BinaryPath)}); if (!Ret.second && !AllowNameConflict) { std::string ErrorMsg = "Binary name conflict: " + BinaryPath.str() + " and " + Ret.first->second.getPath().str() + " \n"; exitWithError(ErrorMsg); } return Ret.first->second; } void PerfReader::updateBinaryAddress(const MMapEvent &Event) { // Load the binary. StringRef BinaryPath = Event.BinaryPath; StringRef BinaryName = llvm::sys::path::filename(BinaryPath); auto I = BinaryTable.find(BinaryName); // Drop the event which doesn't belong to user-provided binaries // or if its image is loaded at the same address if (I == BinaryTable.end() || Event.BaseAddress == I->second.getBaseAddress()) return; ProfiledBinary &Binary = I->second; // A binary image could be uploaded and then reloaded at different // place, so update the address map here AddrToBinaryMap.erase(Binary.getBaseAddress()); AddrToBinaryMap[Event.BaseAddress] = &Binary; // Update binary load address. Binary.setBaseAddress(Event.BaseAddress); } void PerfReader::parseMMap2Event(TraceStream &TraceIt) { // Parse a line like: // PERF_RECORD_MMAP2 2113428/2113428: [0x7fd4efb57000(0x204000) @ 0 // 08:04 19532229 3585508847]: r-xp /usr/lib64/libdl-2.17.so constexpr static const char *const Pattern = "PERF_RECORD_MMAP2 ([0-9]+)/[0-9]+: " "\\[(0x[a-f0-9]+)\\((0x[a-f0-9]+)\\) @ " "(0x[a-f0-9]+|0) .*\\]: [-a-z]+ (.*)"; // Field 0 - whole line // Field 1 - PID // Field 2 - base address // Field 3 - mmapped size // Field 4 - page offset // Field 5 - binary path enum EventIndex { WHOLE_LINE = 0, PID = 1, BASE_ADDRESS = 2, MMAPPED_SIZE = 3, PAGE_OFFSET = 4, BINARY_PATH = 5 }; Regex RegMmap2(Pattern); SmallVector Fields; bool R = RegMmap2.match(TraceIt.getCurrentLine(), &Fields); if (!R) { std::string ErrorMsg = "Cannot parse mmap event: Line" + Twine(TraceIt.getLineNumber()).str() + ": " + TraceIt.getCurrentLine().str() + " \n"; exitWithError(ErrorMsg); } MMapEvent Event; Fields[PID].getAsInteger(10, Event.PID); Fields[BASE_ADDRESS].getAsInteger(0, Event.BaseAddress); Fields[MMAPPED_SIZE].getAsInteger(0, Event.Size); Fields[PAGE_OFFSET].getAsInteger(0, Event.Offset); Event.BinaryPath = Fields[BINARY_PATH]; updateBinaryAddress(Event); if (ShowMmapEvents) { outs() << "Mmap: Binary " << Event.BinaryPath << " loaded at " << format("0x%" PRIx64 ":", Event.BaseAddress) << " \n"; } } void PerfReader::parseEvent(TraceStream &TraceIt) { if (TraceIt.getCurrentLine().startswith("PERF_RECORD_MMAP2")) parseMMap2Event(TraceIt); TraceIt.advance(); } void PerfReader::parseTrace(StringRef Filename) { // Trace line iterator TraceStream TraceIt(Filename); while (!TraceIt.isAtEoF()) { parseEvent(TraceIt); } } void PerfReader::parsePerfTraces(cl::list &PerfTraceFilenames) { // Parse perf traces. for (auto Filename : PerfTraceFilenames) parseTrace(Filename); } } // end namespace sampleprof } // end namespace llvm