/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim:set ts=2 sw=2 sts=2 et cindent: */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #ifndef SHARED_LIBRARIES_H_ #define SHARED_LIBRARIES_H_ #include "BaseProfiler.h" #include #include #include #include #include #include namespace IPC { template struct ParamTraits; } // namespace IPC class SharedLibrary { public: SharedLibrary(uintptr_t aStart, uintptr_t aEnd, uintptr_t aOffset, const std::string& aBreakpadId, const std::string& aCodeId, const std::string& aModuleName, const std::string& aModulePath, const std::string& aDebugName, const std::string& aDebugPath, const std::string& aVersion, const char* aArch) : mStart(aStart), mEnd(aEnd), mOffset(aOffset), mBreakpadId(aBreakpadId), mCodeId(aCodeId), mModuleName(aModuleName), mModulePath(aModulePath), mDebugName(aDebugName), mDebugPath(aDebugPath), mVersion(aVersion), mArch(aArch) {} bool operator==(const SharedLibrary& other) const { return (mStart == other.mStart) && (mEnd == other.mEnd) && (mOffset == other.mOffset) && (mModuleName == other.mModuleName) && (mModulePath == other.mModulePath) && (mDebugName == other.mDebugName) && (mDebugPath == other.mDebugPath) && (mBreakpadId == other.mBreakpadId) && (mCodeId == other.mCodeId) && (mVersion == other.mVersion) && (mArch == other.mArch); } uintptr_t GetStart() const { return mStart; } uintptr_t GetEnd() const { return mEnd; } uintptr_t GetOffset() const { return mOffset; } const std::string& GetBreakpadId() const { return mBreakpadId; } const std::string& GetCodeId() const { return mCodeId; } const std::string& GetModuleName() const { return mModuleName; } const std::string& GetModulePath() const { return mModulePath; } const std::string& GetDebugName() const { return mDebugName; } const std::string& GetDebugPath() const { return mDebugPath; } const std::string& GetVersion() const { return mVersion; } const std::string& GetArch() const { return mArch; } size_t SizeOf() const { return sizeof *this + mBreakpadId.length() + mCodeId.length() + mModuleName.length() + mModulePath.length() + mDebugName.length() + mDebugPath.length() + mVersion.length() + mArch.size(); } SharedLibrary() : mStart{0}, mEnd{0}, mOffset{0} {} private: uintptr_t mStart; uintptr_t mEnd; uintptr_t mOffset; std::string mBreakpadId; // A string carrying an identifier for a binary. // // All platforms have different formats: // - Windows: The code ID for a Windows PE file. // It's the PE timestamp and PE image size. // - macOS: The code ID for a macOS / iOS binary (mach-O). // It's the mach-O UUID without dashes and without the trailing 0 for the // breakpad ID. // - Linux/Android: The code ID for a Linux ELF file. // It's the complete build ID, as hex string. std::string mCodeId; std::string mModuleName; std::string mModulePath; std::string mDebugName; std::string mDebugPath; std::string mVersion; std::string mArch; friend struct IPC::ParamTraits; }; static bool CompareAddresses(const SharedLibrary& first, const SharedLibrary& second) { return first.GetStart() < second.GetStart(); } class SharedLibraryInfo { public: #ifdef MOZ_GECKO_PROFILER MFBT_API static SharedLibraryInfo GetInfoForSelf(); # ifdef XP_WIN MFBT_API static SharedLibraryInfo GetInfoFromPath(const wchar_t* aPath); # endif MFBT_API static void Initialize(); #else static SharedLibraryInfo GetInfoForSelf() { return SharedLibraryInfo(); } # ifdef XP_WIN static SharedLibraryInfo GetInfoFromPath(const wchar_t* aPath) { return SharedLibraryInfo(); } # endif static void Initialize() {} #endif void AddSharedLibrary(SharedLibrary entry) { mEntries.push_back(entry); } void AddAllSharedLibraries(const SharedLibraryInfo& sharedLibraryInfo) { mEntries.insert(mEntries.end(), sharedLibraryInfo.mEntries.begin(), sharedLibraryInfo.mEntries.end()); } const SharedLibrary& GetEntry(size_t i) const { return mEntries[i]; } SharedLibrary& GetMutableEntry(size_t i) { return mEntries[i]; } // Removes items in the range [first, last) // i.e. element at the "last" index is not removed void RemoveEntries(size_t first, size_t last) { mEntries.erase(mEntries.begin() + first, mEntries.begin() + last); } bool Contains(const SharedLibrary& searchItem) const { return (mEntries.end() != std::find(mEntries.begin(), mEntries.end(), searchItem)); } size_t GetSize() const { return mEntries.size(); } void SortByAddress() { std::sort(mEntries.begin(), mEntries.end(), CompareAddresses); } // Remove duplicate entries from the vector. // // We purposefully don't use the operator== implementation of SharedLibrary // because it compares all the fields including mStart, mEnd and mOffset which // are not the same across different processes. void DeduplicateEntries() { static auto cmpSort = [](const SharedLibrary& a, const SharedLibrary& b) { return std::tie(a.GetModuleName(), a.GetBreakpadId()) < std::tie(b.GetModuleName(), b.GetBreakpadId()); }; static auto cmpEqual = [](const SharedLibrary& a, const SharedLibrary& b) { return std::tie(a.GetModuleName(), a.GetBreakpadId()) == std::tie(b.GetModuleName(), b.GetBreakpadId()); }; // std::unique requires the vector to be sorted first. It can only remove // consecutive duplicate elements. std::sort(mEntries.begin(), mEntries.end(), cmpSort); // Remove the duplicates since it's sorted now. mEntries.erase(std::unique(mEntries.begin(), mEntries.end(), cmpEqual), mEntries.end()); } void Clear() { mEntries.clear(); } size_t SizeOf() const { size_t size = 0; for (const auto& item : mEntries) { size += item.SizeOf(); } return size; } private: std::vector mEntries; friend struct IPC::ParamTraits; }; #endif // SHARED_LIBRARIES_H_