/src/immer/extra/fuzzer/map-gc.cpp
Line | Count | Source |
1 | | // |
2 | | // immer: immutable data structures for C++ |
3 | | // Copyright (C) 2016, 2017, 2018 Juan Pedro Bolivar Puente |
4 | | // |
5 | | // This software is distributed under the Boost Software License, Version 1.0. |
6 | | // See accompanying file LICENSE or copy at http://boost.org/LICENSE_1_0.txt |
7 | | // |
8 | | |
9 | | #include "fuzzer_input.hpp" |
10 | | |
11 | | #include <immer/heap/gc_heap.hpp> |
12 | | #include <immer/map.hpp> |
13 | | #include <immer/refcount/no_refcount_policy.hpp> |
14 | | |
15 | | #include <immer/algorithm.hpp> |
16 | | |
17 | | #include <array> |
18 | | |
19 | | using gc_memory = immer::memory_policy<immer::heap_policy<immer::gc_heap>, |
20 | | immer::no_refcount_policy, |
21 | | immer::default_lock_policy, |
22 | | immer::gc_transience_policy, |
23 | | false>; |
24 | | |
25 | | struct colliding_hash_t |
26 | | { |
27 | 4.90M | std::size_t operator()(std::size_t x) const { return x & ~15; } |
28 | | }; |
29 | | |
30 | | extern "C" int LLVMFuzzerTestOneInput(const std::uint8_t* data, |
31 | | std::size_t size) |
32 | 84.8k | { |
33 | 84.8k | auto guard = immer::gc_disable_guard{}; |
34 | | |
35 | 84.8k | constexpr auto var_count = 4; |
36 | | |
37 | 84.8k | using map_t = immer:: |
38 | 84.8k | map<std::size_t, int, colliding_hash_t, std::equal_to<>, gc_memory>; |
39 | | |
40 | 84.8k | auto vars = std::array<map_t, var_count>{}; |
41 | | |
42 | 591k | auto is_valid_var = [&](auto idx) { return idx >= 0 && idx < var_count; }; |
43 | | |
44 | 169k | return fuzzer_input{data, size}.run([&](auto& in) { |
45 | 169k | enum ops |
46 | 169k | { |
47 | 169k | op_set, |
48 | 169k | op_erase, |
49 | 169k | op_set_move, |
50 | 169k | op_erase_move, |
51 | 169k | op_iterate, |
52 | 169k | op_find, |
53 | 169k | op_update, |
54 | 169k | op_update_move, |
55 | 169k | op_update_if_exists, |
56 | 169k | op_update_if_exists_move, |
57 | 169k | op_diff |
58 | 169k | }; |
59 | 169k | auto src = read<char>(in, is_valid_var); |
60 | 169k | auto dst = read<char>(in, is_valid_var); |
61 | 169k | switch (read<char>(in)) { |
62 | 87.8k | case op_set: { |
63 | 87.8k | auto value = read<size_t>(in); |
64 | 87.8k | vars[dst] = vars[src].set(value, 42); |
65 | 87.8k | break; |
66 | 0 | } |
67 | 9.90k | case op_erase: { |
68 | 9.90k | auto value = read<size_t>(in); |
69 | 9.90k | vars[dst] = vars[src].erase(value); |
70 | 9.90k | break; |
71 | 0 | } |
72 | 1.58k | case op_set_move: { |
73 | 1.58k | auto value = read<size_t>(in); |
74 | 1.58k | vars[dst] = std::move(vars[src]).set(value, 42); |
75 | 1.58k | break; |
76 | 0 | } |
77 | 1.23k | case op_erase_move: { |
78 | 1.23k | auto value = read<size_t>(in); |
79 | 1.23k | vars[dst] = std::move(vars[src]).erase(value); |
80 | 1.23k | break; |
81 | 0 | } |
82 | 8.86k | case op_iterate: { |
83 | 8.86k | auto srcv = vars[src]; |
84 | 701k | for (const auto& v : srcv) { |
85 | 701k | vars[dst] = vars[dst].set(v.first, v.second); |
86 | 701k | } |
87 | 8.86k | break; |
88 | 0 | } |
89 | 3.09k | case op_find: { |
90 | 3.09k | auto value = read<size_t>(in); |
91 | 3.09k | auto res = vars[src].find(value); |
92 | 3.09k | if (res != nullptr) { |
93 | 893 | vars[dst] = vars[dst].set(*res, 42); |
94 | 893 | } |
95 | 3.09k | break; |
96 | 0 | } |
97 | 6.06k | case op_update: { |
98 | 6.06k | auto key = read<size_t>(in); |
99 | 6.06k | vars[dst] = vars[src].update(key, [](int x) { return x + 1; }); |
100 | 6.06k | break; |
101 | 0 | } |
102 | 3.01k | case op_update_move: { |
103 | 3.01k | auto key = read<size_t>(in); |
104 | 3.01k | vars[dst] = |
105 | 3.01k | std::move(vars[src]).update(key, [](int x) { return x + 1; }); |
106 | 3.01k | break; |
107 | 0 | } |
108 | 12.3k | case op_update_if_exists: { |
109 | 12.3k | auto key = read<size_t>(in); |
110 | 12.3k | vars[dst] = |
111 | 12.3k | vars[src].update_if_exists(key, [](int x) { return x + 1; }); |
112 | 12.3k | break; |
113 | 0 | } |
114 | 2.71k | case op_update_if_exists_move: { |
115 | 2.71k | auto key = read<size_t>(in); |
116 | 2.71k | vars[dst] = std::move(vars[src]).update_if_exists( |
117 | 2.71k | key, [](int x) { return x + 1; }); |
118 | 2.71k | break; |
119 | 0 | } |
120 | 15.7k | case op_diff: { |
121 | 15.7k | auto&& a = vars[src]; |
122 | 15.7k | auto&& b = vars[dst]; |
123 | 15.7k | diff( |
124 | 15.7k | a, |
125 | 15.7k | b, |
126 | 69.0k | [&](auto&& x) { |
127 | 69.0k | assert(!a.count(x.first)); |
128 | 69.0k | assert(b.count(x.first)); |
129 | 69.0k | }, |
130 | 51.9k | [&](auto&& x) { |
131 | 51.9k | assert(a.count(x.first)); |
132 | 51.9k | assert(!b.count(x.first)); |
133 | 51.9k | }, |
134 | 15.7k | [&](auto&& x, auto&& y) { |
135 | 1.90k | assert(x.first == y.first); |
136 | 1.90k | assert(x.second != y.second); |
137 | 1.90k | }); |
138 | 15.7k | } |
139 | 29.0k | default: |
140 | 29.0k | break; |
141 | 169k | }; |
142 | 165k | return true; |
143 | 169k | }); |
144 | 84.8k | } |