1 /*
2  * Copyright (C) 2019 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 #ifndef ANDROID_FRAMEWORK_ML_NN_RUNTIME_TEST_FUZZING_RANDOM_VARIABLE_H
18 #define ANDROID_FRAMEWORK_ML_NN_RUNTIME_TEST_FUZZING_RANDOM_VARIABLE_H
19 
20 #include <algorithm>
21 #include <iostream>
22 #include <map>
23 #include <memory>
24 #include <numeric>
25 #include <set>
26 #include <string>
27 #include <unordered_map>
28 #include <vector>
29 
30 namespace android {
31 namespace nn {
32 namespace fuzzing_test {
33 
34 static const int kMaxValue = 10000;
35 static const int kInvalidValue = INT_MIN;
36 
37 // Describe the search range for the value of a random variable.
38 class RandomVariableRange {
39    public:
40     RandomVariableRange() = default;
RandomVariableRange(int value)41     explicit RandomVariableRange(int value) : mChoices({value}) {}
RandomVariableRange(int lower,int upper)42     RandomVariableRange(int lower, int upper) : mChoices(upper - lower + 1) {
43         std::iota(mChoices.begin(), mChoices.end(), lower);
44     }
RandomVariableRange(const std::vector<int> & vec)45     explicit RandomVariableRange(const std::vector<int>& vec) : mChoices(vec) {}
RandomVariableRange(const std::set<int> & st)46     explicit RandomVariableRange(const std::set<int>& st) : mChoices(st.begin(), st.end()) {}
47     RandomVariableRange(const RandomVariableRange&) = default;
48     RandomVariableRange& operator=(const RandomVariableRange&) = default;
49 
empty()50     bool empty() const { return mChoices.empty(); }
has(int value)51     bool has(int value) const {
52         return std::binary_search(mChoices.begin(), mChoices.end(), value);
53     }
size()54     size_t size() const { return mChoices.size(); }
min()55     int min() const { return *mChoices.begin(); }
max()56     int max() const { return *mChoices.rbegin(); }
getChoices()57     const std::vector<int>& getChoices() const { return mChoices; }
58 
59     // Narrow down the range to fit [lower, upper]. Use kInvalidValue to indicate unlimited bound.
60     void setRange(int lower, int upper);
61     // Narrow down the range to a random selected choice. Return the chosen value.
62     int toConst();
63 
64     // Calculate the intersection of two ranges.
65     friend RandomVariableRange operator&(const RandomVariableRange& lhs,
66                                          const RandomVariableRange& rhs);
67 
68    private:
69     // Always in ascending order.
70     std::vector<int> mChoices;
71 };
72 
73 // Defines the interface for an operation applying to RandomVariables.
74 class IRandomVariableOp {
75    public:
~IRandomVariableOp()76     virtual ~IRandomVariableOp() {}
77     // Forward evaluation of two values.
78     virtual int eval(int lhs, int rhs) const = 0;
79     // Gets the range of the operation outcomes. The returned range must include all possible
80     // outcomes of this operation, but may contain invalid results.
81     virtual RandomVariableRange getInitRange(const RandomVariableRange& lhs,
82                                              const RandomVariableRange& rhs) const;
83     // Provides faster range evaluation for evalSubnetSingleOpHelper if possible.
84     virtual void eval(const std::set<int>* parent1In, const std::set<int>* parent2In,
85                       const std::set<int>* childIn, std::set<int>* parent1Out,
86                       std::set<int>* parent2Out, std::set<int>* childOut) const;
87     // For debugging purpose.
88     virtual const char* getName() const = 0;
89 };
90 
91 enum class RandomVariableType { FREE = 0, CONST = 1, OP = 2 };
92 
93 struct RandomVariableBase {
94     // Each RandomVariableBase is assigned an unique index for debugging purpose.
95     static unsigned int globalIndex;
96     int index;
97 
98     RandomVariableType type;
99     RandomVariableRange range;
100     int value = 0;
101     std::shared_ptr<const IRandomVariableOp> op = nullptr;
102 
103     // Network structural information.
104     std::shared_ptr<RandomVariableBase> parent1 = nullptr;
105     std::shared_ptr<RandomVariableBase> parent2 = nullptr;
106     std::vector<std::weak_ptr<RandomVariableBase>> children;
107 
108     // The last time that this RandomVariableBase is modified.
109     int timestamp;
110 
111     explicit RandomVariableBase(int value);
112     RandomVariableBase(int lower, int upper);
113     explicit RandomVariableBase(const std::vector<int>& choices);
114     RandomVariableBase(const std::shared_ptr<RandomVariableBase>& lhs,
115                        const std::shared_ptr<RandomVariableBase>& rhs,
116                        const std::shared_ptr<const IRandomVariableOp>& op);
117     RandomVariableBase(const RandomVariableBase&) = delete;
118     RandomVariableBase& operator=(const RandomVariableBase&) = delete;
119 
120     // Freeze FREE RandomVariable to one valid choice.
121     // Should only invoke on FREE RandomVariable.
122     void freeze();
123 
124     // Get CONST value or calculate from parents.
125     // Should not invoke on FREE RandomVariable.
126     int getValue() const;
127 
128     // Update the timestamp to the latest global time.
129     void updateTimestamp();
130 };
131 
132 using RandomVariableNode = std::shared_ptr<RandomVariableBase>;
133 
134 // A wrapper class of RandomVariableBase that manages RandomVariableBase with shared_ptr and
135 // provides useful methods and operator overloading to build the random variable network.
136 class RandomVariable {
137    public:
138     // Construct a dummy RandomVariable with nullptr.
RandomVariable()139     RandomVariable() : mVar(nullptr) {}
140 
141     // Construct a CONST RandomVariable with specified value.
142     /* implicit */ RandomVariable(int value);
143 
144     // Construct a FREE RandomVariable with range [lower, upper].
145     RandomVariable(int lower, int upper);
146 
147     // Construct a FREE RandomVariable with specified value choices.
148     explicit RandomVariable(const std::vector<int>& choices);
149 
150     // This is for RandomVariableType::FREE only.
151     // Construct a FREE RandomVariable with default range [1, defaultValue].
152     /* implicit */ RandomVariable(RandomVariableType type);
153 
154     // RandomVariables share the same RandomVariableBase if copied or copy-assigned.
155     RandomVariable(const RandomVariable& other) = default;
156     RandomVariable& operator=(const RandomVariable& other) = default;
157 
158     // Get the value of the RandomVariable, the value must be deterministic.
getValue()159     int getValue() const { return mVar->getValue(); }
160 
161     // Get the underlying managed RandomVariableNode.
get()162     RandomVariableNode get() const { return mVar; };
163 
164     bool operator==(nullptr_t) const { return mVar == nullptr; }
165     bool operator!=(nullptr_t) const { return mVar != nullptr; }
166 
167     // Arithmetic operators and methods on RandomVariables.
168     friend RandomVariable operator+(const RandomVariable& lhs, const RandomVariable& rhs);
169     friend RandomVariable operator-(const RandomVariable& lhs, const RandomVariable& rhs);
170     friend RandomVariable operator*(const RandomVariable& lhs, const RandomVariable& rhs);
171     friend RandomVariable operator*(const RandomVariable& lhs, const float& rhs);
172     friend RandomVariable operator/(const RandomVariable& lhs, const RandomVariable& rhs);
173     friend RandomVariable operator%(const RandomVariable& lhs, const RandomVariable& rhs);
174     friend RandomVariable max(const RandomVariable& lhs, const RandomVariable& rhs);
175     friend RandomVariable min(const RandomVariable& lhs, const RandomVariable& rhs);
176     RandomVariable exactDiv(const RandomVariable& other);
177 
178     // Set constraints on the RandomVariable. Use kInvalidValue to indicate unlimited bound.
179     void setRange(int lower, int upper);
180     RandomVariable setEqual(const RandomVariable& other) const;
181     RandomVariable setGreaterThan(const RandomVariable& other) const;
182     RandomVariable setGreaterEqual(const RandomVariable& other) const;
183 
184     // A FREE RandomVariable is constructed with default range [1, defaultValue].
185     static int defaultValue;
186 
187    private:
188     // Construct a RandomVariable as the result of an OP between two other RandomVariables.
189     RandomVariable(const RandomVariable& lhs, const RandomVariable& rhs,
190                    const std::shared_ptr<const IRandomVariableOp>& op);
191     RandomVariableNode mVar;
192 };
193 
194 using EvaluationOrder = std::vector<RandomVariableNode>;
195 
196 // The base class of a network consisting of disjoint subnets.
197 class DisjointNetwork {
198    public:
199     // Add a node to the network, join the parent subnets if needed.
200     void add(const RandomVariableNode& var);
201 
202     // Similar to join(int, int), but accept RandomVariableNodes.
join(const RandomVariableNode & var1,const RandomVariableNode & var2)203     int join(const RandomVariableNode& var1, const RandomVariableNode& var2) {
204         return DisjointNetwork::join(mIndexMap[var1], mIndexMap[var2]);
205     }
206 
207    protected:
208     DisjointNetwork() = default;
209     DisjointNetwork(const DisjointNetwork&) = default;
210     DisjointNetwork& operator=(const DisjointNetwork&) = default;
211 
212     // Join two subnets by appending every node in ind2 after ind1, return the resulting subnet
213     // index. Use -1 for invalid subnet index.
214     int join(int ind1, int ind2);
215 
216     // A map from the network node to the corresponding subnet index.
217     std::unordered_map<RandomVariableNode, int> mIndexMap;
218 
219     // A map from the subnet index to the set of nodes within the subnet. The nodes are maintained
220     // in a valid evaluation order, that is, a valid topological sort.
221     std::map<int, EvaluationOrder> mEvalOrderMap;
222 
223     // The next index for a new disjoint subnet component.
224     int mNextIndex = 0;
225 };
226 
227 // Manages the active RandomVariable network. Only one instance of this class will exist.
228 class RandomVariableNetwork : public DisjointNetwork {
229    public:
230     // Returns the singleton network instance.
231     static RandomVariableNetwork* get();
232 
233     // Re-initialization. Should be called every time a new random graph is being generated.
234     void initialize(int defaultValue);
235 
236     // Set the elementwise equality of the two vectors of RandomVariables iff it results in a
237     // soluble network.
238     bool setEqualIfCompatible(const std::vector<RandomVariable>& lhs,
239                               const std::vector<RandomVariable>& rhs);
240 
241     // Freeze all FREE RandomVariables in the network to a random valid combination.
242     bool freeze();
243 
244     // Check if node2 is FREE and can be evaluated after node1.
245     bool isSubordinate(const RandomVariableNode& node1, const RandomVariableNode& node2);
246 
247     // Get and then advance the current global timestamp.
getGlobalTime()248     int getGlobalTime() { return mGlobalTime++; }
249 
250     // Add a special constraint on dimension product.
251     void addDimensionProd(const std::vector<RandomVariable>& dims);
252 
253    private:
254     RandomVariableNetwork() = default;
255     RandomVariableNetwork(const RandomVariableNetwork&) = default;
256     RandomVariableNetwork& operator=(const RandomVariableNetwork&) = default;
257 
258     // A class to revert all the changes made to RandomVariableNetwork since the Reverter object is
259     // constructed. Only used when setEqualIfCompatible results in incompatible.
260     class Reverter;
261 
262     // Find valid choices for all RandomVariables in the network. Update the RandomVariableRange
263     // if the network is soluble, otherwise, return false and leave the ranges unchanged.
264     bool evalRange();
265 
266     int mGlobalTime = 0;
267     int mTimestamp = -1;
268 
269     std::vector<EvaluationOrder> mDimProd;
270 };
271 
272 }  // namespace fuzzing_test
273 }  // namespace nn
274 }  // namespace android
275 
276 #endif  // ANDROID_FRAMEWORK_ML_NN_RUNTIME_TEST_FUZZING_RANDOM_VARIABLE_H
277