VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/fuzz/fuzz.cpp@ 86103

Last change on this file since 86103 was 85121, checked in by vboxsync, 5 years ago

iprt/cdefs.h: Refactored the typedef use of DECLCALLBACK as well as DECLCALLBACKMEMBER to wrap the whole expression, similar to the DECLR?CALLBACKMEMBER macros. This allows adding a throw() at the end when compiling with the VC++ compiler to indicate that the callbacks won't throw anything, so we can stop supressing the C5039 warning about passing functions that can potential throw C++ exceptions to extern C code that can't necessarily cope with such (unwind,++). Introduced a few _EX variations that allows specifying different/no calling convention too, as that's handy when dynamically resolving host APIs. Fixed numerous places missing DECLCALLBACK and such. Left two angry @todos regarding use of CreateThread. bugref:9794

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 82.5 KB
Line 
1/* $Id: fuzz.cpp 85121 2020-07-08 19:33:26Z vboxsync $ */
2/** @file
3 * IPRT - Fuzzing framework API, core.
4 */
5
6/*
7 * Copyright (C) 2018-2020 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#include <iprt/fuzz.h>
32#include "internal/iprt.h"
33
34#include <iprt/asm.h>
35#include <iprt/assert.h>
36#include <iprt/avl.h>
37#include <iprt/critsect.h>
38#include <iprt/ctype.h>
39#include <iprt/dir.h>
40#include <iprt/err.h>
41#include <iprt/file.h>
42#include <iprt/list.h>
43#include <iprt/md5.h>
44#include <iprt/mem.h>
45#include <iprt/path.h>
46#include <iprt/rand.h>
47#include <iprt/semaphore.h>
48#include <iprt/string.h>
49#include <iprt/time.h>
50#include <iprt/vfs.h>
51
52
53/*********************************************************************************************************************************
54* Defined Constants And Macros *
55*********************************************************************************************************************************/
56#define RTFUZZCTX_MAGIC UINT32_C(0xdeadc0de) /** @todo */
57
58
59/*********************************************************************************************************************************
60* Structures and Typedefs *
61*********************************************************************************************************************************/
62/** Pointer to the internal fuzzer state. */
63typedef struct RTFUZZCTXINT *PRTFUZZCTXINT;
64/** Pointer to a fuzzed mutation. */
65typedef struct RTFUZZMUTATION *PRTFUZZMUTATION;
66/** Pointer to a fuzzed mutation pointer. */
67typedef PRTFUZZMUTATION *PPRTFUZZMUTATION;
68/** Pointer to a const mutation. */
69typedef const struct RTFUZZMUTATION *PCRTFUZZMUTATION;
70
71
72/**
73 * Mutator class.
74 */
75typedef enum RTFUZZMUTATORCLASS
76{
77 /** Invalid class, do not use. */
78 RTFUZZMUTATORCLASS_INVALID = 0,
79 /** Mutator operates on single bits. */
80 RTFUZZMUTATORCLASS_BITS,
81 /** Mutator operates on bytes (single or multiple). */
82 RTFUZZMUTATORCLASS_BYTES,
83 /** Mutator interpretes data as integers and operates on them. */
84 RTFUZZMUTATORCLASS_INTEGERS,
85 /** Mutator uses multiple mutations to create new mutations. */
86 RTFUZZMUTATORCLASS_MUTATORS,
87 /** 32bit hack. */
88 RTFUZZMUTATORCLASS_32BIT_HACK = 0x7fffffff
89} RTFUZZMUTATORCLASS;
90
91
92/**
93 * Mutator preparation callback.
94 *
95 * @returns IPRT status code.
96 * @param pThis The fuzzer context instance.
97 * @param offStart Where the mutation should start.
98 * @param pMutationParent The parent mutation to start working from.
99 * @param ppMutation Where to store the created mutation on success.
100 */
101typedef DECLCALLBACKTYPE(int, FNRTFUZZCTXMUTATORPREP,(PRTFUZZCTXINT pThis, uint64_t offStart, PRTFUZZMUTATION pMutationParent,
102 PPRTFUZZMUTATION ppMutation));
103/** Pointer to a mutator preparation callback. */
104typedef FNRTFUZZCTXMUTATORPREP *PFNRTFUZZCTXMUTATORPREP;
105
106
107/**
108 * Mutator execution callback.
109 *
110 * @returns IPRT status code.
111 * @param pThis The fuzzer context instance.
112 * @param pMutation The mutation to work on.
113 * @param pvMutation Mutation dependent data.
114 * @param pbBuf The buffer to work on.
115 * @param cbBuf Size of the remaining buffer.
116 */
117typedef DECLCALLBACKTYPE(int, FNRTFUZZCTXMUTATOREXEC,(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
118 uint8_t *pbBuf, size_t cbBuf));
119/** Pointer to a mutator execution callback. */
120typedef FNRTFUZZCTXMUTATOREXEC *PFNRTFUZZCTXMUTATOREXEC;
121
122
123/**
124 * Mutator export callback.
125 *
126 * @returns IPRT status code.
127 * @param pThis The fuzzer context instance.
128 * @param pMutation The mutation to work on.
129 * @param pvMutation Mutation dependent data.
130 * @param pfnExport The export callback.
131 * @param pvUser Opaque user data to pass to the export callback.
132 */
133typedef DECLCALLBACKTYPE(int, FNRTFUZZCTXMUTATOREXPORT,(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
134 PFNRTFUZZCTXEXPORT pfnExport, void *pvUser));
135/** Pointer to a mutator export callback. */
136typedef FNRTFUZZCTXMUTATOREXPORT *PFNRTFUZZCTXMUTATOREXPORT;
137
138
139/**
140 * Mutator import callback.
141 *
142 * @returns IPRT status code.
143 * @param pThis The fuzzer context instance.
144 * @param pMutation The mutation to work on.
145 * @param pvMutation Mutation dependent data.
146 * @param pfnExport The import callback.
147 * @param pvUser Opaque user data to pass to the import callback.
148 */
149typedef DECLCALLBACKTYPE(int, FNRTFUZZCTXMUTATORIMPORT,(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, void *pvMutation,
150 PFNRTFUZZCTXIMPORT pfnImport, void *pvUser));
151/** Pointer to a mutator import callback. */
152typedef FNRTFUZZCTXMUTATORIMPORT *PFNRTFUZZCTXMUTATORIMPORT;
153
154
155/**
156 * A fuzzing mutator descriptor.
157 */
158typedef struct RTFUZZMUTATOR
159{
160 /** Id of the mutator. */
161 const char *pszId;
162 /** Mutator description. */
163 const char *pszDesc;
164 /** Mutator index. */
165 uint32_t uMutator;
166 /** Mutator class. */
167 RTFUZZMUTATORCLASS enmClass;
168 /** Additional flags for the mutator, controlling the behavior. */
169 uint64_t fFlags;
170 /** The preparation callback. */
171 PFNRTFUZZCTXMUTATORPREP pfnPrep;
172 /** The execution callback. */
173 PFNRTFUZZCTXMUTATOREXEC pfnExec;
174 /** The export callback. */
175 PFNRTFUZZCTXMUTATOREXPORT pfnExport;
176 /** The import callback. */
177 PFNRTFUZZCTXMUTATORIMPORT pfnImport;
178} RTFUZZMUTATOR;
179/** Pointer to a fuzzing mutator descriptor. */
180typedef RTFUZZMUTATOR *PRTFUZZMUTATOR;
181/** Pointer to a const fuzzing mutator descriptor. */
182typedef const RTFUZZMUTATOR *PCRTFUZZMUTATOR;
183
184/** The special corpus mutator. */
185#define RTFUZZMUTATOR_ID_CORPUS UINT32_C(0xffffffff)
186
187/** Mutator always works from the end of the buffer (no starting offset generation). */
188#define RTFUZZMUTATOR_F_END_OF_BUF RT_BIT_64(0)
189/** Default flags. */
190#define RTFUZZMUTATOR_F_DEFAULT (0)
191
192
193/**
194 * A fuzzed mutation.
195 */
196typedef struct RTFUZZMUTATION
197{
198 /** The AVL tree core. */
199 AVLU64NODECORE Core;
200 /** The list node if the mutation has the mutated
201 * data allocated. */
202 RTLISTNODE NdAlloc;
203 /** Magic identifying this structure. */
204 uint32_t u32Magic;
205 /** Reference counter. */
206 volatile uint32_t cRefs;
207 /** The fuzzer this mutation belongs to. */
208 PRTFUZZCTXINT pFuzzer;
209 /** Parent mutation (no reference is held), NULL means root or original data. */
210 PRTFUZZMUTATION pMutationParent;
211 /** Start offset where new mutations are allowed to start. */
212 uint64_t offMutStartNew;
213 /** Size of the range in bytes where mutations are allowed to happen. */
214 uint64_t cbMutNew;
215 /** Mutation level. */
216 uint32_t iLvl;
217 /** The mutator causing this mutation, NULL if original input data. */
218 PCRTFUZZMUTATOR pMutator;
219 /** Byte offset where the mutation starts. */
220 uint64_t offMutation;
221 /** Size of the generated input data in bytes after the mutation was applied. */
222 size_t cbInput;
223 /** Size of the mutation dependent data. */
224 size_t cbMutation;
225 /** Size allocated for the input. */
226 size_t cbAlloc;
227 /** Pointer to the input data if created. */
228 void *pvInput;
229 /** Flag whether the mutation is contained in the tree of the context. */
230 bool fInTree;
231 /** Flag whether the mutation input data is cached. */
232 bool fCached;
233 /** Mutation dependent data, variable in size. */
234 uint8_t abMutation[1];
235} RTFUZZMUTATION;
236
237
238/**
239 * A fuzzing input seed.
240 */
241typedef struct RTFUZZINPUTINT
242{
243 /** Magic identifying this structure. */
244 uint32_t u32Magic;
245 /** Reference counter. */
246 volatile uint32_t cRefs;
247 /** The fuzzer this input belongs to. */
248 PRTFUZZCTXINT pFuzzer;
249 /** The top mutation to work from (reference held). */
250 PRTFUZZMUTATION pMutationTop;
251 /** Fuzzer context type dependent data. */
252 union
253 {
254 /** Blob data. */
255 struct
256 {
257 /** Pointer to the input data if created. */
258 void *pvInput;
259 } Blob;
260 /** Stream state. */
261 struct
262 {
263 /** Number of bytes seen so far. */
264 size_t cbSeen;
265 } Stream;
266 } u;
267} RTFUZZINPUTINT;
268/** Pointer to the internal input state. */
269typedef RTFUZZINPUTINT *PRTFUZZINPUTINT;
270/** Pointer to an internal input state pointer. */
271typedef PRTFUZZINPUTINT *PPRTFUZZINPUTINT;
272
273
274/**
275 * The fuzzer state.
276 */
277typedef struct RTFUZZCTXINT
278{
279 /** Magic value for identification. */
280 uint32_t u32Magic;
281 /** Reference counter. */
282 volatile uint32_t cRefs;
283 /** The random number generator. */
284 RTRAND hRand;
285 /** Fuzzing context type. */
286 RTFUZZCTXTYPE enmType;
287 /** Semaphore protecting the mutations tree. */
288 RTSEMRW hSemRwMutations;
289 /** The AVL tree for indexing the mutations (keyed by counter). */
290 AVLU64TREE TreeMutations;
291 /** Number of inputs currently in the tree. */
292 volatile uint64_t cMutations;
293 /** The maximum size of one input seed to generate. */
294 size_t cbInputMax;
295 /** Behavioral flags. */
296 uint32_t fFlagsBehavioral;
297 /** Number of enabled mutators. */
298 uint32_t cMutators;
299 /** Pointer to the mutator descriptors. */
300 PRTFUZZMUTATOR paMutators;
301 /** Maximum amount of bytes of mutated inputs to cache. */
302 size_t cbMutationsAllocMax;
303 /** Current amount of bytes of cached mutated inputs. */
304 size_t cbMutationsAlloc;
305 /** List of mutators having data allocated currently. */
306 RTLISTANCHOR LstMutationsAlloc;
307 /** Critical section protecting the allocation list. */
308 RTCRITSECT CritSectAlloc;
309 /** Total number of bytes of memory currently allocated in total for this context. */
310 volatile size_t cbMemTotal;
311 /** Start offset in the input where a mutation is allowed to happen. */
312 uint64_t offMutStart;
313 /** size of the range where a mutation can happen. */
314 uint64_t cbMutRange;
315} RTFUZZCTXINT;
316
317
318/**
319 * The fuzzer state to be exported - all members are stored in little endian form.
320 */
321typedef struct RTFUZZCTXSTATE
322{
323 /** Magic value for identification. */
324 uint32_t u32Magic;
325 /** Context type. */
326 uint32_t uCtxType;
327 /** Size of the PRNG state following in bytes. */
328 uint32_t cbPrng;
329 /** Number of mutator descriptors following. */
330 uint32_t cMutators;
331 /** Number of mutation descriptors following. */
332 uint32_t cMutations;
333 /** Behavioral flags. */
334 uint32_t fFlagsBehavioral;
335 /** Maximum input size to generate. */
336 uint64_t cbInputMax;
337} RTFUZZCTXSTATE;
338/** Pointer to a fuzzing context state. */
339typedef RTFUZZCTXSTATE *PRTFUZZCTXSTATE;
340
341/** BLOB context type. */
342#define RTFUZZCTX_STATE_TYPE_BLOB UINT32_C(0)
343/** Stream context type. */
344#define RTFUZZCTX_STATE_TYPE_STREAM UINT32_C(1)
345
346
347/**
348 * The fuzzer mutation state to be exported - all members are stored in little endian form.
349 */
350typedef struct RTFUZZMUTATIONSTATE
351{
352 /** The mutation identifier. */
353 uint64_t u64Id;
354 /** The mutation identifier of the parent, 0 for no parent. */
355 uint64_t u64IdParent;
356 /** The byte offset where the mutation starts. */
357 uint64_t u64OffMutation;
358 /** Size of input data after mutation was applied. */
359 uint64_t cbInput;
360 /** Size of mutation dependent data following. */
361 uint64_t cbMutation;
362 /** The mutator ID. */
363 uint32_t u32IdMutator;
364 /** The mutation level. */
365 uint32_t iLvl;
366 /** Magic value for identification. */
367 uint32_t u32Magic;
368} RTFUZZMUTATIONSTATE;
369
370
371/**
372 * Fuzzing context memory header.
373 */
374typedef struct RTFUZZMEMHDR
375{
376 /** Size of the memory area following. */
377 size_t cb;
378#if HC_ARCH_BITS == 32
379 /** Some padding. */
380 uint32_t uPadding0;
381#elif HC_ARCH_BITS == 64
382 /** Some padding. */
383 uint64_t uPadding0;
384#else
385# error "Port me"
386#endif
387} RTFUZZMEMHDR;
388/** Pointer to a memory header. */
389typedef RTFUZZMEMHDR *PRTFUZZMEMHDR;
390
391
392/**
393 * Fuzzing context export AVL arguments.
394 */
395typedef struct RTFUZZEXPORTARGS
396{
397 /** Pointer to the export callback. */
398 PFNRTFUZZCTXEXPORT pfnExport;
399 /** Opaque user data to pass to the callback. */
400 void *pvUser;
401} RTFUZZEXPORTARGS;
402/** Pointer to the export arguments. */
403typedef RTFUZZEXPORTARGS *PRTFUZZEXPORTARGS;
404/** Pointer to the constant export arguments. */
405typedef const RTFUZZEXPORTARGS *PCRTFUZZEXPORTARGS;
406
407
408/**
409 * Integer replacing mutator additional data.
410 */
411typedef struct RTFUZZMUTATORINTEGER
412{
413 /** The integer class. */
414 uint8_t uIntClass;
415 /** Flag whether to do a byte swap. */
416 bool fByteSwap;
417 /** The index into the class specific array. */
418 uint16_t idxInt;
419} RTFUZZMUTATORINTEGER;
420/** Pointer to additional integer replacing mutator data. */
421typedef RTFUZZMUTATORINTEGER *PRTFUZZMUTATORINTEGER;
422/** Pointer to constant additional integer replacing mutator data. */
423typedef const RTFUZZMUTATORINTEGER *PCRTFUZZMUTATORINTEGER;
424
425
426/*********************************************************************************************************************************
427* Internal Functions *
428*********************************************************************************************************************************/
429static DECLCALLBACK(int) rtFuzzCtxMutatorBitFlipPrep(PRTFUZZCTXINT pThis, uint64_t offStart, PRTFUZZMUTATION pMutationParent,
430 PPRTFUZZMUTATION ppMutation);
431static DECLCALLBACK(int) rtFuzzCtxMutatorByteReplacePrep(PRTFUZZCTXINT pThis, uint64_t offStart, PRTFUZZMUTATION pMutationParent,
432 PPRTFUZZMUTATION ppMutation);
433static DECLCALLBACK(int) rtFuzzCtxMutatorByteInsertPrep(PRTFUZZCTXINT pThis, uint64_t offStart, PRTFUZZMUTATION pMutationParent,
434 PPRTFUZZMUTATION ppMutation);
435static DECLCALLBACK(int) rtFuzzCtxMutatorByteSequenceInsertAppendPrep(PRTFUZZCTXINT pThis, uint64_t offStart, PRTFUZZMUTATION pMutationParent,
436 PPRTFUZZMUTATION ppMutation);
437static DECLCALLBACK(int) rtFuzzCtxMutatorByteDeletePrep(PRTFUZZCTXINT pThis, uint64_t offStart, PRTFUZZMUTATION pMutationParent,
438 PPRTFUZZMUTATION ppMutation);
439static DECLCALLBACK(int) rtFuzzCtxMutatorByteSequenceDeletePrep(PRTFUZZCTXINT pThis, uint64_t offStart, PRTFUZZMUTATION pMutationParent,
440 PPRTFUZZMUTATION ppMutation);
441static DECLCALLBACK(int) rtFuzzCtxMutatorIntegerReplacePrep(PRTFUZZCTXINT pThis, uint64_t offStart, PRTFUZZMUTATION pMutationParent,
442 PPRTFUZZMUTATION ppMutation);
443static DECLCALLBACK(int) rtFuzzCtxMutatorCrossoverPrep(PRTFUZZCTXINT pThis, uint64_t offStart, PRTFUZZMUTATION pMutationParent,
444 PPRTFUZZMUTATION ppMutation);
445
446static DECLCALLBACK(int) rtFuzzCtxMutatorCorpusExec(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
447 uint8_t *pbBuf, size_t cbBuf);
448static DECLCALLBACK(int) rtFuzzCtxMutatorBitFlipExec(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
449 uint8_t *pbBuf, size_t cbBuf);
450static DECLCALLBACK(int) rtFuzzCtxMutatorByteReplaceExec(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
451 uint8_t *pbBuf, size_t cbBuf);
452static DECLCALLBACK(int) rtFuzzCtxMutatorByteInsertExec(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
453 uint8_t *pbBuf, size_t cbBuf);
454static DECLCALLBACK(int) rtFuzzCtxMutatorByteSequenceInsertAppendExec(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
455 uint8_t *pbBuf, size_t cbBuf);
456static DECLCALLBACK(int) rtFuzzCtxMutatorByteDeleteExec(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
457 uint8_t *pbBuf, size_t cbBuf);
458static DECLCALLBACK(int) rtFuzzCtxMutatorByteSequenceDeleteExec(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
459 uint8_t *pbBuf, size_t cbBuf);
460static DECLCALLBACK(int) rtFuzzCtxMutatorIntegerReplaceExec(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
461 uint8_t *pbBuf, size_t cbBuf);
462static DECLCALLBACK(int) rtFuzzCtxMutatorCrossoverExec(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
463 uint8_t *pbBuf, size_t cbBuf);
464
465static DECLCALLBACK(int) rtFuzzCtxMutatorExportDefault(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
466 PFNRTFUZZCTXEXPORT pfnExport, void *pvUser);
467static DECLCALLBACK(int) rtFuzzCtxMutatorImportDefault(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, void *pvMutation,
468 PFNRTFUZZCTXIMPORT pfnImport, void *pvUser);
469
470static DECLCALLBACK(int) rtFuzzCtxMutatorCrossoverExport(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
471 PFNRTFUZZCTXEXPORT pfnExport, void *pvUser);
472static DECLCALLBACK(int) rtFuzzCtxMutatorCrossoverImport(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, void *pvMutation,
473 PFNRTFUZZCTXIMPORT pfnImport, void *pvUser);
474
475
476/*********************************************************************************************************************************
477* Global Variables *
478*********************************************************************************************************************************/
479
480/** Signed 8bit interesting values. */
481static int8_t s_ai8Interesting[] = { INT8_MIN, INT8_MIN + 1, -1, 0, 1, INT8_MAX - 1, INT8_MAX };
482/** Unsigned 8bit interesting values. */
483static uint8_t s_au8Interesting[] = { 0, 1, UINT8_MAX - 1, UINT8_MAX };
484/** Signed 16bit interesting values. */
485static int16_t s_ai16Interesting[] = { INT16_MIN, INT16_MIN + 1, -1, 0, 1, INT16_MAX - 1, INT16_MAX };
486/** Unsigned 16bit interesting values. */
487static uint16_t s_au16Interesting[] = { 0, 1, UINT16_MAX - 1, UINT16_MAX };
488/** Signed 32bit interesting values. */
489static int32_t s_ai32Interesting[] = { INT32_MIN, INT32_MIN + 1, -1, 0, 1, INT32_MAX - 1, INT32_MAX };
490/** Unsigned 32bit interesting values. */
491static uint32_t s_au32Interesting[] = { 0, 1, UINT32_MAX - 1, UINT32_MAX };
492/** Signed 64bit interesting values. */
493static int64_t s_ai64Interesting[] = { INT64_MIN, INT64_MIN + 1, -1, 0, 1, INT64_MAX - 1, INT64_MAX };
494/** Unsigned 64bit interesting values. */
495static uint64_t s_au64Interesting[] = { 0, 1, UINT64_MAX - 1, UINT64_MAX };
496
497
498/**
499 * The special corpus mutator for the original data.
500 */
501static RTFUZZMUTATOR const g_MutatorCorpus =
502{
503 /** pszId */
504 "Corpus",
505 /** pszDesc */
506 "Special mutator, which is assigned to the initial corpus",
507 /** uMutator. */
508 RTFUZZMUTATOR_ID_CORPUS,
509 /** enmClass. */
510 RTFUZZMUTATORCLASS_BYTES,
511 /** fFlags */
512 RTFUZZMUTATOR_F_DEFAULT,
513 /** pfnPrep */
514 NULL,
515 /** pfnExec */
516 rtFuzzCtxMutatorCorpusExec,
517 /** pfnExport */
518 rtFuzzCtxMutatorExportDefault,
519 /** pfnImport */
520 rtFuzzCtxMutatorImportDefault
521};
522
523/**
524 * Array of all available mutators.
525 */
526static RTFUZZMUTATOR const g_aMutators[] =
527{
528 /* pszId pszDesc uMutator enmClass fFlags pfnPrep pfnExec pfnExport pfnImport */
529 { "BitFlip", "Flips a single bit in the input", 0, RTFUZZMUTATORCLASS_BITS, RTFUZZMUTATOR_F_DEFAULT, rtFuzzCtxMutatorBitFlipPrep, rtFuzzCtxMutatorBitFlipExec, rtFuzzCtxMutatorExportDefault, rtFuzzCtxMutatorImportDefault },
530 { "ByteReplace", "Replaces a single byte in the input", 1, RTFUZZMUTATORCLASS_BYTES, RTFUZZMUTATOR_F_DEFAULT, rtFuzzCtxMutatorByteReplacePrep, rtFuzzCtxMutatorByteReplaceExec, rtFuzzCtxMutatorExportDefault, rtFuzzCtxMutatorImportDefault },
531 { "ByteInsert", "Inserts a single byte sequence into the input", 2, RTFUZZMUTATORCLASS_BYTES, RTFUZZMUTATOR_F_DEFAULT, rtFuzzCtxMutatorByteInsertPrep, rtFuzzCtxMutatorByteInsertExec, rtFuzzCtxMutatorExportDefault, rtFuzzCtxMutatorImportDefault },
532 { "ByteSeqIns", "Inserts a byte sequence in the input", 3, RTFUZZMUTATORCLASS_BYTES, RTFUZZMUTATOR_F_DEFAULT, rtFuzzCtxMutatorByteSequenceInsertAppendPrep, rtFuzzCtxMutatorByteSequenceInsertAppendExec, rtFuzzCtxMutatorExportDefault, rtFuzzCtxMutatorImportDefault },
533 { "ByteSeqApp", "Appends a byte sequence to the input", 4, RTFUZZMUTATORCLASS_BYTES, RTFUZZMUTATOR_F_END_OF_BUF, rtFuzzCtxMutatorByteSequenceInsertAppendPrep, rtFuzzCtxMutatorByteSequenceInsertAppendExec, rtFuzzCtxMutatorExportDefault, rtFuzzCtxMutatorImportDefault },
534 { "ByteDelete", "Deletes a single byte sequence from the input", 5, RTFUZZMUTATORCLASS_BYTES, RTFUZZMUTATOR_F_DEFAULT, rtFuzzCtxMutatorByteDeletePrep, rtFuzzCtxMutatorByteDeleteExec, NULL, NULL },
535 { "ByteSeqDel", "Deletes a byte sequence from the input", 6, RTFUZZMUTATORCLASS_BYTES, RTFUZZMUTATOR_F_DEFAULT, rtFuzzCtxMutatorByteSequenceDeletePrep, rtFuzzCtxMutatorByteSequenceDeleteExec, NULL, NULL },
536 { "IntReplace", "Replaces a possible integer with an interesting one", 7, RTFUZZMUTATORCLASS_INTEGERS, RTFUZZMUTATOR_F_DEFAULT, rtFuzzCtxMutatorIntegerReplacePrep, rtFuzzCtxMutatorIntegerReplaceExec, rtFuzzCtxMutatorExportDefault, rtFuzzCtxMutatorImportDefault },
537 { "MutCrossover", "Creates a crossover of two other mutations", 8, RTFUZZMUTATORCLASS_MUTATORS, RTFUZZMUTATOR_F_DEFAULT, rtFuzzCtxMutatorCrossoverPrep, rtFuzzCtxMutatorCrossoverExec, rtFuzzCtxMutatorCrossoverExport, rtFuzzCtxMutatorCrossoverImport }
538};
539
540
541/**
542 * Allocates the given number of bytes.
543 *
544 * @returns Pointer to the allocated memory
545 * @param pThis The fuzzer context instance.
546 * @param cb How much to allocate.
547 */
548static void *rtFuzzCtxMemoryAlloc(PRTFUZZCTXINT pThis, size_t cb)
549{
550 AssertReturn(cb > 0, NULL);
551
552 PRTFUZZMEMHDR pMemHdr = (PRTFUZZMEMHDR)RTMemAllocZ(cb + sizeof(RTFUZZMEMHDR));
553 if (RT_LIKELY(pMemHdr))
554 {
555 pMemHdr->cb = cb;
556 size_t cbIgn = ASMAtomicAddZ(&pThis->cbMemTotal, cb + sizeof(RTFUZZMEMHDR)); RT_NOREF(cbIgn);
557 return pMemHdr + 1;
558 }
559
560 return NULL;
561}
562
563
564/**
565 * Frees the given memory.
566 *
567 * @returns nothing.
568 * @param pThis The fuzzer context instance.
569 * @param pv Pointer to the memory area to free.
570 */
571static void rtFuzzCtxMemoryFree(PRTFUZZCTXINT pThis, void *pv)
572{
573 AssertReturnVoid(pv != NULL);
574 PRTFUZZMEMHDR pMemHdr = ((PRTFUZZMEMHDR)pv) - 1;
575
576 size_t cbIgn = ASMAtomicSubZ(&pThis->cbMemTotal, pMemHdr->cb + sizeof(RTFUZZMEMHDR)); RT_NOREF(cbIgn);
577 RTMemFree(pMemHdr);
578}
579
580
581/**
582 * Frees the cached inputs until the given amount is free.
583 *
584 * @returns Whether the amount of memory is free.
585 * @param pThis The fuzzer context instance.
586 * @param cb How many bytes to reclaim
587 */
588static bool rtFuzzCtxMutationAllocReclaim(PRTFUZZCTXINT pThis, size_t cb)
589{
590 while ( !RTListIsEmpty(&pThis->LstMutationsAlloc)
591 && pThis->cbMutationsAlloc + cb > pThis->cbMutationsAllocMax)
592 {
593 PRTFUZZMUTATION pMutation = RTListGetLast(&pThis->LstMutationsAlloc, RTFUZZMUTATION, NdAlloc);
594 AssertPtr(pMutation);
595 AssertPtr(pMutation->pvInput);
596
597 rtFuzzCtxMemoryFree(pThis, pMutation->pvInput);
598 pThis->cbMutationsAlloc -= pMutation->cbAlloc;
599 pMutation->pvInput = NULL;
600 pMutation->cbAlloc = 0;
601 pMutation->fCached = false;
602 RTListNodeRemove(&pMutation->NdAlloc);
603 }
604
605 return pThis->cbMutationsAlloc + cb <= pThis->cbMutationsAllocMax;
606}
607
608
609/**
610 * Updates the cache status of the given mutation.
611 *
612 * @returns nothing.
613 * @param pThis The fuzzer context instance.
614 * @param pMutation The mutation to update.
615 */
616static void rtFuzzCtxMutationMaybeEnterCache(PRTFUZZCTXINT pThis, PRTFUZZMUTATION pMutation)
617{
618 RTCritSectEnter(&pThis->CritSectAlloc);
619
620 /* Initial corpus mutations are not freed. */
621 if ( pMutation->pvInput
622 && pMutation->pMutator != &g_MutatorCorpus)
623 {
624 Assert(!pMutation->fCached);
625
626 if (rtFuzzCtxMutationAllocReclaim(pThis, pMutation->cbAlloc))
627 {
628 RTListPrepend(&pThis->LstMutationsAlloc, &pMutation->NdAlloc);
629 pThis->cbMutationsAlloc += pMutation->cbAlloc;
630 pMutation->fCached = true;
631 }
632 else
633 {
634 rtFuzzCtxMemoryFree(pThis, pMutation->pvInput);
635 pMutation->pvInput = NULL;
636 pMutation->cbAlloc = 0;
637 pMutation->fCached = false;
638 }
639 }
640 RTCritSectLeave(&pThis->CritSectAlloc);
641}
642
643
644/**
645 * Removes a cached mutation from the cache.
646 *
647 * @returns nothing.
648 * @param pThis The fuzzer context instance.
649 * @param pMutation The mutation to remove.
650 */
651static void rtFuzzCtxMutationCacheRemove(PRTFUZZCTXINT pThis, PRTFUZZMUTATION pMutation)
652{
653 RTCritSectEnter(&pThis->CritSectAlloc);
654 if (pMutation->fCached)
655 {
656 RTListNodeRemove(&pMutation->NdAlloc);
657 pThis->cbMutationsAlloc -= pMutation->cbAlloc;
658 pMutation->fCached = false;
659 }
660 RTCritSectLeave(&pThis->CritSectAlloc);
661}
662
663
664/**
665 * Destroys the given mutation.
666 *
667 * @returns nothing.
668 * @param pMutation The mutation to destroy.
669 */
670static void rtFuzzMutationDestroy(PRTFUZZMUTATION pMutation)
671{
672 if (pMutation->pvInput)
673 {
674 rtFuzzCtxMemoryFree(pMutation->pFuzzer, pMutation->pvInput);
675 if (pMutation->fCached)
676 {
677 RTCritSectEnter(&pMutation->pFuzzer->CritSectAlloc);
678 RTListNodeRemove(&pMutation->NdAlloc);
679 pMutation->pFuzzer->cbMutationsAlloc -= pMutation->cbAlloc;
680 RTCritSectLeave(&pMutation->pFuzzer->CritSectAlloc);
681 }
682 pMutation->pvInput = NULL;
683 pMutation->cbAlloc = 0;
684 pMutation->fCached = false;
685 }
686 rtFuzzCtxMemoryFree(pMutation->pFuzzer, pMutation);
687}
688
689
690/**
691 * Retains an external reference to the given mutation.
692 *
693 * @returns New reference count on success.
694 * @param pMutation The mutation to retain.
695 */
696static uint32_t rtFuzzMutationRetain(PRTFUZZMUTATION pMutation)
697{
698 uint32_t cRefs = ASMAtomicIncU32(&pMutation->cRefs);
699 AssertMsg( ( cRefs > 1
700 || pMutation->fInTree)
701 && cRefs < _1M, ("%#x %p\n", cRefs, pMutation));
702
703 if (cRefs == 1)
704 rtFuzzCtxMutationCacheRemove(pMutation->pFuzzer, pMutation);
705 return cRefs;
706}
707
708
709/**
710 * Releases an external reference from the given mutation.
711 *
712 * @returns New reference count on success.
713 * @param pMutation The mutation to retain.
714 */
715static uint32_t rtFuzzMutationRelease(PRTFUZZMUTATION pMutation)
716{
717 uint32_t cRefs = ASMAtomicDecU32(&pMutation->cRefs);
718 AssertMsg(cRefs < _1M, ("%#x %p\n", cRefs, pMutation));
719
720 if (cRefs == 0)
721 {
722 if (!pMutation->fInTree)
723 rtFuzzMutationDestroy(pMutation);
724 else
725 rtFuzzCtxMutationMaybeEnterCache(pMutation->pFuzzer, pMutation);
726 }
727
728 return cRefs;
729}
730
731
732/**
733 * Adds the given mutation to the corpus of the given fuzzer context.
734 *
735 * @returns IPRT status code.
736 * @param pThis The fuzzer context instance.
737 * @param pMutation The mutation to add.
738 */
739static int rtFuzzCtxMutationAdd(PRTFUZZCTXINT pThis, PRTFUZZMUTATION pMutation)
740{
741 int rc = VINF_SUCCESS;
742
743 pMutation->Core.Key = ASMAtomicIncU64(&pThis->cMutations);
744 rc = RTSemRWRequestWrite(pThis->hSemRwMutations, RT_INDEFINITE_WAIT);
745 AssertRC(rc); RT_NOREF(rc);
746 bool fIns = RTAvlU64Insert(&pThis->TreeMutations, &pMutation->Core);
747 Assert(fIns); RT_NOREF(fIns);
748 rc = RTSemRWReleaseWrite(pThis->hSemRwMutations);
749 AssertRC(rc); RT_NOREF(rc);
750
751 pMutation->fInTree = true;
752 return rc;
753}
754
755
756/**
757 * Locates the mutation with the given key.
758 *
759 * @returns Pointer to the mutation if found or NULL otherwise.
760 * @param pThis The fuzzer context instance.
761 * @param uKey The key to locate.
762 */
763static PRTFUZZMUTATION rtFuzzCtxMutationLocate(PRTFUZZCTXINT pThis, uint64_t uKey)
764{
765 int rc = RTSemRWRequestRead(pThis->hSemRwMutations, RT_INDEFINITE_WAIT);
766 AssertRC(rc); RT_NOREF(rc);
767
768 /*
769 * Using best fit getter here as there might be a racing mutation insertion and the mutation counter has increased
770 * already but the mutation is not yet in the tree.
771 */
772 PRTFUZZMUTATION pMutation = (PRTFUZZMUTATION)RTAvlU64GetBestFit(&pThis->TreeMutations, uKey, false /*fAbove*/);
773 if (RT_LIKELY(pMutation))
774 rtFuzzMutationRetain(pMutation);
775
776 rc = RTSemRWReleaseRead(pThis->hSemRwMutations);
777 AssertRC(rc); RT_NOREF(rc);
778
779 return pMutation;
780}
781
782
783/**
784 * Returns a random mutation from the corpus of the given fuzzer context.
785 *
786 * @returns Pointer to a randomly picked mutation (reference count is increased).
787 * @param pThis The fuzzer context instance.
788 */
789static PRTFUZZMUTATION rtFuzzCtxMutationPickRnd(PRTFUZZCTXINT pThis)
790{
791 uint64_t idxMutation = RTRandAdvU64Ex(pThis->hRand, 1, ASMAtomicReadU64(&pThis->cMutations));
792 return rtFuzzCtxMutationLocate(pThis, idxMutation);
793}
794
795
796/**
797 * Creates a new mutation capable of holding the additional number of bytes - extended version.
798 *
799 * @returns Pointer to the newly created mutation or NULL if out of memory.
800 * @param pThis The fuzzer context instance.
801 * @param offMutation The starting offset for the mutation.
802 * @param pMutationParent The parent mutation, can be NULL.
803 * @param offMuStartNew Offset where descendants of the created mutation can start to mutate.
804 * @param cbMutNew Range in bytes where descendants of the created mutation can mutate.c
805 * @param cbAdditional Additional number of bytes to allocate after the core structure.
806 * @param ppvMutation Where to store the pointer to the mutation dependent data on success.
807 */
808static PRTFUZZMUTATION rtFuzzMutationCreateEx(PRTFUZZCTXINT pThis, uint64_t offMutation, PRTFUZZMUTATION pMutationParent,
809 uint64_t offMutStartNew, uint64_t cbMutNew, size_t cbAdditional, void **ppvMutation)
810{
811 PRTFUZZMUTATION pMutation = (PRTFUZZMUTATION)rtFuzzCtxMemoryAlloc(pThis, sizeof(RTFUZZMUTATION) + cbAdditional);
812 if (RT_LIKELY(pMutation))
813 {
814 pMutation->u32Magic = 0; /** @todo */
815 pMutation->pFuzzer = pThis;
816 pMutation->cRefs = 1;
817 pMutation->iLvl = 0;
818 pMutation->offMutation = offMutation;
819 pMutation->pMutationParent = pMutationParent;
820 pMutation->offMutStartNew = offMutStartNew;
821 pMutation->cbMutNew = cbMutNew;
822 pMutation->cbMutation = cbAdditional;
823 pMutation->fInTree = false;
824 pMutation->fCached = false;
825 pMutation->pvInput = NULL;
826 pMutation->cbInput = 0;
827 pMutation->cbAlloc = 0;
828
829 if (pMutationParent)
830 pMutation->iLvl = pMutationParent->iLvl + 1;
831 if (ppvMutation)
832 *ppvMutation = &pMutation->abMutation[0];
833 }
834
835 return pMutation;
836}
837
838
839/**
840 * Creates a new mutation capable of holding the additional number of bytes.
841 *
842 * @returns Pointer to the newly created mutation or NULL if out of memory.
843 * @param pThis The fuzzer context instance.
844 * @param offMutation The starting offset for the mutation.
845 * @param pMutationParent The parent mutation, can be NULL.
846 * @param cbAdditional Additional number of bytes to allocate after the core structure.
847 * @param ppvMutation Where to store the pointer to the mutation dependent data on success.
848 */
849DECLINLINE(PRTFUZZMUTATION) rtFuzzMutationCreate(PRTFUZZCTXINT pThis, uint64_t offMutation, PRTFUZZMUTATION pMutationParent,
850 size_t cbAdditional, void **ppvMutation)
851{
852 uint64_t offMutNew = pMutationParent ? pMutationParent->offMutStartNew : pThis->offMutStart;
853 uint64_t cbMutNew = pMutationParent ? pMutationParent->cbMutNew : pThis->cbMutRange;
854
855 return rtFuzzMutationCreateEx(pThis, offMutation, pMutationParent, offMutNew, cbMutNew, cbAdditional, ppvMutation);
856}
857
858
859/**
860 * Destroys the given fuzzer context freeing all allocated resources.
861 *
862 * @returns nothing.
863 * @param pThis The fuzzer context instance.
864 */
865static void rtFuzzCtxDestroy(PRTFUZZCTXINT pThis)
866{
867 RT_NOREF(pThis);
868}
869
870
871/**
872 * Creates the final input data applying all accumulated mutations.
873 *
874 * @returns IPRT status code.
875 * @param pMutation The mutation to finalize.
876 */
877static int rtFuzzMutationDataFinalize(PRTFUZZMUTATION pMutation)
878{
879 if (pMutation->pvInput)
880 return VINF_SUCCESS;
881
882 /* Traverse the mutations top to bottom and insert into the array. */
883 int rc = VINF_SUCCESS;
884 uint32_t idx = pMutation->iLvl + 1;
885 PRTFUZZMUTATION *papMutations = (PRTFUZZMUTATION *)RTMemTmpAlloc(idx * sizeof(PCRTFUZZMUTATION));
886 if (RT_LIKELY(papMutations))
887 {
888 PRTFUZZMUTATION pMutationCur = pMutation;
889 size_t cbAlloc = 0;
890
891 /*
892 * As soon as a mutation with allocated input data is encountered the insertion is
893 * stopped as it contains all necessary mutated inputs we can start from.
894 */
895 while (idx > 0)
896 {
897 rtFuzzMutationRetain(pMutationCur);
898 papMutations[idx - 1] = pMutationCur;
899 cbAlloc = RT_MAX(cbAlloc, pMutationCur->cbInput);
900 if (pMutationCur->pvInput)
901 {
902 idx--;
903 break;
904 }
905 pMutationCur = pMutationCur->pMutationParent;
906 idx--;
907 }
908
909 pMutation->cbAlloc = cbAlloc;
910 uint8_t *pbBuf = (uint8_t *)rtFuzzCtxMemoryAlloc(pMutation->pFuzzer, cbAlloc);
911 if (RT_LIKELY(pbBuf))
912 {
913 pMutation->pvInput = pbBuf;
914
915 /* Copy the initial input data. */
916 size_t cbInputNow = papMutations[idx]->cbInput;
917 memcpy(pbBuf, papMutations[idx]->pvInput, cbInputNow);
918 rtFuzzMutationRelease(papMutations[idx]);
919
920 for (uint32_t i = idx + 1; i < pMutation->iLvl + 1; i++)
921 {
922 PRTFUZZMUTATION pCur = papMutations[i];
923 pCur->pMutator->pfnExec(pCur->pFuzzer, pCur, (void *)&pCur->abMutation[0],
924 pbBuf + pCur->offMutation,
925 cbInputNow - pCur->offMutation);
926
927 cbInputNow = pCur->cbInput;
928 rtFuzzMutationRelease(pCur);
929 }
930
931 Assert(cbInputNow == pMutation->cbInput);
932 }
933 else
934 rc = VERR_NO_MEMORY;
935
936 RTMemTmpFree(papMutations);
937 }
938 else
939 rc = VERR_NO_MEMORY;
940
941 return rc;
942}
943
944
945/**
946 * Default mutator export callback (just writing the raw data).
947 */
948static DECLCALLBACK(int) rtFuzzCtxMutatorExportDefault(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
949 PFNRTFUZZCTXEXPORT pfnExport, void *pvUser)
950{
951 return pfnExport(pThis, pvMutation, pMutation->cbMutation, pvUser);
952}
953
954
955/**
956 * Default mutator import callback (just reading the raw data).
957 */
958static DECLCALLBACK(int) rtFuzzCtxMutatorImportDefault(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, void *pvMutation,
959 PFNRTFUZZCTXIMPORT pfnImport, void *pvUser)
960{
961 return pfnImport(pThis, pvMutation, pMutation->cbMutation, NULL, pvUser);
962}
963
964
965static DECLCALLBACK(int) rtFuzzCtxMutatorCorpusExec(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
966 uint8_t *pbBuf, size_t cbBuf)
967{
968 RT_NOREF(pThis, cbBuf, pvMutation);
969 memcpy(pbBuf, pvMutation, pMutation->cbInput);
970 return VINF_SUCCESS;
971}
972
973
974/**
975 * Mutator callback - flips a single bit in the input.
976 */
977static DECLCALLBACK(int) rtFuzzCtxMutatorBitFlipPrep(PRTFUZZCTXINT pThis, uint64_t offStart, PRTFUZZMUTATION pMutationParent,
978 PPRTFUZZMUTATION ppMutation)
979{
980 int rc = VINF_SUCCESS;
981 uint8_t *pidxBitFlip = 0;
982 PRTFUZZMUTATION pMutation = rtFuzzMutationCreate(pThis, offStart, pMutationParent, sizeof(*pidxBitFlip), (void **)&pidxBitFlip);
983 if (RT_LIKELY(pMutation))
984 {
985 pMutation->cbInput = pMutationParent->cbInput; /* Bit flips don't change the input size. */
986 *pidxBitFlip = (uint8_t)RTRandAdvU32Ex(pThis->hRand, 0, sizeof(uint8_t) * 8 - 1);
987 *ppMutation = pMutation;
988 }
989 else
990 rc = VERR_NO_MEMORY;
991
992 return rc;
993}
994
995
996static DECLCALLBACK(int) rtFuzzCtxMutatorBitFlipExec(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
997 uint8_t *pbBuf, size_t cbBuf)
998{
999 RT_NOREF(pThis, cbBuf, pMutation);
1000 uint8_t idxBitFlip = *(uint8_t *)pvMutation;
1001 ASMBitToggle(pbBuf, idxBitFlip);
1002 return VINF_SUCCESS;
1003}
1004
1005
1006/**
1007 * Mutator callback - replaces a single byte in the input.
1008 */
1009static DECLCALLBACK(int) rtFuzzCtxMutatorByteReplacePrep(PRTFUZZCTXINT pThis, uint64_t offStart, PRTFUZZMUTATION pMutationParent,
1010 PPRTFUZZMUTATION ppMutation)
1011{
1012 int rc = VINF_SUCCESS;
1013 uint8_t *pbReplace = 0;
1014 PRTFUZZMUTATION pMutation = rtFuzzMutationCreate(pThis, offStart, pMutationParent, sizeof(*pbReplace), (void **)&pbReplace);
1015 if (RT_LIKELY(pMutation))
1016 {
1017 pMutation->cbInput = pMutationParent->cbInput; /* Byte replacements don't change the input size. */
1018 RTRandAdvBytes(pThis->hRand, pbReplace, 1); /** @todo Filter out same values. */
1019 *ppMutation = pMutation;
1020 }
1021 else
1022 rc = VERR_NO_MEMORY;
1023
1024 return rc;
1025}
1026
1027
1028static DECLCALLBACK(int) rtFuzzCtxMutatorByteReplaceExec(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
1029 uint8_t *pbBuf, size_t cbBuf)
1030{
1031 RT_NOREF(pThis, cbBuf, pMutation);
1032 uint8_t bReplace = *(uint8_t *)pvMutation;
1033 *pbBuf = bReplace;
1034 return VINF_SUCCESS;
1035}
1036
1037
1038/**
1039 * Mutator callback - inserts a single byte into the input.
1040 */
1041static DECLCALLBACK(int) rtFuzzCtxMutatorByteInsertPrep(PRTFUZZCTXINT pThis, uint64_t offStart, PRTFUZZMUTATION pMutationParent,
1042 PPRTFUZZMUTATION ppMutation)
1043{
1044 int rc = VINF_SUCCESS;
1045 uint8_t *pbInsert = 0;
1046 if (pMutationParent->cbInput < pThis->cbInputMax)
1047 {
1048 PRTFUZZMUTATION pMutation = rtFuzzMutationCreate(pThis, offStart, pMutationParent, 1 /*cbAdditional*/, (void **)&pbInsert);
1049 if (RT_LIKELY(pMutation))
1050 {
1051 pMutation->cbInput = pMutationParent->cbInput + 1;
1052 RTRandAdvBytes(pThis->hRand, pbInsert, 1);
1053 *ppMutation = pMutation;
1054 }
1055 else
1056 rc = VERR_NO_MEMORY;
1057 }
1058
1059 return rc;
1060}
1061
1062
1063static DECLCALLBACK(int) rtFuzzCtxMutatorByteInsertExec(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
1064 uint8_t *pbBuf, size_t cbBuf)
1065{
1066 RT_NOREF(pThis, pMutation, pvMutation);
1067
1068 /* Just move the residual data one byte to the back. */
1069 memmove(pbBuf + 1, pbBuf, cbBuf);
1070 *pbBuf = *(uint8_t *)pvMutation;
1071 return VINF_SUCCESS;
1072}
1073
1074
1075/**
1076 * Mutator callback - inserts a byte sequence into the input.
1077 */
1078static DECLCALLBACK(int) rtFuzzCtxMutatorByteSequenceInsertAppendPrep(PRTFUZZCTXINT pThis, uint64_t offStart, PRTFUZZMUTATION pMutationParent,
1079 PPRTFUZZMUTATION ppMutation)
1080{
1081 int rc = VINF_SUCCESS;
1082 if (pMutationParent->cbInput < pThis->cbInputMax)
1083 {
1084 size_t cbInputMutated = (size_t)RTRandAdvU64Ex(pThis->hRand, pMutationParent->cbInput + 1, pThis->cbInputMax);
1085 size_t cbInsert = cbInputMutated - pMutationParent->cbInput;
1086 uint8_t *pbAdd = NULL;
1087
1088 PRTFUZZMUTATION pMutation = rtFuzzMutationCreate(pThis, offStart, pMutationParent, cbInsert, (void **)&pbAdd);
1089 if (RT_LIKELY(pMutation))
1090 {
1091 pMutation->cbInput = cbInputMutated;
1092 RTRandAdvBytes(pThis->hRand, pbAdd, cbInsert);
1093 *ppMutation = pMutation;
1094 }
1095 else
1096 rc = VERR_NO_MEMORY;
1097 }
1098
1099 return rc;
1100}
1101
1102
1103static DECLCALLBACK(int) rtFuzzCtxMutatorByteSequenceInsertAppendExec(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
1104 uint8_t *pbBuf, size_t cbBuf)
1105{
1106 RT_NOREF(pThis);
1107 size_t cbInsert = pMutation->cbInput - pMutation->pMutationParent->cbInput;
1108
1109 /* Move any remaining data to the end. */
1110 if (cbBuf)
1111 memmove(pbBuf + cbInsert, pbBuf, cbBuf);
1112
1113 memcpy(pbBuf, pvMutation, cbInsert);
1114 return VINF_SUCCESS;
1115}
1116
1117
1118/**
1119 * Mutator callback - deletes a single byte in the input.
1120 */
1121static DECLCALLBACK(int) rtFuzzCtxMutatorByteDeletePrep(PRTFUZZCTXINT pThis, uint64_t offStart, PRTFUZZMUTATION pMutationParent,
1122 PPRTFUZZMUTATION ppMutation)
1123{
1124 int rc = VINF_SUCCESS;
1125 if (pMutationParent->cbInput - offStart >= 1)
1126 {
1127 PRTFUZZMUTATION pMutation = rtFuzzMutationCreate(pThis, offStart, pMutationParent, 0 /*cbAdditional*/, NULL);
1128 if (RT_LIKELY(pMutation))
1129 {
1130 pMutation->cbInput = pMutationParent->cbInput - 1;
1131 *ppMutation = pMutation;
1132 }
1133 else
1134 rc = VERR_NO_MEMORY;
1135 }
1136
1137 return rc;
1138}
1139
1140
1141static DECLCALLBACK(int) rtFuzzCtxMutatorByteDeleteExec(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
1142 uint8_t *pbBuf, size_t cbBuf)
1143{
1144 RT_NOREF(pThis, pMutation, pvMutation);
1145
1146 /* Just move the residual data to the front. */
1147 memmove(pbBuf, pbBuf + 1, cbBuf - 1);
1148 return VINF_SUCCESS;
1149}
1150
1151
1152/**
1153 * Mutator callback - deletes a byte sequence in the input.
1154 */
1155static DECLCALLBACK(int) rtFuzzCtxMutatorByteSequenceDeletePrep(PRTFUZZCTXINT pThis, uint64_t offStart, PRTFUZZMUTATION pMutationParent,
1156 PPRTFUZZMUTATION ppMutation)
1157{
1158 int rc = VINF_SUCCESS;
1159 if ( pMutationParent->cbInput > offStart
1160 && pMutationParent->cbInput > 1)
1161 {
1162 size_t cbInputMutated = (size_t)RTRandAdvU64Ex(pThis->hRand, offStart, pMutationParent->cbInput - 1);
1163
1164 PRTFUZZMUTATION pMutation = rtFuzzMutationCreate(pThis, offStart, pMutationParent, 0 /*cbAdditional*/, NULL);
1165 if (RT_LIKELY(pMutation))
1166 {
1167 pMutation->cbInput = cbInputMutated;
1168 *ppMutation = pMutation;
1169 }
1170 else
1171 rc = VERR_NO_MEMORY;
1172 }
1173
1174 return rc;
1175}
1176
1177
1178static DECLCALLBACK(int) rtFuzzCtxMutatorByteSequenceDeleteExec(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
1179 uint8_t *pbBuf, size_t cbBuf)
1180{
1181 RT_NOREF(pThis, pvMutation);
1182 Assert(pMutation->pMutationParent->cbInput > pMutation->cbInput);
1183 size_t cbDel = pMutation->pMutationParent->cbInput - pMutation->cbInput;
1184
1185 /* Just move the residual data to the front. */
1186 memmove(pbBuf, pbBuf + cbDel, cbBuf - cbDel);
1187 return VINF_SUCCESS;
1188}
1189
1190
1191/**
1192 * Mutator callback - replaces a possible integer with something interesting.
1193 */
1194static DECLCALLBACK(int) rtFuzzCtxMutatorIntegerReplacePrep(PRTFUZZCTXINT pThis, uint64_t offStart, PRTFUZZMUTATION pMutationParent,
1195 PPRTFUZZMUTATION ppMutation)
1196{
1197 int rc = VINF_SUCCESS;
1198 PRTFUZZMUTATORINTEGER pMutInt = NULL;
1199 PRTFUZZMUTATION pMutation = rtFuzzMutationCreate(pThis, offStart, pMutationParent, sizeof(*pMutInt), (void **)&pMutInt);
1200 if (RT_LIKELY(pMutation))
1201 {
1202 size_t cbLeft = pMutationParent->cbInput - offStart;
1203 uint32_t uClassMax = 0;
1204
1205 switch (cbLeft)
1206 {
1207 case 1:
1208 uClassMax = 1;
1209 break;
1210 case 2:
1211 case 3:
1212 uClassMax = 3;
1213 break;
1214 case 4:
1215 case 5:
1216 case 6:
1217 case 7:
1218 uClassMax = 5;
1219 break;
1220 default:
1221 uClassMax = 7;
1222 break;
1223 }
1224
1225 pMutInt->uIntClass = (uint8_t)RTRandAdvU32Ex(pThis->hRand, 0, uClassMax);
1226 pMutInt->fByteSwap = RT_BOOL(RTRandAdvU32Ex(pThis->hRand, 0, 1));
1227
1228 switch (pMutInt->uIntClass)
1229 {
1230 case 0:
1231 pMutInt->idxInt = (uint16_t)RTRandAdvU32Ex(pThis->hRand, 0, RT_ELEMENTS(s_ai8Interesting) - 1);
1232 break;
1233 case 1:
1234 pMutInt->idxInt = (uint16_t)RTRandAdvU32Ex(pThis->hRand, 0, RT_ELEMENTS(s_au8Interesting) - 1);
1235 break;
1236 case 2:
1237 pMutInt->idxInt = (uint16_t)RTRandAdvU32Ex(pThis->hRand, 0, RT_ELEMENTS(s_ai16Interesting) - 1);
1238 break;
1239 case 3:
1240 pMutInt->idxInt = (uint16_t)RTRandAdvU32Ex(pThis->hRand, 0, RT_ELEMENTS(s_au16Interesting) - 1);
1241 break;
1242 case 4:
1243 pMutInt->idxInt = (uint16_t)RTRandAdvU32Ex(pThis->hRand, 0, RT_ELEMENTS(s_ai32Interesting) - 1);
1244 break;
1245 case 5:
1246 pMutInt->idxInt = (uint16_t)RTRandAdvU32Ex(pThis->hRand, 0, RT_ELEMENTS(s_au32Interesting) - 1);
1247 break;
1248 case 6:
1249 pMutInt->idxInt = (uint16_t)RTRandAdvU32Ex(pThis->hRand, 0, RT_ELEMENTS(s_ai64Interesting) - 1);
1250 break;
1251 case 7:
1252 pMutInt->idxInt = (uint16_t)RTRandAdvU32Ex(pThis->hRand, 0, RT_ELEMENTS(s_au64Interesting) - 1);
1253 break;
1254 default:
1255 AssertReleaseFailed();
1256 }
1257
1258 pMutation->cbInput = pMutationParent->cbInput;
1259 *ppMutation = pMutation;
1260 }
1261 else
1262 rc = VERR_NO_MEMORY;
1263
1264 return rc;
1265}
1266
1267
1268static DECLCALLBACK(int) rtFuzzCtxMutatorIntegerReplaceExec(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
1269 uint8_t *pbBuf, size_t cbBuf)
1270{
1271 RT_NOREF(pThis, pMutation, cbBuf);
1272 union
1273 {
1274 int8_t i8;
1275 uint8_t u8;
1276 int16_t i16;
1277 uint16_t u16;
1278 int32_t i32;
1279 uint32_t u32;
1280 int64_t i64;
1281 uint64_t u64;
1282 } Int;
1283 PCRTFUZZMUTATORINTEGER pMutInt = (PCRTFUZZMUTATORINTEGER)pvMutation;
1284 size_t cb = 0;
1285
1286 switch (pMutInt->uIntClass)
1287 {
1288 case 0:
1289 Int.i8 = s_ai8Interesting[pMutInt->idxInt];
1290 cb = 1;
1291 break;
1292 case 1:
1293 Int.u8 = s_au8Interesting[pMutInt->idxInt];
1294 cb = 1;
1295 break;
1296 case 2:
1297 Int.i16 = s_ai16Interesting[pMutInt->idxInt];
1298 cb = 2;
1299 if (pMutInt->fByteSwap)
1300 Int.u16 = RT_BSWAP_U16(Int.u16);
1301 break;
1302 case 3:
1303 Int.u16 = s_au16Interesting[pMutInt->idxInt];
1304 cb = 2;
1305 if (pMutInt->fByteSwap)
1306 Int.u16 = RT_BSWAP_U16(Int.u16);
1307 break;
1308 case 4:
1309 Int.i32 = s_ai32Interesting[pMutInt->idxInt];
1310 cb = 4;
1311 if (pMutInt->fByteSwap)
1312 Int.u32 = RT_BSWAP_U32(Int.u32);
1313 break;
1314 case 5:
1315 Int.u32 = s_au32Interesting[pMutInt->idxInt];
1316 cb = 4;
1317 if (pMutInt->fByteSwap)
1318 Int.u32 = RT_BSWAP_U32(Int.u32);
1319 break;
1320 case 6:
1321 Int.i64 = s_ai64Interesting[pMutInt->idxInt];
1322 cb = 8;
1323 if (pMutInt->fByteSwap)
1324 Int.u64 = RT_BSWAP_U64(Int.u64);
1325 break;
1326 case 7:
1327 Int.u64 = s_au64Interesting[pMutInt->idxInt];
1328 cb = 8;
1329 if (pMutInt->fByteSwap)
1330 Int.u64 = RT_BSWAP_U64(Int.u64);
1331 break;
1332 default:
1333 AssertReleaseFailed();
1334 }
1335
1336 memcpy(pbBuf, &Int, cb);
1337 return VINF_SUCCESS;
1338}
1339
1340
1341/**
1342 * Mutator callback - crosses over two mutations at the given point.
1343 */
1344static DECLCALLBACK(int) rtFuzzCtxMutatorCrossoverPrep(PRTFUZZCTXINT pThis, uint64_t offStart, PRTFUZZMUTATION pMutationParent,
1345 PPRTFUZZMUTATION ppMutation)
1346{
1347 int rc = VINF_SUCCESS;
1348
1349 if (pThis->cMutations > 1)
1350 {
1351 uint64_t *pidxMutCrossover = NULL;
1352 PRTFUZZMUTATION pMutation = rtFuzzMutationCreate(pThis, offStart, pMutationParent, sizeof(*pidxMutCrossover), (void **)&pidxMutCrossover);
1353 if (RT_LIKELY(pMutation))
1354 {
1355 uint32_t cTries = 10;
1356 PRTFUZZMUTATION pMutCrossover = NULL;
1357 /*
1358 * Pick a random mutation to crossover with (making sure it is not the current one
1359 * or the crossover point is beyond the end of input).
1360 */
1361 do
1362 {
1363 if (pMutCrossover)
1364 rtFuzzMutationRelease(pMutCrossover);
1365 pMutCrossover = rtFuzzCtxMutationPickRnd(pThis);
1366 cTries--;
1367 } while ( ( pMutCrossover == pMutationParent
1368 || offStart >= pMutCrossover->cbInput)
1369 && cTries > 0);
1370
1371 if (cTries)
1372 {
1373 pMutation->cbInput = pMutCrossover->cbInput;
1374 *pidxMutCrossover = pMutCrossover->Core.Key;
1375 *ppMutation = pMutation;
1376 }
1377 else
1378 rtFuzzMutationDestroy(pMutation);
1379
1380 rtFuzzMutationRelease(pMutCrossover);
1381 }
1382 else
1383 rc = VERR_NO_MEMORY;
1384 }
1385
1386 return rc;
1387}
1388
1389
1390static DECLCALLBACK(int) rtFuzzCtxMutatorCrossoverExec(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
1391 uint8_t *pbBuf, size_t cbBuf)
1392{
1393 RT_NOREF(cbBuf);
1394 uint64_t idxMutCrossover = *(uint64_t *)pvMutation;
1395
1396 PRTFUZZMUTATION pMutCrossover = rtFuzzCtxMutationLocate(pThis, idxMutCrossover);
1397 int rc = rtFuzzMutationDataFinalize(pMutCrossover);
1398 if (RT_SUCCESS(rc))
1399 {
1400 memcpy(pbBuf, (uint8_t *)pMutCrossover->pvInput + pMutation->offMutation,
1401 pMutCrossover->cbInput - pMutation->offMutation);
1402 rtFuzzMutationRelease(pMutCrossover);
1403 }
1404
1405 return rc;
1406}
1407
1408
1409static DECLCALLBACK(int) rtFuzzCtxMutatorCrossoverExport(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, const void *pvMutation,
1410 PFNRTFUZZCTXEXPORT pfnExport, void *pvUser)
1411{
1412 RT_NOREF(pMutation);
1413
1414 uint64_t idxMutCrossover = *(uint64_t *)pvMutation;
1415 idxMutCrossover = RT_H2LE_U64(idxMutCrossover);
1416 return pfnExport(pThis, &idxMutCrossover, sizeof(idxMutCrossover), pvUser);
1417}
1418
1419
1420static DECLCALLBACK(int) rtFuzzCtxMutatorCrossoverImport(PRTFUZZCTXINT pThis, PCRTFUZZMUTATION pMutation, void *pvMutation,
1421 PFNRTFUZZCTXIMPORT pfnImport, void *pvUser)
1422{
1423 RT_NOREF(pMutation);
1424
1425 uint64_t uKey = 0;
1426 int rc = pfnImport(pThis, &uKey, sizeof(uKey), NULL, pvUser);
1427 if (RT_SUCCESS(rc))
1428 {
1429 uKey = RT_LE2H_U64(uKey);
1430 *(uint64_t *)pvMutation = uKey;
1431 }
1432
1433 return rc;
1434}
1435
1436
1437/**
1438 * Creates an empty fuzzing context.
1439 *
1440 * @returns IPRT status code.
1441 * @param ppThis Where to store the pointer to the internal fuzzing context instance on success.
1442 * @param enmType Fuzzing context type.
1443 */
1444static int rtFuzzCtxCreateEmpty(PRTFUZZCTXINT *ppThis, RTFUZZCTXTYPE enmType)
1445{
1446 int rc;
1447 PRTFUZZCTXINT pThis = (PRTFUZZCTXINT)RTMemAllocZ(sizeof(*pThis));
1448 if (RT_LIKELY(pThis))
1449 {
1450 pThis->u32Magic = RTFUZZCTX_MAGIC;
1451 pThis->cRefs = 1;
1452 pThis->enmType = enmType;
1453 pThis->TreeMutations = NULL;
1454 pThis->cbInputMax = UINT32_MAX;
1455 pThis->cMutations = 0;
1456 pThis->fFlagsBehavioral = 0;
1457 pThis->cbMutationsAllocMax = _1G;
1458 pThis->cbMemTotal = 0;
1459 pThis->offMutStart = 0;
1460 pThis->cbMutRange = UINT64_MAX;
1461 RTListInit(&pThis->LstMutationsAlloc);
1462
1463 /* Copy the default mutator descriptors over. */
1464 pThis->paMutators = (PRTFUZZMUTATOR)RTMemAllocZ(RT_ELEMENTS(g_aMutators) * sizeof(RTFUZZMUTATOR));
1465 if (RT_LIKELY(pThis->paMutators))
1466 {
1467 pThis->cMutators = RT_ELEMENTS(g_aMutators);
1468 memcpy(&pThis->paMutators[0], &g_aMutators[0], RT_ELEMENTS(g_aMutators) * sizeof(RTFUZZMUTATOR));
1469
1470 rc = RTSemRWCreate(&pThis->hSemRwMutations);
1471 if (RT_SUCCESS(rc))
1472 {
1473 rc = RTCritSectInit(&pThis->CritSectAlloc);
1474 if (RT_SUCCESS(rc))
1475 {
1476 rc = RTRandAdvCreateParkMiller(&pThis->hRand);
1477 if (RT_SUCCESS(rc))
1478 {
1479 RTRandAdvSeed(pThis->hRand, RTTimeSystemNanoTS());
1480 *ppThis = pThis;
1481 return VINF_SUCCESS;
1482 }
1483
1484 RTCritSectDelete(&pThis->CritSectAlloc);
1485 }
1486
1487 RTSemRWDestroy(pThis->hSemRwMutations);
1488 }
1489 }
1490 else
1491 rc = VERR_NO_MEMORY;
1492
1493 RTMemFree(pThis);
1494 }
1495 else
1496 rc = VERR_NO_MEMORY;
1497
1498 return rc;
1499}
1500
1501
1502/**
1503 * Destroys the given fuzzing input.
1504 *
1505 * @returns nothing.
1506 * @param pThis The fuzzing input to destroy.
1507 */
1508static void rtFuzzInputDestroy(PRTFUZZINPUTINT pThis)
1509{
1510 PRTFUZZCTXINT pFuzzer = pThis->pFuzzer;
1511
1512 rtFuzzMutationRelease(pThis->pMutationTop);
1513 rtFuzzCtxMemoryFree(pFuzzer, pThis);
1514 RTFuzzCtxRelease(pFuzzer);
1515}
1516
1517
1518RTDECL(int) RTFuzzCtxCreate(PRTFUZZCTX phFuzzCtx, RTFUZZCTXTYPE enmType)
1519{
1520 AssertPtrReturn(phFuzzCtx, VERR_INVALID_POINTER);
1521
1522 return rtFuzzCtxCreateEmpty(phFuzzCtx, enmType);
1523}
1524
1525
1526RTDECL(int) RTFuzzCtxCreateFromState(PRTFUZZCTX phFuzzCtx, PFNRTFUZZCTXIMPORT pfnImport, void *pvUser)
1527{
1528 AssertPtrReturn(phFuzzCtx, VERR_INVALID_POINTER);
1529 AssertPtrReturn(pfnImport, VERR_INVALID_POINTER);
1530
1531#if 0
1532 int rc = VINF_SUCCESS;
1533 if (cbState >= sizeof(RTFUZZCTXSTATE))
1534 {
1535 RTFUZZCTXSTATE StateImport;
1536
1537 memcpy(&StateImport, pvState, sizeof(RTFUZZCTXSTATE));
1538 if ( RT_LE2H_U32(StateImport.u32Magic) == RTFUZZCTX_MAGIC
1539 && RT_LE2H_U32(StateImport.cbPrng) <= cbState - sizeof(RTFUZZCTXSTATE))
1540 {
1541 PRTFUZZCTXINT pThis = rtFuzzCtxCreateEmpty();
1542 if (RT_LIKELY(pThis))
1543 {
1544 pThis->cbInputMax = (size_t)RT_LE2H_U64(StateImport.cbInputMax);
1545 pThis->fFlagsBehavioral = RT_LE2H_U32(StateImport.fFlagsBehavioral);
1546
1547 uint8_t *pbState = (uint8_t *)pvState;
1548 uint32_t cInputs = RT_LE2H_U32(StateImport.cInputs);
1549 rc = RTRandAdvRestoreState(pThis->hRand, (const char *)&pbState[sizeof(RTFUZZCTXSTATE)]);
1550 if (RT_SUCCESS(rc))
1551 {
1552 /* Go through the inputs and add them. */
1553 pbState += sizeof(RTFUZZCTXSTATE) + RT_LE2H_U32(StateImport.cbPrng);
1554 cbState -= sizeof(RTFUZZCTXSTATE) + RT_LE2H_U32(StateImport.cbPrng);
1555
1556 uint32_t idx = 0;
1557 while ( idx < cInputs
1558 && RT_SUCCESS(rc))
1559 {
1560 size_t cbInput = 0;
1561 if (cbState >= sizeof(uint32_t))
1562 {
1563 memcpy(&cbInput, pbState, sizeof(uint32_t));
1564 cbInput = RT_LE2H_U32(cbInput);
1565 pbState += sizeof(uint32_t);
1566 }
1567
1568 if ( cbInput
1569 && cbInput <= cbState)
1570 {
1571 PRTFUZZINPUTINT pInput = rtFuzzCtxInputCreate(pThis, cbInput);
1572 if (RT_LIKELY(pInput))
1573 {
1574 memcpy(&pInput->abInput[0], pbState, cbInput);
1575 RTMd5(&pInput->abInput[0], pInput->cbInput, &pInput->abMd5Hash[0]);
1576 rc = rtFuzzCtxInputAdd(pThis, pInput);
1577 if (RT_FAILURE(rc))
1578 RTMemFree(pInput);
1579 pbState += cbInput;
1580 }
1581 }
1582 else
1583 rc = VERR_INVALID_STATE;
1584
1585 idx++;
1586 }
1587
1588 if (RT_SUCCESS(rc))
1589 {
1590 *phFuzzCtx = pThis;
1591 return VINF_SUCCESS;
1592 }
1593 }
1594
1595 rtFuzzCtxDestroy(pThis);
1596 }
1597 else
1598 rc = VERR_NO_MEMORY;
1599 }
1600 else
1601 rc = VERR_INVALID_MAGIC;
1602 }
1603 else
1604 rc = VERR_INVALID_MAGIC;
1605
1606 return rc;
1607#else
1608 RT_NOREF(pvUser);
1609 return VERR_NOT_IMPLEMENTED;
1610#endif
1611}
1612
1613
1614RTDECL(int) RTFuzzCtxCreateFromStateMem(PRTFUZZCTX phFuzzCtx, const void *pvState, size_t cbState)
1615{
1616 AssertPtrReturn(phFuzzCtx, VERR_INVALID_POINTER);
1617 AssertPtrReturn(pvState, VERR_INVALID_POINTER);
1618 AssertPtrReturn(cbState, VERR_INVALID_POINTER);
1619
1620 return VERR_NOT_IMPLEMENTED;
1621}
1622
1623
1624RTDECL(int) RTFuzzCtxCreateFromStateFile(PRTFUZZCTX phFuzzCtx, const char *pszFilename)
1625{
1626 AssertPtrReturn(phFuzzCtx, VERR_INVALID_POINTER);
1627 AssertPtrReturn(pszFilename, VERR_INVALID_POINTER);
1628
1629 void *pv = NULL;
1630 size_t cb = 0;
1631 int rc = RTFileReadAll(pszFilename, &pv, &cb);
1632 if (RT_SUCCESS(rc))
1633 {
1634 rc = RTFuzzCtxCreateFromStateMem(phFuzzCtx, pv, cb);
1635 RTFileReadAllFree(pv, cb);
1636 }
1637
1638 return rc;
1639}
1640
1641
1642RTDECL(uint32_t) RTFuzzCtxRetain(RTFUZZCTX hFuzzCtx)
1643{
1644 PRTFUZZCTXINT pThis = hFuzzCtx;
1645
1646 AssertPtrReturn(pThis, UINT32_MAX);
1647
1648 uint32_t cRefs = ASMAtomicIncU32(&pThis->cRefs);
1649 AssertMsg(cRefs > 1 && cRefs < _1M, ("%#x %p\n", cRefs, pThis));
1650 return cRefs;
1651}
1652
1653
1654RTDECL(uint32_t) RTFuzzCtxRelease(RTFUZZCTX hFuzzCtx)
1655{
1656 PRTFUZZCTXINT pThis = hFuzzCtx;
1657 if (pThis == NIL_RTFUZZCTX)
1658 return 0;
1659 AssertPtrReturn(pThis, UINT32_MAX);
1660
1661 uint32_t cRefs = ASMAtomicDecU32(&pThis->cRefs);
1662 AssertMsg(cRefs < _1M, ("%#x %p\n", cRefs, pThis));
1663 if (cRefs == 0)
1664 rtFuzzCtxDestroy(pThis);
1665 return cRefs;
1666}
1667
1668
1669RTDECL(int) RTFuzzCtxQueryStats(RTFUZZCTX hFuzzCtx, PRTFUZZCTXSTATS pStats)
1670{
1671 PRTFUZZCTXINT pThis = hFuzzCtx;
1672 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1673 AssertPtrReturn(pStats, VERR_INVALID_POINTER);
1674
1675 pStats->cbMemory = ASMAtomicReadZ(&pThis->cbMemTotal);
1676 pStats->cMutations = ASMAtomicReadU64(&pThis->cMutations);
1677 return VINF_SUCCESS;
1678}
1679
1680
1681/**
1682 * Fuzzing context export callback for a single mutation.
1683 */
1684static DECLCALLBACK(int) rtFuzzCtxStateExportMutations(PAVLU64NODECORE pCore, void *pvParam)
1685{
1686 PRTFUZZMUTATION pMutation = (PRTFUZZMUTATION)pCore;
1687 PCRTFUZZMUTATOR pMutator = pMutation->pMutator;
1688 PCRTFUZZEXPORTARGS pArgs = (PCRTFUZZEXPORTARGS)pvParam;
1689 RTFUZZMUTATIONSTATE MutationState;
1690
1691 MutationState.u64Id = RT_H2LE_U64(pMutation->Core.Key);
1692 if (pMutation->pMutationParent)
1693 MutationState.u64IdParent = RT_H2LE_U64(pMutation->pMutationParent->Core.Key);
1694 else
1695 MutationState.u64IdParent = RT_H2LE_U64(0);
1696 MutationState.u64OffMutation = RT_H2LE_U64(pMutation->offMutation);
1697 MutationState.cbInput = RT_H2LE_U64((uint64_t)pMutation->cbInput);
1698 MutationState.cbMutation = RT_H2LE_U64((uint64_t)pMutation->cbMutation);
1699 MutationState.u32IdMutator = RT_H2LE_U32(pMutator->uMutator);
1700 MutationState.iLvl = RT_H2LE_U32(pMutation->iLvl);
1701 MutationState.u32Magic = RT_H2LE_U32(pMutation->u32Magic);
1702
1703 int rc = pArgs->pfnExport(pMutation->pFuzzer, &MutationState, sizeof(MutationState), pArgs->pvUser);
1704 if ( RT_SUCCESS(rc)
1705 && pMutator->pfnExport)
1706 rc = pMutator->pfnExport(pMutation->pFuzzer, pMutation, &pMutation->abMutation[0], pArgs->pfnExport, pArgs->pvUser);
1707 return rc;
1708}
1709
1710
1711RTDECL(int) RTFuzzCtxStateExport(RTFUZZCTX hFuzzCtx, PFNRTFUZZCTXEXPORT pfnExport, void *pvUser)
1712{
1713 PRTFUZZCTXINT pThis = hFuzzCtx;
1714 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1715 AssertPtrReturn(pfnExport, VERR_INVALID_POINTER);
1716
1717 char aszPrngExport[_4K]; /* Should be plenty of room here. */
1718 size_t cbPrng = sizeof(aszPrngExport);
1719 int rc = RTRandAdvSaveState(pThis->hRand, &aszPrngExport[0], &cbPrng);
1720 if (RT_SUCCESS(rc))
1721 {
1722 RTFUZZCTXSTATE StateExport;
1723
1724 StateExport.u32Magic = RT_H2LE_U32(RTFUZZCTX_MAGIC);
1725 switch (pThis->enmType)
1726 {
1727 case RTFUZZCTXTYPE_BLOB:
1728 StateExport.uCtxType = RT_H2LE_U32(RTFUZZCTX_STATE_TYPE_BLOB);
1729 break;
1730 case RTFUZZCTXTYPE_STREAM:
1731 StateExport.uCtxType = RT_H2LE_U32(RTFUZZCTX_STATE_TYPE_STREAM);
1732 break;
1733 default:
1734 AssertFailed();
1735 break;
1736 }
1737 StateExport.cbPrng = RT_H2LE_U32((uint32_t)cbPrng);
1738 StateExport.cMutations = RT_H2LE_U32(pThis->cMutations);
1739 StateExport.cMutators = RT_H2LE_U32(pThis->cMutators);
1740 StateExport.fFlagsBehavioral = RT_H2LE_U32(pThis->fFlagsBehavioral);
1741 StateExport.cbInputMax = RT_H2LE_U64(pThis->cbInputMax);
1742
1743 /* Write the context state and PRNG state first. */
1744 rc = pfnExport(pThis, &StateExport, sizeof(StateExport), pvUser);
1745 if (RT_SUCCESS(rc))
1746 rc = pfnExport(pThis, &aszPrngExport[0], cbPrng, pvUser);
1747 if (RT_SUCCESS(rc))
1748 {
1749 /* Write the mutator descriptors next. */
1750 for (uint32_t i = 0; i < pThis->cMutators && RT_SUCCESS(rc); i++)
1751 {
1752 PRTFUZZMUTATOR pMutator = &pThis->paMutators[i];
1753 uint32_t cchId = (uint32_t)strlen(pMutator->pszId) + 1;
1754 uint32_t cchIdW = RT_H2LE_U32(cchId);
1755
1756 rc = pfnExport(pThis, &cchIdW, sizeof(cchIdW), pvUser);
1757 if (RT_SUCCESS(rc))
1758 rc = pfnExport(pThis, &pMutator->pszId[0], cchId, pvUser);
1759 }
1760 }
1761
1762 /* Write the mutations last. */
1763 if (RT_SUCCESS(rc))
1764 {
1765 RTFUZZEXPORTARGS Args;
1766
1767 Args.pfnExport = pfnExport;
1768 Args.pvUser = pvUser;
1769 rc = RTAvlU64DoWithAll(&pThis->TreeMutations, true /*fFromLeft*/, rtFuzzCtxStateExportMutations, &Args);
1770 }
1771 }
1772
1773 return rc;
1774}
1775
1776
1777RTDECL(int) RTFuzzCtxStateExportToMem(RTFUZZCTX hFuzzCtx, void **ppvState, size_t *pcbState)
1778{
1779 PRTFUZZCTXINT pThis = hFuzzCtx;
1780 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1781 AssertPtrReturn(ppvState, VERR_INVALID_POINTER);
1782 AssertPtrReturn(pcbState, VERR_INVALID_POINTER);
1783
1784 return VERR_NOT_IMPLEMENTED;
1785}
1786
1787
1788/**
1789 * Export to file callback.
1790 */
1791static DECLCALLBACK(int) rtFuzzCtxStateExportFile(RTFUZZCTX hFuzzCtx, const void *pvBuf, size_t cbWrite, void *pvUser)
1792{
1793 RT_NOREF(hFuzzCtx);
1794
1795 RTFILE hFile = (RTFILE)pvUser;
1796 return RTFileWrite(hFile, pvBuf, cbWrite, NULL);
1797}
1798
1799
1800RTDECL(int) RTFuzzCtxStateExportToFile(RTFUZZCTX hFuzzCtx, const char *pszFilename)
1801{
1802 PRTFUZZCTXINT pThis = hFuzzCtx;
1803 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1804 AssertPtrReturn(pszFilename, VERR_INVALID_POINTER);
1805
1806 RTFILE hFile;
1807 int rc = RTFileOpen(&hFile, pszFilename, RTFILE_O_CREATE | RTFILE_O_WRITE | RTFILE_O_DENY_NONE);
1808 if (RT_SUCCESS(rc))
1809 {
1810 rc = RTFuzzCtxStateExport(hFuzzCtx, rtFuzzCtxStateExportFile, hFile);
1811 RTFileClose(hFile);
1812 if (RT_FAILURE(rc))
1813 RTFileDelete(pszFilename);
1814 }
1815
1816 return rc;
1817}
1818
1819
1820RTDECL(int) RTFuzzCtxCorpusInputAdd(RTFUZZCTX hFuzzCtx, const void *pvInput, size_t cbInput)
1821{
1822 PRTFUZZCTXINT pThis = hFuzzCtx;
1823 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1824 AssertPtrReturn(pvInput, VERR_INVALID_POINTER);
1825 AssertReturn(cbInput, VERR_INVALID_POINTER);
1826
1827 return RTFuzzCtxCorpusInputAddEx(hFuzzCtx, pvInput, cbInput, pThis->offMutStart, pThis->cbMutRange);
1828}
1829
1830
1831RTDECL(int) RTFuzzCtxCorpusInputAddEx(RTFUZZCTX hFuzzCtx, const void *pvInput, size_t cbInput,
1832 uint64_t offMutStart, uint64_t cbMutRange)
1833{
1834 PRTFUZZCTXINT pThis = hFuzzCtx;
1835 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1836 AssertPtrReturn(pvInput, VERR_INVALID_POINTER);
1837 AssertReturn(cbInput, VERR_INVALID_POINTER);
1838
1839 int rc = VINF_SUCCESS;
1840 void *pvCorpus = NULL;
1841 PRTFUZZMUTATION pMutation = rtFuzzMutationCreateEx(pThis, 0, NULL, offMutStart, cbMutRange,
1842 cbInput, &pvCorpus);
1843 if (RT_LIKELY(pMutation))
1844 {
1845 pMutation->pMutator = &g_MutatorCorpus;
1846 pMutation->cbInput = cbInput;
1847 pMutation->pvInput = pvCorpus;
1848 memcpy(pvCorpus, pvInput, cbInput);
1849 rc = rtFuzzCtxMutationAdd(pThis, pMutation);
1850 if (RT_FAILURE(rc))
1851 rtFuzzMutationDestroy(pMutation);
1852 }
1853 else
1854 rc = VERR_NO_MEMORY;
1855
1856 return rc;
1857}
1858
1859
1860RTDECL(int) RTFuzzCtxCorpusInputAddFromFile(RTFUZZCTX hFuzzCtx, const char *pszFilename)
1861{
1862 PRTFUZZCTXINT pThis = hFuzzCtx;
1863 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1864 AssertPtrReturn(pszFilename, VERR_INVALID_POINTER);
1865
1866 return RTFuzzCtxCorpusInputAddFromFileEx(hFuzzCtx, pszFilename, pThis->offMutStart, pThis->cbMutRange);
1867}
1868
1869
1870RTDECL(int) RTFuzzCtxCorpusInputAddFromFileEx(RTFUZZCTX hFuzzCtx, const char *pszFilename,
1871 uint64_t offMutStart, uint64_t cbMutRange)
1872{
1873 PRTFUZZCTXINT pThis = hFuzzCtx;
1874 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1875 AssertPtrReturn(pszFilename, VERR_INVALID_POINTER);
1876
1877 void *pv = NULL;
1878 size_t cb = 0;
1879 int rc = RTFileReadAll(pszFilename, &pv, &cb);
1880 if (RT_SUCCESS(rc))
1881 {
1882 rc = RTFuzzCtxCorpusInputAddEx(hFuzzCtx, pv, cb, offMutStart, cbMutRange);
1883 RTFileReadAllFree(pv, cb);
1884 }
1885
1886 return rc;
1887}
1888
1889
1890RTDECL(int) RTFuzzCtxCorpusInputAddFromVfsFile(RTFUZZCTX hFuzzCtx, RTVFSFILE hVfsFile)
1891{
1892 PRTFUZZCTXINT pThis = hFuzzCtx;
1893 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1894 AssertReturn(hVfsFile != NIL_RTVFSFILE, VERR_INVALID_HANDLE);
1895
1896 return RTFuzzCtxCorpusInputAddFromVfsFileEx(hFuzzCtx, hVfsFile, pThis->offMutStart, pThis->cbMutRange);
1897}
1898
1899
1900RTDECL(int) RTFuzzCtxCorpusInputAddFromVfsFileEx(RTFUZZCTX hFuzzCtx, RTVFSFILE hVfsFile,
1901 uint64_t offMutStart, uint64_t cbMutRange)
1902{
1903 PRTFUZZCTXINT pThis = hFuzzCtx;
1904 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1905 AssertReturn(hVfsFile != NIL_RTVFSFILE, VERR_INVALID_HANDLE);
1906
1907 uint64_t cbFile = 0;
1908 void *pvCorpus = NULL;
1909 int rc = RTVfsFileQuerySize(hVfsFile, &cbFile);
1910 if (RT_SUCCESS(rc))
1911 {
1912 PRTFUZZMUTATION pMutation = rtFuzzMutationCreateEx(pThis, 0, NULL, offMutStart, cbMutRange,
1913 cbFile, &pvCorpus);
1914 if (RT_LIKELY(pMutation))
1915 {
1916 pMutation->pMutator = &g_MutatorCorpus;
1917 pMutation->cbInput = cbFile;
1918 pMutation->pvInput = pvCorpus;
1919 rc = RTVfsFileRead(hVfsFile, pvCorpus, cbFile, NULL);
1920 if (RT_SUCCESS(rc))
1921 rc = rtFuzzCtxMutationAdd(pThis, pMutation);
1922
1923 if (RT_FAILURE(rc))
1924 rtFuzzMutationDestroy(pMutation);
1925 }
1926 else
1927 rc = VERR_NO_MEMORY;
1928 }
1929
1930 return rc;
1931}
1932
1933
1934RTDECL(int) RTFuzzCtxCorpusInputAddFromVfsIoStrm(RTFUZZCTX hFuzzCtx, RTVFSIOSTREAM hVfsIos)
1935{
1936 PRTFUZZCTXINT pThis = hFuzzCtx;
1937 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1938 AssertReturn(hVfsIos != NIL_RTVFSIOSTREAM, VERR_INVALID_HANDLE);
1939
1940 return RTFuzzCtxCorpusInputAddFromVfsIoStrmEx(hFuzzCtx, hVfsIos, pThis->offMutStart, pThis->cbMutRange);
1941}
1942
1943RTDECL(int) RTFuzzCtxCorpusInputAddFromVfsIoStrmEx(RTFUZZCTX hFuzzCtx, RTVFSIOSTREAM hVfsIos,
1944 uint64_t offMutStart, uint64_t cbMutRange)
1945{
1946 PRTFUZZCTXINT pThis = hFuzzCtx;
1947 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
1948 AssertReturn(hVfsIos != NIL_RTVFSIOSTREAM, VERR_INVALID_HANDLE);
1949
1950 void *pvCorpus = NULL;
1951 RTFSOBJINFO ObjInfo;
1952 int rc = RTVfsIoStrmQueryInfo(hVfsIos, &ObjInfo, RTFSOBJATTRADD_UNIX);
1953 if (RT_SUCCESS(rc))
1954 {
1955 PRTFUZZMUTATION pMutation = rtFuzzMutationCreateEx(pThis, 0, NULL, offMutStart, cbMutRange,
1956 ObjInfo.cbObject, &pvCorpus);
1957 if (RT_LIKELY(pMutation))
1958 {
1959 pMutation->pMutator = &g_MutatorCorpus;
1960 pMutation->cbInput = ObjInfo.cbObject;
1961 pMutation->pvInput = pvCorpus;
1962 rc = RTVfsIoStrmRead(hVfsIos, pvCorpus, ObjInfo.cbObject, true /*fBlocking*/, NULL);
1963 if (RT_SUCCESS(rc))
1964 rc = rtFuzzCtxMutationAdd(pThis, pMutation);
1965
1966 if (RT_FAILURE(rc))
1967 rtFuzzMutationDestroy(pMutation);
1968 }
1969 else
1970 rc = VERR_NO_MEMORY;
1971 }
1972
1973 return rc;
1974}
1975
1976
1977RTDECL(int) RTFuzzCtxCorpusInputAddFromDirPath(RTFUZZCTX hFuzzCtx, const char *pszDirPath)
1978{
1979 PRTFUZZCTXINT pThis = hFuzzCtx;
1980 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
1981 AssertPtrReturn(pszDirPath, VERR_INVALID_POINTER);
1982
1983 RTDIR hDir;
1984 int rc = RTDirOpen(&hDir, pszDirPath);
1985 if (RT_SUCCESS(rc))
1986 {
1987 for (;;)
1988 {
1989 RTDIRENTRY DirEntry;
1990 rc = RTDirRead(hDir, &DirEntry, NULL);
1991 if (RT_FAILURE(rc))
1992 break;
1993
1994 /* Skip '.', '..' and other non-files. */
1995 if ( DirEntry.enmType != RTDIRENTRYTYPE_UNKNOWN
1996 && DirEntry.enmType != RTDIRENTRYTYPE_FILE)
1997 continue;
1998 if (RTDirEntryIsStdDotLink(&DirEntry))
1999 continue;
2000
2001 /* Compose the full path, result 'unknown' entries and skip non-files. */
2002 char szFile[RTPATH_MAX];
2003 RT_ZERO(szFile);
2004 rc = RTPathJoin(szFile, sizeof(szFile), pszDirPath, DirEntry.szName);
2005 if (RT_FAILURE(rc))
2006 break;
2007
2008 if (DirEntry.enmType == RTDIRENTRYTYPE_UNKNOWN)
2009 {
2010 RTDirQueryUnknownType(szFile, false, &DirEntry.enmType);
2011 if (DirEntry.enmType != RTDIRENTRYTYPE_FILE)
2012 continue;
2013 }
2014
2015 /* Okay, it's a file we can add. */
2016 rc = RTFuzzCtxCorpusInputAddFromFile(hFuzzCtx, szFile);
2017 if (RT_FAILURE(rc))
2018 break;
2019 }
2020 if (rc == VERR_NO_MORE_FILES)
2021 rc = VINF_SUCCESS;
2022 RTDirClose(hDir);
2023 }
2024
2025 return rc;
2026}
2027
2028
2029RTDECL(int) RTFuzzCtxCfgSetInputSeedMaximum(RTFUZZCTX hFuzzCtx, size_t cbMax)
2030{
2031 PRTFUZZCTXINT pThis = hFuzzCtx;
2032 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
2033
2034 pThis->cbInputMax = cbMax;
2035 return VINF_SUCCESS;
2036}
2037
2038
2039RTDECL(size_t) RTFuzzCtxCfgGetInputSeedMaximum(RTFUZZCTX hFuzzCtx)
2040{
2041 PRTFUZZCTXINT pThis = hFuzzCtx;
2042 AssertPtrReturn(pThis, 0);
2043
2044 return pThis->cbInputMax;
2045}
2046
2047
2048RTDECL(int) RTFuzzCtxCfgSetBehavioralFlags(RTFUZZCTX hFuzzCtx, uint32_t fFlags)
2049{
2050 PRTFUZZCTXINT pThis = hFuzzCtx;
2051 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
2052 AssertReturn(!(fFlags & ~RTFUZZCTX_F_BEHAVIORAL_VALID), VERR_INVALID_PARAMETER);
2053
2054 pThis->fFlagsBehavioral = fFlags;
2055 return VINF_SUCCESS;
2056}
2057
2058
2059RTDECL(uint32_t) RTFuzzCfgGetBehavioralFlags(RTFUZZCTX hFuzzCtx)
2060{
2061 PRTFUZZCTXINT pThis = hFuzzCtx;
2062 AssertPtrReturn(pThis, 0);
2063
2064 return pThis->fFlagsBehavioral;
2065}
2066
2067
2068RTDECL(int) RTFuzzCtxCfgSetTmpDirectory(RTFUZZCTX hFuzzCtx, const char *pszPathTmp)
2069{
2070 PRTFUZZCTXINT pThis = hFuzzCtx;
2071 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
2072 AssertPtrReturn(pszPathTmp, VERR_INVALID_POINTER);
2073
2074 return VERR_NOT_IMPLEMENTED;
2075}
2076
2077
2078RTDECL(const char *) RTFuzzCtxCfgGetTmpDirectory(RTFUZZCTX hFuzzCtx)
2079{
2080 PRTFUZZCTXINT pThis = hFuzzCtx;
2081 AssertPtrReturn(pThis, NULL);
2082
2083 return NULL;
2084}
2085
2086
2087RTDECL(int) RTFuzzCtxCfgSetMutationRange(RTFUZZCTX hFuzzCtx, uint64_t offStart, uint64_t cbRange)
2088{
2089 PRTFUZZCTXINT pThis = hFuzzCtx;
2090 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
2091
2092 pThis->offMutStart = offStart;
2093 pThis->cbMutRange = cbRange;
2094 return VINF_SUCCESS;
2095}
2096
2097
2098RTDECL(int) RTFuzzCtxReseed(RTFUZZCTX hFuzzCtx, uint64_t uSeed)
2099{
2100 PRTFUZZCTXINT pThis = hFuzzCtx;
2101 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
2102
2103 RTRandAdvSeed(pThis->hRand, uSeed);
2104 return VINF_SUCCESS;
2105}
2106
2107
2108RTDECL(int) RTFuzzCtxInputGenerate(RTFUZZCTX hFuzzCtx, PRTFUZZINPUT phFuzzInput)
2109{
2110 int rc = VINF_SUCCESS;
2111 PRTFUZZCTXINT pThis = hFuzzCtx;
2112 AssertPtrReturn(pThis, VERR_INVALID_POINTER);
2113 AssertPtrReturn(phFuzzInput, VERR_INVALID_POINTER);
2114
2115 uint32_t cTries = 0;
2116 PRTFUZZMUTATION pMutationParent = rtFuzzCtxMutationPickRnd(pThis);
2117 do
2118 {
2119 uint32_t idxMutator = RTRandAdvU32Ex(pThis->hRand, 0, pThis->cMutators - 1);
2120 PCRTFUZZMUTATOR pMutator = &pThis->paMutators[idxMutator];
2121 PRTFUZZMUTATION pMutation = NULL;
2122
2123 uint64_t offStart = 0;
2124 if (!(pMutator->fFlags & RTFUZZMUTATOR_F_END_OF_BUF))
2125 {
2126 uint64_t offMax = pMutationParent->cbInput - 1;
2127 if ( pMutationParent->cbMutNew != UINT64_MAX
2128 && pMutationParent->offMutStartNew + pMutationParent->cbMutNew < offMax)
2129 offMax = pMutationParent->offMutStartNew + pMutationParent->cbMutNew - 1;
2130
2131 offMax = RT_MAX(pMutationParent->offMutStartNew, offMax);
2132 offStart = RTRandAdvU64Ex(pThis->hRand, pMutationParent->offMutStartNew, offMax);
2133 }
2134 else
2135 offStart = pMutationParent->cbInput;
2136
2137 rc = pMutator->pfnPrep(pThis, offStart, pMutationParent, &pMutation);
2138 if ( RT_SUCCESS(rc)
2139 && VALID_PTR(pMutation))
2140 {
2141 pMutation->pMutator = pMutator;
2142
2143 if (pThis->fFlagsBehavioral & RTFUZZCTX_F_BEHAVIORAL_ADD_INPUT_AUTOMATICALLY_TO_CORPUS)
2144 rtFuzzCtxMutationAdd(pThis, pMutation);
2145
2146 /* Create a new input. */
2147 PRTFUZZINPUTINT pInput = (PRTFUZZINPUTINT)rtFuzzCtxMemoryAlloc(pThis, sizeof(RTFUZZINPUTINT));
2148 if (RT_LIKELY(pInput))
2149 {
2150 pInput->u32Magic = 0; /** @todo */
2151 pInput->cRefs = 1;
2152 pInput->pFuzzer = pThis;
2153 pInput->pMutationTop = pMutation;
2154 RTFuzzCtxRetain(pThis);
2155
2156 rtFuzzMutationRelease(pMutationParent);
2157 *phFuzzInput = pInput;
2158 return rc;
2159 }
2160 else
2161 rc = VERR_NO_MEMORY;
2162 }
2163 } while (++cTries <= 50);
2164
2165 rtFuzzMutationRelease(pMutationParent);
2166 if (RT_SUCCESS(rc))
2167 rc = VERR_INVALID_STATE;
2168
2169 return rc;
2170}
2171
2172
2173RTDECL(int) RTFuzzInputQueryBlobData(RTFUZZINPUT hFuzzInput, void **ppv, size_t *pcb)
2174{
2175 PRTFUZZINPUTINT pThis = hFuzzInput;
2176 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
2177 AssertReturn(pThis->pFuzzer->enmType == RTFUZZCTXTYPE_BLOB, VERR_INVALID_STATE);
2178
2179 int rc = VINF_SUCCESS;
2180 if (!pThis->pMutationTop->pvInput)
2181 rc = rtFuzzMutationDataFinalize(pThis->pMutationTop);
2182
2183 if (RT_SUCCESS(rc))
2184 {
2185 *ppv = pThis->pMutationTop->pvInput;
2186 *pcb = pThis->pMutationTop->cbInput;
2187 }
2188
2189 return rc;
2190}
2191
2192
2193RTDECL(int) RTFuzzInputMutateStreamData(RTFUZZINPUT hFuzzInput, void *pvBuf, size_t cbBuf)
2194{
2195 PRTFUZZINPUTINT pThis = hFuzzInput;
2196 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
2197 AssertReturn(pThis->pFuzzer->enmType == RTFUZZCTXTYPE_STREAM, VERR_INVALID_STATE);
2198
2199 RT_NOREF(pvBuf, cbBuf);
2200 return VERR_NOT_IMPLEMENTED;
2201}
2202
2203
2204RTDECL(uint32_t) RTFuzzInputRetain(RTFUZZINPUT hFuzzInput)
2205{
2206 PRTFUZZINPUTINT pThis = hFuzzInput;
2207
2208 AssertPtrReturn(pThis, UINT32_MAX);
2209
2210 uint32_t cRefs = ASMAtomicIncU32(&pThis->cRefs);
2211 AssertMsg(cRefs > 1 && cRefs < _1M, ("%#x %p\n", cRefs, pThis));
2212 return cRefs;
2213}
2214
2215
2216RTDECL(uint32_t) RTFuzzInputRelease(RTFUZZINPUT hFuzzInput)
2217{
2218 PRTFUZZINPUTINT pThis = hFuzzInput;
2219 if (pThis == NIL_RTFUZZINPUT)
2220 return 0;
2221 AssertPtrReturn(pThis, UINT32_MAX);
2222
2223 uint32_t cRefs = ASMAtomicDecU32(&pThis->cRefs);
2224 AssertMsg(cRefs < _1M, ("%#x %p\n", cRefs, pThis));
2225 if (cRefs == 0)
2226 rtFuzzInputDestroy(pThis);
2227 return cRefs;
2228}
2229
2230
2231RTDECL(int) RTFuzzInputQueryDigestString(RTFUZZINPUT hFuzzInput, char *pszDigest, size_t cchDigest)
2232{
2233 PRTFUZZINPUTINT pThis = hFuzzInput;
2234 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
2235 AssertReturn(pThis->pFuzzer->enmType == RTFUZZCTXTYPE_BLOB, VERR_INVALID_STATE);
2236 AssertPtrReturn(pszDigest, VERR_INVALID_POINTER);
2237 AssertReturn(cchDigest >= RTMD5_STRING_LEN + 1, VERR_INVALID_PARAMETER);
2238
2239 int rc = VINF_SUCCESS;
2240 if (!pThis->pMutationTop->pvInput)
2241 rc = rtFuzzMutationDataFinalize(pThis->pMutationTop);
2242
2243 if (RT_SUCCESS(rc))
2244 {
2245 uint8_t abHash[RTMD5_HASH_SIZE];
2246 RTMd5(pThis->pMutationTop->pvInput, pThis->pMutationTop->cbInput, &abHash[0]);
2247 rc = RTMd5ToString(&abHash[0], pszDigest, cchDigest);
2248 }
2249
2250 return rc;
2251}
2252
2253
2254RTDECL(int) RTFuzzInputWriteToFile(RTFUZZINPUT hFuzzInput, const char *pszFilename)
2255{
2256 PRTFUZZINPUTINT pThis = hFuzzInput;
2257 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
2258 AssertReturn(pThis->pFuzzer->enmType == RTFUZZCTXTYPE_BLOB, VERR_INVALID_STATE);
2259 AssertPtrReturn(pszFilename, VERR_INVALID_POINTER);
2260
2261 int rc = VINF_SUCCESS;
2262 if (!pThis->pMutationTop->pvInput)
2263 rc = rtFuzzMutationDataFinalize(pThis->pMutationTop);
2264
2265 if (RT_SUCCESS(rc))
2266 {
2267 RTFILE hFile;
2268 rc = RTFileOpen(&hFile, pszFilename, RTFILE_O_CREATE | RTFILE_O_WRITE | RTFILE_O_DENY_NONE);
2269 if (RT_SUCCESS(rc))
2270 {
2271 rc = RTFileWrite(hFile, pThis->pMutationTop->pvInput, pThis->pMutationTop->cbInput, NULL);
2272 AssertRC(rc);
2273 RTFileClose(hFile);
2274
2275 if (RT_FAILURE(rc))
2276 RTFileDelete(pszFilename);
2277 }
2278 }
2279
2280 return rc;
2281}
2282
2283
2284RTDECL(int) RTFuzzInputAddToCtxCorpus(RTFUZZINPUT hFuzzInput)
2285{
2286 PRTFUZZINPUTINT pThis = hFuzzInput;
2287 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
2288
2289 return rtFuzzCtxMutationAdd(pThis->pFuzzer, pThis->pMutationTop);
2290}
2291
2292
2293RTDECL(int) RTFuzzInputRemoveFromCtxCorpus(RTFUZZINPUT hFuzzInput)
2294{
2295 PRTFUZZINPUTINT pThis = hFuzzInput;
2296 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
2297
2298#if 0
2299 int rc = VINF_SUCCESS;
2300 PRTFUZZINTERMEDIATE pIntermediate = NULL;
2301 PRTFUZZINPUTINT pInputLoc = rtFuzzCtxInputLocate(pThis->pFuzzer, &pThis->abMd5Hash[0], true /*fExact*/,
2302 &pIntermediate);
2303 if (pInputLoc)
2304 {
2305 AssertPtr(pIntermediate);
2306 Assert(pInputLoc == pThis);
2307
2308 uint64_t u64Md5Low = *(uint64_t *)&pThis->abMd5Hash[0];
2309 RTAvlU64Remove(&pIntermediate->TreeSeedsLow, u64Md5Low);
2310 RTFuzzInputRelease(hFuzzInput);
2311 }
2312 else
2313 rc = VERR_NOT_FOUND;
2314#endif
2315
2316 return VERR_NOT_IMPLEMENTED;
2317}
2318
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette