1 /*
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 */
16
17 #define DEBUG false
18 #include "Log.h"
19
20 #include "anomaly/AlarmMonitor.h"
21 #include "guardrail/StatsdStats.h"
22
23 namespace android {
24 namespace os {
25 namespace statsd {
26
AlarmMonitor(uint32_t minDiffToUpdateRegisteredAlarmTimeSec,const std::function<void (const sp<IStatsCompanionService> &,int64_t)> & updateAlarm,const std::function<void (const sp<IStatsCompanionService> &)> & cancelAlarm)27 AlarmMonitor::AlarmMonitor(
28 uint32_t minDiffToUpdateRegisteredAlarmTimeSec,
29 const std::function<void(const sp<IStatsCompanionService>&, int64_t)>& updateAlarm,
30 const std::function<void(const sp<IStatsCompanionService>&)>& cancelAlarm)
31 : mRegisteredAlarmTimeSec(0), mMinUpdateTimeSec(minDiffToUpdateRegisteredAlarmTimeSec),
32 mUpdateAlarm(updateAlarm),
33 mCancelAlarm(cancelAlarm) {}
34
~AlarmMonitor()35 AlarmMonitor::~AlarmMonitor() {}
36
setStatsCompanionService(sp<IStatsCompanionService> statsCompanionService)37 void AlarmMonitor::setStatsCompanionService(sp<IStatsCompanionService> statsCompanionService) {
38 std::lock_guard<std::mutex> lock(mLock);
39 sp<IStatsCompanionService> tmpForLock = mStatsCompanionService;
40 mStatsCompanionService = statsCompanionService;
41 if (statsCompanionService == nullptr) {
42 VLOG("Erasing link to statsCompanionService");
43 return;
44 }
45 VLOG("Creating link to statsCompanionService");
46 const sp<const InternalAlarm> top = mPq.top();
47 if (top != nullptr) {
48 updateRegisteredAlarmTime_l(top->timestampSec);
49 }
50 }
51
add(sp<const InternalAlarm> alarm)52 void AlarmMonitor::add(sp<const InternalAlarm> alarm) {
53 std::lock_guard<std::mutex> lock(mLock);
54 if (alarm == nullptr) {
55 ALOGW("Asked to add a null alarm.");
56 return;
57 }
58 if (alarm->timestampSec < 1) {
59 // forbidden since a timestamp 0 is used to indicate no alarm registered
60 ALOGW("Asked to add a 0-time alarm.");
61 return;
62 }
63 // TODO: Ensure that refractory period is respected.
64 VLOG("Adding alarm with time %u", alarm->timestampSec);
65 mPq.push(alarm);
66 if (mRegisteredAlarmTimeSec < 1 ||
67 alarm->timestampSec + mMinUpdateTimeSec < mRegisteredAlarmTimeSec) {
68 updateRegisteredAlarmTime_l(alarm->timestampSec);
69 }
70 }
71
remove(sp<const InternalAlarm> alarm)72 void AlarmMonitor::remove(sp<const InternalAlarm> alarm) {
73 std::lock_guard<std::mutex> lock(mLock);
74 if (alarm == nullptr) {
75 ALOGW("Asked to remove a null alarm.");
76 return;
77 }
78 VLOG("Removing alarm with time %u", alarm->timestampSec);
79 bool wasPresent = mPq.remove(alarm);
80 if (!wasPresent) return;
81 if (mPq.empty()) {
82 VLOG("Queue is empty. Cancel any alarm.");
83 cancelRegisteredAlarmTime_l();
84 return;
85 }
86 uint32_t soonestAlarmTimeSec = mPq.top()->timestampSec;
87 VLOG("Soonest alarm is %u", soonestAlarmTimeSec);
88 if (soonestAlarmTimeSec > mRegisteredAlarmTimeSec + mMinUpdateTimeSec) {
89 updateRegisteredAlarmTime_l(soonestAlarmTimeSec);
90 }
91 }
92
93 // More efficient than repeatedly calling remove(mPq.top()) since it batches the
94 // updates to the registered alarm.
popSoonerThan(uint32_t timestampSec)95 unordered_set<sp<const InternalAlarm>, SpHash<InternalAlarm>> AlarmMonitor::popSoonerThan(
96 uint32_t timestampSec) {
97 VLOG("Removing alarms with time <= %u", timestampSec);
98 unordered_set<sp<const InternalAlarm>, SpHash<InternalAlarm>> oldAlarms;
99 std::lock_guard<std::mutex> lock(mLock);
100
101 for (sp<const InternalAlarm> t = mPq.top(); t != nullptr && t->timestampSec <= timestampSec;
102 t = mPq.top()) {
103 oldAlarms.insert(t);
104 mPq.pop(); // remove t
105 }
106 // Always update registered alarm time (if anything has changed).
107 if (!oldAlarms.empty()) {
108 if (mPq.empty()) {
109 VLOG("Queue is empty. Cancel any alarm.");
110 cancelRegisteredAlarmTime_l();
111 } else {
112 // Always update the registered alarm in this case (unlike remove()).
113 updateRegisteredAlarmTime_l(mPq.top()->timestampSec);
114 }
115 }
116 return oldAlarms;
117 }
118
updateRegisteredAlarmTime_l(uint32_t timestampSec)119 void AlarmMonitor::updateRegisteredAlarmTime_l(uint32_t timestampSec) {
120 VLOG("Updating reg alarm time to %u", timestampSec);
121 mRegisteredAlarmTimeSec = timestampSec;
122 mUpdateAlarm(mStatsCompanionService, secToMs(mRegisteredAlarmTimeSec));
123 }
124
cancelRegisteredAlarmTime_l()125 void AlarmMonitor::cancelRegisteredAlarmTime_l() {
126 VLOG("Cancelling reg alarm.");
127 mRegisteredAlarmTimeSec = 0;
128 mCancelAlarm(mStatsCompanionService);
129 }
130
secToMs(uint32_t timeSec)131 int64_t AlarmMonitor::secToMs(uint32_t timeSec) {
132 return ((int64_t)timeSec) * 1000;
133 }
134
135 } // namespace statsd
136 } // namespace os
137 } // namespace android
138