blob: 332d9934e32532e2f9015a3e9107b1e54bf24e9b [file] [log] [blame]
/*
* Copyright (C) 2017 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.
*/
#include "ftrace_procfs.h"
#include <fcntl.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <fstream>
#include <sstream>
#include <string>
#include "perfetto_base/logging.h"
#include "perfetto_base/utils.h"
namespace perfetto {
namespace {
// Reading /trace produces human readable trace output.
// Writing to this file clears all trace buffers for all CPUS.
// Writing to /trace_marker file injects an event into the trace buffer.
// Reading /tracing_on returns 1/0 if tracing is enabled/disabled.
// Writing 1/0 to this file enables/disables tracing.
// Disabling tracing with this file prevents further writes but
// does not clear the buffer.
char ReadOneCharFromFile(const std::string& path) {
base::ScopedFile fd(open(path.c_str(), O_RDONLY));
PERFETTO_CHECK(fd);
char result = '\0';
ssize_t bytes = PERFETTO_EINTR(read(fd.get(), &result, 1));
PERFETTO_CHECK(bytes == 1 || bytes == -1);
return result;
}
std::string ReadFileIntoString(std::string path) {
std::ifstream fin(path, std::ios::in);
if (!fin) {
PERFETTO_DLOG("Could not read '%s'", path.c_str());
return "";
}
std::string str;
// You can't seek or stat the procfs files on Android.
// The vast majority (884/886) of format files are under 4k.
str.reserve(4096);
str.assign(std::istreambuf_iterator<char>(fin),
std::istreambuf_iterator<char>());
return str;
}
} // namespace
FtraceProcfs::FtraceProcfs(const std::string& root) : root_(root) {}
FtraceProcfs::~FtraceProcfs() = default;
bool FtraceProcfs::EnableEvent(const std::string& group,
const std::string& name) {
std::string path = root_ + "events/" + group + "/" + name + "/enable";
return WriteToFile(path, "1");
}
bool FtraceProcfs::DisableEvent(const std::string& group,
const std::string& name) {
std::string path = root_ + "events/" + group + "/" + name + "/enable";
return WriteToFile(path, "0");
}
std::string FtraceProcfs::ReadEventFormat(const std::string& group,
const std::string& name) const {
std::string path = root_ + "events/" + group + "/" + name + "/format";
return ReadFileIntoString(path);
}
std::string FtraceProcfs::ReadAvailableEvents() const {
std::string path = root_ + "available_events";
return ReadFileIntoString(path);
}
size_t FtraceProcfs::NumberOfCpus() const {
static size_t num_cpus = sysconf(_SC_NPROCESSORS_CONF);
return num_cpus;
}
void FtraceProcfs::ClearTrace() {
std::string path = root_ + "trace";
base::ScopedFile fd(open(path.c_str(), O_WRONLY | O_TRUNC));
PERFETTO_CHECK(fd); // Could not clear.
}
bool FtraceProcfs::WriteTraceMarker(const std::string& str) {
std::string path = root_ + "trace_marker";
return WriteToFile(path, str);
}
bool FtraceProcfs::EnableTracing() {
std::string path = root_ + "tracing_on";
return WriteToFile(path, "1");
}
bool FtraceProcfs::DisableTracing() {
std::string path = root_ + "tracing_on";
return WriteToFile(path, "0");
}
bool FtraceProcfs::IsTracingEnabled() {
std::string path = root_ + "tracing_on";
return ReadOneCharFromFile(path) == '1';
}
bool FtraceProcfs::WriteToFile(const std::string& path,
const std::string& str) {
base::ScopedFile fd(open(path.c_str(), O_WRONLY));
if (!fd)
return false;
ssize_t written = PERFETTO_EINTR(write(fd.get(), str.c_str(), str.length()));
ssize_t length = static_cast<ssize_t>(str.length());
// This should either fail or write fully.
PERFETTO_CHECK(written == length || written == -1);
return written == length;
}
base::ScopedFile FtraceProcfs::OpenPipeForCpu(size_t cpu) {
std::string path =
root_ + "per_cpu/cpu" + std::to_string(cpu) + "/trace_pipe_raw";
return base::ScopedFile(open(path.c_str(), O_RDONLY));
}
} // namespace perfetto