// Copyright 2016 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.
#include "dso.h"
#include <elf.h>
#include <fcntl.h>
#include <string.h>
#include <sys/stat.h>
#include <sys/sysmacros.h>
#include <unistd.h>
#include <algorithm>
#include <vector>
#include <build_id.h>
#include <read_elf.h>
#include "base/logging.h"
#include "compat/string.h"
#include "file_reader.h"
namespace quipper {
void InitializeLibelf() {
// Unnecessary.
}
bool ReadElfBuildId(const string &filename, string *buildid) {
BuildId id;
ElfStatus status = GetBuildIdFromElfFile(filename, &id);
if (status == ElfStatus::NO_ERROR) {
*buildid = id.ToString();
return true;
}
return false;
}
bool ReadElfBuildId(int fd, string *buildid) {
// TODO: Implement. b/74410255.
return false;
}
// read /sys/module/<module_name>/notes/.note.gnu.build-id
bool ReadModuleBuildId(const string &module_name, string *buildid) {
string note_filename =
"/sys/module/" + module_name + "/notes/.note.gnu.build-id";
FileReader file(note_filename);
if (!file.IsOpen()) return false;
return ReadBuildIdNote(&file, buildid);
}
bool ReadBuildIdNote(DataReader *data, string *buildid) {
// Non-simpleperf implementation, as a reader is given.
Elf64_Nhdr note_header;
while (data->ReadData(sizeof(note_header), ¬e_header)) {
size_t name_size = Align<4>(note_header.n_namesz);
size_t desc_size = Align<4>(note_header.n_descsz);
string name;
if (!data->ReadString(name_size, &name)) return false;
string desc;
if (!data->ReadDataString(desc_size, &desc)) return false;
if (note_header.n_type == NT_GNU_BUILD_ID && name == ELF_NOTE_GNU) {
*buildid = desc;
return true;
}
}
return false;
}
bool IsKernelNonModuleName(string name) {
// List from kernel: tools/perf/util/dso.c : __kmod_path__parse()
static const std::vector<string> kKernelNonModuleNames{
"[kernel.kallsyms]",
"[guest.kernel.kallsyms",
"[vdso]",
"[vsyscall]",
};
for (const auto &n : kKernelNonModuleNames) {
if (name.compare(0, n.size(), n) == 0) return true;
}
return false;
}
// Do the |DSOInfo| and |struct stat| refer to the same inode?
bool SameInode(const DSOInfo &dso, const struct stat *s) {
return dso.maj == major(s->st_dev) && dso.min == minor(s->st_dev) &&
dso.ino == s->st_ino;
}
} // namespace quipper