184 lines
5.3 KiB
C++
184 lines
5.3 KiB
C++
// 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 <sys/stat.h>
|
|
|
|
#include <cctype>
|
|
#include <cstddef>
|
|
#include <cstdlib>
|
|
#include <cstring>
|
|
#include <fstream> // NOLINT(readability/streams)
|
|
#include <iomanip>
|
|
#include <sstream>
|
|
|
|
#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<event_t*>(calloc(1, size));
|
|
CHECK(event);
|
|
return event;
|
|
}
|
|
|
|
build_id_event* CallocMemoryForBuildID(size_t size) {
|
|
build_id_event* event = reinterpret_cast<build_id_event*>(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<char>* 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<char>& 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
|