You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1629 lines
68 KiB
1629 lines
68 KiB
// 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.
|
|
|
|
#include "statsd_test_util.h"
|
|
|
|
#include <aidl/android/util/StatsEventParcel.h>
|
|
|
|
#include "matchers/SimpleAtomMatchingTracker.h"
|
|
#include "stats_annotations.h"
|
|
#include "stats_event.h"
|
|
|
|
using aidl::android::util::StatsEventParcel;
|
|
using std::shared_ptr;
|
|
|
|
namespace android {
|
|
namespace os {
|
|
namespace statsd {
|
|
|
|
StatsLogReport outputStreamToProto(ProtoOutputStream* proto) {
|
|
vector<uint8_t> bytes;
|
|
bytes.resize(proto->size());
|
|
size_t pos = 0;
|
|
sp<ProtoReader> reader = proto->data();
|
|
|
|
while (reader->readBuffer() != NULL) {
|
|
size_t toRead = reader->currentToRead();
|
|
std::memcpy(&((bytes)[pos]), reader->readBuffer(), toRead);
|
|
pos += toRead;
|
|
reader->move(toRead);
|
|
}
|
|
|
|
StatsLogReport report;
|
|
report.ParseFromArray(bytes.data(), bytes.size());
|
|
return report;
|
|
}
|
|
|
|
AtomMatcher CreateSimpleAtomMatcher(const string& name, int atomId) {
|
|
AtomMatcher atom_matcher;
|
|
atom_matcher.set_id(StringToId(name));
|
|
auto simple_atom_matcher = atom_matcher.mutable_simple_atom_matcher();
|
|
simple_atom_matcher->set_atom_id(atomId);
|
|
return atom_matcher;
|
|
}
|
|
|
|
AtomMatcher CreateTemperatureAtomMatcher() {
|
|
return CreateSimpleAtomMatcher("TemperatureMatcher", util::TEMPERATURE);
|
|
}
|
|
|
|
AtomMatcher CreateScheduledJobStateChangedAtomMatcher(const string& name,
|
|
ScheduledJobStateChanged::State state) {
|
|
AtomMatcher atom_matcher;
|
|
atom_matcher.set_id(StringToId(name));
|
|
auto simple_atom_matcher = atom_matcher.mutable_simple_atom_matcher();
|
|
simple_atom_matcher->set_atom_id(util::SCHEDULED_JOB_STATE_CHANGED);
|
|
auto field_value_matcher = simple_atom_matcher->add_field_value_matcher();
|
|
field_value_matcher->set_field(3); // State field.
|
|
field_value_matcher->set_eq_int(state);
|
|
return atom_matcher;
|
|
}
|
|
|
|
AtomMatcher CreateStartScheduledJobAtomMatcher() {
|
|
return CreateScheduledJobStateChangedAtomMatcher("ScheduledJobStart",
|
|
ScheduledJobStateChanged::STARTED);
|
|
}
|
|
|
|
AtomMatcher CreateFinishScheduledJobAtomMatcher() {
|
|
return CreateScheduledJobStateChangedAtomMatcher("ScheduledJobFinish",
|
|
ScheduledJobStateChanged::FINISHED);
|
|
}
|
|
|
|
AtomMatcher CreateScreenBrightnessChangedAtomMatcher() {
|
|
AtomMatcher atom_matcher;
|
|
atom_matcher.set_id(StringToId("ScreenBrightnessChanged"));
|
|
auto simple_atom_matcher = atom_matcher.mutable_simple_atom_matcher();
|
|
simple_atom_matcher->set_atom_id(util::SCREEN_BRIGHTNESS_CHANGED);
|
|
return atom_matcher;
|
|
}
|
|
|
|
AtomMatcher CreateUidProcessStateChangedAtomMatcher() {
|
|
AtomMatcher atom_matcher;
|
|
atom_matcher.set_id(StringToId("UidProcessStateChanged"));
|
|
auto simple_atom_matcher = atom_matcher.mutable_simple_atom_matcher();
|
|
simple_atom_matcher->set_atom_id(util::UID_PROCESS_STATE_CHANGED);
|
|
return atom_matcher;
|
|
}
|
|
|
|
AtomMatcher CreateWakelockStateChangedAtomMatcher(const string& name,
|
|
WakelockStateChanged::State state) {
|
|
AtomMatcher atom_matcher;
|
|
atom_matcher.set_id(StringToId(name));
|
|
auto simple_atom_matcher = atom_matcher.mutable_simple_atom_matcher();
|
|
simple_atom_matcher->set_atom_id(util::WAKELOCK_STATE_CHANGED);
|
|
auto field_value_matcher = simple_atom_matcher->add_field_value_matcher();
|
|
field_value_matcher->set_field(4); // State field.
|
|
field_value_matcher->set_eq_int(state);
|
|
return atom_matcher;
|
|
}
|
|
|
|
AtomMatcher CreateAcquireWakelockAtomMatcher() {
|
|
return CreateWakelockStateChangedAtomMatcher("AcquireWakelock", WakelockStateChanged::ACQUIRE);
|
|
}
|
|
|
|
AtomMatcher CreateReleaseWakelockAtomMatcher() {
|
|
return CreateWakelockStateChangedAtomMatcher("ReleaseWakelock", WakelockStateChanged::RELEASE);
|
|
}
|
|
|
|
AtomMatcher CreateBatterySaverModeStateChangedAtomMatcher(
|
|
const string& name, BatterySaverModeStateChanged::State state) {
|
|
AtomMatcher atom_matcher;
|
|
atom_matcher.set_id(StringToId(name));
|
|
auto simple_atom_matcher = atom_matcher.mutable_simple_atom_matcher();
|
|
simple_atom_matcher->set_atom_id(util::BATTERY_SAVER_MODE_STATE_CHANGED);
|
|
auto field_value_matcher = simple_atom_matcher->add_field_value_matcher();
|
|
field_value_matcher->set_field(1); // State field.
|
|
field_value_matcher->set_eq_int(state);
|
|
return atom_matcher;
|
|
}
|
|
|
|
AtomMatcher CreateBatterySaverModeStartAtomMatcher() {
|
|
return CreateBatterySaverModeStateChangedAtomMatcher(
|
|
"BatterySaverModeStart", BatterySaverModeStateChanged::ON);
|
|
}
|
|
|
|
|
|
AtomMatcher CreateBatterySaverModeStopAtomMatcher() {
|
|
return CreateBatterySaverModeStateChangedAtomMatcher(
|
|
"BatterySaverModeStop", BatterySaverModeStateChanged::OFF);
|
|
}
|
|
|
|
AtomMatcher CreateBatteryStateChangedAtomMatcher(const string& name,
|
|
BatteryPluggedStateEnum state) {
|
|
AtomMatcher atom_matcher;
|
|
atom_matcher.set_id(StringToId(name));
|
|
auto simple_atom_matcher = atom_matcher.mutable_simple_atom_matcher();
|
|
simple_atom_matcher->set_atom_id(util::PLUGGED_STATE_CHANGED);
|
|
auto field_value_matcher = simple_atom_matcher->add_field_value_matcher();
|
|
field_value_matcher->set_field(1); // State field.
|
|
field_value_matcher->set_eq_int(state);
|
|
return atom_matcher;
|
|
}
|
|
|
|
AtomMatcher CreateBatteryStateNoneMatcher() {
|
|
return CreateBatteryStateChangedAtomMatcher("BatteryPluggedNone",
|
|
BatteryPluggedStateEnum::BATTERY_PLUGGED_NONE);
|
|
}
|
|
|
|
AtomMatcher CreateBatteryStateUsbMatcher() {
|
|
return CreateBatteryStateChangedAtomMatcher("BatteryPluggedUsb",
|
|
BatteryPluggedStateEnum::BATTERY_PLUGGED_USB);
|
|
}
|
|
|
|
AtomMatcher CreateScreenStateChangedAtomMatcher(
|
|
const string& name, android::view::DisplayStateEnum state) {
|
|
AtomMatcher atom_matcher;
|
|
atom_matcher.set_id(StringToId(name));
|
|
auto simple_atom_matcher = atom_matcher.mutable_simple_atom_matcher();
|
|
simple_atom_matcher->set_atom_id(util::SCREEN_STATE_CHANGED);
|
|
auto field_value_matcher = simple_atom_matcher->add_field_value_matcher();
|
|
field_value_matcher->set_field(1); // State field.
|
|
field_value_matcher->set_eq_int(state);
|
|
return atom_matcher;
|
|
}
|
|
|
|
AtomMatcher CreateScreenTurnedOnAtomMatcher() {
|
|
return CreateScreenStateChangedAtomMatcher("ScreenTurnedOn",
|
|
android::view::DisplayStateEnum::DISPLAY_STATE_ON);
|
|
}
|
|
|
|
AtomMatcher CreateScreenTurnedOffAtomMatcher() {
|
|
return CreateScreenStateChangedAtomMatcher("ScreenTurnedOff",
|
|
::android::view::DisplayStateEnum::DISPLAY_STATE_OFF);
|
|
}
|
|
|
|
AtomMatcher CreateSyncStateChangedAtomMatcher(
|
|
const string& name, SyncStateChanged::State state) {
|
|
AtomMatcher atom_matcher;
|
|
atom_matcher.set_id(StringToId(name));
|
|
auto simple_atom_matcher = atom_matcher.mutable_simple_atom_matcher();
|
|
simple_atom_matcher->set_atom_id(util::SYNC_STATE_CHANGED);
|
|
auto field_value_matcher = simple_atom_matcher->add_field_value_matcher();
|
|
field_value_matcher->set_field(3); // State field.
|
|
field_value_matcher->set_eq_int(state);
|
|
return atom_matcher;
|
|
}
|
|
|
|
AtomMatcher CreateSyncStartAtomMatcher() {
|
|
return CreateSyncStateChangedAtomMatcher("SyncStart", SyncStateChanged::ON);
|
|
}
|
|
|
|
AtomMatcher CreateSyncEndAtomMatcher() {
|
|
return CreateSyncStateChangedAtomMatcher("SyncEnd", SyncStateChanged::OFF);
|
|
}
|
|
|
|
AtomMatcher CreateActivityForegroundStateChangedAtomMatcher(
|
|
const string& name, ActivityForegroundStateChanged::State state) {
|
|
AtomMatcher atom_matcher;
|
|
atom_matcher.set_id(StringToId(name));
|
|
auto simple_atom_matcher = atom_matcher.mutable_simple_atom_matcher();
|
|
simple_atom_matcher->set_atom_id(util::ACTIVITY_FOREGROUND_STATE_CHANGED);
|
|
auto field_value_matcher = simple_atom_matcher->add_field_value_matcher();
|
|
field_value_matcher->set_field(4); // Activity field.
|
|
field_value_matcher->set_eq_int(state);
|
|
return atom_matcher;
|
|
}
|
|
|
|
AtomMatcher CreateMoveToBackgroundAtomMatcher() {
|
|
return CreateActivityForegroundStateChangedAtomMatcher(
|
|
"Background", ActivityForegroundStateChanged::BACKGROUND);
|
|
}
|
|
|
|
AtomMatcher CreateMoveToForegroundAtomMatcher() {
|
|
return CreateActivityForegroundStateChangedAtomMatcher(
|
|
"Foreground", ActivityForegroundStateChanged::FOREGROUND);
|
|
}
|
|
|
|
AtomMatcher CreateProcessLifeCycleStateChangedAtomMatcher(
|
|
const string& name, ProcessLifeCycleStateChanged::State state) {
|
|
AtomMatcher atom_matcher;
|
|
atom_matcher.set_id(StringToId(name));
|
|
auto simple_atom_matcher = atom_matcher.mutable_simple_atom_matcher();
|
|
simple_atom_matcher->set_atom_id(util::PROCESS_LIFE_CYCLE_STATE_CHANGED);
|
|
auto field_value_matcher = simple_atom_matcher->add_field_value_matcher();
|
|
field_value_matcher->set_field(3); // Process state field.
|
|
field_value_matcher->set_eq_int(state);
|
|
return atom_matcher;
|
|
}
|
|
|
|
AtomMatcher CreateProcessCrashAtomMatcher() {
|
|
return CreateProcessLifeCycleStateChangedAtomMatcher(
|
|
"Crashed", ProcessLifeCycleStateChanged::CRASHED);
|
|
}
|
|
|
|
void addMatcherToMatcherCombination(const AtomMatcher& matcher, AtomMatcher* combinationMatcher) {
|
|
combinationMatcher->mutable_combination()->add_matcher(matcher.id());
|
|
}
|
|
|
|
Predicate CreateScheduledJobPredicate() {
|
|
Predicate predicate;
|
|
predicate.set_id(StringToId("ScheduledJobRunningPredicate"));
|
|
predicate.mutable_simple_predicate()->set_start(StringToId("ScheduledJobStart"));
|
|
predicate.mutable_simple_predicate()->set_stop(StringToId("ScheduledJobFinish"));
|
|
return predicate;
|
|
}
|
|
|
|
Predicate CreateBatterySaverModePredicate() {
|
|
Predicate predicate;
|
|
predicate.set_id(StringToId("BatterySaverIsOn"));
|
|
predicate.mutable_simple_predicate()->set_start(StringToId("BatterySaverModeStart"));
|
|
predicate.mutable_simple_predicate()->set_stop(StringToId("BatterySaverModeStop"));
|
|
return predicate;
|
|
}
|
|
|
|
Predicate CreateDeviceUnpluggedPredicate() {
|
|
Predicate predicate;
|
|
predicate.set_id(StringToId("DeviceUnplugged"));
|
|
predicate.mutable_simple_predicate()->set_start(StringToId("BatteryPluggedNone"));
|
|
predicate.mutable_simple_predicate()->set_stop(StringToId("BatteryPluggedUsb"));
|
|
return predicate;
|
|
}
|
|
|
|
Predicate CreateScreenIsOnPredicate() {
|
|
Predicate predicate;
|
|
predicate.set_id(StringToId("ScreenIsOn"));
|
|
predicate.mutable_simple_predicate()->set_start(StringToId("ScreenTurnedOn"));
|
|
predicate.mutable_simple_predicate()->set_stop(StringToId("ScreenTurnedOff"));
|
|
return predicate;
|
|
}
|
|
|
|
Predicate CreateScreenIsOffPredicate() {
|
|
Predicate predicate;
|
|
predicate.set_id(1111123);
|
|
predicate.mutable_simple_predicate()->set_start(StringToId("ScreenTurnedOff"));
|
|
predicate.mutable_simple_predicate()->set_stop(StringToId("ScreenTurnedOn"));
|
|
return predicate;
|
|
}
|
|
|
|
Predicate CreateHoldingWakelockPredicate() {
|
|
Predicate predicate;
|
|
predicate.set_id(StringToId("HoldingWakelock"));
|
|
predicate.mutable_simple_predicate()->set_start(StringToId("AcquireWakelock"));
|
|
predicate.mutable_simple_predicate()->set_stop(StringToId("ReleaseWakelock"));
|
|
return predicate;
|
|
}
|
|
|
|
Predicate CreateIsSyncingPredicate() {
|
|
Predicate predicate;
|
|
predicate.set_id(33333333333333);
|
|
predicate.mutable_simple_predicate()->set_start(StringToId("SyncStart"));
|
|
predicate.mutable_simple_predicate()->set_stop(StringToId("SyncEnd"));
|
|
return predicate;
|
|
}
|
|
|
|
Predicate CreateIsInBackgroundPredicate() {
|
|
Predicate predicate;
|
|
predicate.set_id(StringToId("IsInBackground"));
|
|
predicate.mutable_simple_predicate()->set_start(StringToId("Background"));
|
|
predicate.mutable_simple_predicate()->set_stop(StringToId("Foreground"));
|
|
return predicate;
|
|
}
|
|
|
|
State CreateScreenState() {
|
|
State state;
|
|
state.set_id(StringToId("ScreenState"));
|
|
state.set_atom_id(util::SCREEN_STATE_CHANGED);
|
|
return state;
|
|
}
|
|
|
|
State CreateUidProcessState() {
|
|
State state;
|
|
state.set_id(StringToId("UidProcessState"));
|
|
state.set_atom_id(util::UID_PROCESS_STATE_CHANGED);
|
|
return state;
|
|
}
|
|
|
|
State CreateOverlayState() {
|
|
State state;
|
|
state.set_id(StringToId("OverlayState"));
|
|
state.set_atom_id(util::OVERLAY_STATE_CHANGED);
|
|
return state;
|
|
}
|
|
|
|
State CreateScreenStateWithOnOffMap(int64_t screenOnId, int64_t screenOffId) {
|
|
State state;
|
|
state.set_id(StringToId("ScreenStateOnOff"));
|
|
state.set_atom_id(util::SCREEN_STATE_CHANGED);
|
|
|
|
auto map = CreateScreenStateOnOffMap(screenOnId, screenOffId);
|
|
*state.mutable_map() = map;
|
|
|
|
return state;
|
|
}
|
|
|
|
State CreateScreenStateWithSimpleOnOffMap(int64_t screenOnId, int64_t screenOffId) {
|
|
State state;
|
|
state.set_id(StringToId("ScreenStateSimpleOnOff"));
|
|
state.set_atom_id(util::SCREEN_STATE_CHANGED);
|
|
|
|
auto map = CreateScreenStateSimpleOnOffMap(screenOnId, screenOffId);
|
|
*state.mutable_map() = map;
|
|
|
|
return state;
|
|
}
|
|
|
|
StateMap_StateGroup CreateScreenStateOnGroup(int64_t screenOnId) {
|
|
StateMap_StateGroup group;
|
|
group.set_group_id(screenOnId);
|
|
group.add_value(android::view::DisplayStateEnum::DISPLAY_STATE_ON);
|
|
group.add_value(android::view::DisplayStateEnum::DISPLAY_STATE_VR);
|
|
group.add_value(android::view::DisplayStateEnum::DISPLAY_STATE_ON_SUSPEND);
|
|
return group;
|
|
}
|
|
|
|
StateMap_StateGroup CreateScreenStateOffGroup(int64_t screenOffId) {
|
|
StateMap_StateGroup group;
|
|
group.set_group_id(screenOffId);
|
|
group.add_value(android::view::DisplayStateEnum::DISPLAY_STATE_OFF);
|
|
group.add_value(android::view::DisplayStateEnum::DISPLAY_STATE_DOZE);
|
|
group.add_value(android::view::DisplayStateEnum::DISPLAY_STATE_DOZE_SUSPEND);
|
|
return group;
|
|
}
|
|
|
|
StateMap_StateGroup CreateScreenStateSimpleOnGroup(int64_t screenOnId) {
|
|
StateMap_StateGroup group;
|
|
group.set_group_id(screenOnId);
|
|
group.add_value(android::view::DisplayStateEnum::DISPLAY_STATE_ON);
|
|
return group;
|
|
}
|
|
|
|
StateMap_StateGroup CreateScreenStateSimpleOffGroup(int64_t screenOffId) {
|
|
StateMap_StateGroup group;
|
|
group.set_group_id(screenOffId);
|
|
group.add_value(android::view::DisplayStateEnum::DISPLAY_STATE_OFF);
|
|
return group;
|
|
}
|
|
|
|
StateMap CreateScreenStateOnOffMap(int64_t screenOnId, int64_t screenOffId) {
|
|
StateMap map;
|
|
*map.add_group() = CreateScreenStateOnGroup(screenOnId);
|
|
*map.add_group() = CreateScreenStateOffGroup(screenOffId);
|
|
return map;
|
|
}
|
|
|
|
StateMap CreateScreenStateSimpleOnOffMap(int64_t screenOnId, int64_t screenOffId) {
|
|
StateMap map;
|
|
*map.add_group() = CreateScreenStateSimpleOnGroup(screenOnId);
|
|
*map.add_group() = CreateScreenStateSimpleOffGroup(screenOffId);
|
|
return map;
|
|
}
|
|
|
|
void addPredicateToPredicateCombination(const Predicate& predicate,
|
|
Predicate* combinationPredicate) {
|
|
combinationPredicate->mutable_combination()->add_predicate(predicate.id());
|
|
}
|
|
|
|
FieldMatcher CreateAttributionUidDimensions(const int atomId,
|
|
const std::vector<Position>& positions) {
|
|
FieldMatcher dimensions;
|
|
dimensions.set_field(atomId);
|
|
for (const auto position : positions) {
|
|
auto child = dimensions.add_child();
|
|
child->set_field(1);
|
|
child->set_position(position);
|
|
child->add_child()->set_field(1);
|
|
}
|
|
return dimensions;
|
|
}
|
|
|
|
FieldMatcher CreateAttributionUidAndTagDimensions(const int atomId,
|
|
const std::vector<Position>& positions) {
|
|
FieldMatcher dimensions;
|
|
dimensions.set_field(atomId);
|
|
for (const auto position : positions) {
|
|
auto child = dimensions.add_child();
|
|
child->set_field(1);
|
|
child->set_position(position);
|
|
child->add_child()->set_field(1);
|
|
child->add_child()->set_field(2);
|
|
}
|
|
return dimensions;
|
|
}
|
|
|
|
FieldMatcher CreateDimensions(const int atomId, const std::vector<int>& fields) {
|
|
FieldMatcher dimensions;
|
|
dimensions.set_field(atomId);
|
|
for (const int field : fields) {
|
|
dimensions.add_child()->set_field(field);
|
|
}
|
|
return dimensions;
|
|
}
|
|
|
|
FieldMatcher CreateAttributionUidAndOtherDimensions(const int atomId,
|
|
const std::vector<Position>& positions,
|
|
const std::vector<int>& fields) {
|
|
FieldMatcher dimensions = CreateAttributionUidDimensions(atomId, positions);
|
|
|
|
for (const int field : fields) {
|
|
dimensions.add_child()->set_field(field);
|
|
}
|
|
return dimensions;
|
|
}
|
|
|
|
EventMetric createEventMetric(const string& name, const int64_t what,
|
|
const optional<int64_t>& condition) {
|
|
EventMetric metric;
|
|
metric.set_id(StringToId(name));
|
|
metric.set_what(what);
|
|
if (condition) {
|
|
metric.set_condition(condition.value());
|
|
}
|
|
return metric;
|
|
}
|
|
|
|
CountMetric createCountMetric(const string& name, const int64_t what,
|
|
const optional<int64_t>& condition, const vector<int64_t>& states) {
|
|
CountMetric metric;
|
|
metric.set_id(StringToId(name));
|
|
metric.set_what(what);
|
|
metric.set_bucket(TEN_MINUTES);
|
|
if (condition) {
|
|
metric.set_condition(condition.value());
|
|
}
|
|
for (const int64_t state : states) {
|
|
metric.add_slice_by_state(state);
|
|
}
|
|
return metric;
|
|
}
|
|
|
|
DurationMetric createDurationMetric(const string& name, const int64_t what,
|
|
const optional<int64_t>& condition,
|
|
const vector<int64_t>& states) {
|
|
DurationMetric metric;
|
|
metric.set_id(StringToId(name));
|
|
metric.set_what(what);
|
|
metric.set_bucket(TEN_MINUTES);
|
|
if (condition) {
|
|
metric.set_condition(condition.value());
|
|
}
|
|
for (const int64_t state : states) {
|
|
metric.add_slice_by_state(state);
|
|
}
|
|
return metric;
|
|
}
|
|
|
|
GaugeMetric createGaugeMetric(const string& name, const int64_t what,
|
|
const GaugeMetric::SamplingType samplingType,
|
|
const optional<int64_t>& condition,
|
|
const optional<int64_t>& triggerEvent) {
|
|
GaugeMetric metric;
|
|
metric.set_id(StringToId(name));
|
|
metric.set_what(what);
|
|
metric.set_bucket(TEN_MINUTES);
|
|
metric.set_sampling_type(samplingType);
|
|
if (condition) {
|
|
metric.set_condition(condition.value());
|
|
}
|
|
if (triggerEvent) {
|
|
metric.set_trigger_event(triggerEvent.value());
|
|
}
|
|
metric.mutable_gauge_fields_filter()->set_include_all(true);
|
|
return metric;
|
|
}
|
|
|
|
ValueMetric createValueMetric(const string& name, const AtomMatcher& what, const int valueField,
|
|
const optional<int64_t>& condition, const vector<int64_t>& states) {
|
|
ValueMetric metric;
|
|
metric.set_id(StringToId(name));
|
|
metric.set_what(what.id());
|
|
metric.set_bucket(TEN_MINUTES);
|
|
metric.mutable_value_field()->set_field(what.simple_atom_matcher().atom_id());
|
|
metric.mutable_value_field()->add_child()->set_field(valueField);
|
|
if (condition) {
|
|
metric.set_condition(condition.value());
|
|
}
|
|
for (const int64_t state : states) {
|
|
metric.add_slice_by_state(state);
|
|
}
|
|
return metric;
|
|
}
|
|
|
|
Alert createAlert(const string& name, const int64_t metricId, const int buckets,
|
|
const int64_t triggerSum) {
|
|
Alert alert;
|
|
alert.set_id(StringToId(name));
|
|
alert.set_metric_id(metricId);
|
|
alert.set_num_buckets(buckets);
|
|
alert.set_trigger_if_sum_gt(triggerSum);
|
|
return alert;
|
|
}
|
|
|
|
Alarm createAlarm(const string& name, const int64_t offsetMillis, const int64_t periodMillis) {
|
|
Alarm alarm;
|
|
alarm.set_id(StringToId(name));
|
|
alarm.set_offset_millis(offsetMillis);
|
|
alarm.set_period_millis(periodMillis);
|
|
return alarm;
|
|
}
|
|
|
|
Subscription createSubscription(const string& name, const Subscription_RuleType type,
|
|
const int64_t ruleId) {
|
|
Subscription subscription;
|
|
subscription.set_id(StringToId(name));
|
|
subscription.set_rule_type(type);
|
|
subscription.set_rule_id(ruleId);
|
|
subscription.mutable_broadcast_subscriber_details();
|
|
return subscription;
|
|
}
|
|
|
|
// START: get primary key functions
|
|
void getUidProcessKey(int uid, HashableDimensionKey* key) {
|
|
int pos1[] = {1, 0, 0};
|
|
Field field1(27 /* atom id */, pos1, 0 /* depth */);
|
|
Value value1((int32_t)uid);
|
|
|
|
key->addValue(FieldValue(field1, value1));
|
|
}
|
|
|
|
void getOverlayKey(int uid, string packageName, HashableDimensionKey* key) {
|
|
int pos1[] = {1, 0, 0};
|
|
int pos2[] = {2, 0, 0};
|
|
|
|
Field field1(59 /* atom id */, pos1, 0 /* depth */);
|
|
Field field2(59 /* atom id */, pos2, 0 /* depth */);
|
|
|
|
Value value1((int32_t)uid);
|
|
Value value2(packageName);
|
|
|
|
key->addValue(FieldValue(field1, value1));
|
|
key->addValue(FieldValue(field2, value2));
|
|
}
|
|
|
|
void getPartialWakelockKey(int uid, const std::string& tag, HashableDimensionKey* key) {
|
|
int pos1[] = {1, 1, 1};
|
|
int pos3[] = {2, 0, 0};
|
|
int pos4[] = {3, 0, 0};
|
|
|
|
Field field1(10 /* atom id */, pos1, 2 /* depth */);
|
|
|
|
Field field3(10 /* atom id */, pos3, 0 /* depth */);
|
|
Field field4(10 /* atom id */, pos4, 0 /* depth */);
|
|
|
|
Value value1((int32_t)uid);
|
|
Value value3((int32_t)1 /*partial*/);
|
|
Value value4(tag);
|
|
|
|
key->addValue(FieldValue(field1, value1));
|
|
key->addValue(FieldValue(field3, value3));
|
|
key->addValue(FieldValue(field4, value4));
|
|
}
|
|
|
|
void getPartialWakelockKey(int uid, HashableDimensionKey* key) {
|
|
int pos1[] = {1, 1, 1};
|
|
int pos3[] = {2, 0, 0};
|
|
|
|
Field field1(10 /* atom id */, pos1, 2 /* depth */);
|
|
Field field3(10 /* atom id */, pos3, 0 /* depth */);
|
|
|
|
Value value1((int32_t)uid);
|
|
Value value3((int32_t)1 /*partial*/);
|
|
|
|
key->addValue(FieldValue(field1, value1));
|
|
key->addValue(FieldValue(field3, value3));
|
|
}
|
|
// END: get primary key functions
|
|
|
|
void writeAttribution(AStatsEvent* statsEvent, const vector<int>& attributionUids,
|
|
const vector<string>& attributionTags) {
|
|
vector<const char*> cTags(attributionTags.size());
|
|
for (int i = 0; i < cTags.size(); i++) {
|
|
cTags[i] = attributionTags[i].c_str();
|
|
}
|
|
|
|
AStatsEvent_writeAttributionChain(statsEvent,
|
|
reinterpret_cast<const uint32_t*>(attributionUids.data()),
|
|
cTags.data(), attributionUids.size());
|
|
}
|
|
|
|
void parseStatsEventToLogEvent(AStatsEvent* statsEvent, LogEvent* logEvent) {
|
|
AStatsEvent_build(statsEvent);
|
|
|
|
size_t size;
|
|
uint8_t* buf = AStatsEvent_getBuffer(statsEvent, &size);
|
|
logEvent->parseBuffer(buf, size);
|
|
|
|
AStatsEvent_release(statsEvent);
|
|
}
|
|
|
|
void CreateTwoValueLogEvent(LogEvent* logEvent, int atomId, int64_t eventTimeNs, int32_t value1,
|
|
int32_t value2) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, atomId);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, eventTimeNs);
|
|
|
|
AStatsEvent_writeInt32(statsEvent, value1);
|
|
AStatsEvent_writeInt32(statsEvent, value2);
|
|
|
|
parseStatsEventToLogEvent(statsEvent, logEvent);
|
|
}
|
|
|
|
shared_ptr<LogEvent> CreateTwoValueLogEvent(int atomId, int64_t eventTimeNs, int32_t value1,
|
|
int32_t value2) {
|
|
shared_ptr<LogEvent> logEvent = std::make_shared<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
CreateTwoValueLogEvent(logEvent.get(), atomId, eventTimeNs, value1, value2);
|
|
return logEvent;
|
|
}
|
|
|
|
void CreateThreeValueLogEvent(LogEvent* logEvent, int atomId, int64_t eventTimeNs, int32_t value1,
|
|
int32_t value2, int32_t value3) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, atomId);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, eventTimeNs);
|
|
|
|
AStatsEvent_writeInt32(statsEvent, value1);
|
|
AStatsEvent_writeInt32(statsEvent, value2);
|
|
AStatsEvent_writeInt32(statsEvent, value3);
|
|
|
|
parseStatsEventToLogEvent(statsEvent, logEvent);
|
|
}
|
|
|
|
shared_ptr<LogEvent> CreateThreeValueLogEvent(int atomId, int64_t eventTimeNs, int32_t value1,
|
|
int32_t value2, int32_t value3) {
|
|
shared_ptr<LogEvent> logEvent = std::make_shared<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
CreateThreeValueLogEvent(logEvent.get(), atomId, eventTimeNs, value1, value2, value3);
|
|
return logEvent;
|
|
}
|
|
|
|
void CreateRepeatedValueLogEvent(LogEvent* logEvent, int atomId, int64_t eventTimeNs,
|
|
int32_t value) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, atomId);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, eventTimeNs);
|
|
|
|
AStatsEvent_writeInt32(statsEvent, value);
|
|
AStatsEvent_writeInt32(statsEvent, value);
|
|
|
|
parseStatsEventToLogEvent(statsEvent, logEvent);
|
|
}
|
|
|
|
shared_ptr<LogEvent> CreateRepeatedValueLogEvent(int atomId, int64_t eventTimeNs, int32_t value) {
|
|
shared_ptr<LogEvent> logEvent = std::make_shared<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
CreateRepeatedValueLogEvent(logEvent.get(), atomId, eventTimeNs, value);
|
|
return logEvent;
|
|
}
|
|
|
|
void CreateNoValuesLogEvent(LogEvent* logEvent, int atomId, int64_t eventTimeNs) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, atomId);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, eventTimeNs);
|
|
|
|
parseStatsEventToLogEvent(statsEvent, logEvent);
|
|
}
|
|
|
|
shared_ptr<LogEvent> CreateNoValuesLogEvent(int atomId, int64_t eventTimeNs) {
|
|
shared_ptr<LogEvent> logEvent = std::make_shared<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
CreateNoValuesLogEvent(logEvent.get(), atomId, eventTimeNs);
|
|
return logEvent;
|
|
}
|
|
|
|
shared_ptr<LogEvent> makeUidLogEvent(int atomId, int64_t eventTimeNs, int uid, int data1,
|
|
int data2) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, atomId);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, eventTimeNs);
|
|
|
|
AStatsEvent_writeInt32(statsEvent, uid);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_IS_UID, true);
|
|
AStatsEvent_writeInt32(statsEvent, data1);
|
|
AStatsEvent_writeInt32(statsEvent, data2);
|
|
|
|
shared_ptr<LogEvent> logEvent = std::make_shared<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
parseStatsEventToLogEvent(statsEvent, logEvent.get());
|
|
return logEvent;
|
|
}
|
|
|
|
shared_ptr<LogEvent> makeAttributionLogEvent(int atomId, int64_t eventTimeNs,
|
|
const vector<int>& uids, const vector<string>& tags,
|
|
int data1, int data2) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, atomId);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, eventTimeNs);
|
|
|
|
writeAttribution(statsEvent, uids, tags);
|
|
AStatsEvent_writeInt32(statsEvent, data1);
|
|
AStatsEvent_writeInt32(statsEvent, data2);
|
|
|
|
shared_ptr<LogEvent> logEvent = std::make_shared<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
parseStatsEventToLogEvent(statsEvent, logEvent.get());
|
|
return logEvent;
|
|
}
|
|
|
|
sp<MockUidMap> makeMockUidMapForOneHost(int hostUid, const vector<int>& isolatedUids) {
|
|
sp<MockUidMap> uidMap = new NaggyMock<MockUidMap>();
|
|
EXPECT_CALL(*uidMap, getHostUidOrSelf(_)).WillRepeatedly(ReturnArg<0>());
|
|
for (const int isolatedUid : isolatedUids) {
|
|
EXPECT_CALL(*uidMap, getHostUidOrSelf(isolatedUid)).WillRepeatedly(Return(hostUid));
|
|
}
|
|
|
|
return uidMap;
|
|
}
|
|
|
|
sp<MockUidMap> makeMockUidMapForPackage(const string& pkg, const set<int32_t>& uids) {
|
|
sp<MockUidMap> uidMap = new StrictMock<MockUidMap>();
|
|
EXPECT_CALL(*uidMap, getAppUid(_)).Times(AnyNumber());
|
|
EXPECT_CALL(*uidMap, getAppUid(pkg)).WillRepeatedly(Return(uids));
|
|
|
|
return uidMap;
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateScreenStateChangedEvent(uint64_t timestampNs,
|
|
const android::view::DisplayStateEnum state,
|
|
int loggerUid) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, util::SCREEN_STATE_CHANGED);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, timestampNs);
|
|
AStatsEvent_writeInt32(statsEvent, state);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_EXCLUSIVE_STATE, true);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_STATE_NESTED, false);
|
|
|
|
std::unique_ptr<LogEvent> logEvent = std::make_unique<LogEvent>(loggerUid, /*pid=*/0);
|
|
parseStatsEventToLogEvent(statsEvent, logEvent.get());
|
|
return logEvent;
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateBatterySaverOnEvent(uint64_t timestampNs) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, util::BATTERY_SAVER_MODE_STATE_CHANGED);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, timestampNs);
|
|
AStatsEvent_writeInt32(statsEvent, BatterySaverModeStateChanged::ON);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_EXCLUSIVE_STATE, true);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_STATE_NESTED, false);
|
|
|
|
std::unique_ptr<LogEvent> logEvent = std::make_unique<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
parseStatsEventToLogEvent(statsEvent, logEvent.get());
|
|
return logEvent;
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateBatterySaverOffEvent(uint64_t timestampNs) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, util::BATTERY_SAVER_MODE_STATE_CHANGED);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, timestampNs);
|
|
AStatsEvent_writeInt32(statsEvent, BatterySaverModeStateChanged::OFF);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_EXCLUSIVE_STATE, true);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_STATE_NESTED, false);
|
|
|
|
std::unique_ptr<LogEvent> logEvent = std::make_unique<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
parseStatsEventToLogEvent(statsEvent, logEvent.get());
|
|
return logEvent;
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateBatteryStateChangedEvent(const uint64_t timestampNs, const BatteryPluggedStateEnum state) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, util::PLUGGED_STATE_CHANGED);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, timestampNs);
|
|
AStatsEvent_writeInt32(statsEvent, state);
|
|
|
|
std::unique_ptr<LogEvent> logEvent = std::make_unique<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
parseStatsEventToLogEvent(statsEvent, logEvent.get());
|
|
return logEvent;
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateScreenBrightnessChangedEvent(uint64_t timestampNs, int level) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, util::SCREEN_BRIGHTNESS_CHANGED);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, timestampNs);
|
|
AStatsEvent_writeInt32(statsEvent, level);
|
|
|
|
std::unique_ptr<LogEvent> logEvent = std::make_unique<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
parseStatsEventToLogEvent(statsEvent, logEvent.get());
|
|
return logEvent;
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateScheduledJobStateChangedEvent(
|
|
const vector<int>& attributionUids, const vector<string>& attributionTags,
|
|
const string& jobName, const ScheduledJobStateChanged::State state, uint64_t timestampNs) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, util::SCHEDULED_JOB_STATE_CHANGED);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, timestampNs);
|
|
|
|
writeAttribution(statsEvent, attributionUids, attributionTags);
|
|
AStatsEvent_writeString(statsEvent, jobName.c_str());
|
|
AStatsEvent_writeInt32(statsEvent, state);
|
|
|
|
std::unique_ptr<LogEvent> logEvent = std::make_unique<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
parseStatsEventToLogEvent(statsEvent, logEvent.get());
|
|
return logEvent;
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateStartScheduledJobEvent(uint64_t timestampNs,
|
|
const vector<int>& attributionUids,
|
|
const vector<string>& attributionTags,
|
|
const string& jobName) {
|
|
return CreateScheduledJobStateChangedEvent(attributionUids, attributionTags, jobName,
|
|
ScheduledJobStateChanged::STARTED, timestampNs);
|
|
}
|
|
|
|
// Create log event when scheduled job finishes.
|
|
std::unique_ptr<LogEvent> CreateFinishScheduledJobEvent(uint64_t timestampNs,
|
|
const vector<int>& attributionUids,
|
|
const vector<string>& attributionTags,
|
|
const string& jobName) {
|
|
return CreateScheduledJobStateChangedEvent(attributionUids, attributionTags, jobName,
|
|
ScheduledJobStateChanged::FINISHED, timestampNs);
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateWakelockStateChangedEvent(uint64_t timestampNs,
|
|
const vector<int>& attributionUids,
|
|
const vector<string>& attributionTags,
|
|
const string& wakelockName,
|
|
const WakelockStateChanged::State state) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, util::WAKELOCK_STATE_CHANGED);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, timestampNs);
|
|
|
|
writeAttribution(statsEvent, attributionUids, attributionTags);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_PRIMARY_FIELD_FIRST_UID, true);
|
|
AStatsEvent_writeInt32(statsEvent, android::os::WakeLockLevelEnum::PARTIAL_WAKE_LOCK);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_PRIMARY_FIELD, true);
|
|
AStatsEvent_writeString(statsEvent, wakelockName.c_str());
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_PRIMARY_FIELD, true);
|
|
AStatsEvent_writeInt32(statsEvent, state);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_EXCLUSIVE_STATE, true);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_STATE_NESTED, true);
|
|
|
|
std::unique_ptr<LogEvent> logEvent = std::make_unique<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
parseStatsEventToLogEvent(statsEvent, logEvent.get());
|
|
return logEvent;
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateAcquireWakelockEvent(uint64_t timestampNs,
|
|
const vector<int>& attributionUids,
|
|
const vector<string>& attributionTags,
|
|
const string& wakelockName) {
|
|
return CreateWakelockStateChangedEvent(timestampNs, attributionUids, attributionTags,
|
|
wakelockName, WakelockStateChanged::ACQUIRE);
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateReleaseWakelockEvent(uint64_t timestampNs,
|
|
const vector<int>& attributionUids,
|
|
const vector<string>& attributionTags,
|
|
const string& wakelockName) {
|
|
return CreateWakelockStateChangedEvent(timestampNs, attributionUids, attributionTags,
|
|
wakelockName, WakelockStateChanged::RELEASE);
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateActivityForegroundStateChangedEvent(
|
|
uint64_t timestampNs, const int uid, const ActivityForegroundStateChanged::State state) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, util::ACTIVITY_FOREGROUND_STATE_CHANGED);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, timestampNs);
|
|
|
|
AStatsEvent_writeInt32(statsEvent, uid);
|
|
AStatsEvent_writeString(statsEvent, "pkg_name");
|
|
AStatsEvent_writeString(statsEvent, "class_name");
|
|
AStatsEvent_writeInt32(statsEvent, state);
|
|
|
|
std::unique_ptr<LogEvent> logEvent = std::make_unique<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
parseStatsEventToLogEvent(statsEvent, logEvent.get());
|
|
return logEvent;
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateMoveToBackgroundEvent(uint64_t timestampNs, const int uid) {
|
|
return CreateActivityForegroundStateChangedEvent(timestampNs, uid,
|
|
ActivityForegroundStateChanged::BACKGROUND);
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateMoveToForegroundEvent(uint64_t timestampNs, const int uid) {
|
|
return CreateActivityForegroundStateChangedEvent(timestampNs, uid,
|
|
ActivityForegroundStateChanged::FOREGROUND);
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateSyncStateChangedEvent(uint64_t timestampNs,
|
|
const vector<int>& attributionUids,
|
|
const vector<string>& attributionTags,
|
|
const string& name,
|
|
const SyncStateChanged::State state) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, util::SYNC_STATE_CHANGED);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, timestampNs);
|
|
|
|
writeAttribution(statsEvent, attributionUids, attributionTags);
|
|
AStatsEvent_writeString(statsEvent, name.c_str());
|
|
AStatsEvent_writeInt32(statsEvent, state);
|
|
|
|
std::unique_ptr<LogEvent> logEvent = std::make_unique<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
parseStatsEventToLogEvent(statsEvent, logEvent.get());
|
|
return logEvent;
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateSyncStartEvent(uint64_t timestampNs,
|
|
const vector<int>& attributionUids,
|
|
const vector<string>& attributionTags,
|
|
const string& name) {
|
|
return CreateSyncStateChangedEvent(timestampNs, attributionUids, attributionTags, name,
|
|
SyncStateChanged::ON);
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateSyncEndEvent(uint64_t timestampNs,
|
|
const vector<int>& attributionUids,
|
|
const vector<string>& attributionTags,
|
|
const string& name) {
|
|
return CreateSyncStateChangedEvent(timestampNs, attributionUids, attributionTags, name,
|
|
SyncStateChanged::OFF);
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateProcessLifeCycleStateChangedEvent(
|
|
uint64_t timestampNs, const int uid, const ProcessLifeCycleStateChanged::State state) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, util::PROCESS_LIFE_CYCLE_STATE_CHANGED);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, timestampNs);
|
|
|
|
AStatsEvent_writeInt32(statsEvent, uid);
|
|
AStatsEvent_writeString(statsEvent, "");
|
|
AStatsEvent_writeInt32(statsEvent, state);
|
|
|
|
std::unique_ptr<LogEvent> logEvent = std::make_unique<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
parseStatsEventToLogEvent(statsEvent, logEvent.get());
|
|
return logEvent;
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateAppCrashEvent(uint64_t timestampNs, const int uid) {
|
|
return CreateProcessLifeCycleStateChangedEvent(timestampNs, uid,
|
|
ProcessLifeCycleStateChanged::CRASHED);
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateAppCrashOccurredEvent(uint64_t timestampNs, const int uid) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, util::APP_CRASH_OCCURRED);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, timestampNs);
|
|
|
|
AStatsEvent_writeInt32(statsEvent, uid);
|
|
AStatsEvent_writeString(statsEvent, "eventType");
|
|
AStatsEvent_writeString(statsEvent, "processName");
|
|
|
|
std::unique_ptr<LogEvent> logEvent = std::make_unique<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
parseStatsEventToLogEvent(statsEvent, logEvent.get());
|
|
return logEvent;
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateIsolatedUidChangedEvent(uint64_t timestampNs, int hostUid,
|
|
int isolatedUid, bool is_create) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, util::ISOLATED_UID_CHANGED);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, timestampNs);
|
|
|
|
AStatsEvent_writeInt32(statsEvent, hostUid);
|
|
AStatsEvent_writeInt32(statsEvent, isolatedUid);
|
|
AStatsEvent_writeInt32(statsEvent, is_create);
|
|
|
|
std::unique_ptr<LogEvent> logEvent = std::make_unique<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
parseStatsEventToLogEvent(statsEvent, logEvent.get());
|
|
return logEvent;
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateUidProcessStateChangedEvent(
|
|
uint64_t timestampNs, int uid, const android::app::ProcessStateEnum state) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, util::UID_PROCESS_STATE_CHANGED);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, timestampNs);
|
|
|
|
AStatsEvent_writeInt32(statsEvent, uid);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_IS_UID, true);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_PRIMARY_FIELD, true);
|
|
AStatsEvent_writeInt32(statsEvent, state);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_EXCLUSIVE_STATE, true);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_STATE_NESTED, false);
|
|
|
|
std::unique_ptr<LogEvent> logEvent = std::make_unique<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
parseStatsEventToLogEvent(statsEvent, logEvent.get());
|
|
return logEvent;
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateBleScanStateChangedEvent(uint64_t timestampNs,
|
|
const vector<int>& attributionUids,
|
|
const vector<string>& attributionTags,
|
|
const BleScanStateChanged::State state,
|
|
const bool filtered, const bool firstMatch,
|
|
const bool opportunistic) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, util::BLE_SCAN_STATE_CHANGED);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, timestampNs);
|
|
|
|
writeAttribution(statsEvent, attributionUids, attributionTags);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_PRIMARY_FIELD_FIRST_UID, true);
|
|
AStatsEvent_writeInt32(statsEvent, state);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_EXCLUSIVE_STATE, true);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_STATE_NESTED, true);
|
|
if (state == util::BLE_SCAN_STATE_CHANGED__STATE__RESET) {
|
|
AStatsEvent_addInt32Annotation(statsEvent, ANNOTATION_ID_TRIGGER_STATE_RESET,
|
|
util::BLE_SCAN_STATE_CHANGED__STATE__OFF);
|
|
}
|
|
AStatsEvent_writeBool(statsEvent, filtered); // filtered
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_PRIMARY_FIELD, true);
|
|
AStatsEvent_writeBool(statsEvent, firstMatch); // first match
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_PRIMARY_FIELD, true);
|
|
AStatsEvent_writeBool(statsEvent, opportunistic); // opportunistic
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_PRIMARY_FIELD, true);
|
|
|
|
std::unique_ptr<LogEvent> logEvent = std::make_unique<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
parseStatsEventToLogEvent(statsEvent, logEvent.get());
|
|
return logEvent;
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateOverlayStateChangedEvent(int64_t timestampNs, const int32_t uid,
|
|
const string& packageName,
|
|
const bool usingAlertWindow,
|
|
const OverlayStateChanged::State state) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, util::OVERLAY_STATE_CHANGED);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, timestampNs);
|
|
|
|
AStatsEvent_writeInt32(statsEvent, uid);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_IS_UID, true);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_PRIMARY_FIELD, true);
|
|
AStatsEvent_writeString(statsEvent, packageName.c_str());
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_PRIMARY_FIELD, true);
|
|
AStatsEvent_writeBool(statsEvent, usingAlertWindow);
|
|
AStatsEvent_writeInt32(statsEvent, state);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_EXCLUSIVE_STATE, true);
|
|
AStatsEvent_addBoolAnnotation(statsEvent, ANNOTATION_ID_STATE_NESTED, false);
|
|
|
|
std::unique_ptr<LogEvent> logEvent = std::make_unique<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
parseStatsEventToLogEvent(statsEvent, logEvent.get());
|
|
return logEvent;
|
|
}
|
|
|
|
std::unique_ptr<LogEvent> CreateAppStartOccurredEvent(
|
|
uint64_t timestampNs, const int uid, const string& pkgName,
|
|
AppStartOccurred::TransitionType type, const string& activityName,
|
|
const string& callingPkgName, const bool isInstantApp, int64_t activityStartMs) {
|
|
AStatsEvent* statsEvent = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(statsEvent, util::APP_START_OCCURRED);
|
|
AStatsEvent_overwriteTimestamp(statsEvent, timestampNs);
|
|
|
|
AStatsEvent_writeInt32(statsEvent, uid);
|
|
AStatsEvent_writeString(statsEvent, pkgName.c_str());
|
|
AStatsEvent_writeInt32(statsEvent, type);
|
|
AStatsEvent_writeString(statsEvent, activityName.c_str());
|
|
AStatsEvent_writeString(statsEvent, callingPkgName.c_str());
|
|
AStatsEvent_writeInt32(statsEvent, isInstantApp);
|
|
AStatsEvent_writeInt32(statsEvent, activityStartMs);
|
|
|
|
std::unique_ptr<LogEvent> logEvent = std::make_unique<LogEvent>(/*uid=*/0, /*pid=*/0);
|
|
parseStatsEventToLogEvent(statsEvent, logEvent.get());
|
|
return logEvent;
|
|
}
|
|
|
|
sp<StatsLogProcessor> CreateStatsLogProcessor(const int64_t timeBaseNs, const int64_t currentTimeNs,
|
|
const StatsdConfig& config, const ConfigKey& key,
|
|
const shared_ptr<IPullAtomCallback>& puller,
|
|
const int32_t atomTag, const sp<UidMap> uidMap) {
|
|
sp<StatsPullerManager> pullerManager = new StatsPullerManager();
|
|
if (puller != nullptr) {
|
|
pullerManager->RegisterPullAtomCallback(/*uid=*/0, atomTag, NS_PER_SEC, NS_PER_SEC * 10, {},
|
|
puller);
|
|
}
|
|
sp<AlarmMonitor> anomalyAlarmMonitor =
|
|
new AlarmMonitor(1,
|
|
[](const shared_ptr<IStatsCompanionService>&, int64_t){},
|
|
[](const shared_ptr<IStatsCompanionService>&){});
|
|
sp<AlarmMonitor> periodicAlarmMonitor =
|
|
new AlarmMonitor(1,
|
|
[](const shared_ptr<IStatsCompanionService>&, int64_t){},
|
|
[](const shared_ptr<IStatsCompanionService>&){});
|
|
sp<StatsLogProcessor> processor =
|
|
new StatsLogProcessor(uidMap, pullerManager, anomalyAlarmMonitor, periodicAlarmMonitor,
|
|
timeBaseNs, [](const ConfigKey&) { return true; },
|
|
[](const int&, const vector<int64_t>&) {return true;});
|
|
processor->OnConfigUpdated(currentTimeNs, key, config);
|
|
return processor;
|
|
}
|
|
|
|
void sortLogEventsByTimestamp(std::vector<std::unique_ptr<LogEvent>> *events) {
|
|
std::sort(events->begin(), events->end(),
|
|
[](const std::unique_ptr<LogEvent>& a, const std::unique_ptr<LogEvent>& b) {
|
|
return a->GetElapsedTimestampNs() < b->GetElapsedTimestampNs();
|
|
});
|
|
}
|
|
|
|
int64_t StringToId(const string& str) {
|
|
return static_cast<int64_t>(std::hash<std::string>()(str));
|
|
}
|
|
|
|
sp<EventMatcherWizard> createEventMatcherWizard(
|
|
int tagId, int matcherIndex, const vector<FieldValueMatcher>& fieldValueMatchers) {
|
|
sp<UidMap> uidMap = new UidMap();
|
|
SimpleAtomMatcher atomMatcher;
|
|
atomMatcher.set_atom_id(tagId);
|
|
for (const FieldValueMatcher& fvm : fieldValueMatchers) {
|
|
*atomMatcher.add_field_value_matcher() = fvm;
|
|
}
|
|
uint64_t matcherHash = 0x12345678;
|
|
int64_t matcherId = 678;
|
|
return new EventMatcherWizard({new SimpleAtomMatchingTracker(
|
|
matcherId, matcherIndex, matcherHash, atomMatcher, uidMap)});
|
|
}
|
|
|
|
StatsDimensionsValueParcel CreateAttributionUidDimensionsValueParcel(const int atomId,
|
|
const int uid) {
|
|
StatsDimensionsValueParcel root;
|
|
root.field = atomId;
|
|
root.valueType = STATS_DIMENSIONS_VALUE_TUPLE_TYPE;
|
|
StatsDimensionsValueParcel attrNode;
|
|
attrNode.field = 1;
|
|
attrNode.valueType = STATS_DIMENSIONS_VALUE_TUPLE_TYPE;
|
|
StatsDimensionsValueParcel posInAttrChain;
|
|
posInAttrChain.field = 1;
|
|
posInAttrChain.valueType = STATS_DIMENSIONS_VALUE_TUPLE_TYPE;
|
|
StatsDimensionsValueParcel uidNode;
|
|
uidNode.field = 1;
|
|
uidNode.valueType = STATS_DIMENSIONS_VALUE_INT_TYPE;
|
|
uidNode.intValue = uid;
|
|
posInAttrChain.tupleValue.push_back(uidNode);
|
|
attrNode.tupleValue.push_back(posInAttrChain);
|
|
root.tupleValue.push_back(attrNode);
|
|
return root;
|
|
}
|
|
|
|
void ValidateUidDimension(const DimensionsValue& value, int atomId, int uid) {
|
|
EXPECT_EQ(value.field(), atomId);
|
|
ASSERT_EQ(value.value_tuple().dimensions_value_size(), 1);
|
|
EXPECT_EQ(value.value_tuple().dimensions_value(0).field(), 1);
|
|
EXPECT_EQ(value.value_tuple().dimensions_value(0).value_int(), uid);
|
|
}
|
|
|
|
void ValidateWakelockAttributionUidAndTagDimension(const DimensionsValue& value, const int atomId,
|
|
const int uid, const string& tag) {
|
|
EXPECT_EQ(value.field(), atomId);
|
|
ASSERT_EQ(value.value_tuple().dimensions_value_size(), 2);
|
|
// Attribution field.
|
|
EXPECT_EQ(value.value_tuple().dimensions_value(0).field(), 1);
|
|
// Uid field.
|
|
ASSERT_EQ(value.value_tuple().dimensions_value(0).value_tuple().dimensions_value_size(), 1);
|
|
EXPECT_EQ(value.value_tuple().dimensions_value(0).value_tuple().dimensions_value(0).field(), 1);
|
|
EXPECT_EQ(value.value_tuple().dimensions_value(0).value_tuple().dimensions_value(0).value_int(),
|
|
uid);
|
|
// Tag field.
|
|
EXPECT_EQ(value.value_tuple().dimensions_value(1).field(), 3);
|
|
EXPECT_EQ(value.value_tuple().dimensions_value(1).value_str(), tag);
|
|
}
|
|
|
|
void ValidateAttributionUidDimension(const DimensionsValue& value, int atomId, int uid) {
|
|
EXPECT_EQ(value.field(), atomId);
|
|
ASSERT_EQ(value.value_tuple().dimensions_value_size(), 1);
|
|
// Attribution field.
|
|
EXPECT_EQ(value.value_tuple().dimensions_value(0).field(), 1);
|
|
// Uid only.
|
|
EXPECT_EQ(value.value_tuple().dimensions_value(0)
|
|
.value_tuple().dimensions_value_size(), 1);
|
|
EXPECT_EQ(value.value_tuple().dimensions_value(0)
|
|
.value_tuple().dimensions_value(0).field(), 1);
|
|
EXPECT_EQ(value.value_tuple().dimensions_value(0)
|
|
.value_tuple().dimensions_value(0).value_int(), uid);
|
|
}
|
|
|
|
void ValidateUidDimension(const DimensionsValue& value, int node_idx, int atomId, int uid) {
|
|
EXPECT_EQ(value.field(), atomId);
|
|
ASSERT_GT(value.value_tuple().dimensions_value_size(), node_idx);
|
|
// Attribution field.
|
|
EXPECT_EQ(value.value_tuple().dimensions_value(node_idx).field(), 1);
|
|
EXPECT_EQ(value.value_tuple().dimensions_value(node_idx)
|
|
.value_tuple().dimensions_value(0).field(), 1);
|
|
EXPECT_EQ(value.value_tuple().dimensions_value(node_idx)
|
|
.value_tuple().dimensions_value(0).value_int(), uid);
|
|
}
|
|
|
|
void ValidateAttributionUidAndTagDimension(
|
|
const DimensionsValue& value, int node_idx, int atomId, int uid, const std::string& tag) {
|
|
EXPECT_EQ(value.field(), atomId);
|
|
ASSERT_GT(value.value_tuple().dimensions_value_size(), node_idx);
|
|
// Attribution field.
|
|
EXPECT_EQ(1, value.value_tuple().dimensions_value(node_idx).field());
|
|
// Uid only.
|
|
EXPECT_EQ(2, value.value_tuple().dimensions_value(node_idx)
|
|
.value_tuple().dimensions_value_size());
|
|
EXPECT_EQ(1, value.value_tuple().dimensions_value(node_idx)
|
|
.value_tuple().dimensions_value(0).field());
|
|
EXPECT_EQ(uid, value.value_tuple().dimensions_value(node_idx)
|
|
.value_tuple().dimensions_value(0).value_int());
|
|
EXPECT_EQ(2, value.value_tuple().dimensions_value(node_idx)
|
|
.value_tuple().dimensions_value(1).field());
|
|
EXPECT_EQ(tag, value.value_tuple().dimensions_value(node_idx)
|
|
.value_tuple().dimensions_value(1).value_str());
|
|
}
|
|
|
|
void ValidateAttributionUidAndTagDimension(
|
|
const DimensionsValue& value, int atomId, int uid, const std::string& tag) {
|
|
EXPECT_EQ(value.field(), atomId);
|
|
ASSERT_EQ(1, value.value_tuple().dimensions_value_size());
|
|
// Attribution field.
|
|
EXPECT_EQ(1, value.value_tuple().dimensions_value(0).field());
|
|
// Uid only.
|
|
EXPECT_EQ(value.value_tuple().dimensions_value(0)
|
|
.value_tuple().dimensions_value_size(), 2);
|
|
EXPECT_EQ(value.value_tuple().dimensions_value(0)
|
|
.value_tuple().dimensions_value(0).field(), 1);
|
|
EXPECT_EQ(value.value_tuple().dimensions_value(0)
|
|
.value_tuple().dimensions_value(0).value_int(), uid);
|
|
EXPECT_EQ(value.value_tuple().dimensions_value(0)
|
|
.value_tuple().dimensions_value(1).field(), 2);
|
|
EXPECT_EQ(value.value_tuple().dimensions_value(0)
|
|
.value_tuple().dimensions_value(1).value_str(), tag);
|
|
}
|
|
|
|
void ValidateStateValue(const google::protobuf::RepeatedPtrField<StateValue>& stateValues,
|
|
int atomId, int64_t value) {
|
|
ASSERT_EQ(stateValues.size(), 1);
|
|
ASSERT_EQ(stateValues[0].atom_id(), atomId);
|
|
switch (stateValues[0].contents_case()) {
|
|
case StateValue::ContentsCase::kValue:
|
|
EXPECT_EQ(stateValues[0].value(), (int32_t)value);
|
|
break;
|
|
case StateValue::ContentsCase::kGroupId:
|
|
EXPECT_EQ(stateValues[0].group_id(), value);
|
|
break;
|
|
default:
|
|
FAIL() << "State value should have either a value or a group id";
|
|
}
|
|
}
|
|
|
|
void ValidateCountBucket(const CountBucketInfo& countBucket, int64_t startTimeNs, int64_t endTimeNs,
|
|
int64_t count) {
|
|
EXPECT_EQ(countBucket.start_bucket_elapsed_nanos(), startTimeNs);
|
|
EXPECT_EQ(countBucket.end_bucket_elapsed_nanos(), endTimeNs);
|
|
EXPECT_EQ(countBucket.count(), count);
|
|
}
|
|
|
|
void ValidateDurationBucket(const DurationBucketInfo& bucket, int64_t startTimeNs,
|
|
int64_t endTimeNs, int64_t durationNs) {
|
|
EXPECT_EQ(bucket.start_bucket_elapsed_nanos(), startTimeNs);
|
|
EXPECT_EQ(bucket.end_bucket_elapsed_nanos(), endTimeNs);
|
|
EXPECT_EQ(bucket.duration_nanos(), durationNs);
|
|
}
|
|
|
|
void ValidateGaugeBucketTimes(const GaugeBucketInfo& gaugeBucket, int64_t startTimeNs,
|
|
int64_t endTimeNs, vector<int64_t> eventTimesNs) {
|
|
EXPECT_EQ(gaugeBucket.start_bucket_elapsed_nanos(), startTimeNs);
|
|
EXPECT_EQ(gaugeBucket.end_bucket_elapsed_nanos(), endTimeNs);
|
|
EXPECT_EQ(gaugeBucket.elapsed_timestamp_nanos_size(), eventTimesNs.size());
|
|
for (int i = 0; i < eventTimesNs.size(); i++) {
|
|
EXPECT_EQ(gaugeBucket.elapsed_timestamp_nanos(i), eventTimesNs[i]);
|
|
}
|
|
}
|
|
|
|
void ValidateValueBucket(const ValueBucketInfo& bucket, int64_t startTimeNs, int64_t endTimeNs,
|
|
const vector<int64_t>& values, int64_t conditionTrueNs) {
|
|
EXPECT_EQ(bucket.start_bucket_elapsed_nanos(), startTimeNs);
|
|
EXPECT_EQ(bucket.end_bucket_elapsed_nanos(), endTimeNs);
|
|
ASSERT_EQ(bucket.values_size(), values.size());
|
|
for (int i = 0; i < values.size(); ++i) {
|
|
if (bucket.values(i).has_value_double()) {
|
|
EXPECT_EQ((int64_t)bucket.values(i).value_double(), values[i]);
|
|
} else {
|
|
EXPECT_EQ(bucket.values(i).value_long(), values[i]);
|
|
}
|
|
}
|
|
if (conditionTrueNs > 0) {
|
|
EXPECT_EQ(bucket.condition_true_nanos(), conditionTrueNs);
|
|
}
|
|
}
|
|
|
|
bool EqualsTo(const DimensionsValue& s1, const DimensionsValue& s2) {
|
|
if (s1.field() != s2.field()) {
|
|
return false;
|
|
}
|
|
if (s1.value_case() != s2.value_case()) {
|
|
return false;
|
|
}
|
|
switch (s1.value_case()) {
|
|
case DimensionsValue::ValueCase::kValueStr:
|
|
return (s1.value_str() == s2.value_str());
|
|
case DimensionsValue::ValueCase::kValueInt:
|
|
return s1.value_int() == s2.value_int();
|
|
case DimensionsValue::ValueCase::kValueLong:
|
|
return s1.value_long() == s2.value_long();
|
|
case DimensionsValue::ValueCase::kValueBool:
|
|
return s1.value_bool() == s2.value_bool();
|
|
case DimensionsValue::ValueCase::kValueFloat:
|
|
return s1.value_float() == s2.value_float();
|
|
case DimensionsValue::ValueCase::kValueTuple: {
|
|
if (s1.value_tuple().dimensions_value_size() !=
|
|
s2.value_tuple().dimensions_value_size()) {
|
|
return false;
|
|
}
|
|
bool allMatched = true;
|
|
for (int i = 0; allMatched && i < s1.value_tuple().dimensions_value_size(); ++i) {
|
|
allMatched &= EqualsTo(s1.value_tuple().dimensions_value(i),
|
|
s2.value_tuple().dimensions_value(i));
|
|
}
|
|
return allMatched;
|
|
}
|
|
case DimensionsValue::ValueCase::VALUE_NOT_SET:
|
|
default:
|
|
return true;
|
|
}
|
|
}
|
|
|
|
bool LessThan(const google::protobuf::RepeatedPtrField<StateValue>& s1,
|
|
const google::protobuf::RepeatedPtrField<StateValue>& s2) {
|
|
if (s1.size() != s2.size()) {
|
|
return s1.size() < s2.size();
|
|
}
|
|
for (int i = 0; i < s1.size(); i++) {
|
|
const StateValue& state1 = s1[i];
|
|
const StateValue& state2 = s2[i];
|
|
if (state1.atom_id() != state2.atom_id()) {
|
|
return state1.atom_id() < state2.atom_id();
|
|
}
|
|
if (state1.value() != state2.value()) {
|
|
return state1.value() < state2.value();
|
|
}
|
|
if (state1.group_id() != state2.group_id()) {
|
|
return state1.group_id() < state2.group_id();
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool LessThan(const DimensionsValue& s1, const DimensionsValue& s2) {
|
|
if (s1.field() != s2.field()) {
|
|
return s1.field() < s2.field();
|
|
}
|
|
if (s1.value_case() != s2.value_case()) {
|
|
return s1.value_case() < s2.value_case();
|
|
}
|
|
switch (s1.value_case()) {
|
|
case DimensionsValue::ValueCase::kValueStr:
|
|
return s1.value_str() < s2.value_str();
|
|
case DimensionsValue::ValueCase::kValueInt:
|
|
return s1.value_int() < s2.value_int();
|
|
case DimensionsValue::ValueCase::kValueLong:
|
|
return s1.value_long() < s2.value_long();
|
|
case DimensionsValue::ValueCase::kValueBool:
|
|
return (int)s1.value_bool() < (int)s2.value_bool();
|
|
case DimensionsValue::ValueCase::kValueFloat:
|
|
return s1.value_float() < s2.value_float();
|
|
case DimensionsValue::ValueCase::kValueTuple: {
|
|
if (s1.value_tuple().dimensions_value_size() !=
|
|
s2.value_tuple().dimensions_value_size()) {
|
|
return s1.value_tuple().dimensions_value_size() <
|
|
s2.value_tuple().dimensions_value_size();
|
|
}
|
|
for (int i = 0; i < s1.value_tuple().dimensions_value_size(); ++i) {
|
|
if (EqualsTo(s1.value_tuple().dimensions_value(i),
|
|
s2.value_tuple().dimensions_value(i))) {
|
|
continue;
|
|
} else {
|
|
return LessThan(s1.value_tuple().dimensions_value(i),
|
|
s2.value_tuple().dimensions_value(i));
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
case DimensionsValue::ValueCase::VALUE_NOT_SET:
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool LessThan(const DimensionsPair& s1, const DimensionsPair& s2) {
|
|
if (LessThan(s1.dimInWhat, s2.dimInWhat)) {
|
|
return true;
|
|
} else if (LessThan(s2.dimInWhat, s1.dimInWhat)) {
|
|
return false;
|
|
}
|
|
|
|
return LessThan(s1.stateValues, s2.stateValues);
|
|
}
|
|
|
|
void backfillStringInDimension(const std::map<uint64_t, string>& str_map,
|
|
DimensionsValue* dimension) {
|
|
if (dimension->has_value_str_hash()) {
|
|
auto it = str_map.find((uint64_t)(dimension->value_str_hash()));
|
|
if (it != str_map.end()) {
|
|
dimension->clear_value_str_hash();
|
|
dimension->set_value_str(it->second);
|
|
} else {
|
|
ALOGE("Can not find the string hash: %llu",
|
|
(unsigned long long)dimension->value_str_hash());
|
|
}
|
|
} else if (dimension->has_value_tuple()) {
|
|
auto value_tuple = dimension->mutable_value_tuple();
|
|
for (int i = 0; i < value_tuple->dimensions_value_size(); ++i) {
|
|
backfillStringInDimension(str_map, value_tuple->mutable_dimensions_value(i));
|
|
}
|
|
}
|
|
}
|
|
|
|
void backfillStringInReport(ConfigMetricsReport *config_report) {
|
|
std::map<uint64_t, string> str_map;
|
|
for (const auto& str : config_report->strings()) {
|
|
uint64_t hash = Hash64(str);
|
|
if (str_map.find(hash) != str_map.end()) {
|
|
ALOGE("String hash conflicts: %s %s", str.c_str(), str_map[hash].c_str());
|
|
}
|
|
str_map[hash] = str;
|
|
}
|
|
for (int i = 0; i < config_report->metrics_size(); ++i) {
|
|
auto metric_report = config_report->mutable_metrics(i);
|
|
if (metric_report->has_count_metrics()) {
|
|
backfillStringInDimension(str_map, metric_report->mutable_count_metrics());
|
|
} else if (metric_report->has_duration_metrics()) {
|
|
backfillStringInDimension(str_map, metric_report->mutable_duration_metrics());
|
|
} else if (metric_report->has_gauge_metrics()) {
|
|
backfillStringInDimension(str_map, metric_report->mutable_gauge_metrics());
|
|
} else if (metric_report->has_value_metrics()) {
|
|
backfillStringInDimension(str_map, metric_report->mutable_value_metrics());
|
|
}
|
|
}
|
|
// Backfill the package names.
|
|
for (int i = 0 ; i < config_report->uid_map().snapshots_size(); ++i) {
|
|
auto snapshot = config_report->mutable_uid_map()->mutable_snapshots(i);
|
|
for (int j = 0 ; j < snapshot->package_info_size(); ++j) {
|
|
auto package_info = snapshot->mutable_package_info(j);
|
|
if (package_info->has_name_hash()) {
|
|
auto it = str_map.find((uint64_t)(package_info->name_hash()));
|
|
if (it != str_map.end()) {
|
|
package_info->clear_name_hash();
|
|
package_info->set_name(it->second);
|
|
} else {
|
|
ALOGE("Can not find the string package name hash: %llu",
|
|
(unsigned long long)package_info->name_hash());
|
|
}
|
|
|
|
}
|
|
}
|
|
}
|
|
// Backfill the app name in app changes.
|
|
for (int i = 0 ; i < config_report->uid_map().changes_size(); ++i) {
|
|
auto change = config_report->mutable_uid_map()->mutable_changes(i);
|
|
if (change->has_app_hash()) {
|
|
auto it = str_map.find((uint64_t)(change->app_hash()));
|
|
if (it != str_map.end()) {
|
|
change->clear_app_hash();
|
|
change->set_app(it->second);
|
|
} else {
|
|
ALOGE("Can not find the string change app name hash: %llu",
|
|
(unsigned long long)change->app_hash());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void backfillStringInReport(ConfigMetricsReportList *config_report_list) {
|
|
for (int i = 0; i < config_report_list->reports_size(); ++i) {
|
|
backfillStringInReport(config_report_list->mutable_reports(i));
|
|
}
|
|
}
|
|
|
|
bool backfillDimensionPath(const DimensionsValue& path,
|
|
const google::protobuf::RepeatedPtrField<DimensionsValue>& leafValues,
|
|
int* leafIndex,
|
|
DimensionsValue* dimension) {
|
|
dimension->set_field(path.field());
|
|
if (path.has_value_tuple()) {
|
|
for (int i = 0; i < path.value_tuple().dimensions_value_size(); ++i) {
|
|
if (!backfillDimensionPath(
|
|
path.value_tuple().dimensions_value(i), leafValues, leafIndex,
|
|
dimension->mutable_value_tuple()->add_dimensions_value())) {
|
|
return false;
|
|
}
|
|
}
|
|
} else {
|
|
if (*leafIndex < 0 || *leafIndex >= leafValues.size()) {
|
|
return false;
|
|
}
|
|
dimension->MergeFrom(leafValues.Get(*leafIndex));
|
|
(*leafIndex)++;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool backfillDimensionPath(const DimensionsValue& path,
|
|
const google::protobuf::RepeatedPtrField<DimensionsValue>& leafValues,
|
|
DimensionsValue* dimension) {
|
|
int leafIndex = 0;
|
|
return backfillDimensionPath(path, leafValues, &leafIndex, dimension);
|
|
}
|
|
|
|
void backfillDimensionPath(StatsLogReport* report) {
|
|
if (report->has_dimensions_path_in_what() || report->has_dimensions_path_in_condition()) {
|
|
auto whatPath = report->dimensions_path_in_what();
|
|
auto conditionPath = report->dimensions_path_in_condition();
|
|
if (report->has_count_metrics()) {
|
|
backfillDimensionPath(whatPath, conditionPath, report->mutable_count_metrics());
|
|
} else if (report->has_duration_metrics()) {
|
|
backfillDimensionPath(whatPath, conditionPath, report->mutable_duration_metrics());
|
|
} else if (report->has_gauge_metrics()) {
|
|
backfillDimensionPath(whatPath, conditionPath, report->mutable_gauge_metrics());
|
|
} else if (report->has_value_metrics()) {
|
|
backfillDimensionPath(whatPath, conditionPath, report->mutable_value_metrics());
|
|
}
|
|
report->clear_dimensions_path_in_what();
|
|
report->clear_dimensions_path_in_condition();
|
|
}
|
|
}
|
|
|
|
void backfillDimensionPath(ConfigMetricsReport* config_report) {
|
|
for (int i = 0; i < config_report->metrics_size(); ++i) {
|
|
backfillDimensionPath(config_report->mutable_metrics(i));
|
|
}
|
|
}
|
|
|
|
void backfillDimensionPath(ConfigMetricsReportList* config_report_list) {
|
|
for (int i = 0; i < config_report_list->reports_size(); ++i) {
|
|
backfillDimensionPath(config_report_list->mutable_reports(i));
|
|
}
|
|
}
|
|
|
|
void backfillStartEndTimestamp(StatsLogReport *report) {
|
|
const int64_t timeBaseNs = report->time_base_elapsed_nano_seconds();
|
|
const int64_t bucketSizeNs = report->bucket_size_nano_seconds();
|
|
if (report->has_count_metrics()) {
|
|
backfillStartEndTimestampForMetrics(
|
|
timeBaseNs, bucketSizeNs, report->mutable_count_metrics());
|
|
} else if (report->has_duration_metrics()) {
|
|
backfillStartEndTimestampForMetrics(
|
|
timeBaseNs, bucketSizeNs, report->mutable_duration_metrics());
|
|
} else if (report->has_gauge_metrics()) {
|
|
backfillStartEndTimestampForMetrics(
|
|
timeBaseNs, bucketSizeNs, report->mutable_gauge_metrics());
|
|
if (report->gauge_metrics().skipped_size() > 0) {
|
|
backfillStartEndTimestampForSkippedBuckets(
|
|
timeBaseNs, report->mutable_gauge_metrics());
|
|
}
|
|
} else if (report->has_value_metrics()) {
|
|
backfillStartEndTimestampForMetrics(
|
|
timeBaseNs, bucketSizeNs, report->mutable_value_metrics());
|
|
if (report->value_metrics().skipped_size() > 0) {
|
|
backfillStartEndTimestampForSkippedBuckets(
|
|
timeBaseNs, report->mutable_value_metrics());
|
|
}
|
|
}
|
|
}
|
|
|
|
void backfillStartEndTimestamp(ConfigMetricsReport *config_report) {
|
|
for (int j = 0; j < config_report->metrics_size(); ++j) {
|
|
backfillStartEndTimestamp(config_report->mutable_metrics(j));
|
|
}
|
|
}
|
|
|
|
void backfillStartEndTimestamp(ConfigMetricsReportList *config_report_list) {
|
|
for (int i = 0; i < config_report_list->reports_size(); ++i) {
|
|
backfillStartEndTimestamp(config_report_list->mutable_reports(i));
|
|
}
|
|
}
|
|
|
|
Status FakeSubsystemSleepCallback::onPullAtom(int atomTag,
|
|
const shared_ptr<IPullAtomResultReceiver>& resultReceiver) {
|
|
// Convert stats_events into StatsEventParcels.
|
|
std::vector<StatsEventParcel> parcels;
|
|
for (int i = 1; i < 3; i++) {
|
|
AStatsEvent* event = AStatsEvent_obtain();
|
|
AStatsEvent_setAtomId(event, atomTag);
|
|
std::string subsystemName = "subsystem_name_";
|
|
subsystemName = subsystemName + std::to_string(i);
|
|
AStatsEvent_writeString(event, subsystemName.c_str());
|
|
AStatsEvent_writeString(event, "subsystem_subname foo");
|
|
AStatsEvent_writeInt64(event, /*count= */ i);
|
|
AStatsEvent_writeInt64(event, /*time_millis= */ pullNum * pullNum * 100 + i);
|
|
AStatsEvent_build(event);
|
|
size_t size;
|
|
uint8_t* buffer = AStatsEvent_getBuffer(event, &size);
|
|
|
|
StatsEventParcel p;
|
|
// vector.assign() creates a copy, but this is inevitable unless
|
|
// stats_event.h/c uses a vector as opposed to a buffer.
|
|
p.buffer.assign(buffer, buffer + size);
|
|
parcels.push_back(std::move(p));
|
|
AStatsEvent_release(event);
|
|
}
|
|
pullNum++;
|
|
resultReceiver->pullFinished(atomTag, /*success=*/true, parcels);
|
|
return Status::ok();
|
|
}
|
|
|
|
} // namespace statsd
|
|
} // namespace os
|
|
} // namespace android
|