Aaron Ballman | ef11698 | 2015-01-29 16:58:29 +0000 | [diff] [blame] | 1 | //===- FuzzerMutate.cpp - Mutate a test input -----------------------------===// |
| 2 | // |
| 3 | // The LLVM Compiler Infrastructure |
| 4 | // |
| 5 | // This file is distributed under the University of Illinois Open Source |
| 6 | // License. See LICENSE.TXT for details. |
| 7 | // |
| 8 | //===----------------------------------------------------------------------===// |
| 9 | // Mutate a test input. |
| 10 | //===----------------------------------------------------------------------===// |
| 11 | |
Kostya Serebryany | f342459 | 2015-05-22 22:35:31 +0000 | [diff] [blame] | 12 | #include <cstring> |
| 13 | |
Aaron Ballman | ef11698 | 2015-01-29 16:58:29 +0000 | [diff] [blame] | 14 | #include "FuzzerInternal.h" |
| 15 | |
| 16 | namespace fuzzer { |
| 17 | |
Kostya Serebryany | 404c69f | 2015-07-24 01:06:40 +0000 | [diff] [blame] | 18 | static char FlipRandomBit(char X, FuzzerRandomBase &Rand) { |
| 19 | int Bit = Rand(8); |
Aaron Ballman | ef11698 | 2015-01-29 16:58:29 +0000 | [diff] [blame] | 20 | char Mask = 1 << Bit; |
| 21 | char R; |
| 22 | if (X & (1 << Bit)) |
| 23 | R = X & ~Mask; |
| 24 | else |
| 25 | R = X | Mask; |
| 26 | assert(R != X); |
| 27 | return R; |
| 28 | } |
| 29 | |
Kostya Serebryany | 404c69f | 2015-07-24 01:06:40 +0000 | [diff] [blame] | 30 | static char RandCh(FuzzerRandomBase &Rand) { |
| 31 | if (Rand.RandBool()) return Rand(256); |
Aaron Ballman | ef11698 | 2015-01-29 16:58:29 +0000 | [diff] [blame] | 32 | const char *Special = "!*'();:@&=+$,/?%#[]123ABCxyz-`~."; |
Kostya Serebryany | 404c69f | 2015-07-24 01:06:40 +0000 | [diff] [blame] | 33 | return Special[Rand(sizeof(Special) - 1)]; |
Aaron Ballman | ef11698 | 2015-01-29 16:58:29 +0000 | [diff] [blame] | 34 | } |
| 35 | |
Kostya Serebryany | 8ce7424 | 2015-08-01 01:42:51 +0000 | [diff] [blame^] | 36 | size_t Mutate_EraseByte(uint8_t *Data, size_t Size, size_t MaxSize, |
| 37 | FuzzerRandomBase &Rand) { |
| 38 | assert(Size); |
| 39 | if (Size == 1) return Size; |
| 40 | size_t Idx = Rand(Size); |
| 41 | // Erase Data[Idx]. |
| 42 | memmove(Data + Idx, Data + Idx + 1, Size - Idx - 1); |
| 43 | return Size - 1; |
| 44 | } |
| 45 | |
Kostya Serebryany | f342459 | 2015-05-22 22:35:31 +0000 | [diff] [blame] | 46 | // Mutates Data in place, returns new size. |
Kostya Serebryany | 404c69f | 2015-07-24 01:06:40 +0000 | [diff] [blame] | 47 | size_t Mutate(uint8_t *Data, size_t Size, size_t MaxSize, |
| 48 | FuzzerRandomBase &Rand) { |
Kostya Serebryany | f342459 | 2015-05-22 22:35:31 +0000 | [diff] [blame] | 49 | assert(MaxSize > 0); |
| 50 | assert(Size <= MaxSize); |
| 51 | if (Size == 0) { |
| 52 | for (size_t i = 0; i < MaxSize; i++) |
Kostya Serebryany | 404c69f | 2015-07-24 01:06:40 +0000 | [diff] [blame] | 53 | Data[i] = RandCh(Rand); |
Kostya Serebryany | f342459 | 2015-05-22 22:35:31 +0000 | [diff] [blame] | 54 | return MaxSize; |
Kostya Serebryany | 5b266a8 | 2015-02-04 19:10:20 +0000 | [diff] [blame] | 55 | } |
Kostya Serebryany | f342459 | 2015-05-22 22:35:31 +0000 | [diff] [blame] | 56 | assert(Size > 0); |
Kostya Serebryany | 404c69f | 2015-07-24 01:06:40 +0000 | [diff] [blame] | 57 | size_t Idx = Rand(Size); |
| 58 | switch (Rand(3)) { |
Kostya Serebryany | 8ce7424 | 2015-08-01 01:42:51 +0000 | [diff] [blame^] | 59 | case 0: Size = Mutate_EraseByte(Data, Size, MaxSize, Rand); break; |
Aaron Ballman | ef11698 | 2015-01-29 16:58:29 +0000 | [diff] [blame] | 60 | case 1: |
Kostya Serebryany | f342459 | 2015-05-22 22:35:31 +0000 | [diff] [blame] | 61 | if (Size < MaxSize) { |
| 62 | // Insert new value at Data[Idx]. |
| 63 | memmove(Data + Idx + 1, Data + Idx, Size - Idx); |
Kostya Serebryany | 404c69f | 2015-07-24 01:06:40 +0000 | [diff] [blame] | 64 | Data[Idx] = RandCh(Rand); |
Aaron Ballman | ef11698 | 2015-01-29 16:58:29 +0000 | [diff] [blame] | 65 | } |
Kostya Serebryany | 404c69f | 2015-07-24 01:06:40 +0000 | [diff] [blame] | 66 | Data[Idx] = RandCh(Rand); |
Aaron Ballman | ef11698 | 2015-01-29 16:58:29 +0000 | [diff] [blame] | 67 | break; |
Kostya Serebryany | f342459 | 2015-05-22 22:35:31 +0000 | [diff] [blame] | 68 | case 2: |
Kostya Serebryany | 404c69f | 2015-07-24 01:06:40 +0000 | [diff] [blame] | 69 | Data[Idx] = FlipRandomBit(Data[Idx], Rand); |
Aaron Ballman | ef11698 | 2015-01-29 16:58:29 +0000 | [diff] [blame] | 70 | break; |
| 71 | } |
Kostya Serebryany | f342459 | 2015-05-22 22:35:31 +0000 | [diff] [blame] | 72 | assert(Size > 0); |
| 73 | return Size; |
Aaron Ballman | ef11698 | 2015-01-29 16:58:29 +0000 | [diff] [blame] | 74 | } |
| 75 | |
| 76 | } // namespace fuzzer |