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 #pragma once
18 
19 #include <aidl/android/os/StatsDimensionsValueParcel.h>
20 #include <utils/JenkinsHash.h>
21 #include <vector>
22 #include "android-base/stringprintf.h"
23 #include "FieldValue.h"
24 #include "logd/LogEvent.h"
25 
26 namespace android {
27 namespace os {
28 namespace statsd {
29 
30 using ::aidl::android::os::StatsDimensionsValueParcel;
31 
32 // These constants must be kept in sync with those in StatsDimensionsValue.java
33 inline constexpr int STATS_DIMENSIONS_VALUE_STRING_TYPE = 2;
34 inline constexpr int STATS_DIMENSIONS_VALUE_INT_TYPE = 3;
35 inline constexpr int STATS_DIMENSIONS_VALUE_LONG_TYPE = 4;
36 // inline constexpr int STATS_DIMENSIONS_VALUE_BOOL_TYPE = 5; (commented out because
37 // unused -- statsd does not correctly support bool types)
38 inline constexpr int STATS_DIMENSIONS_VALUE_FLOAT_TYPE = 6;
39 inline constexpr int STATS_DIMENSIONS_VALUE_TUPLE_TYPE = 7;
40 
41 struct Metric2Condition {
42     int64_t conditionId;
43     std::vector<Matcher> metricFields;
44     std::vector<Matcher> conditionFields;
45 };
46 
47 struct Metric2State {
48     int32_t stateAtomId;
49     std::vector<Matcher> metricFields;
50     std::vector<Matcher> stateFields;
51 };
52 
53 class HashableDimensionKey {
54 public:
55     explicit HashableDimensionKey(const std::vector<FieldValue>& values) {
56         mValues = values;
57     }
58 
59     HashableDimensionKey() {};
60 
61     HashableDimensionKey(const HashableDimensionKey& that) : mValues(that.getValues()){};
62 
63     inline void addValue(const FieldValue& value) {
64         mValues.push_back(value);
65     }
66 
67     inline const std::vector<FieldValue>& getValues() const {
68         return mValues;
69     }
70 
71     inline std::vector<FieldValue>* mutableValues() {
72         return &mValues;
73     }
74 
75     inline FieldValue* mutableValue(size_t i) {
76         if (i >= 0 && i < mValues.size()) {
77             return &(mValues[i]);
78         }
79         return nullptr;
80     }
81 
82     StatsDimensionsValueParcel toStatsDimensionsValueParcel() const;
83 
84     std::string toString() const;
85 
86     bool operator!=(const HashableDimensionKey& that) const;
87 
88     bool operator==(const HashableDimensionKey& that) const;
89 
90     bool operator<(const HashableDimensionKey& that) const;
91 
92     bool contains(const HashableDimensionKey& that) const;
93 
94 private:
95     std::vector<FieldValue> mValues;
96 };
97 
98 class MetricDimensionKey {
99 public:
100     explicit MetricDimensionKey(const HashableDimensionKey& dimensionKeyInWhat,
101                                 const HashableDimensionKey& stateValuesKey)
102         : mDimensionKeyInWhat(dimensionKeyInWhat), mStateValuesKey(stateValuesKey){};
103 
104     MetricDimensionKey(){};
105 
106     MetricDimensionKey(const MetricDimensionKey& that)
107         : mDimensionKeyInWhat(that.getDimensionKeyInWhat()),
108           mStateValuesKey(that.getStateValuesKey()){};
109 
110     MetricDimensionKey& operator=(const MetricDimensionKey& from) = default;
111 
112     std::string toString() const;
113 
114     inline const HashableDimensionKey& getDimensionKeyInWhat() const {
115         return mDimensionKeyInWhat;
116     }
117 
118     inline const HashableDimensionKey& getStateValuesKey() const {
119         return mStateValuesKey;
120     }
121 
122     inline HashableDimensionKey* getMutableStateValuesKey() {
123         return &mStateValuesKey;
124     }
125 
126     inline void setStateValuesKey(const HashableDimensionKey& key) {
127         mStateValuesKey = key;
128     }
129 
130     bool hasStateValuesKey() const {
131         return mStateValuesKey.getValues().size() > 0;
132     }
133 
134     bool operator==(const MetricDimensionKey& that) const;
135 
136     bool operator<(const MetricDimensionKey& that) const;
137 
138 private:
139     HashableDimensionKey mDimensionKeyInWhat;
140     HashableDimensionKey mStateValuesKey;
141 };
142 
143 android::hash_t hashDimension(const HashableDimensionKey& key);
144 
145 /**
146  * Returns true if a FieldValue field matches the matcher field.
147  * The value of the FieldValue is output.
148  */
149 bool filterValues(const Matcher& matcherField, const std::vector<FieldValue>& values,
150                   FieldValue* output);
151 
152 /**
153  * Creating HashableDimensionKeys from FieldValues using matcher.
154  *
155  * This function may make modifications to the Field if the matcher has Position=FIRST,LAST or ALL
156  * in it. This is because: for example, when we create dimension from last uid in attribution chain,
157  * In one event, uid 1000 is at position 5 and it's the last
158  * In another event, uid 1000 is at position 6, and it's the last
159  * these 2 events should be mapped to the same dimension.  So we will remove the original position
160  * from the dimension key for the uid field (by applying 0x80 bit mask).
161  */
162 bool filterValues(const std::vector<Matcher>& matcherFields, const std::vector<FieldValue>& values,
163                   HashableDimensionKey* output);
164 
165 /**
166  * Creating HashableDimensionKeys from State Primary Keys in FieldValues.
167  *
168  * This function may make modifications to the Field if the matcher has Position=FIRST,LAST or ALL
169  * in it. This is because: for example, when we create dimension from last uid in attribution chain,
170  * In one event, uid 1000 is at position 5 and it's the last
171  * In another event, uid 1000 is at position 6, and it's the last
172  * these 2 events should be mapped to the same dimension.  So we will remove the original position
173  * from the dimension key for the uid field (by applying 0x80 bit mask).
174  */
175 bool filterPrimaryKey(const std::vector<FieldValue>& values, HashableDimensionKey* output);
176 
177 /**
178  * Filter the values from FieldValues using the matchers.
179  *
180  * In contrast to the above function, this function will not do any modification to the original
181  * data. Considering it as taking a snapshot on the atom event.
182  */
183 void filterGaugeValues(const std::vector<Matcher>& matchers, const std::vector<FieldValue>& values,
184                        std::vector<FieldValue>* output);
185 
186 void getDimensionForCondition(const std::vector<FieldValue>& eventValues,
187                               const Metric2Condition& links,
188                               HashableDimensionKey* conditionDimension);
189 
190 /**
191  * Get dimension values using metric's "what" fields and fill statePrimaryKey's
192  * mField information using "state" fields.
193  */
194 void getDimensionForState(const std::vector<FieldValue>& eventValues, const Metric2State& link,
195                           HashableDimensionKey* statePrimaryKey);
196 
197 /**
198  * Returns true if the primaryKey values are a subset of the whatKey values.
199  * The values from the primaryKey come from the state atom, so we need to
200  * check that a link exists between the state atom field and what atom field.
201  *
202  * Example:
203  * whatKey = [Atom: 10, {uid: 1005, wakelock_name: "compose"}]
204  * statePrimaryKey = [Atom: 27, {uid: 1005}]
205  * Returns true IF one of the Metric2State links Atom 10's uid to Atom 27's uid
206  *
207  * Example:
208  * whatKey = [Atom: 10, {uid: 1005, wakelock_name: "compose"}]
209  * statePrimaryKey = [Atom: 59, {uid: 1005, package_name: "system"}]
210  * Returns false
211  */
212 bool containsLinkedStateValues(const HashableDimensionKey& whatKey,
213                                const HashableDimensionKey& primaryKey,
214                                const std::vector<Metric2State>& stateLinks,
215                                const int32_t stateAtomId);
216 
217 /**
218  * Returns true if there is a Metric2State link that links the stateField and
219  * the metricField (they are equal fields from different atoms).
220  */
221 bool linked(const std::vector<Metric2State>& stateLinks, const int32_t stateAtomId,
222             const Field& stateField, const Field& metricField);
223 }  // namespace statsd
224 }  // namespace os
225 }  // namespace android
226 
227 namespace std {
228 
229 using android::os::statsd::HashableDimensionKey;
230 using android::os::statsd::MetricDimensionKey;
231 
232 template <>
233 struct hash<HashableDimensionKey> {
234     std::size_t operator()(const HashableDimensionKey& key) const {
235         return hashDimension(key);
236     }
237 };
238 
239 template <>
240 struct hash<MetricDimensionKey> {
241     std::size_t operator()(const MetricDimensionKey& key) const {
242         android::hash_t hash = hashDimension(key.getDimensionKeyInWhat());
243         hash = android::JenkinsHashMix(hash, hashDimension(key.getStateValuesKey()));
244         return android::JenkinsHashWhiten(hash);
245     }
246 };
247 }  // namespace std
248