// Copyright (c) 2012 The Chromium OS Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #define LOG_TAG "perf_reader" #include "perf_utils.h" #include #include #include #include #include #include // NOLINT(readability/streams) #include #include #include "base/logging.h" #include "base/macros.h" namespace { // Number of hex digits in a byte. const int kNumHexDigitsInByte = 2; } // namespace namespace quipper { event_t* CallocMemoryForEvent(size_t size) { event_t* event = reinterpret_cast(calloc(1, size)); CHECK(event); return event; } build_id_event* CallocMemoryForBuildID(size_t size) { build_id_event* event = reinterpret_cast(calloc(1, size)); CHECK(event); return event; } string HexToString(const u8* array, size_t length) { // Convert the bytes to hex digits one at a time. // There will be kNumHexDigitsInByte hex digits, and 1 char for NUL. char buffer[kNumHexDigitsInByte + 1]; string result = ""; for (size_t i = 0; i < length; ++i) { snprintf(buffer, sizeof(buffer), "%02x", array[i]); result += buffer; } return result; } bool StringToHex(const string& str, u8* array, size_t length) { const int kHexRadix = 16; char* err; // Loop through kNumHexDigitsInByte characters at a time (to get one byte) // Stop when there are no more characters, or the array has been filled. for (size_t i = 0; (i + 1) * kNumHexDigitsInByte <= str.size() && i < length; ++i) { string one_byte = str.substr(i * kNumHexDigitsInByte, kNumHexDigitsInByte); array[i] = strtol(one_byte.c_str(), &err, kHexRadix); if (*err) return false; } return true; } uint64_t AlignSize(uint64_t size, uint32_t align_size) { return ((size + align_size - 1) / align_size) * align_size; } // In perf data, strings are packed into the smallest number of 8-byte blocks // possible, including the null terminator. // e.g. // "0123" -> 5 bytes -> packed into 8 bytes // "0123456" -> 8 bytes -> packed into 8 bytes // "01234567" -> 9 bytes -> packed into 16 bytes // "0123456789abcd" -> 15 bytes -> packed into 16 bytes // "0123456789abcde" -> 16 bytes -> packed into 16 bytes // "0123456789abcdef" -> 17 bytes -> packed into 24 bytes // // Returns the size of the 8-byte-aligned memory for storing |string|. size_t GetUint64AlignedStringLength(const string& str) { return AlignSize(str.size() + 1, sizeof(uint64_t)); } uint64_t GetSampleFieldsForEventType(uint32_t event_type, uint64_t sample_type) { uint64_t mask = kuint64max; switch (event_type) { case PERF_RECORD_MMAP: case PERF_RECORD_LOST: case PERF_RECORD_COMM: case PERF_RECORD_EXIT: case PERF_RECORD_THROTTLE: case PERF_RECORD_UNTHROTTLE: case PERF_RECORD_FORK: case PERF_RECORD_READ: case PERF_RECORD_MMAP2: // See perf_event.h "struct" sample_id and sample_id_all. mask = PERF_SAMPLE_TID | PERF_SAMPLE_TIME | PERF_SAMPLE_ID | PERF_SAMPLE_STREAM_ID | PERF_SAMPLE_CPU | PERF_SAMPLE_IDENTIFIER; break; case PERF_RECORD_SAMPLE: case SIMPLE_PERF_RECORD_KERNEL_SYMBOL: break; default: LOG(FATAL) << "Unknown event type " << event_type; } return sample_type & mask; } uint64_t GetPerfSampleDataOffset(const event_t& event) { uint64_t offset = kuint64max; switch (event.header.type) { case PERF_RECORD_SAMPLE: offset = offsetof(event_t, sample.array); break; case PERF_RECORD_MMAP: offset = sizeof(event.mmap) - sizeof(event.mmap.filename) + GetUint64AlignedStringLength(event.mmap.filename); break; case PERF_RECORD_FORK: case PERF_RECORD_EXIT: offset = sizeof(event.fork); break; case PERF_RECORD_COMM: offset = sizeof(event.comm) - sizeof(event.comm.comm) + GetUint64AlignedStringLength(event.comm.comm); break; case PERF_RECORD_LOST: offset = sizeof(event.lost); break; case PERF_RECORD_READ: offset = sizeof(event.read); break; case PERF_RECORD_MMAP2: offset = sizeof(event.mmap2) - sizeof(event.mmap2.filename) + GetUint64AlignedStringLength(event.mmap2.filename); break; case SIMPLE_PERF_RECORD_KERNEL_SYMBOL: offset = 0; break; default: LOG(FATAL) << "Unknown/unsupported event type " << event.header.type; break; } // Make sure the offset was valid CHECK_NE(offset, kuint64max); CHECK_EQ(offset % sizeof(uint64_t), 0U); return offset; } bool ReadFileToData(const string& filename, std::vector* data) { std::ifstream in(filename.c_str(), std::ios::binary); if (!in.good()) { LOG(ERROR) << "Failed to open file " << filename; return false; } in.seekg(0, in.end); size_t length = in.tellg(); in.seekg(0, in.beg); data->resize(length); in.read(&(*data)[0], length); if (!in.good()) { LOG(ERROR) << "Error reading from file " << filename; return false; } return true; } bool WriteDataToFile(const std::vector& data, const string& filename) { std::ofstream out(filename.c_str(), std::ios::binary); out.seekp(0, std::ios::beg); out.write(&data[0], data.size()); return out.good(); } } // namespace quipper