blob: 1ca59a366fcb17bdd63f43c8755cee8405f9ed98 [file] [log] [blame]
Yao Chen44cf27c2017-09-14 22:32:50 -07001/*
2 * Copyright (C) 2017 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 */
Yao Chen8a8d16c2018-02-08 14:50:40 -080016#define DEBUG true // STOPSHIP if true
Joe Onorato9fc9edf2017-10-15 20:08:52 -070017#include "Log.h"
Yao Chen44cf27c2017-09-14 22:32:50 -070018#include "MetricsManager.h"
Yao Chend10f7b12017-12-18 12:53:50 -080019#include "statslog.h"
Yao Chen93fe3a32017-11-02 13:52:59 -070020
Yao Chen44cf27c2017-09-14 22:32:50 -070021#include "CountMetricProducer.h"
Yao Chen93fe3a32017-11-02 13:52:59 -070022#include "condition/CombinationConditionTracker.h"
23#include "condition/SimpleConditionTracker.h"
Yao Chenb3561512017-11-21 18:07:17 -080024#include "guardrail/StatsdStats.h"
Yao Chen93fe3a32017-11-02 13:52:59 -070025#include "matchers/CombinationLogMatchingTracker.h"
26#include "matchers/SimpleLogMatchingTracker.h"
Yao Chencaf339d2017-10-06 16:01:10 -070027#include "metrics_manager_util.h"
28#include "stats_util.h"
Yangster-mac330af582018-02-08 15:24:38 -080029#include "stats_log_util.h"
Yao Chen44cf27c2017-09-14 22:32:50 -070030
Yao Chen93fe3a32017-11-02 13:52:59 -070031#include <log/logprint.h>
Bookatz6f197902018-02-05 12:30:14 -080032#include <private/android_filesystem_config.h>
David Chen16049572018-02-01 18:27:51 -080033#include <utils/SystemClock.h>
Yao Chen288c6002017-12-12 13:43:18 -080034
35using android::util::FIELD_COUNT_REPEATED;
36using android::util::FIELD_TYPE_MESSAGE;
37using android::util::ProtoOutputStream;
38
Yao Chen44cf27c2017-09-14 22:32:50 -070039using std::make_unique;
40using std::set;
41using std::string;
Yao Chen44cf27c2017-09-14 22:32:50 -070042using std::unordered_map;
43using std::vector;
44
45namespace android {
46namespace os {
47namespace statsd {
48
Yao Chen288c6002017-12-12 13:43:18 -080049const int FIELD_ID_METRICS = 1;
50
Yao Chend10f7b12017-12-18 12:53:50 -080051MetricsManager::MetricsManager(const ConfigKey& key, const StatsdConfig& config,
Yangster-mac932ecec2018-02-01 10:23:52 -080052 const long timeBaseSec,
53 const sp<UidMap> &uidMap,
54 const sp<AlarmMonitor>& anomalyAlarmMonitor,
55 const sp<AlarmMonitor>& periodicAlarmMonitor)
Yangster-mac330af582018-02-08 15:24:38 -080056 : mConfigKey(key), mUidMap(uidMap), mLastReportTimeNs(timeBaseSec * NS_PER_SEC) {
Yao Chenb3561512017-11-21 18:07:17 -080057 mConfigValid =
Yangster-mac932ecec2018-02-01 10:23:52 -080058 initStatsdConfig(key, config, *uidMap, anomalyAlarmMonitor, periodicAlarmMonitor,
59 timeBaseSec, mTagIds, mAllAtomMatchers,
60 mAllConditionTrackers, mAllMetricProducers, mAllAnomalyTrackers,
61 mAllPeriodicAlarmTrackers, mConditionToMetricMap, mTrackerToMetricMap,
62 mTrackerToConditionMap, mNoReportMetricIds);
Yao Chenb3561512017-11-21 18:07:17 -080063
Yao Chen147ce602017-12-22 14:35:34 -080064 if (config.allowed_log_source_size() == 0) {
Yao Chend10f7b12017-12-18 12:53:50 -080065 // TODO(b/70794411): uncomment the following line and remove the hard coded log source
66 // after all configs have the log source added.
67 // mConfigValid = false;
68 // ALOGE("Log source white list is empty! This config won't get any data.");
69
Bookatz6f197902018-02-05 12:30:14 -080070 mAllowedUid.push_back(AID_ROOT);
71 mAllowedUid.push_back(AID_STATSD);
72 mAllowedUid.push_back(AID_SYSTEM);
Yao Chend10f7b12017-12-18 12:53:50 -080073 mAllowedLogSources.insert(mAllowedUid.begin(), mAllowedUid.end());
74 } else {
Yao Chen147ce602017-12-22 14:35:34 -080075 for (const auto& source : config.allowed_log_source()) {
76 auto it = UidMap::sAidToUidMapping.find(source);
77 if (it != UidMap::sAidToUidMapping.end()) {
78 mAllowedUid.push_back(it->second);
79 } else {
80 mAllowedPkg.push_back(source);
81 }
82 }
Yao Chend10f7b12017-12-18 12:53:50 -080083
84 if (mAllowedUid.size() + mAllowedPkg.size() > StatsdStats::kMaxLogSourceCount) {
85 ALOGE("Too many log sources. This is likely to be an error in the config.");
86 mConfigValid = false;
87 } else {
88 initLogSourceWhiteList();
89 }
90 }
91
Yao Chenb3561512017-11-21 18:07:17 -080092 // Guardrail. Reject the config if it's too big.
93 if (mAllMetricProducers.size() > StatsdStats::kMaxMetricCountPerConfig ||
94 mAllConditionTrackers.size() > StatsdStats::kMaxConditionCountPerConfig ||
Stefan Lafonb8c9aa82017-12-03 14:27:25 -080095 mAllAtomMatchers.size() > StatsdStats::kMaxMatcherCountPerConfig) {
Yao Chenb3561512017-11-21 18:07:17 -080096 ALOGE("This config is too big! Reject!");
97 mConfigValid = false;
98 }
Bookatz1476ef22018-02-13 12:26:01 -080099 if (mAllAnomalyTrackers.size() > StatsdStats::kMaxAlertCountPerConfig) {
100 ALOGE("This config has too many alerts! Reject!");
101 mConfigValid = false;
102 }
Yao Chenf09569f2017-12-13 17:00:51 -0800103 // no matter whether this config is valid, log it in the stats.
104 StatsdStats::getInstance().noteConfigReceived(key, mAllMetricProducers.size(),
105 mAllConditionTrackers.size(),
Bookatzaea8d10b2018-03-02 10:37:24 -0800106 mAllAtomMatchers.size(),
107 mAllAnomalyTrackers.size(),
108 mConfigValid);
Yao Chen44cf27c2017-09-14 22:32:50 -0700109}
110
111MetricsManager::~MetricsManager() {
Bookatzd3606c72017-10-19 10:13:49 -0700112 VLOG("~MetricsManager()");
Yao Chen44cf27c2017-09-14 22:32:50 -0700113}
114
Yao Chend10f7b12017-12-18 12:53:50 -0800115void MetricsManager::initLogSourceWhiteList() {
116 std::lock_guard<std::mutex> lock(mAllowedLogSourcesMutex);
117 mAllowedLogSources.clear();
118 mAllowedLogSources.insert(mAllowedUid.begin(), mAllowedUid.end());
119
120 for (const auto& pkg : mAllowedPkg) {
121 auto uids = mUidMap->getAppUid(pkg);
122 mAllowedLogSources.insert(uids.begin(), uids.end());
123 }
124 if (DEBUG) {
125 for (const auto& uid : mAllowedLogSources) {
126 VLOG("Allowed uid %d", uid);
127 }
128 }
129}
130
Yao Chencaf339d2017-10-06 16:01:10 -0700131bool MetricsManager::isConfigValid() const {
132 return mConfigValid;
133}
134
David Chen27785a82018-01-19 17:06:45 -0800135void MetricsManager::notifyAppUpgrade(const uint64_t& eventTimeNs, const string& apk, const int uid,
136 const int64_t version) {
Yao Chend10f7b12017-12-18 12:53:50 -0800137 // check if we care this package
138 if (std::find(mAllowedPkg.begin(), mAllowedPkg.end(), apk) == mAllowedPkg.end()) {
139 return;
140 }
141 // We will re-initialize the whole list because we don't want to keep the multi mapping of
142 // UID<->pkg inside MetricsManager to reduce the memory usage.
143 initLogSourceWhiteList();
144}
145
David Chen27785a82018-01-19 17:06:45 -0800146void MetricsManager::notifyAppRemoved(const uint64_t& eventTimeNs, const string& apk,
147 const int uid) {
Yao Chend10f7b12017-12-18 12:53:50 -0800148 // check if we care this package
149 if (std::find(mAllowedPkg.begin(), mAllowedPkg.end(), apk) == mAllowedPkg.end()) {
150 return;
151 }
152 // We will re-initialize the whole list because we don't want to keep the multi mapping of
153 // UID<->pkg inside MetricsManager to reduce the memory usage.
154 initLogSourceWhiteList();
155}
156
David Chen27785a82018-01-19 17:06:45 -0800157void MetricsManager::onUidMapReceived(const uint64_t& eventTimeNs) {
Yao Chend10f7b12017-12-18 12:53:50 -0800158 if (mAllowedPkg.size() == 0) {
159 return;
160 }
161 initLogSourceWhiteList();
162}
163
Yao Chen884c8c12018-01-26 10:36:25 -0800164void MetricsManager::dumpStates(FILE* out, bool verbose) {
165 fprintf(out, "ConfigKey %s, allowed source:", mConfigKey.ToString().c_str());
166 {
167 std::lock_guard<std::mutex> lock(mAllowedLogSourcesMutex);
168 for (const auto& source : mAllowedLogSources) {
169 fprintf(out, "%d ", source);
170 }
171 }
172 fprintf(out, "\n");
173 for (const auto& producer : mAllMetricProducers) {
174 producer->dumpStates(out, verbose);
175 }
176}
177
Yao Chen06dba5d2018-01-26 13:38:16 -0800178void MetricsManager::dropData(const uint64_t dropTimeNs) {
179 for (const auto& producer : mAllMetricProducers) {
180 producer->dropData(dropTimeNs);
181 }
182}
183
Yao Chen8a8d16c2018-02-08 14:50:40 -0800184void MetricsManager::onDumpReport(const uint64_t dumpTimeStampNs, ProtoOutputStream* protoOutput) {
Yao Chen729093d2017-10-16 10:33:26 -0700185 VLOG("=========================Metric Reports Start==========================");
186 // one StatsLogReport per MetricProduer
Yangster-mac94e197c2018-01-02 16:03:03 -0800187 for (const auto& producer : mAllMetricProducers) {
188 if (mNoReportMetricIds.find(producer->getMetricId()) == mNoReportMetricIds.end()) {
Yi Jin5ee07872018-03-05 18:18:27 -0800189 uint64_t token =
Yangster-mac94e197c2018-01-02 16:03:03 -0800190 protoOutput->start(FIELD_TYPE_MESSAGE | FIELD_COUNT_REPEATED | FIELD_ID_METRICS);
191 producer->onDumpReport(dumpTimeStampNs, protoOutput);
192 protoOutput->end(token);
193 }
Yao Chen729093d2017-10-16 10:33:26 -0700194 }
Yangster-mac330af582018-02-08 15:24:38 -0800195 mLastReportTimeNs = dumpTimeStampNs;
Yao Chen729093d2017-10-16 10:33:26 -0700196 VLOG("=========================Metric Reports End==========================");
Yao Chen729093d2017-10-16 10:33:26 -0700197}
198
Yao Chen44cf27c2017-09-14 22:32:50 -0700199// Consume the stats log if it's interesting to this metric.
Joe Onoratoc4dfae52017-10-17 23:38:21 -0700200void MetricsManager::onLogEvent(const LogEvent& event) {
Yao Chencaf339d2017-10-06 16:01:10 -0700201 if (!mConfigValid) {
202 return;
203 }
204
David Chenb639d142018-02-14 17:29:54 -0800205 if (event.GetTagId() == android::util::APP_BREADCRUMB_REPORTED) {
206 // Check that app breadcrumb reported fields are valid.
207 // TODO: Find a way to make these checks easier to maintain.
David Chendaa9f3a2017-12-28 16:52:22 -0800208 status_t err = NO_ERROR;
Bookatzb223c4e2018-02-01 15:35:04 -0800209
210 // Uid is 3rd from last field and must match the caller's uid,
211 // unless that caller is statsd itself (statsd is allowed to spoof uids).
212 long appHookUid = event.GetLong(event.size()-2, &err);
David Chen77ef6712018-02-23 18:23:42 -0800213 if (err != NO_ERROR ) {
214 VLOG("APP_BREADCRUMB_REPORTED had error when parsing the uid");
215 return;
216 }
Bookatzb223c4e2018-02-01 15:35:04 -0800217 int32_t loggerUid = event.GetUid();
David Chen77ef6712018-02-23 18:23:42 -0800218 if (loggerUid != appHookUid && loggerUid != AID_STATSD) {
219 VLOG("APP_BREADCRUMB_REPORTED has invalid uid: claimed %ld but caller is %d",
220 appHookUid, loggerUid);
David Chendaa9f3a2017-12-28 16:52:22 -0800221 return;
222 }
Bookatzb223c4e2018-02-01 15:35:04 -0800223
224 // Label is 2nd from last field and must be from [0, 15].
225 long appHookLabel = event.GetLong(event.size()-1, &err);
David Chen77ef6712018-02-23 18:23:42 -0800226 if (err != NO_ERROR ) {
227 VLOG("APP_BREADCRUMB_REPORTED had error when parsing the label field");
228 return;
229 } else if (appHookLabel < 0 || appHookLabel > 15) {
230 VLOG("APP_BREADCRUMB_REPORTED does not have valid label %ld", appHookLabel);
Bookatzb223c4e2018-02-01 15:35:04 -0800231 return;
232 }
233
David Chendaa9f3a2017-12-28 16:52:22 -0800234 // The state must be from 0,3. This part of code must be manually updated.
Bookatzb223c4e2018-02-01 15:35:04 -0800235 long appHookState = event.GetLong(event.size(), &err);
David Chen77ef6712018-02-23 18:23:42 -0800236 if (err != NO_ERROR ) {
237 VLOG("APP_BREADCRUMB_REPORTED had error when parsing the state field");
238 return;
239 } else if (appHookState < 0 || appHookState > 3) {
240 VLOG("APP_BREADCRUMB_REPORTED does not have valid state %ld", appHookState);
David Chendaa9f3a2017-12-28 16:52:22 -0800241 return;
242 }
Tej Singhbb8554a2018-01-26 11:59:14 -0800243 } else if (event.GetTagId() == android::util::DAVEY_OCCURRED) {
244 // Daveys can be logged from any app since they are logged in libs/hwui/JankTracker.cpp.
245 // Check that the davey duration is reasonable. Max length check is for privacy.
246 status_t err = NO_ERROR;
David Chen77ef6712018-02-23 18:23:42 -0800247
248 // Uid is the first field provided.
249 long jankUid = event.GetLong(1, &err);
250 if (err != NO_ERROR ) {
251 VLOG("Davey occurred had error when parsing the uid");
252 return;
253 }
254 int32_t loggerUid = event.GetUid();
255 if (loggerUid != jankUid && loggerUid != AID_STATSD) {
256 VLOG("DAVEY_OCCURRED has invalid uid: claimed %ld but caller is %d", jankUid,
257 loggerUid);
258 return;
259 }
260
Tej Singhbb8554a2018-01-26 11:59:14 -0800261 long duration = event.GetLong(event.size(), &err);
David Chen77ef6712018-02-23 18:23:42 -0800262 if (err != NO_ERROR ) {
263 VLOG("Davey occurred had error when parsing the duration");
264 return;
265 } else if (duration > 100000) {
Tej Singhbb8554a2018-01-26 11:59:14 -0800266 VLOG("Davey duration is unreasonably long: %ld", duration);
267 return;
268 }
269 } else {
270 std::lock_guard<std::mutex> lock(mAllowedLogSourcesMutex);
271 if (mAllowedLogSources.find(event.GetUid()) == mAllowedLogSources.end()) {
272 VLOG("log source %d not on the whitelist", event.GetUid());
273 return;
274 }
Yao Chend10f7b12017-12-18 12:53:50 -0800275 }
276
Joe Onoratoc4dfae52017-10-17 23:38:21 -0700277 int tagId = event.GetTagId();
Yangster-mac330af582018-02-08 15:24:38 -0800278 uint64_t eventTime = event.GetElapsedTimestampNs();
Yao Chen44cf27c2017-09-14 22:32:50 -0700279 if (mTagIds.find(tagId) == mTagIds.end()) {
280 // not interesting...
281 return;
282 }
Yao Chen44cf27c2017-09-14 22:32:50 -0700283
Stefan Lafonb8c9aa82017-12-03 14:27:25 -0800284 vector<MatchingState> matcherCache(mAllAtomMatchers.size(), MatchingState::kNotComputed);
Yao Chencaf339d2017-10-06 16:01:10 -0700285
Stefan Lafonb8c9aa82017-12-03 14:27:25 -0800286 for (auto& matcher : mAllAtomMatchers) {
287 matcher->onLogEvent(event, mAllAtomMatchers, matcherCache);
Yao Chen44cf27c2017-09-14 22:32:50 -0700288 }
289
Yao Chencaf339d2017-10-06 16:01:10 -0700290 // A bitmap to see which ConditionTracker needs to be re-evaluated.
291 vector<bool> conditionToBeEvaluated(mAllConditionTrackers.size(), false);
292
293 for (const auto& pair : mTrackerToConditionMap) {
294 if (matcherCache[pair.first] == MatchingState::kMatched) {
295 const auto& conditionList = pair.second;
296 for (const int conditionIndex : conditionList) {
297 conditionToBeEvaluated[conditionIndex] = true;
298 }
299 }
300 }
301
302 vector<ConditionState> conditionCache(mAllConditionTrackers.size(),
303 ConditionState::kNotEvaluated);
304 // A bitmap to track if a condition has changed value.
305 vector<bool> changedCache(mAllConditionTrackers.size(), false);
306 for (size_t i = 0; i < mAllConditionTrackers.size(); i++) {
307 if (conditionToBeEvaluated[i] == false) {
308 continue;
309 }
Yao Chencaf339d2017-10-06 16:01:10 -0700310 sp<ConditionTracker>& condition = mAllConditionTrackers[i];
311 condition->evaluateCondition(event, matcherCache, mAllConditionTrackers, conditionCache,
Yao Chen967b2052017-11-07 16:36:43 -0800312 changedCache);
Yao Chen729093d2017-10-16 10:33:26 -0700313 }
314
315 for (size_t i = 0; i < mAllConditionTrackers.size(); i++) {
Yao Chen967b2052017-11-07 16:36:43 -0800316 if (changedCache[i] == false) {
Yao Chen729093d2017-10-16 10:33:26 -0700317 continue;
318 }
319 auto pair = mConditionToMetricMap.find(i);
320 if (pair != mConditionToMetricMap.end()) {
321 auto& metricList = pair->second;
322 for (auto metricIndex : metricList) {
323 // metric cares about non sliced condition, and it's changed.
324 // Push the new condition to it directly.
Yao Chen967b2052017-11-07 16:36:43 -0800325 if (!mAllMetricProducers[metricIndex]->isConditionSliced()) {
Yao Chen5154a3792017-10-30 22:57:06 -0700326 mAllMetricProducers[metricIndex]->onConditionChanged(conditionCache[i],
327 eventTime);
Yao Chen729093d2017-10-16 10:33:26 -0700328 // metric cares about sliced conditions, and it may have changed. Send
329 // notification, and the metric can query the sliced conditions that are
330 // interesting to it.
Yangsterf2bee6f2017-11-29 12:01:05 -0800331 } else {
Yao Chen5154a3792017-10-30 22:57:06 -0700332 mAllMetricProducers[metricIndex]->onSlicedConditionMayChange(eventTime);
Yao Chen44cf27c2017-09-14 22:32:50 -0700333 }
Yao Chencaf339d2017-10-06 16:01:10 -0700334 }
335 }
336 }
337
Stefan Lafonb8c9aa82017-12-03 14:27:25 -0800338 // For matched AtomMatchers, tell relevant metrics that a matched event has come.
339 for (size_t i = 0; i < mAllAtomMatchers.size(); i++) {
Yao Chencaf339d2017-10-06 16:01:10 -0700340 if (matcherCache[i] == MatchingState::kMatched) {
Yao Chenb3561512017-11-21 18:07:17 -0800341 StatsdStats::getInstance().noteMatcherMatched(mConfigKey,
Yangster-mac94e197c2018-01-02 16:03:03 -0800342 mAllAtomMatchers[i]->getId());
Yao Chencaf339d2017-10-06 16:01:10 -0700343 auto pair = mTrackerToMetricMap.find(i);
344 if (pair != mTrackerToMetricMap.end()) {
345 auto& metricList = pair->second;
346 for (const int metricIndex : metricList) {
Chenjie Yub3dda412017-10-24 13:41:59 -0700347 // pushed metrics are never scheduled pulls
Chenjie Yua7259ab2017-12-10 08:31:05 -0800348 mAllMetricProducers[metricIndex]->onMatchedLogEvent(i, event);
Yao Chencaf339d2017-10-06 16:01:10 -0700349 }
Yao Chen44cf27c2017-09-14 22:32:50 -0700350 }
351 }
352 }
353}
354
Yangster-mac932ecec2018-02-01 10:23:52 -0800355void MetricsManager::onAnomalyAlarmFired(
356 const uint64_t timestampNs,
357 unordered_set<sp<const InternalAlarm>, SpHash<InternalAlarm>>& alarmSet) {
Yangster-mace2cd6d52017-11-09 20:38:30 -0800358 for (const auto& itr : mAllAnomalyTrackers) {
Yangster-mac932ecec2018-02-01 10:23:52 -0800359 itr->informAlarmsFired(timestampNs, alarmSet);
Yangster-mace2cd6d52017-11-09 20:38:30 -0800360 }
361}
362
Yangster-mac932ecec2018-02-01 10:23:52 -0800363void MetricsManager::onPeriodicAlarmFired(
364 const uint64_t timestampNs,
365 unordered_set<sp<const InternalAlarm>, SpHash<InternalAlarm>>& alarmSet) {
366 for (const auto& itr : mAllPeriodicAlarmTrackers) {
367 itr->informAlarmsFired(timestampNs, alarmSet);
Yangster-mace2cd6d52017-11-09 20:38:30 -0800368 }
369}
370
yro69007c82017-10-26 20:42:57 -0700371// Returns the total byte size of all metrics managed by a single config source.
372size_t MetricsManager::byteSize() {
373 size_t totalSize = 0;
374 for (auto metricProducer : mAllMetricProducers) {
375 totalSize += metricProducer->byteSize();
376 }
377 return totalSize;
378}
379
Yao Chen44cf27c2017-09-14 22:32:50 -0700380} // namespace statsd
381} // namespace os
382} // namespace android