/* * Copyright (C) 2017 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #define STATSD_DEBUG false // STOPSHIP if true #include "Log.h" #include "DurationAnomalyTracker.h" #include "guardrail/StatsdStats.h" namespace android { namespace os { namespace statsd { DurationAnomalyTracker::DurationAnomalyTracker(const Alert& alert, const ConfigKey& configKey, const sp& alarmMonitor) : AnomalyTracker(alert, configKey), mAlarmMonitor(alarmMonitor) { VLOG("DurationAnomalyTracker() called"); } DurationAnomalyTracker::~DurationAnomalyTracker() { VLOG("~DurationAnomalyTracker() called"); cancelAllAlarms(); } void DurationAnomalyTracker::startAlarm(const MetricDimensionKey& dimensionKey, const int64_t timestampNs) { // Alarms are stored in secs. Must round up, since if it fires early, it is ignored completely. uint32_t timestampSec = static_cast((timestampNs -1) / NS_PER_SEC) + 1; // round up if (isInRefractoryPeriod(timestampNs, dimensionKey)) { VLOG("Not setting anomaly alarm since it would fall in the refractory period."); return; } auto itr = mAlarms.find(dimensionKey); if (itr != mAlarms.end() && mAlarmMonitor != nullptr) { mAlarmMonitor->remove(itr->second); } sp alarm = new InternalAlarm{timestampSec}; mAlarms[dimensionKey] = alarm; if (mAlarmMonitor != nullptr) { mAlarmMonitor->add(alarm); } } void DurationAnomalyTracker::stopAlarm(const MetricDimensionKey& dimensionKey, const int64_t timestampNs) { const auto itr = mAlarms.find(dimensionKey); if (itr == mAlarms.end()) { return; } // If the alarm is set in the past but hasn't fired yet (due to lag), catch it now. if (itr->second != nullptr && timestampNs >= (int64_t)NS_PER_SEC * itr->second->timestampSec) { declareAnomaly(timestampNs, mAlert.metric_id(), dimensionKey, mAlert.trigger_if_sum_gt() + (timestampNs / NS_PER_SEC) - itr->second->timestampSec); } if (mAlarmMonitor != nullptr) { mAlarmMonitor->remove(itr->second); } mAlarms.erase(dimensionKey); } void DurationAnomalyTracker::cancelAllAlarms() { if (mAlarmMonitor != nullptr) { for (const auto& itr : mAlarms) { mAlarmMonitor->remove(itr.second); } } mAlarms.clear(); } void DurationAnomalyTracker::informAlarmsFired( const int64_t timestampNs, unordered_set, SpHash>& firedAlarms) { if (firedAlarms.empty() || mAlarms.empty()) return; // Find the intersection of firedAlarms and mAlarms. // The for loop is inefficient, since it loops over all keys, but that's okay since it is very // seldomly called. The alternative would be having InternalAlarms store information about the // DurationAnomalyTracker and key, but that's a lot of data overhead to speed up something that // is rarely ever called. std::unordered_map> matchedAlarms; for (const auto& kv : mAlarms) { if (firedAlarms.count(kv.second) > 0) { matchedAlarms.insert({kv.first, kv.second}); } } // Now declare each of these alarms to have fired. for (const auto& kv : matchedAlarms) { declareAnomaly( timestampNs, mAlert.metric_id(), kv.first, mAlert.trigger_if_sum_gt() + (timestampNs / NS_PER_SEC) - kv.second->timestampSec); mAlarms.erase(kv.first); firedAlarms.erase(kv.second); // No one else can also own it, so we're done with it. } } } // namespace statsd } // namespace os } // namespace android