普通文本  |  137行  |  3.87 KB

// Copyright 2017 The Chromium 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 "procfs_utils.h"

#include <stdio.h>
#include <string.h>

#include "file_utils.h"
#include "logging.h"

using file_utils::ForEachPidInProcPath;
using file_utils::ReadProcFile;
using file_utils::ReadProcFileTrimmed;

namespace procfs_utils {

namespace {

const char kJavaAppPrefix[] = "/system/bin/app_process";
const char kZygotePrefix[] = "zygote";

inline void ReadProcString(int pid, const char* path, char* buf, size_t size) {
  if (!file_utils::ReadProcFileTrimmed(pid, path, buf, size))
    buf[0] = '\0';
}

inline void ReadExePath(int pid, char* buf, size_t size) {
  char exe_path[64];
  sprintf(exe_path, "/proc/%d/exe", pid);
  ssize_t res = readlink(exe_path, buf, size - 1);
  if (res >= 0)
    buf[res] = '\0';
  else
    buf[0] = '\0';
}

inline bool IsApp(const char* name, const char* exe) {
  return strncmp(exe, kJavaAppPrefix, sizeof(kJavaAppPrefix) - 1) == 0 &&
         strncmp(name, kZygotePrefix, sizeof(kZygotePrefix) - 1) != 0;
}

}  // namespace

int ReadTgid(int pid) {
  static const char kTgid[] = "\nTgid:";
  char buf[512];
  ssize_t rsize = ReadProcFile(pid, "status", buf, sizeof(buf));
  if (rsize <= 0)
    return -1;
  const char* tgid_line = strstr(buf, kTgid);
  CHECK(tgid_line);
  return atoi(tgid_line + sizeof(kTgid) - 1);
}

std::unique_ptr<ProcessInfo> ReadProcessInfo(int pid) {
  ProcessInfo* process = new ProcessInfo();
  process->pid = pid;
  ReadProcString(pid, "cmdline", process->name, sizeof(process->name));
  if (process->name[0] != 0) {
    ReadExePath(pid, process->exe, sizeof(process->exe));
    process->is_app = IsApp(process->name, process->exe);
  } else {
    ReadProcString(pid, "comm", process->name, sizeof(process->name));
    CHECK(process->name[0]);
    process->in_kernel = true;
  }
  return std::unique_ptr<ProcessInfo>(process);
}

void ReadProcessThreads(ProcessInfo* process) {
  if (process->in_kernel)
    return;

  char tasks_path[64];
  sprintf(tasks_path, "/proc/%d/task", process->pid);
  ForEachPidInProcPath(tasks_path, [process](int tid) {
    if (process->threads.count(tid))
      return;
    ThreadInfo thread = { tid, "" };
    char task_comm[64];
    sprintf(task_comm, "task/%d/comm", tid);
    ReadProcString(process->pid, task_comm, thread.name, sizeof(thread.name));
    if (thread.name[0] == '\0' && process->is_app)
      strcpy(thread.name, "UI Thread");
    process->threads[tid] = thread;
  });
}

bool ReadOomStats(ProcessSnapshot* snapshot) {
  char buf[64];
  if (ReadProcFileTrimmed(snapshot->pid, "oom_score", buf, sizeof(buf)))
    snapshot->oom_score = atoi(buf);
  else
    return false;
  if (ReadProcFileTrimmed(snapshot->pid, "oom_score_adj", buf, sizeof(buf)))
    snapshot->oom_score_adj = atoi(buf);
  else
    return false;
  return true;
}

bool ReadPageFaultsAndCpuTimeStats(ProcessSnapshot* snapshot) {
  char buf[512];
  if (!ReadProcFileTrimmed(snapshot->pid, "stat", buf, sizeof(buf)))
    return false;
  int ret = sscanf(buf,
      "%*d (%*[^)]) %*c %*d %*d %*d %*d %*d %*u %lu %*lu %lu %*lu %lu %lu",
      &snapshot->minor_faults, &snapshot->major_faults,
      &snapshot->utime, &snapshot->stime);
  CHECK(ret == 4);
  return true;
}

bool ReadMemInfoStats(std::map<std::string, uint64_t>* mem_info) {
  char buf[1024];
  ssize_t rsize = file_utils::ReadFile("/proc/meminfo", buf, sizeof(buf));
  if (rsize <= 0)
    return false;

  file_utils::LineReader reader(buf, rsize);
  for (const char* line = reader.NextLine();
       line && line[0];
       line = reader.NextLine()) {

    const char* pos_colon = strstr(line, ":");
    if (pos_colon == nullptr)
      continue;  // Should not happen.
    std::string name(line, pos_colon - line);
    (*mem_info)[name] = strtoull(&pos_colon[1], nullptr, 10);
  }
  return true;
}

}  // namespace procfs_utils