blob: f6df05135a0b62a51dec0b127669ffce1602313b [file] [log] [blame]
/*
* Copyright 2020 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 "fuzzer/FuzzedDataProvider.h"
#include "utils/Vector.h"
static constexpr uint16_t MAX_VEC_SIZE = 5000;
void runVectorFuzz(const uint8_t* data, size_t size) {
FuzzedDataProvider dataProvider(data, size);
android::Vector<uint8_t> vec = android::Vector<uint8_t>();
// We want to test handling of sizeof as well.
android::Vector<uint32_t> vec32 = android::Vector<uint32_t>();
// We're going to generate two vectors of this size
size_t vectorSize = dataProvider.ConsumeIntegralInRange<size_t>(0, MAX_VEC_SIZE);
vec.setCapacity(vectorSize);
vec32.setCapacity(vectorSize);
for (size_t i = 0; i < vectorSize; i++) {
uint8_t count = dataProvider.ConsumeIntegralInRange<uint8_t>(1, 5);
vec.insertAt((uint8_t)i, i, count);
vec32.insertAt((uint32_t)i, i, count);
vec.push_front(i);
vec32.push(i);
}
// Now we'll perform some test operations with any remaining data
// Index to perform operations at
size_t index = dataProvider.ConsumeIntegralInRange<size_t>(0, vec.size());
std::vector<uint8_t> remainingVec = dataProvider.ConsumeRemainingBytes<uint8_t>();
// Insert an array and vector
vec.insertArrayAt(remainingVec.data(), index, remainingVec.size());
android::Vector<uint8_t> vecCopy = android::Vector<uint8_t>(vec);
vec.insertVectorAt(vecCopy, index);
// Same thing for 32 bit vector
android::Vector<uint32_t> vec32Copy = android::Vector<uint32_t>(vec32);
vec32.insertArrayAt(vec32Copy.array(), index, vec32.size());
vec32.insertVectorAt(vec32Copy, index);
// Replace single character
if (remainingVec.size() > 0) {
vec.replaceAt(remainingVec[0], index);
vec32.replaceAt(static_cast<uint32_t>(remainingVec[0]), index);
} else {
vec.replaceAt(0, index);
vec32.replaceAt(0, index);
}
// Add any remaining bytes
for (uint8_t i : remainingVec) {
vec.add(i);
vec32.add(static_cast<uint32_t>(i));
}
// Shrink capactiy
vec.setCapacity(remainingVec.size());
vec32.setCapacity(remainingVec.size());
// Iterate through each pointer
size_t sum = 0;
for (auto& it : vec) {
sum += it;
}
for (auto& it : vec32) {
sum += it;
}
// Cleanup
vec.clear();
vecCopy.clear();
vec32.clear();
vec32Copy.clear();
}
extern "C" int LLVMFuzzerTestOneInput(const uint8_t* data, size_t size) {
runVectorFuzz(data, size);
return 0;
}