xref: /aosp_15_r20/external/perfetto/src/profiling/common/profiler_guardrails.cc (revision 6dbdd20afdafa5e3ca9b8809fa73465d530080dc)
1*6dbdd20aSAndroid Build Coastguard Worker /*
2*6dbdd20aSAndroid Build Coastguard Worker  * Copyright (C) 2020 The Android Open Source Project
3*6dbdd20aSAndroid Build Coastguard Worker  *
4*6dbdd20aSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*6dbdd20aSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*6dbdd20aSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*6dbdd20aSAndroid Build Coastguard Worker  *
8*6dbdd20aSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*6dbdd20aSAndroid Build Coastguard Worker  *
10*6dbdd20aSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*6dbdd20aSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*6dbdd20aSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*6dbdd20aSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*6dbdd20aSAndroid Build Coastguard Worker  * limitations under the License.
15*6dbdd20aSAndroid Build Coastguard Worker  */
16*6dbdd20aSAndroid Build Coastguard Worker 
17*6dbdd20aSAndroid Build Coastguard Worker #include "src/profiling/common/profiler_guardrails.h"
18*6dbdd20aSAndroid Build Coastguard Worker 
19*6dbdd20aSAndroid Build Coastguard Worker #include <unistd.h>
20*6dbdd20aSAndroid Build Coastguard Worker #include <algorithm>
21*6dbdd20aSAndroid Build Coastguard Worker #include <optional>
22*6dbdd20aSAndroid Build Coastguard Worker 
23*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/file_utils.h"
24*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/scoped_file.h"
25*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/watchdog_posix.h"
26*6dbdd20aSAndroid Build Coastguard Worker 
27*6dbdd20aSAndroid Build Coastguard Worker namespace perfetto {
28*6dbdd20aSAndroid Build Coastguard Worker namespace profiling {
29*6dbdd20aSAndroid Build Coastguard Worker 
GetCputimeSecForCurrentProcess()30*6dbdd20aSAndroid Build Coastguard Worker std::optional<uint64_t> GetCputimeSecForCurrentProcess() {
31*6dbdd20aSAndroid Build Coastguard Worker   return GetCputimeSecForCurrentProcess(
32*6dbdd20aSAndroid Build Coastguard Worker       base::OpenFile("/proc/self/stat", O_RDONLY));
33*6dbdd20aSAndroid Build Coastguard Worker }
34*6dbdd20aSAndroid Build Coastguard Worker 
GetCputimeSecForCurrentProcess(base::ScopedFile stat_fd)35*6dbdd20aSAndroid Build Coastguard Worker std::optional<uint64_t> GetCputimeSecForCurrentProcess(
36*6dbdd20aSAndroid Build Coastguard Worker     base::ScopedFile stat_fd) {
37*6dbdd20aSAndroid Build Coastguard Worker   if (!stat_fd)
38*6dbdd20aSAndroid Build Coastguard Worker     return std::nullopt;
39*6dbdd20aSAndroid Build Coastguard Worker   base::ProcStat stat;
40*6dbdd20aSAndroid Build Coastguard Worker   if (!ReadProcStat(stat_fd.get(), &stat)) {
41*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_ELOG("Failed to read stat file to enforce guardrails.");
42*6dbdd20aSAndroid Build Coastguard Worker     return std::nullopt;
43*6dbdd20aSAndroid Build Coastguard Worker   }
44*6dbdd20aSAndroid Build Coastguard Worker   return (stat.utime + stat.stime) /
45*6dbdd20aSAndroid Build Coastguard Worker          static_cast<unsigned long>(sysconf(_SC_CLK_TCK));
46*6dbdd20aSAndroid Build Coastguard Worker }
47*6dbdd20aSAndroid Build Coastguard Worker 
ProfilerMemoryGuardrails()48*6dbdd20aSAndroid Build Coastguard Worker ProfilerMemoryGuardrails::ProfilerMemoryGuardrails()
49*6dbdd20aSAndroid Build Coastguard Worker     : ProfilerMemoryGuardrails(base::OpenFile("/proc/self/status", O_RDONLY)) {}
50*6dbdd20aSAndroid Build Coastguard Worker 
ProfilerMemoryGuardrails(base::ScopedFile status_fd)51*6dbdd20aSAndroid Build Coastguard Worker ProfilerMemoryGuardrails::ProfilerMemoryGuardrails(base::ScopedFile status_fd) {
52*6dbdd20aSAndroid Build Coastguard Worker   std::string status;
53*6dbdd20aSAndroid Build Coastguard Worker   if (base::ReadFileDescriptor(*status_fd, &status))
54*6dbdd20aSAndroid Build Coastguard Worker     anon_and_swap_ = GetRssAnonAndSwap(status);
55*6dbdd20aSAndroid Build Coastguard Worker 
56*6dbdd20aSAndroid Build Coastguard Worker   if (!anon_and_swap_) {
57*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_ELOG("Failed to read memory usage.");
58*6dbdd20aSAndroid Build Coastguard Worker     return;
59*6dbdd20aSAndroid Build Coastguard Worker   }
60*6dbdd20aSAndroid Build Coastguard Worker }
61*6dbdd20aSAndroid Build Coastguard Worker 
IsOverMemoryThreshold(const GuardrailConfig & ds)62*6dbdd20aSAndroid Build Coastguard Worker bool ProfilerMemoryGuardrails::IsOverMemoryThreshold(
63*6dbdd20aSAndroid Build Coastguard Worker     const GuardrailConfig& ds) {
64*6dbdd20aSAndroid Build Coastguard Worker   uint32_t ds_max_mem = ds.memory_guardrail_kb;
65*6dbdd20aSAndroid Build Coastguard Worker   if (!ds_max_mem || !anon_and_swap_)
66*6dbdd20aSAndroid Build Coastguard Worker     return false;
67*6dbdd20aSAndroid Build Coastguard Worker 
68*6dbdd20aSAndroid Build Coastguard Worker   if (ds_max_mem > 0 && *anon_and_swap_ > ds_max_mem) {
69*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_ELOG("Exceeded data-source memory guardrail (%" PRIu32
70*6dbdd20aSAndroid Build Coastguard Worker                   " > %" PRIu32 "). Shutting down.",
71*6dbdd20aSAndroid Build Coastguard Worker                   *anon_and_swap_, ds_max_mem);
72*6dbdd20aSAndroid Build Coastguard Worker     return true;
73*6dbdd20aSAndroid Build Coastguard Worker   }
74*6dbdd20aSAndroid Build Coastguard Worker   return false;
75*6dbdd20aSAndroid Build Coastguard Worker }
76*6dbdd20aSAndroid Build Coastguard Worker 
ProfilerCpuGuardrails()77*6dbdd20aSAndroid Build Coastguard Worker ProfilerCpuGuardrails::ProfilerCpuGuardrails() {
78*6dbdd20aSAndroid Build Coastguard Worker   opt_cputime_sec_ = GetCputimeSecForCurrentProcess();
79*6dbdd20aSAndroid Build Coastguard Worker   if (!opt_cputime_sec_) {
80*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_ELOG("Failed to get CPU time.");
81*6dbdd20aSAndroid Build Coastguard Worker   }
82*6dbdd20aSAndroid Build Coastguard Worker }
83*6dbdd20aSAndroid Build Coastguard Worker 
84*6dbdd20aSAndroid Build Coastguard Worker // For testing.
ProfilerCpuGuardrails(base::ScopedFile stat_fd)85*6dbdd20aSAndroid Build Coastguard Worker ProfilerCpuGuardrails::ProfilerCpuGuardrails(base::ScopedFile stat_fd) {
86*6dbdd20aSAndroid Build Coastguard Worker   opt_cputime_sec_ = GetCputimeSecForCurrentProcess(std::move(stat_fd));
87*6dbdd20aSAndroid Build Coastguard Worker   if (!opt_cputime_sec_) {
88*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_ELOG("Failed to get CPU time.");
89*6dbdd20aSAndroid Build Coastguard Worker   }
90*6dbdd20aSAndroid Build Coastguard Worker }
91*6dbdd20aSAndroid Build Coastguard Worker 
IsOverCpuThreshold(const GuardrailConfig & ds)92*6dbdd20aSAndroid Build Coastguard Worker bool ProfilerCpuGuardrails::IsOverCpuThreshold(const GuardrailConfig& ds) {
93*6dbdd20aSAndroid Build Coastguard Worker   uint64_t ds_max_cpu = ds.cpu_guardrail_sec;
94*6dbdd20aSAndroid Build Coastguard Worker   if (!ds_max_cpu || !opt_cputime_sec_)
95*6dbdd20aSAndroid Build Coastguard Worker     return false;
96*6dbdd20aSAndroid Build Coastguard Worker   uint64_t cputime_sec = *opt_cputime_sec_;
97*6dbdd20aSAndroid Build Coastguard Worker 
98*6dbdd20aSAndroid Build Coastguard Worker   auto start_cputime_sec = ds.cpu_start_secs;
99*6dbdd20aSAndroid Build Coastguard Worker   // We reject data-sources with CPU guardrails if we cannot read the
100*6dbdd20aSAndroid Build Coastguard Worker   // initial value, which means we get a non-nullopt value here.
101*6dbdd20aSAndroid Build Coastguard Worker   PERFETTO_CHECK(start_cputime_sec);
102*6dbdd20aSAndroid Build Coastguard Worker   uint64_t cpu_diff = cputime_sec - *start_cputime_sec;
103*6dbdd20aSAndroid Build Coastguard Worker   if (cputime_sec > *start_cputime_sec && cpu_diff > ds_max_cpu) {
104*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_ELOG("Exceeded data-source CPU guardrail (%" PRIu64 " > %" PRIu64
105*6dbdd20aSAndroid Build Coastguard Worker                   "). Shutting down.",
106*6dbdd20aSAndroid Build Coastguard Worker                   cpu_diff, ds_max_cpu);
107*6dbdd20aSAndroid Build Coastguard Worker     return true;
108*6dbdd20aSAndroid Build Coastguard Worker   }
109*6dbdd20aSAndroid Build Coastguard Worker   return false;
110*6dbdd20aSAndroid Build Coastguard Worker }
111*6dbdd20aSAndroid Build Coastguard Worker 
112*6dbdd20aSAndroid Build Coastguard Worker }  // namespace profiling
113*6dbdd20aSAndroid Build Coastguard Worker }  // namespace perfetto
114