1 /*
2 * Copyright (C) 2021 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 #include "reporter.h"
18
19 #include <android-base/parseint.h>
20
21 #include <algorithm>
22
23 #include "base/flags.h"
24 #include "base/stl_util.h"
25 #include "oat/oat_file_manager.h"
26 #include "runtime.h"
27 #include "runtime_options.h"
28 #include "statsd.h"
29 #include "thread-current-inl.h"
30
31 #pragma clang diagnostic push
32 #pragma clang diagnostic error "-Wconversion"
33
34 namespace art HIDDEN {
35 namespace metrics {
36
Create(const ReportingConfig & config,Runtime * runtime)37 std::unique_ptr<MetricsReporter> MetricsReporter::Create(
38 const ReportingConfig& config, Runtime* runtime) {
39 // We can't use std::make_unique here because the MetricsReporter constructor is private.
40 return std::unique_ptr<MetricsReporter>{new MetricsReporter{std::move(config), runtime}};
41 }
42
MetricsReporter(const ReportingConfig & config,Runtime * runtime)43 MetricsReporter::MetricsReporter(const ReportingConfig& config, Runtime* runtime)
44 : config_{config},
45 runtime_{runtime},
46 startup_reported_{false},
47 report_interval_index_{0} {}
48
~MetricsReporter()49 MetricsReporter::~MetricsReporter() { MaybeStopBackgroundThread(); }
50
ReloadConfig(const ReportingConfig & config)51 void MetricsReporter::ReloadConfig(const ReportingConfig& config) {
52 DCHECK(!thread_.has_value()) << "The config cannot be reloaded after the background "
53 "reporting thread is started.";
54 config_ = config;
55 }
56
IsMetricsReportingEnabled(const SessionData & session_data) const57 bool MetricsReporter::IsMetricsReportingEnabled(const SessionData& session_data) const {
58 return session_data.session_id % config_.reporting_num_mods < config_.reporting_mods;
59 }
60
MaybeStartBackgroundThread(SessionData session_data)61 bool MetricsReporter::MaybeStartBackgroundThread(SessionData session_data) {
62 CHECK(!thread_.has_value());
63
64 session_data_ = session_data;
65 LOG_STREAM(DEBUG) << "Received session metadata: " << session_data_.session_id;
66
67 if (!IsMetricsReportingEnabled(session_data_)) {
68 return false;
69 }
70
71 thread_.emplace(&MetricsReporter::BackgroundThreadRun, this);
72 return true;
73 }
74
MaybeStopBackgroundThread()75 void MetricsReporter::MaybeStopBackgroundThread() {
76 if (thread_.has_value()) {
77 messages_.SendMessage(ShutdownRequestedMessage{});
78 thread_->join();
79 thread_.reset();
80 }
81 }
82
NotifyStartupCompleted()83 void MetricsReporter::NotifyStartupCompleted() {
84 if (ShouldReportAtStartup() && thread_.has_value()) {
85 messages_.SendMessage(StartupCompletedMessage{});
86 }
87 }
88
NotifyAppInfoUpdated(AppInfo * app_info)89 void MetricsReporter::NotifyAppInfoUpdated(AppInfo* app_info) {
90 std::string compilation_reason;
91 std::string compiler_filter;
92
93 app_info->GetPrimaryApkOptimizationStatus(
94 &compiler_filter, &compilation_reason);
95
96 SetCompilationInfo(
97 CompilationReasonFromName(compilation_reason),
98 CompilerFilterReportingFromName(compiler_filter));
99 }
100
RequestMetricsReport(bool synchronous)101 void MetricsReporter::RequestMetricsReport(bool synchronous) {
102 if (thread_.has_value()) {
103 messages_.SendMessage(RequestMetricsReportMessage{synchronous});
104 if (synchronous) {
105 thread_to_host_messages_.ReceiveMessage();
106 }
107 }
108 }
109
SetCompilationInfo(CompilationReason compilation_reason,CompilerFilterReporting compiler_filter)110 void MetricsReporter::SetCompilationInfo(CompilationReason compilation_reason,
111 CompilerFilterReporting compiler_filter) {
112 if (thread_.has_value()) {
113 messages_.SendMessage(CompilationInfoMessage{compilation_reason, compiler_filter});
114 }
115 }
116
BackgroundThreadRun()117 void MetricsReporter::BackgroundThreadRun() {
118 LOG_STREAM(DEBUG) << "Metrics reporting thread started";
119
120 // AttachCurrentThread is needed so we can safely use the ART concurrency primitives within the
121 // messages_ MessageQueue.
122 const bool attached = runtime_->AttachCurrentThread(kBackgroundThreadName,
123 /*as_daemon=*/true,
124 runtime_->GetSystemThreadGroup(),
125 /*create_peer=*/true);
126 bool running = true;
127
128 // Configure the backends
129 if (config_.dump_to_logcat) {
130 backends_.emplace_back(new LogBackend(std::make_unique<TextFormatter>(), LogSeverity::INFO));
131 }
132 if (config_.dump_to_file.has_value()) {
133 std::unique_ptr<MetricsFormatter> formatter;
134 if (config_.metrics_format == "xml") {
135 formatter = std::make_unique<XmlFormatter>();
136 } else {
137 formatter = std::make_unique<TextFormatter>();
138 }
139
140 backends_.emplace_back(new FileBackend(std::move(formatter), config_.dump_to_file.value()));
141 }
142 if (config_.dump_to_statsd) {
143 auto backend = CreateStatsdBackend();
144 if (backend != nullptr) {
145 backends_.emplace_back(std::move(backend));
146 }
147 }
148
149 MaybeResetTimeout();
150
151 while (running) {
152 messages_.SwitchReceive(
153 [&]([[maybe_unused]] ShutdownRequestedMessage message) {
154 LOG_STREAM(DEBUG) << "Shutdown request received " << session_data_.session_id;
155 running = false;
156
157 ReportMetrics();
158 },
159 [&](RequestMetricsReportMessage message) {
160 LOG_STREAM(DEBUG) << "Explicit report request received " << session_data_.session_id;
161 ReportMetrics();
162 if (message.synchronous) {
163 thread_to_host_messages_.SendMessage(ReportCompletedMessage{});
164 }
165 },
166 [&]([[maybe_unused]] TimeoutExpiredMessage message) {
167 LOG_STREAM(DEBUG) << "Timer expired, reporting metrics " << session_data_.session_id;
168
169 ReportMetrics();
170 MaybeResetTimeout();
171 },
172 [&]([[maybe_unused]] StartupCompletedMessage message) {
173 LOG_STREAM(DEBUG) << "App startup completed, reporting metrics "
174 << session_data_.session_id;
175 ReportMetrics();
176 startup_reported_ = true;
177 MaybeResetTimeout();
178 },
179 [&](CompilationInfoMessage message) {
180 LOG_STREAM(DEBUG) << "Compilation info received " << session_data_.session_id;
181 session_data_.compilation_reason = message.compilation_reason;
182 session_data_.compiler_filter = message.compiler_filter;
183
184 UpdateSessionInBackends();
185 });
186 }
187
188 if (attached) {
189 runtime_->DetachCurrentThread();
190 }
191 LOG_STREAM(DEBUG) << "Metrics reporting thread terminating " << session_data_.session_id;
192 }
193
MaybeResetTimeout()194 void MetricsReporter::MaybeResetTimeout() {
195 if (ShouldContinueReporting()) {
196 messages_.SetTimeout(SecondsToMs(GetNextPeriodSeconds()));
197 }
198 }
199
GetMetrics()200 ArtMetrics* MetricsReporter::GetMetrics() { return runtime_->GetMetrics(); }
201
ReportMetrics()202 void MetricsReporter::ReportMetrics() {
203 ArtMetrics* metrics = GetMetrics();
204
205 if (!session_started_) {
206 for (auto& backend : backends_) {
207 backend->BeginOrUpdateSession(session_data_);
208 }
209 session_started_ = true;
210 }
211
212 metrics->ReportAllMetricsAndResetValueMetrics(MakeNonOwningPointerVector(backends_));
213 }
214
UpdateSessionInBackends()215 void MetricsReporter::UpdateSessionInBackends() {
216 if (session_started_) {
217 for (auto& backend : backends_) {
218 backend->BeginOrUpdateSession(session_data_);
219 }
220 }
221 }
222
ShouldReportAtStartup() const223 bool MetricsReporter::ShouldReportAtStartup() const {
224 return IsMetricsReportingEnabled(session_data_) &&
225 config_.period_spec.has_value() &&
226 config_.period_spec->report_startup_first;
227 }
228
ShouldContinueReporting() const229 bool MetricsReporter::ShouldContinueReporting() const {
230 bool result =
231 // Only if the reporting is enabled
232 IsMetricsReportingEnabled(session_data_) &&
233 // and if we have period spec
234 config_.period_spec.has_value() &&
235 // and the periods are non empty
236 !config_.period_spec->periods_seconds.empty() &&
237 // and we already reported startup or not required to report startup
238 (startup_reported_ || !config_.period_spec->report_startup_first) &&
239 // and we still have unreported intervals or we are asked to report continuously.
240 (config_.period_spec->continuous_reporting ||
241 (report_interval_index_ < config_.period_spec->periods_seconds.size()));
242 return result;
243 }
244
GetNextPeriodSeconds()245 uint32_t MetricsReporter::GetNextPeriodSeconds() {
246 DCHECK(ShouldContinueReporting());
247
248 // The index is either the current report_interval_index or the last index
249 // if we are in continuous mode and reached the end.
250 uint32_t index = std::min(
251 report_interval_index_,
252 static_cast<uint32_t>(config_.period_spec->periods_seconds.size() - 1));
253
254 uint32_t result = config_.period_spec->periods_seconds[index];
255
256 // Advance the index if we didn't get to the end.
257 if (report_interval_index_ < config_.period_spec->periods_seconds.size()) {
258 report_interval_index_++;
259 }
260 return result;
261 }
262
FromFlags(bool is_system_server)263 ReportingConfig ReportingConfig::FromFlags(bool is_system_server) {
264 std::optional<std::string> spec_str = is_system_server
265 ? gFlags.MetricsReportingSpecSystemServer.GetValueOptional()
266 : gFlags.MetricsReportingSpec.GetValueOptional();
267
268 std::optional<ReportingPeriodSpec> period_spec = std::nullopt;
269
270 if (spec_str.has_value()) {
271 std::string error;
272 period_spec = ReportingPeriodSpec::Parse(spec_str.value(), &error);
273 if (!period_spec.has_value()) {
274 LOG(ERROR) << "Failed to create metrics reporting spec from: " << spec_str.value()
275 << " with error: " << error;
276 }
277 }
278
279 uint32_t reporting_num_mods = is_system_server
280 ? gFlags.MetricsReportingNumModsServer()
281 : gFlags.MetricsReportingNumMods();
282 uint32_t reporting_mods = is_system_server
283 ? gFlags.MetricsReportingModsServer()
284 : gFlags.MetricsReportingMods();
285
286 if (reporting_mods > reporting_num_mods || reporting_num_mods == 0) {
287 LOG(ERROR) << "Invalid metrics reporting mods: " << reporting_mods
288 << " num modes=" << reporting_num_mods
289 << ". The reporting is disabled";
290 reporting_mods = 0;
291 reporting_num_mods = 100;
292 }
293
294 return {
295 .dump_to_logcat = gFlags.MetricsWriteToLogcat(),
296 .dump_to_statsd = gFlags.MetricsWriteToStatsd(),
297 .dump_to_file = gFlags.MetricsWriteToFile.GetValueOptional(),
298 .metrics_format = gFlags.MetricsFormat(),
299 .period_spec = period_spec,
300 .reporting_mods = reporting_mods,
301 .reporting_num_mods = reporting_num_mods,
302 };
303 }
304
Parse(const std::string & spec_str,std::string * error_msg)305 std::optional<ReportingPeriodSpec> ReportingPeriodSpec::Parse(
306 const std::string& spec_str, std::string* error_msg) {
307 *error_msg = "";
308 if (spec_str.empty()) {
309 *error_msg = "Invalid empty spec.";
310 return std::nullopt;
311 }
312
313 // Split the string. Each element is separated by comma.
314 std::vector<std::string> elems;
315 Split(spec_str, ',', &elems);
316
317 // Check the startup marker (front) and the continuous one (back).
318 std::optional<ReportingPeriodSpec> spec = std::make_optional(ReportingPeriodSpec());
319 spec->spec = spec_str;
320 spec->report_startup_first = elems.front() == "S";
321 spec->continuous_reporting = elems.back() == "*";
322
323 // Compute the indices for the period values.
324 size_t start_interval_idx = spec->report_startup_first ? 1 : 0;
325 size_t end_interval_idx = spec->continuous_reporting ? (elems.size() - 1) : elems.size();
326
327 // '*' needs a numeric interval before in order to be valid.
328 if (spec->continuous_reporting &&
329 end_interval_idx == start_interval_idx) {
330 *error_msg = "Invalid period value in spec: " + spec_str;
331 return std::nullopt;
332 }
333
334 // Parse the periods.
335 for (size_t i = start_interval_idx; i < end_interval_idx; i++) {
336 uint32_t period;
337 if (!android::base::ParseUint(elems[i], &period)) {
338 *error_msg = "Invalid period value in spec: " + spec_str;
339 return std::nullopt;
340 }
341 spec->periods_seconds.push_back(period);
342 }
343
344 return spec;
345 }
346
347 } // namespace metrics
348 } // namespace art
349
350 #pragma clang diagnostic pop // -Wconversion
351