blob: 088f607ecfce9d2a25d59e15ce9295dc466b0522 [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 */
Yangster-mac754e29e2018-05-02 12:23:17 -070016#define DEBUG false // STOPSHIP if true
Joe Onorato9fc9edf2017-10-15 20:08:52 -070017#include "Log.h"
tsaichristined21aacf2019-10-07 14:47:38 -070018
Yao Chen44cf27c2017-09-14 22:32:50 -070019#include "MetricsManager.h"
tsaichristined21aacf2019-10-07 14:47:38 -070020
21#include <log/logprint.h>
22#include <private/android_filesystem_config.h>
23#include <utils/SystemClock.h>
Yao Chen93fe3a32017-11-02 13:52:59 -070024
Yao Chen44cf27c2017-09-14 22:32:50 -070025#include "CountMetricProducer.h"
Muhammad Qureshic8e22662019-11-20 17:18:03 -080026#include "atoms_info.h"
Yao Chen93fe3a32017-11-02 13:52:59 -070027#include "condition/CombinationConditionTracker.h"
28#include "condition/SimpleConditionTracker.h"
Yao Chenb3561512017-11-21 18:07:17 -080029#include "guardrail/StatsdStats.h"
Yao Chen93fe3a32017-11-02 13:52:59 -070030#include "matchers/CombinationLogMatchingTracker.h"
31#include "matchers/SimpleLogMatchingTracker.h"
Yao Chencaf339d2017-10-06 16:01:10 -070032#include "metrics_manager_util.h"
tsaichristined21aacf2019-10-07 14:47:38 -070033#include "state/StateManager.h"
Yangster-mac330af582018-02-08 15:24:38 -080034#include "stats_log_util.h"
tsaichristined21aacf2019-10-07 14:47:38 -070035#include "stats_util.h"
36#include "statslog.h"
Yao Chen288c6002017-12-12 13:43:18 -080037
38using android::util::FIELD_COUNT_REPEATED;
David Chenfaa1af52018-03-30 15:14:04 -070039using android::util::FIELD_TYPE_INT32;
40using android::util::FIELD_TYPE_INT64;
Yao Chen288c6002017-12-12 13:43:18 -080041using android::util::FIELD_TYPE_MESSAGE;
Yangster-mac9def8e32018-04-17 13:55:51 -070042using android::util::FIELD_TYPE_STRING;
Yao Chen288c6002017-12-12 13:43:18 -080043using android::util::ProtoOutputStream;
44
Yao Chen44cf27c2017-09-14 22:32:50 -070045using std::make_unique;
46using std::set;
47using std::string;
Yao Chen44cf27c2017-09-14 22:32:50 -070048using std::unordered_map;
49using std::vector;
50
51namespace android {
52namespace os {
53namespace statsd {
54
Yao Chen288c6002017-12-12 13:43:18 -080055const int FIELD_ID_METRICS = 1;
David Chenfaa1af52018-03-30 15:14:04 -070056const int FIELD_ID_ANNOTATIONS = 7;
57const int FIELD_ID_ANNOTATIONS_INT64 = 1;
58const int FIELD_ID_ANNOTATIONS_INT32 = 2;
Yao Chen288c6002017-12-12 13:43:18 -080059
Muhammad Qureshi844694b2019-04-05 10:10:40 -070060// for ActiveConfig
61const int FIELD_ID_ACTIVE_CONFIG_ID = 1;
62const int FIELD_ID_ACTIVE_CONFIG_UID = 2;
63const int FIELD_ID_ACTIVE_CONFIG_METRIC = 3;
64
Yao Chend10f7b12017-12-18 12:53:50 -080065MetricsManager::MetricsManager(const ConfigKey& key, const StatsdConfig& config,
Yangster-mac15f6bbc2018-04-08 11:52:26 -070066 const int64_t timeBaseNs, const int64_t currentTimeNs,
Chenjie Yue2219202018-06-08 10:07:51 -070067 const sp<UidMap>& uidMap,
68 const sp<StatsPullerManager>& pullerManager,
Yangster-mac932ecec2018-02-01 10:23:52 -080069 const sp<AlarmMonitor>& anomalyAlarmMonitor,
70 const sp<AlarmMonitor>& periodicAlarmMonitor)
Chenjie Yue2219202018-06-08 10:07:51 -070071 : mConfigKey(key),
72 mUidMap(uidMap),
Yangster-macb142cc82018-03-30 15:22:08 -070073 mTtlNs(config.has_ttl_in_seconds() ? config.ttl_in_seconds() * NS_PER_SEC : -1),
74 mTtlEndNs(-1),
David Chen48944902018-05-03 10:29:11 -070075 mLastReportTimeNs(currentTimeNs),
Yao Chen9a43b4f2019-04-10 10:43:20 -070076 mLastReportWallClockNs(getWallClockNs()),
77 mShouldPersistHistory(config.persist_locally()) {
Yangster-macb142cc82018-03-30 15:22:08 -070078 // Init the ttl end timestamp.
Yangster-mac15f6bbc2018-04-08 11:52:26 -070079 refreshTtl(timeBaseNs);
Yangster-macb142cc82018-03-30 15:22:08 -070080
Chenjie Yue2219202018-06-08 10:07:51 -070081 mConfigValid = initStatsdConfig(
82 key, config, *uidMap, pullerManager, anomalyAlarmMonitor, periodicAlarmMonitor,
83 timeBaseNs, currentTimeNs, mTagIds, mAllAtomMatchers, mAllConditionTrackers,
84 mAllMetricProducers, mAllAnomalyTrackers, mAllPeriodicAlarmTrackers,
Yangster-mac849dfdc22018-10-12 15:41:45 -070085 mConditionToMetricMap, mTrackerToMetricMap, mTrackerToConditionMap,
Muhammad Qureshi3a5ebf52019-03-28 12:38:21 -070086 mActivationAtomTrackerToMetricMap, mDeactivationAtomTrackerToMetricMap,
87 mMetricIndexesWithActivation, mNoReportMetricIds);
Yao Chenb3561512017-11-21 18:07:17 -080088
Yangster-mac1c58f042018-05-17 15:52:51 -070089 mHashStringsInReport = config.hash_strings_in_metric_report();
dwchen730403e2018-10-29 11:41:56 -070090 mVersionStringsInReport = config.version_strings_in_metric_report();
91 mInstallerInReport = config.installer_in_metric_report();
Yangster-mac1c58f042018-05-17 15:52:51 -070092
Yao Chen147ce602017-12-22 14:35:34 -080093 if (config.allowed_log_source_size() == 0) {
David Chen8faaa012018-02-28 15:54:36 -080094 mConfigValid = false;
95 ALOGE("Log source whitelist is empty! This config won't get any data. Suggest adding at "
96 "least AID_SYSTEM and AID_STATSD to the allowed_log_source field.");
Yao Chend10f7b12017-12-18 12:53:50 -080097 } else {
Yao Chen147ce602017-12-22 14:35:34 -080098 for (const auto& source : config.allowed_log_source()) {
99 auto it = UidMap::sAidToUidMapping.find(source);
100 if (it != UidMap::sAidToUidMapping.end()) {
101 mAllowedUid.push_back(it->second);
102 } else {
103 mAllowedPkg.push_back(source);
104 }
105 }
Yao Chend10f7b12017-12-18 12:53:50 -0800106
107 if (mAllowedUid.size() + mAllowedPkg.size() > StatsdStats::kMaxLogSourceCount) {
108 ALOGE("Too many log sources. This is likely to be an error in the config.");
109 mConfigValid = false;
110 } else {
111 initLogSourceWhiteList();
112 }
113 }
114
David Chenfaa1af52018-03-30 15:14:04 -0700115 // Store the sub-configs used.
116 for (const auto& annotation : config.annotation()) {
117 mAnnotations.emplace_back(annotation.field_int64(), annotation.field_int32());
118 }
119
Yao Chenb3561512017-11-21 18:07:17 -0800120 // Guardrail. Reject the config if it's too big.
121 if (mAllMetricProducers.size() > StatsdStats::kMaxMetricCountPerConfig ||
122 mAllConditionTrackers.size() > StatsdStats::kMaxConditionCountPerConfig ||
Stefan Lafonb8c9aa82017-12-03 14:27:25 -0800123 mAllAtomMatchers.size() > StatsdStats::kMaxMatcherCountPerConfig) {
Yao Chenb3561512017-11-21 18:07:17 -0800124 ALOGE("This config is too big! Reject!");
125 mConfigValid = false;
126 }
Bookatz1476ef22018-02-13 12:26:01 -0800127 if (mAllAnomalyTrackers.size() > StatsdStats::kMaxAlertCountPerConfig) {
128 ALOGE("This config has too many alerts! Reject!");
129 mConfigValid = false;
130 }
Tej Singh6ede28b2019-01-29 17:06:54 -0800131
132 mIsAlwaysActive = (mMetricIndexesWithActivation.size() != mAllMetricProducers.size()) ||
133 (mAllMetricProducers.size() == 0);
134 bool isActive = mIsAlwaysActive;
135 for (int metric : mMetricIndexesWithActivation) {
136 isActive |= mAllMetricProducers[metric]->isActive();
137 }
138 mIsActive = isActive;
139 VLOG("mIsActive is initialized to %d", mIsActive)
140
Yao Chenf09569f2017-12-13 17:00:51 -0800141 // no matter whether this config is valid, log it in the stats.
David Chenfaa1af52018-03-30 15:14:04 -0700142 StatsdStats::getInstance().noteConfigReceived(
143 key, mAllMetricProducers.size(), mAllConditionTrackers.size(), mAllAtomMatchers.size(),
144 mAllAnomalyTrackers.size(), mAnnotations, mConfigValid);
Chenjie Yuc7939cb2019-02-04 17:25:45 -0800145 // Check active
146 for (const auto& metric : mAllMetricProducers) {
147 if (metric->isActive()) {
148 mIsActive = true;
149 break;
150 }
151 }
Yao Chen44cf27c2017-09-14 22:32:50 -0700152}
153
154MetricsManager::~MetricsManager() {
tsaichristined21aacf2019-10-07 14:47:38 -0700155 for (auto it : mAllMetricProducers) {
156 for (int atomId : it->getSlicedStateAtoms()) {
157 StateManager::getInstance().unregisterListener(atomId, it);
158 }
159 }
160
Bookatzd3606c72017-10-19 10:13:49 -0700161 VLOG("~MetricsManager()");
Yao Chen44cf27c2017-09-14 22:32:50 -0700162}
163
Yao Chend10f7b12017-12-18 12:53:50 -0800164void MetricsManager::initLogSourceWhiteList() {
165 std::lock_guard<std::mutex> lock(mAllowedLogSourcesMutex);
166 mAllowedLogSources.clear();
167 mAllowedLogSources.insert(mAllowedUid.begin(), mAllowedUid.end());
168
169 for (const auto& pkg : mAllowedPkg) {
170 auto uids = mUidMap->getAppUid(pkg);
171 mAllowedLogSources.insert(uids.begin(), uids.end());
172 }
173 if (DEBUG) {
174 for (const auto& uid : mAllowedLogSources) {
175 VLOG("Allowed uid %d", uid);
176 }
177 }
178}
179
Yao Chencaf339d2017-10-06 16:01:10 -0700180bool MetricsManager::isConfigValid() const {
181 return mConfigValid;
182}
183
Yangster-macb142cc82018-03-30 15:22:08 -0700184void MetricsManager::notifyAppUpgrade(const int64_t& eventTimeNs, const string& apk, const int uid,
David Chen27785a82018-01-19 17:06:45 -0800185 const int64_t version) {
Tej Singh9ec159a2019-11-14 11:59:48 -0800186 // Inform all metric producers.
187 for (auto it : mAllMetricProducers) {
188 it->notifyAppUpgrade(eventTimeNs, apk, uid, version);
189 }
Yao Chend10f7b12017-12-18 12:53:50 -0800190 // check if we care this package
191 if (std::find(mAllowedPkg.begin(), mAllowedPkg.end(), apk) == mAllowedPkg.end()) {
192 return;
193 }
194 // We will re-initialize the whole list because we don't want to keep the multi mapping of
195 // UID<->pkg inside MetricsManager to reduce the memory usage.
196 initLogSourceWhiteList();
197}
198
Yangster-macb142cc82018-03-30 15:22:08 -0700199void MetricsManager::notifyAppRemoved(const int64_t& eventTimeNs, const string& apk,
David Chen27785a82018-01-19 17:06:45 -0800200 const int uid) {
Tej Singh9ec159a2019-11-14 11:59:48 -0800201 // Inform all metric producers.
202 for (auto it : mAllMetricProducers) {
203 it->notifyAppRemoved(eventTimeNs, apk, uid);
204 }
Yao Chend10f7b12017-12-18 12:53:50 -0800205 // check if we care this package
206 if (std::find(mAllowedPkg.begin(), mAllowedPkg.end(), apk) == mAllowedPkg.end()) {
207 return;
208 }
209 // We will re-initialize the whole list because we don't want to keep the multi mapping of
210 // UID<->pkg inside MetricsManager to reduce the memory usage.
211 initLogSourceWhiteList();
212}
213
Yangster-macb142cc82018-03-30 15:22:08 -0700214void MetricsManager::onUidMapReceived(const int64_t& eventTimeNs) {
Tej Singh9ec159a2019-11-14 11:59:48 -0800215 // Purposefully don't inform metric producers on a new snapshot
216 // because we don't need to flush partial buckets.
217 // This occurs if a new user is added/removed or statsd crashes.
Yao Chend10f7b12017-12-18 12:53:50 -0800218 if (mAllowedPkg.size() == 0) {
219 return;
220 }
221 initLogSourceWhiteList();
222}
223
Yao Chen884c8c12018-01-26 10:36:25 -0800224void MetricsManager::dumpStates(FILE* out, bool verbose) {
225 fprintf(out, "ConfigKey %s, allowed source:", mConfigKey.ToString().c_str());
226 {
227 std::lock_guard<std::mutex> lock(mAllowedLogSourcesMutex);
228 for (const auto& source : mAllowedLogSources) {
229 fprintf(out, "%d ", source);
230 }
231 }
232 fprintf(out, "\n");
233 for (const auto& producer : mAllMetricProducers) {
234 producer->dumpStates(out, verbose);
235 }
236}
237
Yangster-macb142cc82018-03-30 15:22:08 -0700238void MetricsManager::dropData(const int64_t dropTimeNs) {
Yao Chen06dba5d2018-01-26 13:38:16 -0800239 for (const auto& producer : mAllMetricProducers) {
240 producer->dropData(dropTimeNs);
241 }
242}
243
Yangster-mace68f3a52018-04-04 00:01:43 -0700244void MetricsManager::onDumpReport(const int64_t dumpTimeStampNs,
245 const bool include_current_partial_bucket,
Bookatzff71cad2018-09-20 17:17:49 -0700246 const bool erase_data,
Olivier Gaillard6c75ecd2019-02-20 09:57:33 +0000247 const DumpLatency dumpLatency,
Yangster-mac9def8e32018-04-17 13:55:51 -0700248 std::set<string> *str_set,
Yangster-mace68f3a52018-04-04 00:01:43 -0700249 ProtoOutputStream* protoOutput) {
Yao Chen729093d2017-10-16 10:33:26 -0700250 VLOG("=========================Metric Reports Start==========================");
251 // one StatsLogReport per MetricProduer
Yangster-mac94e197c2018-01-02 16:03:03 -0800252 for (const auto& producer : mAllMetricProducers) {
253 if (mNoReportMetricIds.find(producer->getMetricId()) == mNoReportMetricIds.end()) {
Yangster-mac9def8e32018-04-17 13:55:51 -0700254 uint64_t token = protoOutput->start(
255 FIELD_TYPE_MESSAGE | FIELD_COUNT_REPEATED | FIELD_ID_METRICS);
Yangster-mac1c58f042018-05-17 15:52:51 -0700256 if (mHashStringsInReport) {
Bookatzff71cad2018-09-20 17:17:49 -0700257 producer->onDumpReport(dumpTimeStampNs, include_current_partial_bucket, erase_data,
Olivier Gaillard6c75ecd2019-02-20 09:57:33 +0000258 dumpLatency, str_set, protoOutput);
Yangster-mac1c58f042018-05-17 15:52:51 -0700259 } else {
Bookatzff71cad2018-09-20 17:17:49 -0700260 producer->onDumpReport(dumpTimeStampNs, include_current_partial_bucket, erase_data,
Olivier Gaillard6c75ecd2019-02-20 09:57:33 +0000261 dumpLatency, nullptr, protoOutput);
Yangster-mac1c58f042018-05-17 15:52:51 -0700262 }
Yangster-mac94e197c2018-01-02 16:03:03 -0800263 protoOutput->end(token);
Yangster-maca802d732018-04-24 07:50:38 -0700264 } else {
265 producer->clearPastBuckets(dumpTimeStampNs);
Yangster-mac94e197c2018-01-02 16:03:03 -0800266 }
Yao Chen729093d2017-10-16 10:33:26 -0700267 }
David Chenfaa1af52018-03-30 15:14:04 -0700268 for (const auto& annotation : mAnnotations) {
269 uint64_t token = protoOutput->start(FIELD_TYPE_MESSAGE | FIELD_COUNT_REPEATED |
270 FIELD_ID_ANNOTATIONS);
271 protoOutput->write(FIELD_TYPE_INT64 | FIELD_ID_ANNOTATIONS_INT64,
272 (long long)annotation.first);
273 protoOutput->write(FIELD_TYPE_INT32 | FIELD_ID_ANNOTATIONS_INT32, annotation.second);
274 protoOutput->end(token);
275 }
Yangster-mac9def8e32018-04-17 13:55:51 -0700276
Yangster-mac330af582018-02-08 15:24:38 -0800277 mLastReportTimeNs = dumpTimeStampNs;
Yangster-mac3fa5d7f2018-03-10 21:50:27 -0800278 mLastReportWallClockNs = getWallClockNs();
Yao Chen729093d2017-10-16 10:33:26 -0700279 VLOG("=========================Metric Reports End==========================");
Yao Chen729093d2017-10-16 10:33:26 -0700280}
281
Yao Chencaf339d2017-10-06 16:01:10 -0700282
Andrei Oneada01ea52019-01-30 15:28:36 +0000283bool MetricsManager::checkLogCredentials(const LogEvent& event) {
284 if (android::util::AtomsInfo::kWhitelistedAtoms.find(event.GetTagId()) !=
Muhammad Qureshi3a5ebf52019-03-28 12:38:21 -0700285 android::util::AtomsInfo::kWhitelistedAtoms.end())
Andrei Oneada01ea52019-01-30 15:28:36 +0000286 {
287 return true;
288 }
289 std::lock_guard<std::mutex> lock(mAllowedLogSourcesMutex);
290 if (mAllowedLogSources.find(event.GetUid()) == mAllowedLogSources.end()) {
291 VLOG("log source %d not on the whitelist", event.GetUid());
292 return false;
293 }
294 return true;
295}
296
297bool MetricsManager::eventSanityCheck(const LogEvent& event) {
David Chenb639d142018-02-14 17:29:54 -0800298 if (event.GetTagId() == android::util::APP_BREADCRUMB_REPORTED) {
299 // Check that app breadcrumb reported fields are valid.
David Chendaa9f3a2017-12-28 16:52:22 -0800300 status_t err = NO_ERROR;
Bookatzb223c4e2018-02-01 15:35:04 -0800301
302 // Uid is 3rd from last field and must match the caller's uid,
303 // unless that caller is statsd itself (statsd is allowed to spoof uids).
304 long appHookUid = event.GetLong(event.size()-2, &err);
David Chen77ef6712018-02-23 18:23:42 -0800305 if (err != NO_ERROR ) {
306 VLOG("APP_BREADCRUMB_REPORTED had error when parsing the uid");
Andrei Oneada01ea52019-01-30 15:28:36 +0000307 return false;
David Chen77ef6712018-02-23 18:23:42 -0800308 }
Bookatzb223c4e2018-02-01 15:35:04 -0800309 int32_t loggerUid = event.GetUid();
David Chen77ef6712018-02-23 18:23:42 -0800310 if (loggerUid != appHookUid && loggerUid != AID_STATSD) {
311 VLOG("APP_BREADCRUMB_REPORTED has invalid uid: claimed %ld but caller is %d",
312 appHookUid, loggerUid);
Andrei Oneada01ea52019-01-30 15:28:36 +0000313 return false;
David Chendaa9f3a2017-12-28 16:52:22 -0800314 }
Bookatzb223c4e2018-02-01 15:35:04 -0800315
David Chendaa9f3a2017-12-28 16:52:22 -0800316 // The state must be from 0,3. This part of code must be manually updated.
Bookatzb223c4e2018-02-01 15:35:04 -0800317 long appHookState = event.GetLong(event.size(), &err);
David Chen77ef6712018-02-23 18:23:42 -0800318 if (err != NO_ERROR ) {
319 VLOG("APP_BREADCRUMB_REPORTED had error when parsing the state field");
Andrei Oneada01ea52019-01-30 15:28:36 +0000320 return false;
David Chen77ef6712018-02-23 18:23:42 -0800321 } else if (appHookState < 0 || appHookState > 3) {
322 VLOG("APP_BREADCRUMB_REPORTED does not have valid state %ld", appHookState);
Andrei Oneada01ea52019-01-30 15:28:36 +0000323 return false;
David Chendaa9f3a2017-12-28 16:52:22 -0800324 }
Tej Singhbb8554a2018-01-26 11:59:14 -0800325 } else if (event.GetTagId() == android::util::DAVEY_OCCURRED) {
326 // Daveys can be logged from any app since they are logged in libs/hwui/JankTracker.cpp.
327 // Check that the davey duration is reasonable. Max length check is for privacy.
328 status_t err = NO_ERROR;
David Chen77ef6712018-02-23 18:23:42 -0800329
330 // Uid is the first field provided.
331 long jankUid = event.GetLong(1, &err);
332 if (err != NO_ERROR ) {
333 VLOG("Davey occurred had error when parsing the uid");
Andrei Oneada01ea52019-01-30 15:28:36 +0000334 return false;
David Chen77ef6712018-02-23 18:23:42 -0800335 }
336 int32_t loggerUid = event.GetUid();
337 if (loggerUid != jankUid && loggerUid != AID_STATSD) {
338 VLOG("DAVEY_OCCURRED has invalid uid: claimed %ld but caller is %d", jankUid,
339 loggerUid);
Andrei Oneada01ea52019-01-30 15:28:36 +0000340 return false;
David Chen77ef6712018-02-23 18:23:42 -0800341 }
342
Tej Singhbb8554a2018-01-26 11:59:14 -0800343 long duration = event.GetLong(event.size(), &err);
David Chen77ef6712018-02-23 18:23:42 -0800344 if (err != NO_ERROR ) {
345 VLOG("Davey occurred had error when parsing the duration");
Andrei Oneada01ea52019-01-30 15:28:36 +0000346 return false;
David Chen77ef6712018-02-23 18:23:42 -0800347 } else if (duration > 100000) {
Tej Singhbb8554a2018-01-26 11:59:14 -0800348 VLOG("Davey duration is unreasonably long: %ld", duration);
Andrei Oneada01ea52019-01-30 15:28:36 +0000349 return false;
Tej Singhbb8554a2018-01-26 11:59:14 -0800350 }
Andrei Oneada01ea52019-01-30 15:28:36 +0000351 }
352
353 return true;
354}
355
356// Consume the stats log if it's interesting to this metric.
357void MetricsManager::onLogEvent(const LogEvent& event) {
358 if (!mConfigValid) {
359 return;
360 }
361
362 if (!checkLogCredentials(event)) {
363 return;
364 }
365
366 if (!eventSanityCheck(event)) {
367 return;
Yao Chend10f7b12017-12-18 12:53:50 -0800368 }
369
Joe Onoratoc4dfae52017-10-17 23:38:21 -0700370 int tagId = event.GetTagId();
Yangster-mac849dfdc22018-10-12 15:41:45 -0700371 int64_t eventTimeNs = event.GetElapsedTimestampNs();
372
Tej Singh6ede28b2019-01-29 17:06:54 -0800373 bool isActive = mIsAlwaysActive;
Muhammad Qureshi3a5ebf52019-03-28 12:38:21 -0700374
375 // Set of metrics that are still active after flushing.
376 unordered_set<int> activeMetricsIndices;
377
378 // Update state of all metrics w/ activation conditions as of eventTimeNs.
379 for (int metricIndex : mMetricIndexesWithActivation) {
380 const sp<MetricProducer>& metric = mAllMetricProducers[metricIndex];
381 metric->flushIfExpire(eventTimeNs);
382 if (metric->isActive()) {
383 // If this metric w/ activation condition is still active after
384 // flushing, remember it.
385 activeMetricsIndices.insert(metricIndex);
386 }
Yangster-mac849dfdc22018-10-12 15:41:45 -0700387 }
388
Muhammad Qureshi3a5ebf52019-03-28 12:38:21 -0700389 mIsActive = isActive || !activeMetricsIndices.empty();
Tej Singh6ede28b2019-01-29 17:06:54 -0800390
Yao Chen44cf27c2017-09-14 22:32:50 -0700391 if (mTagIds.find(tagId) == mTagIds.end()) {
Muhammad Qureshi3a5ebf52019-03-28 12:38:21 -0700392 // Not interesting...
Yao Chen44cf27c2017-09-14 22:32:50 -0700393 return;
394 }
Yao Chen44cf27c2017-09-14 22:32:50 -0700395
Stefan Lafonb8c9aa82017-12-03 14:27:25 -0800396 vector<MatchingState> matcherCache(mAllAtomMatchers.size(), MatchingState::kNotComputed);
Yao Chencaf339d2017-10-06 16:01:10 -0700397
Muhammad Qureshi3a5ebf52019-03-28 12:38:21 -0700398 // Evaluate all atom matchers.
Stefan Lafonb8c9aa82017-12-03 14:27:25 -0800399 for (auto& matcher : mAllAtomMatchers) {
400 matcher->onLogEvent(event, mAllAtomMatchers, matcherCache);
Yao Chen44cf27c2017-09-14 22:32:50 -0700401 }
402
Muhammad Qureshi3a5ebf52019-03-28 12:38:21 -0700403 // Set of metrics that received an activation cancellation.
404 unordered_set<int> metricIndicesWithCanceledActivations;
405
406 // Determine which metric activations received a cancellation and cancel them.
407 for (const auto& it : mDeactivationAtomTrackerToMetricMap) {
408 if (matcherCache[it.first] == MatchingState::kMatched) {
409 for (int metricIndex : it.second) {
410 mAllMetricProducers[metricIndex]->cancelEventActivation(it.first);
411 metricIndicesWithCanceledActivations.insert(metricIndex);
412 }
413 }
414 }
415
416 // Determine whether any metrics are no longer active after cancelling metric activations.
417 for (const int metricIndex : metricIndicesWithCanceledActivations) {
418 const sp<MetricProducer>& metric = mAllMetricProducers[metricIndex];
419 metric->flushIfExpire(eventTimeNs);
420 if (!metric->isActive()) {
421 activeMetricsIndices.erase(metricIndex);
422 }
423 }
424
425 isActive |= !activeMetricsIndices.empty();
426
427
428 // Determine which metric activations should be turned on and turn them on
Yangster-mac849dfdc22018-10-12 15:41:45 -0700429 for (const auto& it : mActivationAtomTrackerToMetricMap) {
430 if (matcherCache[it.first] == MatchingState::kMatched) {
431 for (int metricIndex : it.second) {
432 mAllMetricProducers[metricIndex]->activate(it.first, eventTimeNs);
Chenjie Yuc7939cb2019-02-04 17:25:45 -0800433 isActive |= mAllMetricProducers[metricIndex]->isActive();
Yangster-mac849dfdc22018-10-12 15:41:45 -0700434 }
435 }
436 }
437
Chenjie Yuc7939cb2019-02-04 17:25:45 -0800438 mIsActive = isActive;
439
Yao Chencaf339d2017-10-06 16:01:10 -0700440 // A bitmap to see which ConditionTracker needs to be re-evaluated.
441 vector<bool> conditionToBeEvaluated(mAllConditionTrackers.size(), false);
442
443 for (const auto& pair : mTrackerToConditionMap) {
444 if (matcherCache[pair.first] == MatchingState::kMatched) {
445 const auto& conditionList = pair.second;
446 for (const int conditionIndex : conditionList) {
447 conditionToBeEvaluated[conditionIndex] = true;
448 }
449 }
450 }
451
452 vector<ConditionState> conditionCache(mAllConditionTrackers.size(),
453 ConditionState::kNotEvaluated);
454 // A bitmap to track if a condition has changed value.
455 vector<bool> changedCache(mAllConditionTrackers.size(), false);
456 for (size_t i = 0; i < mAllConditionTrackers.size(); i++) {
457 if (conditionToBeEvaluated[i] == false) {
458 continue;
459 }
Yao Chencaf339d2017-10-06 16:01:10 -0700460 sp<ConditionTracker>& condition = mAllConditionTrackers[i];
461 condition->evaluateCondition(event, matcherCache, mAllConditionTrackers, conditionCache,
Yao Chen967b2052017-11-07 16:36:43 -0800462 changedCache);
Yao Chen729093d2017-10-16 10:33:26 -0700463 }
464
465 for (size_t i = 0; i < mAllConditionTrackers.size(); i++) {
Yao Chen967b2052017-11-07 16:36:43 -0800466 if (changedCache[i] == false) {
Yao Chen729093d2017-10-16 10:33:26 -0700467 continue;
468 }
469 auto pair = mConditionToMetricMap.find(i);
470 if (pair != mConditionToMetricMap.end()) {
471 auto& metricList = pair->second;
472 for (auto metricIndex : metricList) {
Muhammad Qureshi3a5ebf52019-03-28 12:38:21 -0700473 // Metric cares about non sliced condition, and it's changed.
Yao Chen729093d2017-10-16 10:33:26 -0700474 // Push the new condition to it directly.
Yao Chen967b2052017-11-07 16:36:43 -0800475 if (!mAllMetricProducers[metricIndex]->isConditionSliced()) {
Yao Chen5154a3792017-10-30 22:57:06 -0700476 mAllMetricProducers[metricIndex]->onConditionChanged(conditionCache[i],
Yangster-mac849dfdc22018-10-12 15:41:45 -0700477 eventTimeNs);
Muhammad Qureshi3a5ebf52019-03-28 12:38:21 -0700478 // Metric cares about sliced conditions, and it may have changed. Send
Yao Chen729093d2017-10-16 10:33:26 -0700479 // notification, and the metric can query the sliced conditions that are
480 // interesting to it.
Yangsterf2bee6f2017-11-29 12:01:05 -0800481 } else {
Yao Chen427d3722018-03-22 15:21:52 -0700482 mAllMetricProducers[metricIndex]->onSlicedConditionMayChange(conditionCache[i],
Yangster-mac849dfdc22018-10-12 15:41:45 -0700483 eventTimeNs);
Yao Chen44cf27c2017-09-14 22:32:50 -0700484 }
Yao Chencaf339d2017-10-06 16:01:10 -0700485 }
486 }
487 }
488
Stefan Lafonb8c9aa82017-12-03 14:27:25 -0800489 // For matched AtomMatchers, tell relevant metrics that a matched event has come.
490 for (size_t i = 0; i < mAllAtomMatchers.size(); i++) {
Yao Chencaf339d2017-10-06 16:01:10 -0700491 if (matcherCache[i] == MatchingState::kMatched) {
Yao Chenb3561512017-11-21 18:07:17 -0800492 StatsdStats::getInstance().noteMatcherMatched(mConfigKey,
Yangster-mac94e197c2018-01-02 16:03:03 -0800493 mAllAtomMatchers[i]->getId());
Yao Chencaf339d2017-10-06 16:01:10 -0700494 auto pair = mTrackerToMetricMap.find(i);
495 if (pair != mTrackerToMetricMap.end()) {
496 auto& metricList = pair->second;
497 for (const int metricIndex : metricList) {
Chenjie Yub3dda412017-10-24 13:41:59 -0700498 // pushed metrics are never scheduled pulls
Chenjie Yua7259ab2017-12-10 08:31:05 -0800499 mAllMetricProducers[metricIndex]->onMatchedLogEvent(i, event);
Yao Chencaf339d2017-10-06 16:01:10 -0700500 }
Yao Chen44cf27c2017-09-14 22:32:50 -0700501 }
502 }
503 }
504}
505
Yangster-mac932ecec2018-02-01 10:23:52 -0800506void MetricsManager::onAnomalyAlarmFired(
Yangster-macb142cc82018-03-30 15:22:08 -0700507 const int64_t& timestampNs,
Yangster-mac932ecec2018-02-01 10:23:52 -0800508 unordered_set<sp<const InternalAlarm>, SpHash<InternalAlarm>>& alarmSet) {
Yangster-mace2cd6d52017-11-09 20:38:30 -0800509 for (const auto& itr : mAllAnomalyTrackers) {
Yangster-mac932ecec2018-02-01 10:23:52 -0800510 itr->informAlarmsFired(timestampNs, alarmSet);
Yangster-mace2cd6d52017-11-09 20:38:30 -0800511 }
512}
513
Yangster-mac932ecec2018-02-01 10:23:52 -0800514void MetricsManager::onPeriodicAlarmFired(
Yangster-macb142cc82018-03-30 15:22:08 -0700515 const int64_t& timestampNs,
Yangster-mac932ecec2018-02-01 10:23:52 -0800516 unordered_set<sp<const InternalAlarm>, SpHash<InternalAlarm>>& alarmSet) {
517 for (const auto& itr : mAllPeriodicAlarmTrackers) {
518 itr->informAlarmsFired(timestampNs, alarmSet);
Yangster-mace2cd6d52017-11-09 20:38:30 -0800519 }
520}
521
yro69007c82017-10-26 20:42:57 -0700522// Returns the total byte size of all metrics managed by a single config source.
523size_t MetricsManager::byteSize() {
524 size_t totalSize = 0;
Chih-Hung Hsieha1b644e2018-12-11 11:09:20 -0800525 for (const auto& metricProducer : mAllMetricProducers) {
yro69007c82017-10-26 20:42:57 -0700526 totalSize += metricProducer->byteSize();
527 }
528 return totalSize;
529}
530
Muhammad Qureshi844694b2019-04-05 10:10:40 -0700531void MetricsManager::loadActiveConfig(const ActiveConfig& config, int64_t currentTimeNs) {
532 if (config.metric_size() == 0) {
Chenjie Yuc7939cb2019-02-04 17:25:45 -0800533 ALOGW("No active metric for config %s", mConfigKey.ToString().c_str());
534 return;
535 }
536
Muhammad Qureshi844694b2019-04-05 10:10:40 -0700537 for (int i = 0; i < config.metric_size(); i++) {
538 const auto& activeMetric = config.metric(i);
539 for (int metricIndex : mMetricIndexesWithActivation) {
540 const auto& metric = mAllMetricProducers[metricIndex];
541 if (metric->getMetricId() == activeMetric.id()) {
542 VLOG("Setting active metric: %lld", (long long)metric->getMetricId());
543 metric->loadActiveMetric(activeMetric, currentTimeNs);
Tej Singh16ca28f2019-06-24 11:58:23 -0700544 if (!mIsActive && metric->isActive()) {
545 StatsdStats::getInstance().noteActiveStatusChanged(mConfigKey,
546 /*activate=*/ true);
547 }
Muhammad Qureshi844694b2019-04-05 10:10:40 -0700548 mIsActive |= metric->isActive();
Chenjie Yuc7939cb2019-02-04 17:25:45 -0800549 }
550 }
551 }
552}
553
Muhammad Qureshi844694b2019-04-05 10:10:40 -0700554void MetricsManager::writeActiveConfigToProtoOutputStream(
Tej Singhf53d4452019-05-09 18:17:59 -0700555 int64_t currentTimeNs, const DumpReportReason reason, ProtoOutputStream* proto) {
Muhammad Qureshi844694b2019-04-05 10:10:40 -0700556 proto->write(FIELD_TYPE_INT64 | FIELD_ID_ACTIVE_CONFIG_ID, (long long)mConfigKey.GetId());
557 proto->write(FIELD_TYPE_INT32 | FIELD_ID_ACTIVE_CONFIG_UID, mConfigKey.GetUid());
558 for (int metricIndex : mMetricIndexesWithActivation) {
559 const auto& metric = mAllMetricProducers[metricIndex];
560 const uint64_t metricToken = proto->start(FIELD_TYPE_MESSAGE | FIELD_COUNT_REPEATED |
561 FIELD_ID_ACTIVE_CONFIG_METRIC);
Tej Singhf53d4452019-05-09 18:17:59 -0700562 metric->writeActiveMetricToProtoOutputStream(currentTimeNs, reason, proto);
Muhammad Qureshi844694b2019-04-05 10:10:40 -0700563 proto->end(metricToken);
564 }
565}
566
567
568
569
Yao Chen44cf27c2017-09-14 22:32:50 -0700570} // namespace statsd
571} // namespace os
572} // namespace android