blob: 452ea75971ec6b47b94df91186d3b23851822c45 [file]
/*
* Copyright 2016, 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 "util.h"
#include <dirent.h>
#include <stdio.h>
#include <string.h>
#include <time.h>
#include <unistd.h>
#include <functional>
#include <string>
#include <unordered_map>
#include <utility>
#include <vector>
#include <android-base/file.h>
#include <android-base/properties.h>
#include <android-base/stringprintf.h>
#include <android-base/strings.h>
#include "protocol.h"
constexpr const char kUnknown[] = "<unknown>";
std::vector<std::string> get_command_line(pid_t pid) {
std::vector<std::string> result;
std::string cmdline;
android::base::ReadFileToString(android::base::StringPrintf("/proc/%d/cmdline", pid), &cmdline);
auto it = cmdline.cbegin();
while (it != cmdline.cend()) {
// string::iterator is a wrapped type, not a raw char*.
auto terminator = std::find(it, cmdline.cend(), '\0');
result.emplace_back(it, terminator);
it = std::find_if(terminator, cmdline.cend(), [](char c) { return c != '\0'; });
}
if (result.empty()) {
return {kUnknown};
}
return result;
}
std::string get_process_name(pid_t pid) {
std::string result(kUnknown);
android::base::ReadFileToString(android::base::StringPrintf("/proc/%d/cmdline", pid), &result);
// We only want the name, not the whole command line, so truncate at the first NUL.
return result.c_str();
}
std::string get_thread_name(pid_t tid) {
std::string result(kUnknown);
android::base::ReadFileToString(android::base::StringPrintf("/proc/%d/comm", tid), &result);
return android::base::Trim(result);
}
std::string get_executable_name(pid_t pid) {
std::string result;
if (!android::base::Readlink(android::base::StringPrintf("/proc/%d/exe", pid), &result)) {
return kUnknown;
}
return result;
}
std::string get_timestamp() {
timespec ts;
clock_gettime(CLOCK_REALTIME, &ts);
tm tm;
localtime_r(&ts.tv_sec, &tm);
char buf[strlen("1970-01-01 00:00:00.123456789+0830") + 1];
char* s = buf;
size_t sz = sizeof(buf), n;
n = strftime(s, sz, "%F %H:%M", &tm), s += n, sz -= n;
n = snprintf(s, sz, ":%02d.%09ld", tm.tm_sec, ts.tv_nsec), s += n, sz -= n;
n = strftime(s, sz, "%z", &tm), s += n, sz -= n;
return buf;
}
bool iterate_tids(pid_t pid, std::function<void(pid_t)> callback) {
char buf[BUFSIZ];
snprintf(buf, sizeof(buf), "/proc/%d/task", pid);
std::unique_ptr<DIR, int (*)(DIR*)> dir(opendir(buf), closedir);
if (dir == nullptr) {
return false;
}
struct dirent* entry;
while ((entry = readdir(dir.get())) != nullptr) {
pid_t tid = atoi(entry->d_name);
if (tid == 0) {
continue;
}
callback(tid);
}
return true;
}
bool is_microdroid() {
return android::base::GetProperty("ro.hardware", "") == "microdroid";
}
void get_vmflags(pid_t pid, std::unordered_map<uint64_t, std::string>& vmflags) {
// See if any maps have other flags enabled. The only way to get this
// information is to read the smaps file and check the VmFlags field.
std::string smaps_str(android::base::StringPrintf("/proc/%d/smaps", pid));
FILE* smaps_file = fopen(smaps_str.c_str(), "re");
if (smaps_file == nullptr) {
return;
}
// Choose a size that should be big enough for all lines and avoid ever
// doing a realloc.
constexpr size_t kLineSize = 200;
char* buffer = reinterpret_cast<char*>(malloc(kLineSize));
size_t allocated_length = kLineSize;
uint64_t map_start = 0;
while (true) {
ssize_t buffer_len = getline(&buffer, &allocated_length, smaps_file);
if (buffer_len <= 0) {
break;
}
char* end = nullptr;
uint64_t tmp_start = strtoul(buffer, &end, 16);
if (end != nullptr && *end == '-') {
map_start = tmp_start;
continue;
}
constexpr char kVmFlagsStr[] = "VmFlags:";
constexpr size_t kVmFlagsStrLen = std::char_traits<char>::length(kVmFlagsStr);
if (strncmp(buffer, kVmFlagsStr, kVmFlagsStrLen) == 0) {
// Skip spaces in front of string.
size_t flag_start = kVmFlagsStrLen + 1;
while (isspace(buffer[flag_start])) {
++flag_start;
}
// Remove spaces at end of string.
while (isspace(buffer[--buffer_len])) {
buffer[buffer_len] = '\0';
}
vmflags[map_start] = std::string(&buffer[flag_start]);
map_start = 0;
}
}
free(buffer);
fclose(smaps_file);
}