Coverage Report

Created: 2026-06-30 07:33

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/kea-fuzzer/fuzz_encode.cc
Line
Count
Source
1
// Copyright (C) 2025 Ada Logics Ltd.
2
//
3
// This Source Code Form is subject to the terms of the Mozilla Public
4
// License, v. 2.0. If a copy of the MPL was not distributed with this
5
// file, You can obtain one at http://mozilla.org/MPL/2.0/.
6
////////////////////////////////////////////////////////////////////////////////
7
#include <config.h>
8
#include <fuzzer/FuzzedDataProvider.h>
9
10
#include <exceptions/exceptions.h>
11
#include <util/encode/encode.h>
12
13
#include <string>
14
#include <vector>
15
#include <cstddef>
16
17
using namespace isc::util::encode;
18
19
1.36k
extern "C" int LLVMFuzzerTestOneInput(const uint8_t* data, size_t size) {
20
1.36k
    if (size < 2) {
21
1
        return 0;
22
1
    }
23
24
1.36k
    FuzzedDataProvider fdp(data, size);
25
    
26
    // Choose which encoding/decoding path to test
27
1.36k
    uint8_t path = fdp.ConsumeIntegralInRange<uint8_t>(0, 11);
28
    
29
1.36k
    std::vector<uint8_t> binary_data;
30
1.36k
    std::string encoded_str;
31
1.36k
    std::vector<uint8_t> decoded_output;
32
    
33
1.36k
    switch (path) {
34
125
        case 0: {
35
            // Test Base64 encoding from binary data
36
125
            try {
37
125
                size_t bin_size = fdp.ConsumeIntegralInRange<size_t>(0, size);
38
125
                binary_data = fdp.ConsumeBytes<uint8_t>(bin_size);
39
125
                encoded_str = encodeBase64(binary_data);
40
                // Verify round-trip
41
125
                decodeBase64(encoded_str, decoded_output);
42
125
            } catch (const isc::Exception&) {
43
                // Expected for invalid input
44
0
            }
45
125
            break;
46
125
        }
47
        
48
92
        case 1: {
49
            // Test Base64 decoding from string
50
92
            try {
51
92
                encoded_str = fdp.ConsumeRemainingBytesAsString();
52
92
                decodeBase64(encoded_str, decoded_output);
53
92
            } catch (const isc::Exception&) {
54
                // Expected for invalid Base64 strings
55
91
            }
56
92
            break;
57
92
        }
58
        
59
93
        case 2: {
60
            // Test Base32Hex encoding from binary data
61
93
            try {
62
93
                size_t bin_size = fdp.ConsumeIntegralInRange<size_t>(0, size);
63
93
                binary_data = fdp.ConsumeBytes<uint8_t>(bin_size);
64
93
                encoded_str = encodeBase32Hex(binary_data);
65
                // Verify round-trip
66
93
                decodeBase32Hex(encoded_str, decoded_output);
67
93
            } catch (const isc::Exception&) {
68
                // Expected for invalid input
69
0
            }
70
93
            break;
71
93
        }
72
        
73
75
        case 3: {
74
            // Test Base32Hex decoding from string
75
75
            try {
76
75
                encoded_str = fdp.ConsumeRemainingBytesAsString();
77
75
                decodeBase32Hex(encoded_str, decoded_output);
78
75
            } catch (const isc::Exception&) {
79
                // Expected for invalid Base32Hex strings
80
73
            }
81
75
            break;
82
75
        }
83
        
84
111
        case 4: {
85
            // Test Base16 (hex) encoding from binary data
86
111
            try {
87
111
                size_t bin_size = fdp.ConsumeIntegralInRange<size_t>(0, size);
88
111
                binary_data = fdp.ConsumeBytes<uint8_t>(bin_size);
89
111
                encoded_str = encodeHex(binary_data);
90
                // Verify round-trip
91
111
                decodeHex(encoded_str, decoded_output);
92
111
            } catch (const isc::Exception&) {
93
                // Expected for invalid input
94
0
            }
95
111
            break;
96
111
        }
97
        
98
70
        case 5: {
99
            // Test Base16 (hex) decoding from string
100
70
            try {
101
70
                encoded_str = fdp.ConsumeRemainingBytesAsString();
102
70
                decodeHex(encoded_str, decoded_output);
103
70
            } catch (const isc::Exception&) {
104
                // Expected for invalid hex strings
105
69
            }
106
70
            break;
107
70
        }
108
        
109
148
        case 6: {
110
            // Test Base64 with various padding scenarios
111
148
            try {
112
148
                std::string test_str = fdp.ConsumeRandomLengthString();
113
                // Add various padding permutations
114
148
                test_str += fdp.ConsumeBool() ? "=" : "";
115
148
                test_str += fdp.ConsumeBool() ? "=" : "";
116
148
                decodeBase64(test_str, decoded_output);
117
148
            } catch (const isc::Exception&) {
118
                // Expected for invalid padding
119
111
            }
120
148
            break;
121
148
        }
122
        
123
142
        case 7: {
124
            // Test Base32Hex with various padding scenarios
125
142
            try {
126
142
                std::string test_str = fdp.ConsumeRandomLengthString();
127
                // Add various padding permutations
128
282
                for (int i = 0; i < fdp.ConsumeIntegralInRange(0, 6); i++) {
129
140
                    test_str += "=";
130
140
                }
131
142
                decodeBase32Hex(test_str, decoded_output);
132
142
            } catch (const isc::Exception&) {
133
                // Expected for invalid padding
134
122
            }
135
142
            break;
136
142
        }
137
        
138
65
        case 8: {
139
            // Test mixed case Base64 (should be case-sensitive)
140
65
            try {
141
65
                encoded_str = fdp.ConsumeRemainingBytesAsString();
142
                // Mix uppercase and lowercase
143
8.38M
                for (auto& c : encoded_str) {
144
8.38M
                    if (fdp.ConsumeBool() && isalpha(c)) {
145
0
                        c = (isupper(c)) ? tolower(c) : toupper(c);
146
0
                    }
147
8.38M
                }
148
65
                decodeBase64(encoded_str, decoded_output);
149
65
            } catch (const isc::Exception&) {
150
                // Expected for case errors
151
64
            }
152
65
            break;
153
65
        }
154
        
155
62
        case 9: {
156
            // Test mixed case Base32Hex (case-insensitive)
157
62
            try {
158
62
                encoded_str = fdp.ConsumeRemainingBytesAsString();
159
                // Mix uppercase and lowercase
160
6.35M
                for (auto& c : encoded_str) {
161
6.35M
                    if (fdp.ConsumeBool() && isalpha(c)) {
162
0
                        c = (isupper(c)) ? tolower(c) : toupper(c);
163
0
                    }
164
6.35M
                }
165
62
                decodeBase32Hex(encoded_str, decoded_output);
166
62
            } catch (const isc::Exception&) {
167
                // May succeed due to case-insensitivity
168
61
            }
169
62
            break;
170
62
        }
171
        
172
67
        case 10: {
173
            // Test Base16 with mixed case (case-insensitive)
174
67
            try {
175
67
                encoded_str = fdp.ConsumeRemainingBytesAsString();
176
                // Mix uppercase and lowercase
177
7.36M
                for (auto& c : encoded_str) {
178
7.36M
                    if (fdp.ConsumeBool() && isalpha(c)) {
179
0
                        c = (isupper(c)) ? tolower(c) : toupper(c);
180
0
                    }
181
7.36M
                }
182
67
                decodeHex(encoded_str, decoded_output);
183
67
            } catch (const isc::Exception&) {
184
                // May succeed due to case-insensitivity
185
66
            }
186
67
            break;
187
67
        }
188
        
189
316
        case 11: {
190
            // Test encoding/decoding with whitespace injection
191
316
            try {
192
316
                std::string test_str = fdp.ConsumeRandomLengthString();
193
                // Inject whitespace characters
194
316
                size_t insertions = fdp.ConsumeIntegralInRange<size_t>(0, 10);
195
1.34k
                for (size_t i = 0; i < insertions && test_str.size() > 0; i++) {
196
1.02k
                    size_t pos = fdp.ConsumeIntegralInRange<size_t>(0, test_str.size());
197
1.02k
                    char ws = fdp.PickValueInArray({' ', '\t', '\n', '\r'});
198
1.02k
                    test_str.insert(pos, 1, ws);
199
1.02k
                }
200
                
201
                // Try decoding with all encoders
202
316
                try { decodeBase64(test_str, decoded_output); } catch (...) {}
203
316
                try { decodeBase32Hex(test_str, decoded_output); } catch (...) {}
204
316
                try { decodeHex(test_str, decoded_output); } catch (...) {}
205
316
            } catch (const isc::Exception&) {
206
                // Expected for whitespace handling
207
0
            }
208
316
            break;
209
316
        }
210
1.36k
    }
211
    
212
1.36k
    return 0;
213
1.36k
}