/* * 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. */ #pragma once #include #include "config/ConfigListener.h" #include "logd/LogEvent.h" #include "metrics/MetricsManager.h" #include "packages/UidMap.h" #include "external/StatsPullerManager.h" #include "src/statsd_config.pb.h" #include "src/statsd_metadata.pb.h" #include #include namespace android { namespace os { namespace statsd { class StatsLogProcessor : public ConfigListener, public virtual PackageInfoListener { public: StatsLogProcessor(const sp& uidMap, const sp& pullerManager, const sp& anomalyAlarmMonitor, const sp& subscriberTriggerAlarmMonitor, const int64_t timeBaseNs, const std::function& sendBroadcast, const std::function&)>& sendActivationBroadcast); virtual ~StatsLogProcessor(); void OnLogEvent(LogEvent* event); void OnConfigUpdated(const int64_t timestampNs, const ConfigKey& key, const StatsdConfig& config, bool modularUpdate = true); void OnConfigRemoved(const ConfigKey& key); size_t GetMetricsSize(const ConfigKey& key) const; void GetActiveConfigs(const int uid, vector& outActiveConfigs); void onDumpReport(const ConfigKey& key, const int64_t dumpTimeNs, const bool include_current_partial_bucket, const bool erase_data, const DumpReportReason dumpReportReason, const DumpLatency dumpLatency, vector* outData); void onDumpReport(const ConfigKey& key, const int64_t dumpTimeNs, const bool include_current_partial_bucket, const bool erase_data, const DumpReportReason dumpReportReason, const DumpLatency dumpLatency, ProtoOutputStream* proto); /* Tells MetricsManager that the alarms in alarmSet have fired. Modifies periodic alarmSet. */ void onPeriodicAlarmFired( const int64_t& timestampNs, unordered_set, SpHash> alarmSet); /* Flushes data to disk. Data on memory will be gone after written to disk. */ void WriteDataToDisk(const DumpReportReason dumpReportReason, const DumpLatency dumpLatency, const int64_t elapsedRealtimeNs); /* Persist configs containing metrics with active activations to disk. */ void SaveActiveConfigsToDisk(int64_t currentTimeNs); /* Writes the current active status/ttl for all configs and metrics to ProtoOutputStream. */ void WriteActiveConfigsToProtoOutputStream( int64_t currentTimeNs, const DumpReportReason reason, ProtoOutputStream* proto); /* Load configs containing metrics with active activations from disk. */ void LoadActiveConfigsFromDisk(); /* Persist metadata for configs and metrics to disk. */ void SaveMetadataToDisk(int64_t currentWallClockTimeNs, int64_t systemElapsedTimeNs); /* Writes the statsd metadata for all configs and metrics to StatsMetadataList. */ void WriteMetadataToProto(int64_t currentWallClockTimeNs, int64_t systemElapsedTimeNs, metadata::StatsMetadataList* metadataList); /* Load stats metadata for configs and metrics from disk. */ void LoadMetadataFromDisk(int64_t currentWallClockTimeNs, int64_t systemElapsedTimeNs); /* Sets the metadata for all configs and metrics */ void SetMetadataState(const metadata::StatsMetadataList& statsMetadataList, int64_t currentWallClockTimeNs, int64_t systemElapsedTimeNs); /* Sets the active status/ttl for all configs and metrics to the status in ActiveConfigList. */ void SetConfigsActiveState(const ActiveConfigList& activeConfigList, int64_t currentTimeNs); /* Notify all MetricsManagers of app upgrades */ void notifyAppUpgrade(const int64_t& eventTimeNs, const string& apk, const int uid, const int64_t version) override; /* Notify all MetricsManagers of app removals */ void notifyAppRemoved(const int64_t& eventTimeNs, const string& apk, const int uid) override; /* Notify all MetricsManagers of uid map snapshots received */ void onUidMapReceived(const int64_t& eventTimeNs) override; /* Notify all metrics managers of boot completed * This will force a bucket split when the boot is finished. */ void onStatsdInitCompleted(const int64_t& elapsedTimeNs); // Reset all configs. void resetConfigs(); inline sp getUidMap() { return mUidMap; } void dumpStates(int outFd, bool verbose); void informPullAlarmFired(const int64_t timestampNs); int64_t getLastReportTimeNs(const ConfigKey& key); inline void setPrintLogs(bool enabled) { std::lock_guard lock(mMetricsMutex); mPrintAllLogs = enabled; } // Add a specific config key to the possible configs to dump ASAP. void noteOnDiskData(const ConfigKey& key); void setAnomalyAlarm(const int64_t timeMillis); void cancelAnomalyAlarm(); private: // For testing only. inline sp getAnomalyAlarmMonitor() const { return mAnomalyAlarmMonitor; } inline sp getPeriodicAlarmMonitor() const { return mPeriodicAlarmMonitor; } mutable mutex mMetricsMutex; // Guards mNextAnomalyAlarmTime. A separate mutex is needed because alarms are set/cancelled // in the onLogEvent code path, which is locked by mMetricsMutex. // DO NOT acquire mMetricsMutex while holding mAnomalyAlarmMutex. This can lead to a deadlock. mutable mutex mAnomalyAlarmMutex; std::unordered_map> mMetricsManagers; std::unordered_map mLastBroadcastTimes; // Last time we sent a broadcast to this uid that the active configs had changed. std::unordered_map mLastActivationBroadcastTimes; // Tracks when we last checked the bytes consumed for each config key. std::unordered_map mLastByteSizeTimes; // Tracks which config keys has metric reports on disk std::set mOnDiskDataConfigs; sp mUidMap; // Reference to the UidMap to lookup app name and version for each uid. sp mPullerManager; // Reference to StatsPullerManager sp mAnomalyAlarmMonitor; sp mPeriodicAlarmMonitor; void OnLogEvent(LogEvent* event, int64_t elapsedRealtimeNs); void resetIfConfigTtlExpiredLocked(const int64_t eventTimeNs); void OnConfigUpdatedLocked(const int64_t currentTimestampNs, const ConfigKey& key, const StatsdConfig& config, bool modularUpdate); void GetActiveConfigsLocked(const int uid, vector& outActiveConfigs); void WriteActiveConfigsToProtoOutputStreamLocked( int64_t currentTimeNs, const DumpReportReason reason, ProtoOutputStream* proto); void SetConfigsActiveStateLocked(const ActiveConfigList& activeConfigList, int64_t currentTimeNs); void SetMetadataStateLocked(const metadata::StatsMetadataList& statsMetadataList, int64_t currentWallClockTimeNs, int64_t systemElapsedTimeNs); void WriteMetadataToProtoLocked(int64_t currentWallClockTimeNs, int64_t systemElapsedTimeNs, metadata::StatsMetadataList* metadataList); void WriteDataToDiskLocked(const DumpReportReason dumpReportReason, const DumpLatency dumpLatency, const int64_t elapsedRealtimeNs); void WriteDataToDiskLocked(const ConfigKey& key, const int64_t timestampNs, const DumpReportReason dumpReportReason, const DumpLatency dumpLatency); void onConfigMetricsReportLocked( const ConfigKey& key, const int64_t dumpTimeStampNs, const bool include_current_partial_bucket, const bool erase_data, const DumpReportReason dumpReportReason, const DumpLatency dumpLatency, /*if dataSavedToDisk is true, it indicates the caller will write the data to disk (e.g., before reboot). So no need to further persist local history.*/ const bool dataSavedToDisk, vector* proto); /* Check if we should send a broadcast if approaching memory limits and if we're over, we * actually delete the data. */ void flushIfNecessaryLocked(const ConfigKey& key, MetricsManager& metricsManager); // Maps the isolated uid in the log event to host uid if the log event contains uid fields. void mapIsolatedUidToHostUidIfNecessaryLocked(LogEvent* event) const; // Handler over the isolated uid change event. void onIsolatedUidChangedEventLocked(const LogEvent& event); // Handler over the binary push state changed event. void onBinaryPushStateChangedEventLocked(LogEvent* event); // Handler over the watchdog rollback occurred event. void onWatchdogRollbackOccurredLocked(LogEvent* event); // Updates train info on disk based on binary push state changed info and // write disk info into parameters. void getAndUpdateTrainInfoOnDisk(bool is_rollback, InstallTrainInfo* trainInfoIn); // Gets experiment ids on disk for associated train and updates them // depending on rollback type. Then writes them back to disk and returns // them. std::vector processWatchdogRollbackOccurred(const int32_t rollbackTypeIn, const string& packageName); // Reset all configs. void resetConfigsLocked(const int64_t timestampNs); // Reset the specified configs. void resetConfigsLocked(const int64_t timestampNs, const std::vector& configs); // An anomaly alarm should have fired. // Check with anomaly alarm manager to find the alarms and process the result. void informAnomalyAlarmFiredLocked(const int64_t elapsedTimeMillis); /* Tells MetricsManager that the alarms in alarmSet have fired. Modifies anomaly alarmSet. */ void processFiredAnomalyAlarmsLocked( const int64_t& timestampNs, unordered_set, SpHash> alarmSet); // Function used to send a broadcast so that receiver for the config key can call getData // to retrieve the stored data. std::function mSendBroadcast; // Function used to send a broadcast so that receiver can be notified of which configs // are currently active. std::function& configIds)> mSendActivationBroadcast; const int64_t mTimeBaseNs; // Largest timestamp of the events that we have processed. int64_t mLargestTimestampSeen = 0; int64_t mLastTimestampSeen = 0; int64_t mLastPullerCacheClearTimeSec = 0; // Last time we wrote data to disk. int64_t mLastWriteTimeNs = 0; // Last time we wrote active metrics to disk. int64_t mLastActiveMetricsWriteNs = 0; //Last time we wrote metadata to disk. int64_t mLastMetadataWriteNs = 0; // The time for the next anomaly alarm for alerts. int64_t mNextAnomalyAlarmTime = 0; bool mPrintAllLogs = false; FRIEND_TEST(StatsLogProcessorTest, TestOutOfOrderLogs); FRIEND_TEST(StatsLogProcessorTest, TestRateLimitByteSize); FRIEND_TEST(StatsLogProcessorTest, TestRateLimitBroadcast); FRIEND_TEST(StatsLogProcessorTest, TestDropWhenByteSizeTooLarge); FRIEND_TEST(StatsLogProcessorTest, InvalidConfigRemoved); FRIEND_TEST(StatsLogProcessorTest, TestActiveConfigMetricDiskWriteRead); FRIEND_TEST(StatsLogProcessorTest, TestActivationOnBoot); FRIEND_TEST(StatsLogProcessorTest, TestActivationOnBootMultipleActivations); FRIEND_TEST(StatsLogProcessorTest, TestActivationOnBootMultipleActivationsDifferentActivationTypes); FRIEND_TEST(StatsLogProcessorTest, TestActivationsPersistAcrossSystemServerRestart); FRIEND_TEST(WakelockDurationE2eTest, TestAggregatedPredicateDimensionsForSumDuration1); FRIEND_TEST(WakelockDurationE2eTest, TestAggregatedPredicateDimensionsForSumDuration2); FRIEND_TEST(WakelockDurationE2eTest, TestAggregatedPredicateDimensionsForSumDuration3); FRIEND_TEST(WakelockDurationE2eTest, TestAggregatedPredicateDimensionsForMaxDuration1); FRIEND_TEST(WakelockDurationE2eTest, TestAggregatedPredicateDimensionsForMaxDuration2); FRIEND_TEST(WakelockDurationE2eTest, TestAggregatedPredicateDimensionsForMaxDuration3); FRIEND_TEST(MetricConditionLinkE2eTest, TestMultiplePredicatesAndLinks1); FRIEND_TEST(MetricConditionLinkE2eTest, TestMultiplePredicatesAndLinks2); FRIEND_TEST(AttributionE2eTest, TestAttributionMatchAndSliceByFirstUid); FRIEND_TEST(AttributionE2eTest, TestAttributionMatchAndSliceByChain); FRIEND_TEST(GaugeMetricE2eTest, TestMultipleFieldsForPushedEvent); FRIEND_TEST(GaugeMetricE2eTest, TestRandomSamplePulledEvents); FRIEND_TEST(GaugeMetricE2eTest, TestRandomSamplePulledEvent_LateAlarm); FRIEND_TEST(GaugeMetricE2eTest, TestRandomSamplePulledEventsWithActivation); FRIEND_TEST(GaugeMetricE2eTest, TestRandomSamplePulledEventsNoCondition); FRIEND_TEST(GaugeMetricE2eTest, TestConditionChangeToTrueSamplePulledEvents); FRIEND_TEST(AnomalyDetectionE2eTest, TestSlicedCountMetric_single_bucket); FRIEND_TEST(AnomalyDetectionE2eTest, TestSlicedCountMetric_multiple_buckets); FRIEND_TEST(AnomalyDetectionE2eTest, TestCountMetric_save_refractory_to_disk_no_data_written); FRIEND_TEST(AnomalyDetectionE2eTest, TestCountMetric_save_refractory_to_disk); FRIEND_TEST(AnomalyDetectionE2eTest, TestCountMetric_load_refractory_from_disk); FRIEND_TEST(AnomalyDetectionE2eTest, TestDurationMetric_SUM_single_bucket); FRIEND_TEST(AnomalyDetectionE2eTest, TestDurationMetric_SUM_partial_bucket); FRIEND_TEST(AnomalyDetectionE2eTest, TestDurationMetric_SUM_multiple_buckets); FRIEND_TEST(AnomalyDetectionE2eTest, TestDurationMetric_SUM_long_refractory_period); FRIEND_TEST(AlarmE2eTest, TestMultipleAlarms); FRIEND_TEST(ConfigTtlE2eTest, TestCountMetric); FRIEND_TEST(MetricActivationE2eTest, TestCountMetric); FRIEND_TEST(MetricActivationE2eTest, TestCountMetricWithOneDeactivation); FRIEND_TEST(MetricActivationE2eTest, TestCountMetricWithTwoDeactivations); FRIEND_TEST(MetricActivationE2eTest, TestCountMetricWithSameDeactivation); FRIEND_TEST(MetricActivationE2eTest, TestCountMetricWithTwoMetricsTwoDeactivations); FRIEND_TEST(ConfigUpdateE2eTest, TestAlarms); FRIEND_TEST(ConfigUpdateE2eTest, TestGaugeMetric); FRIEND_TEST(ConfigUpdateE2eTest, TestValueMetric); FRIEND_TEST(ConfigUpdateE2eTest, TestAnomalyDurationMetric); FRIEND_TEST(ConfigUpdateE2eAbTest, TestHashStrings); FRIEND_TEST(ConfigUpdateE2eAbTest, TestUidMapVersionStringInstaller); FRIEND_TEST(ConfigUpdateE2eAbTest, TestConfigTtl); FRIEND_TEST(CountMetricE2eTest, TestInitialConditionChanges); FRIEND_TEST(CountMetricE2eTest, TestSlicedState); FRIEND_TEST(CountMetricE2eTest, TestSlicedStateWithMap); FRIEND_TEST(CountMetricE2eTest, TestMultipleSlicedStates); FRIEND_TEST(CountMetricE2eTest, TestSlicedStateWithPrimaryFields); FRIEND_TEST(DurationMetricE2eTest, TestOneBucket); FRIEND_TEST(DurationMetricE2eTest, TestTwoBuckets); FRIEND_TEST(DurationMetricE2eTest, TestWithActivation); FRIEND_TEST(DurationMetricE2eTest, TestWithCondition); FRIEND_TEST(DurationMetricE2eTest, TestWithSlicedCondition); FRIEND_TEST(DurationMetricE2eTest, TestWithActivationAndSlicedCondition); FRIEND_TEST(DurationMetricE2eTest, TestWithSlicedState); FRIEND_TEST(DurationMetricE2eTest, TestWithConditionAndSlicedState); FRIEND_TEST(DurationMetricE2eTest, TestWithSlicedStateMapped); FRIEND_TEST(DurationMetricE2eTest, TestSlicedStatePrimaryFieldsNotSubsetDimInWhat); FRIEND_TEST(DurationMetricE2eTest, TestWithSlicedStatePrimaryFieldsSubset); FRIEND_TEST(DurationMetricE2eTest, TestUploadThreshold); FRIEND_TEST(ValueMetricE2eTest, TestInitialConditionChanges); FRIEND_TEST(ValueMetricE2eTest, TestPulledEvents); FRIEND_TEST(ValueMetricE2eTest, TestPulledEvents_LateAlarm); FRIEND_TEST(ValueMetricE2eTest, TestPulledEvents_WithActivation); FRIEND_TEST(ValueMetricE2eTest, TestInitWithSlicedState); FRIEND_TEST(ValueMetricE2eTest, TestInitWithSlicedState_WithDimensions); FRIEND_TEST(ValueMetricE2eTest, TestInitWithSlicedState_WithIncorrectDimensions); }; } // namespace statsd } // namespace os } // namespace android