123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137 |
- //===- FuzzerDataFlowTrace.h - Internal header for the Fuzzer ---*- C++ -* ===//
- //
- // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
- // See https://llvm.org/LICENSE.txt for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- //
- //===----------------------------------------------------------------------===//
- // fuzzer::DataFlowTrace; reads and handles a data-flow trace.
- //
- // A data flow trace is generated by e.g. dataflow/DataFlow.cpp
- // and is stored on disk in a separate directory.
- //
- // The trace dir contains a file 'functions.txt' which lists function names,
- // oner per line, e.g.
- // ==> functions.txt <==
- // Func2
- // LLVMFuzzerTestOneInput
- // Func1
- //
- // All other files in the dir are the traces, see dataflow/DataFlow.cpp.
- // The name of the file is sha1 of the input used to generate the trace.
- //
- // Current status:
- // the data is parsed and the summary is printed, but the data is not yet
- // used in any other way.
- //===----------------------------------------------------------------------===//
- #ifndef LLVM_FUZZER_DATA_FLOW_TRACE
- #define LLVM_FUZZER_DATA_FLOW_TRACE
- #include "FuzzerDefs.h"
- #include "FuzzerIO.h"
- #include <unordered_map>
- #include <unordered_set>
- #include <vector>
- #include <string>
- namespace fuzzer {
- int CollectDataFlow(const std::string &DFTBinary, const std::string &DirPath,
- const std::vector<SizedFile> &CorporaFiles);
- class BlockCoverage {
- public:
- // These functions guarantee no CoverageVector is longer than UINT32_MAX.
- bool AppendCoverage(std::istream &IN);
- bool AppendCoverage(const std::string &S);
- size_t NumCoveredFunctions() const { return Functions.size(); }
- uint32_t GetCounter(size_t FunctionId, size_t BasicBlockId) {
- auto It = Functions.find(FunctionId);
- if (It == Functions.end())
- return 0;
- const auto &Counters = It->second;
- if (BasicBlockId < Counters.size())
- return Counters[BasicBlockId];
- return 0;
- }
- uint32_t GetNumberOfBlocks(size_t FunctionId) {
- auto It = Functions.find(FunctionId);
- if (It == Functions.end()) return 0;
- const auto &Counters = It->second;
- return static_cast<uint32_t>(Counters.size());
- }
- uint32_t GetNumberOfCoveredBlocks(size_t FunctionId) {
- auto It = Functions.find(FunctionId);
- if (It == Functions.end()) return 0;
- const auto &Counters = It->second;
- uint32_t Result = 0;
- for (auto Cnt: Counters)
- if (Cnt)
- Result++;
- return Result;
- }
- std::vector<double> FunctionWeights(size_t NumFunctions) const;
- void clear() { Functions.clear(); }
- private:
- typedef std::vector<uint32_t> CoverageVector;
- uint32_t NumberOfCoveredBlocks(const CoverageVector &Counters) const {
- uint32_t Res = 0;
- for (auto Cnt : Counters)
- if (Cnt)
- Res++;
- return Res;
- }
- uint32_t NumberOfUncoveredBlocks(const CoverageVector &Counters) const {
- return static_cast<uint32_t>(Counters.size()) -
- NumberOfCoveredBlocks(Counters);
- }
- uint32_t SmallestNonZeroCounter(const CoverageVector &Counters) const {
- assert(!Counters.empty());
- uint32_t Res = Counters[0];
- for (auto Cnt : Counters)
- if (Cnt)
- Res = Min(Res, Cnt);
- assert(Res);
- return Res;
- }
- // Function ID => vector of counters.
- // Each counter represents how many input files trigger the given basic block.
- std::unordered_map<size_t, CoverageVector> Functions;
- // Functions that have DFT entry.
- std::unordered_set<size_t> FunctionsWithDFT;
- };
- class DataFlowTrace {
- public:
- void ReadCoverage(const std::string &DirPath);
- bool Init(const std::string &DirPath, std::string *FocusFunction,
- std::vector<SizedFile> &CorporaFiles, Random &Rand);
- void Clear() { Traces.clear(); }
- const std::vector<uint8_t> *Get(const std::string &InputSha1) const {
- auto It = Traces.find(InputSha1);
- if (It != Traces.end())
- return &It->second;
- return nullptr;
- }
- private:
- // Input's sha1 => DFT for the FocusFunction.
- std::unordered_map<std::string, std::vector<uint8_t>> Traces;
- BlockCoverage Coverage;
- std::unordered_set<std::string> CorporaHashes;
- };
- } // namespace fuzzer
- #endif // LLVM_FUZZER_DATA_FLOW_TRACE
|