/* * 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. */ #ifndef SIMPLE_PERF_UTILS_H_ #define SIMPLE_PERF_UTILS_H_ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace simpleperf { static constexpr size_t kKilobyte = 1024; static constexpr size_t kMegabyte = 1024 * kKilobyte; static constexpr uint64_t kGigabyte = 1024 * kMegabyte; static inline uint64_t AlignDown(uint64_t value, uint64_t alignment) { return value & ~(alignment - 1); } static inline uint64_t Align(uint64_t value, uint64_t alignment) { return AlignDown(value + alignment - 1, alignment); } #ifdef _WIN32 #define CLOSE_ON_EXEC_MODE "" #define OS_PATH_SEPARATOR '\\' #else #define CLOSE_ON_EXEC_MODE "e" #define OS_PATH_SEPARATOR '/' #endif // OneTimeAllocator is used to allocate memory many times and free only once at the end. // It reduces the cost to free each allocated memory. class OneTimeFreeAllocator { public: explicit OneTimeFreeAllocator(size_t unit_size = 8192u) : unit_size_(unit_size), cur_(nullptr), end_(nullptr) {} ~OneTimeFreeAllocator() { Clear(); } void Clear(); const char* AllocateString(std::string_view s); private: const size_t unit_size_; std::vector v_; char* cur_; char* end_; }; class LineReader { public: explicit LineReader(std::string_view file_path) : ifs_(file_path) {} // Return true if open file successfully. bool Ok() const { return ifs_.good(); } // If available, return next line content with new line, otherwise return nullptr. std::string* ReadLine() { return (std::getline(ifs_, buf_)) ? &buf_ : nullptr; } private: std::ifstream ifs_; std::string buf_; }; class FileHelper { public: static android::base::unique_fd OpenReadOnly(const std::string& filename); static android::base::unique_fd OpenWriteOnly(const std::string& filename); }; class ArchiveHelper { public: static std::unique_ptr CreateInstance(const std::string& filename); ~ArchiveHelper(); // Iterate each entry in the zip file. Break the iteration when callback returns false. bool IterateEntries(const std::function& callback); bool FindEntry(const std::string& name, ZipEntry* entry); bool GetEntryData(ZipEntry& entry, std::vector* data); int GetFd(); private: ArchiveHelper(ZipArchiveHandle handle, const std::string& filename) : handle_(handle), filename_(filename) {} ZipArchiveHandle handle_; std::string filename_; DISALLOW_COPY_AND_ASSIGN(ArchiveHelper); }; template void MoveFromBinaryFormat(T& data, const char*& p) { static_assert(std::is_standard_layout::value, "not standard layout"); memcpy(&data, p, sizeof(T)); p += sizeof(T); } template void MoveFromBinaryFormat(T& data, char*& p) { static_assert(std::is_standard_layout::value, "not standard layout"); memcpy(&data, p, sizeof(T)); p += sizeof(T); } template void MoveFromBinaryFormat(T* data_p, size_t n, const char*& p) { static_assert(std::is_standard_layout::value, "not standard layout"); size_t size = n * sizeof(T); memcpy(data_p, p, size); p += size; } template void MoveToBinaryFormat(const T& data, char*& p) { static_assert(std::is_standard_layout::value, "not standard layout"); memcpy(p, &data, sizeof(T)); p += sizeof(T); } template void MoveToBinaryFormat(const T* data_p, size_t n, char*& p) { static_assert(std::is_standard_layout::value, "not standard layout"); size_t size = n * sizeof(T); memcpy(p, data_p, size); p += size; } // Read info from binary data. struct BinaryReader { public: BinaryReader(const char* head, size_t size) : head(head), end(head + size), error(false) {} size_t LeftSize() const { return end - head; } bool CheckLeftSize(size_t size) { if (UNLIKELY(error)) { return false; } if (UNLIKELY(LeftSize() < size)) { error = true; return false; } return true; } void Move(size_t size) { if (CheckLeftSize(size)) { head += size; } } template void Read(T& data) { static_assert(std::is_standard_layout::value, "not standard layout"); if (UNLIKELY(error)) { return; } if (UNLIKELY(LeftSize() < sizeof(T))) { error = true; } else { memcpy(&data, head, sizeof(T)); head += sizeof(T); } } template void Read(T* data_p, size_t n) { static_assert(std::is_standard_layout::value, "not standard layout"); if (UNLIKELY(error)) { return; } size_t size; if (UNLIKELY(__builtin_mul_overflow(n, sizeof(T), &size) || LeftSize() < size)) { error = true; } else { memcpy(data_p, head, size); head += size; } } // Read a string ending with '\0'. std::string ReadString() { if (UNLIKELY(error)) { return ""; } std::string result; while (head < end && *head != '\0') { result.push_back(*head++); } if (LIKELY(head < end && *head == '\0')) { head++; return result; } error = true; return ""; } const char* head; const char* end; bool error; }; void PrintIndented(size_t indent, const char* fmt, ...); void FprintIndented(FILE* fp, size_t indent, const char* fmt, ...); bool IsPowerOfTwo(uint64_t value); std::vector GetEntriesInDir(const std::string& dirpath); std::vector GetSubDirs(const std::string& dirpath); bool IsDir(const std::string& dirpath); bool IsRegularFile(const std::string& filename); uint64_t GetFileSize(const std::string& filename); bool MkdirWithParents(const std::string& path); bool XzDecompress(const std::string& compressed_data, std::string* decompressed_data); bool GetLogSeverity(const std::string& name, android::base::LogSeverity* severity); std::string GetLogSeverityName(); bool IsRoot(); size_t GetPageSize(); uint64_t ConvertBytesToValue(const char* bytes, uint32_t size); timeval SecondToTimeval(double time_in_sec); std::string GetSimpleperfVersion(); std::optional> GetCpusFromString(const std::string& s); std::optional> GetTidsFromString(const std::string& s, bool check_if_exists); std::optional> GetPidsFromStrings(const std::vector& strs, bool check_if_exists, bool support_progress_name_regex); template std::optional> ParseUintVector(const std::string& s) { std::set result; T value; for (const auto& p : android::base::Split(s, ",")) { if (!android::base::ParseUint(p.c_str(), &value, std::numeric_limits::max())) { LOG(ERROR) << "Invalid Uint '" << p << "' in " << s; return std::nullopt; } result.insert(value); } return result; } // from boost::hash_combine template static inline void HashCombine(size_t& seed, const T& val) { seed ^= std::hash()(val) + 0x9e3779b9 + (seed << 6) + (seed >> 2); } size_t SafeStrlen(const char* s, const char* end); struct OverflowResult { bool overflow = false; uint64_t value = 0; }; OverflowResult SafeAdd(uint64_t a, uint64_t b); void OverflowSafeAdd(uint64_t& dest, uint64_t add); std::string ReadableCount(uint64_t count); } // namespace simpleperf #endif // SIMPLE_PERF_UTILS_H_