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