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 #include "fuzzing/RandomGraphGenerator.h"
18 
19 #include <gtest/gtest.h>
20 
21 #include <algorithm>
22 #include <cmath>
23 #include <memory>
24 #include <set>
25 #include <string>
26 #include <unordered_map>
27 #include <utility>
28 #include <vector>
29 
30 #include "TestHarness.h"
31 #include "TestNeuralNetworksWrapper.h"
32 #include "fuzzing/OperationManager.h"
33 #include "fuzzing/RandomGraphGeneratorUtils.h"
34 #include "fuzzing/RandomVariable.h"
35 
36 namespace android {
37 namespace nn {
38 namespace fuzzing_test {
39 
40 using test_wrapper::Result;
41 using namespace test_helper;
42 
43 // Construct a RandomOperand from OperandSignature.
RandomOperand(const OperandSignature & operand,TestOperandType dataType,uint32_t rank)44 RandomOperand::RandomOperand(const OperandSignature& operand, TestOperandType dataType,
45                              uint32_t rank)
46     : type(operand.type), finalizer(operand.finalizer) {
47     NN_FUZZER_LOG << "Operand: " << toString(type);
48     if (operand.constructor) operand.constructor(dataType, rank, this);
49 }
50 
getDimensions() const51 std::vector<uint32_t> RandomOperand::getDimensions() const {
52     std::vector<uint32_t> result(dimensions.size(), 0);
53     for (uint32_t i = 0; i < dimensions.size(); i++) result[i] = dimensions[i].getValue();
54     return result;
55 }
56 
areValuePropertiesCompatible(int guaranteed,int required)57 static bool areValuePropertiesCompatible(int guaranteed, int required) {
58     return !(~guaranteed & required);
59 }
60 
61 // Check if an edge between [this, other] is valid. If yes, add the edge.
createEdgeIfValid(const RandomOperand & other) const62 bool RandomOperand::createEdgeIfValid(const RandomOperand& other) const {
63     if (other.type != RandomOperandType::INPUT) return false;
64     if (dataType != other.dataType || dimensions.size() != other.dimensions.size() ||
65         scale != other.scale || zeroPoint != other.zeroPoint || doNotConnect ||
66         other.doNotConnect || !areValuePropertiesCompatible(valueProperties, other.valueProperties))
67         return false;
68     return RandomVariableNetwork::get()->setEqualIfCompatible(dimensions, other.dimensions);
69 }
70 
getNumberOfElements() const71 uint32_t RandomOperand::getNumberOfElements() const {
72     uint32_t num = 1;
73     for (const auto& d : dimensions) num *= d.getValue();
74     return num;
75 }
76 
getBufferSize() const77 size_t RandomOperand::getBufferSize() const {
78     return kSizeOfDataType[static_cast<int32_t>(dataType)] * getNumberOfElements();
79 }
80 
81 // Construct a RandomOperation from OperationSignature.
RandomOperation(const OperationSignature & operation)82 RandomOperation::RandomOperation(const OperationSignature& operation)
83     : opType(operation.opType), finalizer(operation.finalizer) {
84     NN_FUZZER_LOG << "Operation: " << toString(opType);
85 
86     // Determine the data type and rank of the operation and invoke the constructor.
87     TestOperandType dataType = getRandomChoice(operation.supportedDataTypes);
88     uint32_t rank = getRandomChoice(operation.supportedRanks);
89 
90     // Initialize operands and operation.
91     for (const auto& op : operation.inputs) {
92         inputs.emplace_back(new RandomOperand(op, dataType, rank));
93     }
94     for (const auto& op : operation.outputs) {
95         outputs.emplace_back(new RandomOperand(op, dataType, rank));
96     }
97     if (operation.constructor) operation.constructor(dataType, rank, this);
98 
99     // Add constraints on the number of elements.
100     if (RandomVariable::defaultValue > 10) {
101         for (auto in : inputs) RandomVariableNetwork::get()->addDimensionProd(in->dimensions);
102         for (auto out : outputs) RandomVariableNetwork::get()->addDimensionProd(out->dimensions);
103     }
104     // The output operands should have dimensions larger than 0.
105     for (auto out : outputs) {
106         for (auto& dim : out->dimensions) dim.setRange(1, kInvalidValue);
107     }
108 }
109 
generate(uint32_t seed,uint32_t numOperations,uint32_t dimensionRange)110 bool RandomGraph::generate(uint32_t seed, uint32_t numOperations, uint32_t dimensionRange) {
111     RandomNumberGenerator::generator.seed(seed);
112     // The generator may (with low probability) end up with an invalid graph.
113     // If so, regenerate the graph until a valid one is produced.
114     while (true) {
115         RandomVariableNetwork::get()->initialize(dimensionRange);
116         mOperations.clear();
117         mOperands.clear();
118         if (generateGraph(numOperations) && generateValue()) break;
119         std::cout << "[ Retry    ]   The RandomGraphGenerator produces an invalid graph.\n";
120     }
121     return true;
122 }
123 
generateGraph(uint32_t numOperations)124 bool RandomGraph::generateGraph(uint32_t numOperations) {
125     NN_FUZZER_LOG << "Generate Graph";
126     // Randomly generate a vector of operations, this is a valid topological sort.
127     for (uint32_t i = 0; i < numOperations; i++) {
128         mOperations.emplace_back(OperationManager::get()->getRandomOperation());
129     }
130 
131     // Randomly add edges from the output of one operation to the input of another operation
132     // with larger positional index.
133     for (uint32_t i = 0; i < numOperations; i++) {
134         for (auto& output : mOperations[i].outputs) {
135             for (uint32_t j = i + 1; j < numOperations; j++) {
136                 for (auto& input : mOperations[j].inputs) {
137                     // For each [output, input] pair, add an edge with probability prob.
138                     // TODO: Find a better formula by first defining what "better" is.
139                     float prob = 0.1f;
140                     if (getBernoulli(prob)) {
141                         if (output->createEdgeIfValid(*input)) {
142                             NN_FUZZER_LOG << "Add edge: operation " << i << " -> " << j;
143                             input = output;
144                             output->type = RandomOperandType::INTERNAL;
145                         }
146                     }
147                 }
148             }
149         }
150     }
151     return true;
152 }
153 
asConstant(const std::shared_ptr<RandomOperand> & operand,float prob=0.5f)154 static bool asConstant(const std::shared_ptr<RandomOperand>& operand, float prob = 0.5f) {
155     if (operand->type == RandomOperandType::CONST) return true;
156     if (operand->type != RandomOperandType::INPUT) return false;
157     // Force all scalars to be CONST.
158     if (kScalarDataType[static_cast<int32_t>(operand->dataType)]) return true;
159     if (getBernoulli(prob)) return true;
160     return false;
161 }
162 
163 // Freeze the dimensions to a random but valid combination.
164 // Generate random buffer values for model inputs and constants.
generateValue()165 bool RandomGraph::generateValue() {
166     NN_FUZZER_LOG << "Generate Value";
167     if (!RandomVariableNetwork::get()->freeze()) return false;
168 
169     // Fill all unique operands into mOperands.
170     std::set<std::shared_ptr<RandomOperand>> seen;
171     auto fillOperands = [&seen, this](const std::vector<std::shared_ptr<RandomOperand>>& ops) {
172         for (const auto& op : ops) {
173             if (seen.find(op) == seen.end()) {
174                 seen.insert(op);
175                 mOperands.push_back(op);
176             }
177         }
178     };
179     for (const auto& operation : mOperations) {
180         fillOperands(operation.inputs);
181         fillOperands(operation.outputs);
182     }
183 
184     // Count the number of INPUTs.
185     uint32_t numInputs = 0;
186     for (auto& operand : mOperands) {
187         if (operand->type == RandomOperandType::INPUT) numInputs++;
188     }
189 
190     for (auto& operand : mOperands) {
191         // Turn INPUT into CONST with probability prob. Need to keep at least one INPUT.
192         float prob = 0.5f;
193         if (asConstant(operand, prob) && numInputs > 1) {
194             if (operand->type == RandomOperandType::INPUT) numInputs--;
195             operand->type = RandomOperandType::CONST;
196         }
197         if (operand->type != RandomOperandType::INTERNAL &&
198             operand->type != RandomOperandType::NO_VALUE) {
199             if (operand->buffer.empty()) operand->resizeBuffer<uint8_t>(operand->getBufferSize());
200             // If operand is set by randomBuffer, copy the frozen values into buffer.
201             if (!operand->randomBuffer.empty()) {
202                 int32_t* data = reinterpret_cast<int32_t*>(operand->buffer.data());
203                 for (uint32_t i = 0; i < operand->randomBuffer.size(); i++) {
204                     data[i] = operand->randomBuffer[i].getValue();
205                 }
206             }
207             if (operand->finalizer) operand->finalizer(operand.get());
208         }
209     }
210 
211     for (auto& operation : mOperations) {
212         if (operation.finalizer) operation.finalizer(&operation);
213     }
214     return true;
215 }
216 
convertToTestOperandLifeTime(RandomOperandType type)217 static TestOperandLifeTime convertToTestOperandLifeTime(RandomOperandType type) {
218     switch (type) {
219         case RandomOperandType::INPUT:
220             return TestOperandLifeTime::SUBGRAPH_INPUT;
221         case RandomOperandType::OUTPUT:
222             return TestOperandLifeTime::SUBGRAPH_OUTPUT;
223         case RandomOperandType::INTERNAL:
224             return TestOperandLifeTime::TEMPORARY_VARIABLE;
225         case RandomOperandType::CONST:
226             return TestOperandLifeTime::CONSTANT_COPY;
227         case RandomOperandType::NO_VALUE:
228             return TestOperandLifeTime::NO_VALUE;
229     }
230 }
231 
createTestModel()232 TestModel RandomGraph::createTestModel() {
233     NN_FUZZER_LOG << "Create Test Model";
234     TestModel testModel;
235 
236     // Set model operands.
237     for (auto& operand : mOperands) {
238         operand->opIndex = testModel.main.operands.size();
239         TestOperand testOperand = {
240                 .type = static_cast<TestOperandType>(operand->dataType),
241                 .dimensions = operand->getDimensions(),
242                 // It is safe to always set numberOfConsumers to 0 here because
243                 // this field is not used in NDK.
244                 .numberOfConsumers = 0,
245                 .scale = operand->scale,
246                 .zeroPoint = operand->zeroPoint,
247                 .lifetime = convertToTestOperandLifeTime(operand->type),
248                 .isIgnored = operand->doNotCheckAccuracy,
249         };
250 
251         // Test buffers.
252         switch (testOperand.lifetime) {
253             case TestOperandLifeTime::SUBGRAPH_OUTPUT:
254                 testOperand.data = TestBuffer(operand->getBufferSize());
255                 break;
256             case TestOperandLifeTime::SUBGRAPH_INPUT:
257             case TestOperandLifeTime::CONSTANT_COPY:
258             case TestOperandLifeTime::CONSTANT_REFERENCE:
259                 testOperand.data = TestBuffer(operand->getBufferSize(), operand->buffer.data());
260                 break;
261             case TestOperandLifeTime::TEMPORARY_VARIABLE:
262             case TestOperandLifeTime::NO_VALUE:
263                 break;
264             default:
265                 NN_FUZZER_CHECK(false) << "Unknown lifetime";
266         }
267 
268         // Input/Output indexes.
269         if (testOperand.lifetime == TestOperandLifeTime::SUBGRAPH_INPUT) {
270             testModel.main.inputIndexes.push_back(operand->opIndex);
271         } else if (testOperand.lifetime == TestOperandLifeTime::SUBGRAPH_OUTPUT) {
272             testModel.main.outputIndexes.push_back(operand->opIndex);
273         }
274         testModel.main.operands.push_back(std::move(testOperand));
275     }
276 
277     // Set model operations.
278     for (auto& operation : mOperations) {
279         NN_FUZZER_LOG << "Operation: " << toString(operation.opType);
280         TestOperation testOperation = {.type = static_cast<TestOperationType>(operation.opType)};
281         for (auto& op : operation.inputs) {
282             NN_FUZZER_LOG << toString(*op);
283             testOperation.inputs.push_back(op->opIndex);
284         }
285         for (auto& op : operation.outputs) {
286             NN_FUZZER_LOG << toString(*op);
287             testOperation.outputs.push_back(op->opIndex);
288         }
289         testModel.main.operations.push_back(std::move(testOperation));
290     }
291     return testModel;
292 }
293 
294 }  // namespace fuzzing_test
295 }  // namespace nn
296 }  // namespace android
297