• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

randombit / botan / 26205954101

19 May 2026 08:32PM UTC coverage: 89.343%. Remained the same
26205954101

push

github

web-flow
Merge pull request #5609 from randombit/jack/improve-http

Improve the HTTP 1.0 client used for OCSP/CRL

109341 of 122383 relevant lines covered (89.34%)

11271652.81 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

95.62
/src/tests/test_utils.cpp
1
/*
2
* (C) 2015,2018,2024 Jack Lloyd
3
* (C) 2016 Daniel Neus, Rohde & Schwarz Cybersecurity
4
* (C) 2017 René Korthaus, Rohde & Schwarz Cybersecurity
5
*
6
* Botan is released under the Simplified BSD License (see license.txt)
7
*/
8

9
#include "tests.h"
10

11
#include "test_arb_eq.h"
12
#include <botan/hex.h>
13
#include <botan/rng.h>
14
#include <botan/version.h>
15
#include <botan/internal/bit_ops.h>
16
#include <botan/internal/calendar.h>
17
#include <botan/internal/charset.h>
18
#include <botan/internal/concat_util.h>
19
#include <botan/internal/ct_utils.h>
20
#include <botan/internal/fmt.h>
21
#include <botan/internal/int_utils.h>
22
#include <botan/internal/loadstor.h>
23
#include <botan/internal/parsing.h>
24
#include <botan/internal/rounding.h>
25
#include <botan/internal/target_info.h>
26

27
#include <bit>
28

29
#if defined(BOTAN_HAS_CPUID)
30
   #include <botan/internal/cpuid.h>
31
#endif
32

33
#if defined(BOTAN_HAS_IPV4_ADDRESS)
34
   #include <botan/ipv4_address.h>
35
#endif
36

37
#if defined(BOTAN_HAS_IPV6_ADDRESS)
38
   #include <botan/ipv6_address.h>
39
#endif
40

41
#if defined(BOTAN_HAS_POLY_DBL)
42
   #include <botan/internal/poly_dbl.h>
43
#endif
44

45
#if defined(BOTAN_HAS_UUID)
46
   #include <botan/uuid.h>
47
#endif
48

49
namespace Botan_Tests {
50

51
namespace {
52

53
class Utility_Function_Tests final : public Test {
1✔
54
   public:
55
      std::vector<Test::Result> run() override {
1✔
56
         std::vector<Test::Result> results;
1✔
57

58
         results.push_back(test_checked_add());
2✔
59
         results.push_back(test_checked_mul());
2✔
60
         results.push_back(test_checked_cast());
2✔
61
         results.push_back(test_round_up());
2✔
62
         results.push_back(test_loadstore());
2✔
63
         results.push_back(test_loadstore_ambiguity());
2✔
64
         results.push_back(test_loadstore_fallback());
2✔
65
         results.push_back(test_loadstore_constexpr());
2✔
66
         return Botan::concat(results, test_copy_out_be_le());
3✔
67
      }
1✔
68

69
   private:
70
      Test::Result test_checked_add() {
1✔
71
         Test::Result result("checked_add");
1✔
72

73
         const size_t large = static_cast<size_t>(-5);
1✔
74
         const size_t zero = 0;
1✔
75

76
         for(int si = -15; si != 15; ++si) {
31✔
77
            const size_t i = static_cast<size_t>(si);
30✔
78
            auto sum1 = Botan::checked_add<size_t>(i, zero, zero, zero, large);
30✔
79
            auto sum2 = Botan::checked_add<size_t>(large, zero, zero, zero, i);
30✔
80

81
            result.test_is_true("checked_add looks at all args", sum1 == sum2);
60✔
82

83
            if(i < 5) {
30✔
84
               result.test_sz_eq("checked_add worked", sum1.value(), i + large);
5✔
85
            } else {
86
               result.test_is_true("checked_add did not return a result", !sum1.has_value());
25✔
87
            }
88
         }
89

90
         auto& rng = Test::rng();
1✔
91

92
         for(size_t i = 0; i != 100; ++i) {
101✔
93
            const uint16_t x = Botan::make_uint16(rng.next_byte(), rng.next_byte());
100✔
94
            const uint16_t y = Botan::make_uint16(rng.next_byte(), rng.next_byte());
100✔
95

96
            const uint32_t ref = static_cast<uint32_t>(x) + y;
100✔
97

98
            if(auto z = Botan::checked_add(x, y)) {
200✔
99
               result.test_u32_eq("checked_add adds", static_cast<uint32_t>(z.value()), ref);
47✔
100
            } else {
101
               result.test_is_true("checked_add checks", (ref >> 16) > 0);
53✔
102
            }
103
         }
104

105
         return result;
1✔
106
      }
×
107

108
      Test::Result test_checked_mul() {
1✔
109
         Test::Result result("checked_mul");
1✔
110

111
         auto& rng = Test::rng();
1✔
112

113
         for(size_t i = 0; i != 100; ++i) {
101✔
114
            const uint16_t x = Botan::make_uint16(rng.next_byte(), rng.next_byte());
100✔
115
            const uint16_t y = Botan::make_uint16(rng.next_byte(), rng.next_byte());
100✔
116

117
            const uint32_t ref = static_cast<uint32_t>(x) * y;
100✔
118

119
            if(auto z = Botan::checked_mul(x, y)) {
200✔
120
               result.test_u32_eq("checked_mul multiplies", static_cast<uint32_t>(z.value()), ref);
×
121
            } else {
122
               result.test_is_true("checked_mul checks", (ref >> 16) > 0);
100✔
123
            }
124
         }
125

126
         return result;
1✔
127
      }
×
128

129
      Test::Result test_checked_cast() {
1✔
130
         Test::Result result("checked_cast");
1✔
131

132
         const uint32_t large = static_cast<uint32_t>(-1);
1✔
133
         const uint32_t is_16_bits = 0x8123;
1✔
134
         const uint32_t is_8_bits = 0x89;
1✔
135

136
         result.test_throws("checked_cast checks", [&] { Botan::checked_cast_to<uint16_t>(large); });
2✔
137
         result.test_throws("checked_cast checks", [&] { Botan::checked_cast_to<uint8_t>(large); });
2✔
138

139
         result.test_u32_eq("checked_cast converts", Botan::checked_cast_to<uint32_t>(large), large);
1✔
140
         result.test_u16_eq("checked_cast converts", Botan::checked_cast_to<uint16_t>(is_16_bits), 0x8123);
2✔
141
         result.test_u8_eq("checked_cast converts", Botan::checked_cast_to<uint8_t>(is_8_bits), 0x89);
1✔
142

143
         return result;
1✔
144
      }
×
145

146
      Test::Result test_round_up() {
1✔
147
         Test::Result result("Util round_up");
1✔
148

149
         // clang-format off
150
         const std::vector<size_t> inputs = {
1✔
151
            0, 1, 2, 3, 4, 9, 10, 32, 99, 100, 101, 255, 256, 1000, 10000,
152
            65535, 65536, 65537,
153
         };
1✔
154

155
         const std::vector<size_t> alignments = {
1✔
156
            1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 16, 32, 50, 64, 100, 512, 521,
157
            1000, 1023, 1024, 1025, 10000, 65535, 65536
158
         };
1✔
159
         // clang-format on
160

161
         for(const size_t i : inputs) {
19✔
162
            for(const size_t m : alignments) {
450✔
163
               try {
432✔
164
                  const size_t z = Botan::round_up(i, m);
432✔
165

166
                  result.test_is_true("z % m == 0", z % m == 0);
432✔
167
                  result.test_is_true("z >= i", z >= i);
432✔
168
                  result.test_is_true("z <= i + m", z <= i + m);
432✔
169
               } catch(Botan::Exception& e) {
×
170
                  result.test_failure(Botan::fmt("round_up({},{})", i, m), e.what());
×
171
               }
×
172
            }
173
         }
174

175
         result.test_throws("Integer overflow is detected", []() { Botan::round_up(static_cast<size_t>(-1), 1024); });
1✔
176

177
         return result;
1✔
178
      }
1✔
179

180
      using TestInt64 = Botan::Strong<uint64_t, struct TestInt64_>;
181
      using TestInt32 = Botan::Strong<uint32_t, struct TestInt64_>;
182
      using TestVectorSink = Botan::Strong<std::vector<uint8_t>, struct TestVectorSink_>;
183

184
      enum class TestEnum64 : uint64_t {
185
         _1 = 0x1234567890ABCDEF,
186
         _2 = 0xEFCDAB9078563412,
187
      };
188

189
      enum class TestEnum32 : uint32_t {
190
         _1 = 0x12345678,
191
         _2 = 0x78563412,
192
      };
193

194
      static Test::Result test_loadstore() {
1✔
195
         Test::Result result("Util load/store");
1✔
196

197
         const std::vector<uint8_t> membuf = Botan::hex_decode("00112233445566778899AABBCCDDEEFF");
1✔
198
         const uint8_t* mem = membuf.data();
1✔
199

200
         const uint16_t in16 = 0x1234;
1✔
201
         const uint32_t in32 = 0xA0B0C0D0;
1✔
202
         const uint64_t in64 = 0xABCDEF0123456789;
1✔
203

204
         result.test_u8_eq(Botan::get_byte<0>(in32), 0xA0);
1✔
205
         result.test_u8_eq(Botan::get_byte<1>(in32), 0xB0);
1✔
206
         result.test_u8_eq(Botan::get_byte<2>(in32), 0xC0);
1✔
207
         result.test_u8_eq(Botan::get_byte<3>(in32), 0xD0);
1✔
208

209
         result.test_u16_eq(Botan::make_uint16(0xAA, 0xBB), 0xAABB);
1✔
210
         result.test_u32_eq(Botan::make_uint32(0x01, 0x02, 0x03, 0x04), 0x01020304);
1✔
211

212
         result.test_u16_eq(Botan::load_be<uint16_t>(mem, 0), 0x0011);
1✔
213
         result.test_u16_eq(Botan::load_be<uint16_t>(mem, 1), 0x2233);
1✔
214
         result.test_u16_eq(Botan::load_be<uint16_t>(mem, 2), 0x4455);
1✔
215
         result.test_u16_eq(Botan::load_be<uint16_t>(mem, 3), 0x6677);
1✔
216

217
         result.test_u16_eq(Botan::load_le<uint16_t>(mem, 0), 0x1100);
1✔
218
         result.test_u16_eq(Botan::load_le<uint16_t>(mem, 1), 0x3322);
1✔
219
         result.test_u16_eq(Botan::load_le<uint16_t>(mem, 2), 0x5544);
1✔
220
         result.test_u16_eq(Botan::load_le<uint16_t>(mem, 3), 0x7766);
1✔
221

222
         result.test_u32_eq(Botan::load_be<uint32_t>(mem, 0), 0x00112233);
1✔
223
         result.test_u32_eq(Botan::load_be<uint32_t>(mem, 1), 0x44556677);
1✔
224
         result.test_u32_eq(Botan::load_be<uint32_t>(mem, 2), 0x8899AABB);
1✔
225
         result.test_u32_eq(Botan::load_be<uint32_t>(mem, 3), 0xCCDDEEFF);
1✔
226

227
         result.test_u32_eq(Botan::load_le<uint32_t>(mem, 0), 0x33221100);
1✔
228
         result.test_u32_eq(Botan::load_le<uint32_t>(mem, 1), 0x77665544);
1✔
229
         result.test_u32_eq(Botan::load_le<uint32_t>(mem, 2), 0xBBAA9988);
1✔
230
         result.test_u32_eq(Botan::load_le<uint32_t>(mem, 3), 0xFFEEDDCC);
1✔
231

232
         result.test_u64_eq(Botan::load_be<uint64_t>(mem, 0), 0x0011223344556677);
1✔
233
         result.test_u64_eq(Botan::load_be<uint64_t>(mem, 1), 0x8899AABBCCDDEEFF);
1✔
234

235
         result.test_u64_eq(Botan::load_le<uint64_t>(mem, 0), 0x7766554433221100);
1✔
236
         result.test_u64_eq(Botan::load_le<uint64_t>(mem, 1), 0xFFEEDDCCBBAA9988);
1✔
237

238
         // Check misaligned loads:
239
         result.test_u16_eq(Botan::load_be<uint16_t>(mem + 1, 0), 0x1122);
1✔
240
         result.test_u16_eq(Botan::load_le<uint16_t>(mem + 3, 0), 0x4433);
1✔
241

242
         result.test_u32_eq(Botan::load_be<uint32_t>(mem + 1, 1), 0x55667788);
1✔
243
         result.test_u32_eq(Botan::load_le<uint32_t>(mem + 3, 1), 0xAA998877);
1✔
244

245
         result.test_u64_eq(Botan::load_be<uint64_t>(mem + 1, 0), 0x1122334455667788);
1✔
246
         result.test_u64_eq(Botan::load_le<uint64_t>(mem + 7, 0), 0xEEDDCCBBAA998877);
1✔
247
         result.test_u64_eq(Botan::load_le<uint64_t>(mem + 5, 0), 0xCCBBAA9988776655);
1✔
248

249
         uint8_t outbuf[16] = {0};
1✔
250

251
         for(size_t offset = 0; offset != 7; ++offset) {
8✔
252
            uint8_t* out = outbuf + offset;  // NOLINT(*-const-correctness) clang-tidy bug
7✔
253

254
            Botan::store_be(in16, out);
7✔
255
            result.test_u8_eq(out[0], 0x12);
7✔
256
            result.test_u8_eq(out[1], 0x34);
7✔
257

258
            Botan::store_le(in16, out);
7✔
259
            result.test_u8_eq(out[0], 0x34);
7✔
260
            result.test_u8_eq(out[1], 0x12);
7✔
261

262
            Botan::store_be(in32, out);
7✔
263
            result.test_u8_eq(out[0], 0xA0);
7✔
264
            result.test_u8_eq(out[1], 0xB0);
7✔
265
            result.test_u8_eq(out[2], 0xC0);
7✔
266
            result.test_u8_eq(out[3], 0xD0);
7✔
267

268
            Botan::store_le(in32, out);
7✔
269
            result.test_u8_eq(out[0], 0xD0);
7✔
270
            result.test_u8_eq(out[1], 0xC0);
7✔
271
            result.test_u8_eq(out[2], 0xB0);
7✔
272
            result.test_u8_eq(out[3], 0xA0);
7✔
273

274
            Botan::store_be(in64, out);
7✔
275
            result.test_u8_eq(out[0], 0xAB);
7✔
276
            result.test_u8_eq(out[1], 0xCD);
7✔
277
            result.test_u8_eq(out[2], 0xEF);
7✔
278
            result.test_u8_eq(out[3], 0x01);
7✔
279
            result.test_u8_eq(out[4], 0x23);
7✔
280
            result.test_u8_eq(out[5], 0x45);
7✔
281
            result.test_u8_eq(out[6], 0x67);
7✔
282
            result.test_u8_eq(out[7], 0x89);
7✔
283

284
            Botan::store_le(in64, out);
7✔
285
            result.test_u8_eq(out[0], 0x89);
7✔
286
            result.test_u8_eq(out[1], 0x67);
7✔
287
            result.test_u8_eq(out[2], 0x45);
7✔
288
            result.test_u8_eq(out[3], 0x23);
7✔
289
            result.test_u8_eq(out[4], 0x01);
7✔
290
            result.test_u8_eq(out[5], 0xEF);
7✔
291
            result.test_u8_eq(out[6], 0xCD);
7✔
292
            result.test_u8_eq(out[7], 0xAB);
7✔
293
         }
294

295
         std::array<uint8_t, 8> outarr{};
1✔
296
         uint16_t i0 = 0;
1✔
297
         uint16_t i1 = 0;
1✔
298
         uint16_t i2 = 0;
1✔
299
         uint16_t i3 = 0;
1✔
300
         Botan::store_be(in64, outarr);
1✔
301

302
         Botan::load_be(outarr, i0, i1, i2, i3);
1✔
303
         result.test_u16_eq(i0, 0xABCD);
1✔
304
         result.test_u16_eq(i1, 0xEF01);
1✔
305
         result.test_u16_eq(i2, 0x2345);
1✔
306
         result.test_u16_eq(i3, 0x6789);
1✔
307

308
         Botan::load_le(std::span{outarr}.first<6>(), i0, i1, i2);
1✔
309
         result.test_u16_eq(i0, 0xCDAB);
1✔
310
         result.test_u16_eq(i1, 0x01EF);
1✔
311
         result.test_u16_eq(i2, 0x4523);
1✔
312
         result.test_u16_eq(i3, 0x6789);  // remains unchanged
1✔
313

314
         Botan::store_le(in64, outarr);
1✔
315

316
         Botan::load_le(outarr, i0, i1, i2, i3);
1✔
317
         result.test_u16_eq(i0, 0x6789);
1✔
318
         result.test_u16_eq(i1, 0x2345);
1✔
319
         result.test_u16_eq(i2, 0xEF01);
1✔
320
         result.test_u16_eq(i3, 0xABCD);
1✔
321

322
         Botan::load_be(std::span{outarr}.first<6>(), i0, i1, i2);
1✔
323
         result.test_u16_eq(i0, 0x8967);
1✔
324
         result.test_u16_eq(i1, 0x4523);
1✔
325
         result.test_u16_eq(i2, 0x01EF);
1✔
326
         result.test_u16_eq(i3, 0xABCD);  // remains unchanged
1✔
327

328
         i0 = 0xAA11;
1✔
329
         i1 = 0xBB22;
1✔
330
         i2 = 0xCC33;
1✔
331
         i3 = 0xDD44;
1✔
332
         Botan::store_be(outarr, i0, i1, i2, i3);
1✔
333
         result.test_bin_eq("store_be", outarr, "AA11BB22CC33DD44");
1✔
334
         std::vector<uint8_t> outvec(8);
1✔
335
         Botan::store_be(outvec, i0, i1, i2, i3);
1✔
336
         result.test_bin_eq("store_be", outvec, "AA11BB22CC33DD44");
1✔
337

338
         Botan::store_le(outarr, i0, i1, i2, i3);
1✔
339
         result.test_bin_eq("store_le(arr)", outarr, "11AA22BB33CC44DD");
1✔
340
         Botan::store_le(outvec, i0, i1, i2, i3);
1✔
341
         result.test_bin_eq("store_le", outvec, "11AA22BB33CC44DD");
1✔
342

343
#if !defined(BOTAN_TERMINATE_ON_ASSERTS)
344
         std::vector<uint8_t> sink56bits(7);
345
         std::vector<uint8_t> sink72bits(9);
346
         result.test_throws("store_le with a buffer that is too small",
347
                            [&] { Botan::store_le(sink56bits, i0, i1, i2, i3); });
348
         result.test_throws("store_le with a buffer that is too big",
349
                            [&] { Botan::store_le(sink72bits, i0, i1, i2, i3); });
350
         result.test_throws("store_be with a buffer that is too small",
351
                            [&] { Botan::store_be(sink56bits, i0, i1, i2, i3); });
352
         result.test_throws("store_be with a buffer that is too big",
353
                            [&] { Botan::store_be(sink72bits, i0, i1, i2, i3); });
354
#endif
355

356
         // can store multiple values straight into a collection
357
         auto out64_array_be = Botan::store_be(i0, i1, i2, i3);
1✔
358
         auto out64_vec_be = Botan::store_be<std::vector<uint8_t>>(i0, i1, i2, i3);
1✔
359
         auto out64_strong_be = Botan::store_be<TestVectorSink>(i0, i1, i2, i3);
1✔
360
         result.test_bin_eq("store_be(arr)", out64_array_be, "AA11BB22CC33DD44");
1✔
361
         result.test_bin_eq("store_be(vec)", out64_vec_be, "AA11BB22CC33DD44");
1✔
362
         result.test_bin_eq("store_be(strong)", out64_strong_be, "AA11BB22CC33DD44");
1✔
363
         auto out64_array_le = Botan::store_le(i0, i1, i2, i3);
1✔
364
         auto out64_vec_le = Botan::store_le<std::vector<uint8_t>>(i0, i1, i2, i3);
1✔
365
         auto out64_strong_le = Botan::store_le<TestVectorSink>(i0, i1, i2, i3);
1✔
366
         result.test_bin_eq("store_le(arr)", out64_array_le, "11AA22BB33CC44DD");
1✔
367
         result.test_bin_eq("store_le(vec)", out64_vec_le, "11AA22BB33CC44DD");
1✔
368
         result.test_bin_eq("store_le(strong)", out64_strong_le, "11AA22BB33CC44DD");
1✔
369

370
         result.test_u16_eq(in16, Botan::load_be(Botan::store_be(in16)));
1✔
371
         result.test_u32_eq(in32, Botan::load_be(Botan::store_be(in32)));
1✔
372
         result.test_u64_eq(in64, Botan::load_be(Botan::store_be(in64)));
1✔
373

374
         result.test_u16_eq(in16, Botan::load_le(Botan::store_le(in16)));
1✔
375
         result.test_u32_eq(in32, Botan::load_le(Botan::store_le(in32)));
1✔
376
         result.test_u64_eq(in64, Botan::load_le(Botan::store_le(in64)));
1✔
377

378
         // Test that the runtime detects incompatible range sizes
379
#if !defined(BOTAN_TERMINATE_ON_ASSERTS)
380
         std::vector<uint16_t> too_big16(4);
381
         std::vector<uint16_t> too_small16(1);
382
         result.test_throws("load_le with incompatible buffers",
383
                            [&] { Botan::load_le(too_big16, Botan::hex_decode("BAADB00B")); });
384
         result.test_throws("load_le with incompatible buffers",
385
                            [&] { Botan::load_le(too_small16, Botan::hex_decode("BAADB00B")); });
386
         result.test_throws("load_be with incompatible buffers",
387
                            [&] { Botan::load_be(too_big16, Botan::hex_decode("BAADB00B")); });
388
         result.test_throws("load_be with incompatible buffers",
389
                            [&] { Botan::load_be(too_small16, Botan::hex_decode("BAADB00B")); });
390

391
         std::vector<uint8_t> too_big8(4);
392
         std::vector<uint8_t> too_small8(1);
393
         result.test_throws("store_le with incompatible buffers",
394
                            [&] { Botan::store_le(too_big8, std::array<uint16_t, 1>{}); });
395
         result.test_throws("store_le with incompatible buffers",
396
                            [&] { Botan::store_le(too_small8, std::array<uint16_t, 1>{}); });
397
         result.test_throws("store_be with incompatible buffers",
398
                            [&] { Botan::store_be(too_big8, std::array<uint16_t, 1>{}); });
399
         result.test_throws("store_be with incompatible buffers",
400
                            [&] { Botan::store_be(too_small8, std::array<uint16_t, 1>{}); });
401
#endif
402

403
         // Test store of entire ranges
404
         const std::array<uint16_t, 2> in16_array = {0x0A0B, 0x0C0D};
1✔
405
         result.test_bin_eq("store_be(vec)", Botan::store_be<std::vector<uint8_t>>(in16_array), "0A0B0C0D");
1✔
406
         result.test_bin_eq("store_le(vec)", Botan::store_le<std::vector<uint8_t>>(in16_array), "0B0A0D0C");
1✔
407

408
         const std::vector<uint16_t> in16_vector = {0x0A0B, 0x0C0D};
1✔
409
         result.test_bin_eq("store_be(vec)", Botan::store_be<std::vector<uint8_t>>(in16_vector), "0A0B0C0D");
1✔
410
         result.test_bin_eq("store_le(vec)", Botan::store_le<std::vector<uint8_t>>(in16_vector), "0B0A0D0C");
1✔
411

412
         std::array<uint8_t, 4> out_array{};
1✔
413
         Botan::store_be(out_array, in16_array);
1✔
414
         result.test_bin_eq("store_be(arr)", out_array, "0A0B0C0D");
1✔
415
         Botan::store_le(out_array, in16_array);
1✔
416
         result.test_bin_eq("store_le(arr)", out_array, "0B0A0D0C");
1✔
417

418
         const auto be_inferred = Botan::store_be(in16_array);
1✔
419
         result.test_bin_eq("store_be(arr)", be_inferred, "0A0B0C0D");
1✔
420
         const auto le_inferred = Botan::store_le(in16_array);
1✔
421
         result.test_bin_eq("store_le(arr)", le_inferred, "0B0A0D0C");
1✔
422

423
         // Test load of entire ranges
424
         const auto in_buffer = Botan::hex_decode("AABBCCDD");
1✔
425
         auto out16_array_be = Botan::load_be<std::array<uint16_t, 2>>(in_buffer);
1✔
426
         result.test_u16_eq(out16_array_be[0], 0xAABB);
1✔
427
         result.test_u16_eq(out16_array_be[1], 0xCCDD);
1✔
428
         auto out16_vec_be = Botan::load_be<std::vector<uint16_t>>(in_buffer);
1✔
429
         result.test_sz_eq("be-vector has expected size", out16_vec_be.size(), 2);
1✔
430
         result.test_u16_eq(out16_vec_be[0], 0xAABB);
1✔
431
         result.test_u16_eq(out16_vec_be[1], 0xCCDD);
1✔
432

433
         auto out16_array_le = Botan::load_le<std::array<uint16_t, 2>>(in_buffer);
1✔
434
         result.test_u16_eq(out16_array_le[0], 0xBBAA);
1✔
435
         result.test_u16_eq(out16_array_le[1], 0xDDCC);
1✔
436
         auto out16_vec_le = Botan::load_le<Botan::secure_vector<uint16_t>>(in_buffer);
1✔
437
         result.test_sz_eq("le-vector has expected size", out16_vec_be.size(), 2);
1✔
438
         result.test_u16_eq(out16_vec_le[0], 0xBBAA);
1✔
439
         result.test_u16_eq(out16_vec_le[1], 0xDDCC);
1✔
440

441
         // Test loading/storing of strong type integers
442
         const TestInt64 in64_strong{0xABCDEF0123456789};
1✔
443
         const TestInt32 in32_strong{0xABCDEF01};
1✔
444

445
         result.test_bin_eq(
1✔
446
            "store_be(u64,strong)", Botan::store_be<std::vector<uint8_t>>(in64_strong), "ABCDEF0123456789");
×
447
         result.test_bin_eq(
1✔
448
            "store_le(u64,strong)", Botan::store_le<std::vector<uint8_t>>(in64_strong), "8967452301EFCDAB");
×
449
         result.test_bin_eq("store_be(u32,strong)", Botan::store_be<std::vector<uint8_t>>(in32_strong), "ABCDEF01");
1✔
450
         result.test_bin_eq("store_le(u32,strong)", Botan::store_le<std::vector<uint8_t>>(in32_strong), "01EFCDAB");
1✔
451

452
         test_arb_eq(
2✔
453
            result, "load_be(strong64)", Botan::load_be<TestInt64>(Botan::hex_decode("ABCDEF0123456789")), in64_strong);
2✔
454
         test_arb_eq(
2✔
455
            result, "load_le(strong64)", Botan::load_le<TestInt64>(Botan::hex_decode("8967452301EFCDAB")), in64_strong);
2✔
456
         test_arb_eq(
2✔
457
            result, "load_be(strong32)", Botan::load_be<TestInt32>(Botan::hex_decode("ABCDEF01")), in32_strong);
2✔
458
         test_arb_eq(
2✔
459
            result, "load_le(strong32)", Botan::load_le<TestInt32>(Botan::hex_decode("01EFCDAB")), in32_strong);
2✔
460

461
         const std::vector<TestInt64> some_in64_strongs{TestInt64{0xABCDEF0123456789}, TestInt64{0x0123456789ABCDEF}};
1✔
462
         result.test_bin_eq("store_be(vector,strong)",
1✔
463
                            Botan::store_be<std::vector<uint8_t>>(some_in64_strongs),
×
464
                            "ABCDEF01234567890123456789ABCDEF");
465
         result.test_bin_eq("store_le(vector,strong)",
1✔
466
                            Botan::store_le<std::vector<uint8_t>>(some_in64_strongs),
×
467
                            "8967452301EFCDABEFCDAB8967452301");
468

469
         const auto in64_strongs_le =
1✔
470
            Botan::load_le<std::array<TestInt64, 2>>(Botan::hex_decode("8967452301EFCDABEFCDAB8967452301"));
2✔
471
         test_arb_eq(result, "load_le(strong arr)", in64_strongs_le[0], TestInt64{0xABCDEF0123456789});
1✔
472
         test_arb_eq(result, "load_le(strong arr)", in64_strongs_le[1], TestInt64{0x0123456789ABCDEF});
1✔
473

474
         const auto in64_strongs_be =
1✔
475
            Botan::load_be<std::vector<TestInt64>>(Botan::hex_decode("ABCDEF01234567890123456789ABCDEF"));
2✔
476
         test_arb_eq(result, "load_be(strong arr)", in64_strongs_be[0], TestInt64{0xABCDEF0123456789});
1✔
477
         test_arb_eq(result, "load_be(strong arr)", in64_strongs_be[1], TestInt64{0x0123456789ABCDEF});
1✔
478

479
         // Test loading/storing of enum types with different endianness
480
         const auto in64_enum_le = Botan::load_le<TestEnum64>(Botan::hex_decode("1234567890ABCDEF"));
2✔
481
         result.test_enum_eq("load_le(enum64)", in64_enum_le, TestEnum64::_2);
1✔
482
         const auto in64_enum_be = Botan::load_be<TestEnum64>(Botan::hex_decode("1234567890ABCDEF"));
2✔
483
         result.test_enum_eq("load_be(enum64)", in64_enum_be, TestEnum64::_1);
1✔
484
         result.test_bin_eq(
1✔
485
            "store_be(enum64)", Botan::store_le<std::vector<uint8_t>>(TestEnum64::_1), "EFCDAB9078563412");
1✔
486
         result.test_bin_eq("store_be(enum64)", Botan::store_be(TestEnum64::_2), "EFCDAB9078563412");
1✔
487

488
         const auto in32_enum_le = Botan::load_le<TestEnum32>(Botan::hex_decode("78563412"));
2✔
489
         result.test_enum_eq("load_le(enum32)", in32_enum_le, TestEnum32::_1);
1✔
490
         const auto in32_enum_be = Botan::load_be<TestEnum32>(Botan::hex_decode("78563412"));
2✔
491
         result.test_enum_eq("load_be(enum32)", in32_enum_be, TestEnum32::_2);
1✔
492
         result.test_bin_eq("store_le(enum32)", Botan::store_le<std::vector<uint8_t>>(TestEnum32::_1), "78563412");
1✔
493
         result.test_bin_eq("store_be(enum32)", Botan::store_be(TestEnum32::_2), "78563412");
1✔
494

495
         return result;
2✔
496
      }
10✔
497

498
      template <std::unsigned_integral T>
499
      static T fb_load_be(std::array<const uint8_t, sizeof(T)> in) {
3✔
500
         return Botan::detail::fallback_load_any<std::endian::big, T>(in);
1✔
501
      }
502

503
      template <std::unsigned_integral T>
504
      static T fb_load_le(std::array<const uint8_t, sizeof(T)> in) {
3✔
505
         return Botan::detail::fallback_load_any<std::endian::little, T>(in);
1✔
506
      }
507

508
      template <std::unsigned_integral T>
509
      static decltype(auto) fb_store_be(const T in) {
3✔
510
         std::array<uint8_t, sizeof(T)> out{};
2✔
511
         Botan::detail::fallback_store_any<std::endian::big, T>(in, out);
1✔
512
         return out;
2✔
513
      }
514

515
      template <std::unsigned_integral T>
516
      static decltype(auto) fb_store_le(const T in) {
3✔
517
         std::array<uint8_t, sizeof(T)> out{};
2✔
518
         Botan::detail::fallback_store_any<std::endian::little, T>(in, out);
1✔
519
         return out;
2✔
520
      }
521

522
      template <size_t N>
523
      using a = std::array<uint8_t, N>;
524

525
      static Test::Result test_loadstore_ambiguity() {
1✔
526
         // This is a regression test for a (probable) compiler bug in Xcode 15
527
         // where it would fail to compile the load/store functions for size_t
528
         //
529
         // It seems that this platform defines uint64_t as "unsigned long long"
530
         // and size_t as "unsigned long". Both are 64-bits but the compiler
531
         // was unable to disambiguate the two in reverse_bytes in bswap.h
532

533
         const uint32_t in32 = 0x01234567;
1✔
534
         const uint64_t in64 = 0x0123456789ABCDEF;
1✔
535
         const size_t inszt = 0x87654321;
1✔
536

537
         Test::Result result("Util load/store ambiguity");
1✔
538
         const auto out_be_32 = Botan::store_be(in32);
1✔
539
         const auto out_le_32 = Botan::store_le(in32);
1✔
540
         const auto out_be_64 = Botan::store_be(in64);
1✔
541
         const auto out_le_64 = Botan::store_le(in64);
1✔
542
         const auto out_be_szt = Botan::store_be(inszt);
1✔
543
         const auto out_le_szt = Botan::store_le(inszt);
1✔
544

545
         result.test_u32_eq("be 32", Botan::load_be<uint32_t>(out_be_32), in32);
1✔
546
         result.test_u32_eq("le 32", Botan::load_le<uint32_t>(out_le_32), in32);
1✔
547
         result.test_u64_eq("be 64", Botan::load_be<uint64_t>(out_be_64), in64);
1✔
548
         result.test_u64_eq("le 64", Botan::load_le<uint64_t>(out_le_64), in64);
1✔
549
         result.test_sz_eq("be sz", Botan::load_be<size_t>(out_be_szt), inszt);
1✔
550
         result.test_sz_eq("le sz", Botan::load_le<size_t>(out_le_szt), inszt);
1✔
551

552
         return result;
1✔
553
      }
×
554

555
      static Test::Result test_loadstore_fallback() {
1✔
556
         // The fallback implementation is only used if we don't know the
557
         // endianness of the target at compile time. This makes sure that the
558
         // fallback implementation is correct. On all typical platforms it
559
         // won't be called in production.
560
         Test::Result result("Util load/store fallback");
1✔
561

562
         result.test_u16_eq("lLE 16", fb_load_le<uint16_t>({1, 2}), 0x0201);
1✔
563
         result.test_u32_eq("lLE 32", fb_load_le<uint32_t>({1, 2, 3, 4}), 0x04030201);
1✔
564
         result.test_u64_eq("lLE 64", fb_load_le<uint64_t>({1, 2, 3, 4, 5, 6, 7, 8}), 0x0807060504030201);
1✔
565

566
         result.test_u16_eq("lBE 16", fb_load_be<uint16_t>({1, 2}), 0x0102);
1✔
567
         result.test_u32_eq("lBE 32", fb_load_be<uint32_t>({1, 2, 3, 4}), 0x01020304);
1✔
568
         result.test_u64_eq("lBE 64", fb_load_be<uint64_t>({1, 2, 3, 4, 5, 6, 7, 8}), 0x0102030405060708);
1✔
569

570
         result.test_bin_eq("sLE 16", fb_store_le<uint16_t>(0x0201), "0102");
1✔
571
         result.test_bin_eq("sLE 32", fb_store_le<uint32_t>(0x04030201), "01020304");
1✔
572
         result.test_bin_eq("sLE 64", fb_store_le<uint64_t>(0x0807060504030201), "0102030405060708");
1✔
573

574
         result.test_bin_eq("sBE 16", fb_store_be<uint16_t>(0x0102), "0102");
1✔
575
         result.test_bin_eq("sBE 32", fb_store_be<uint32_t>(0x01020304), "01020304");
1✔
576
         result.test_bin_eq("sBE 64", fb_store_be<uint64_t>(0x0102030405060708), "0102030405060708");
1✔
577

578
         return result;
1✔
579
      }
×
580

581
      static Test::Result test_loadstore_constexpr() {
1✔
582
         Test::Result result("Util load/store constexpr");
1✔
583

584
         constexpr uint16_t in16 = 0x1234;
1✔
585
         constexpr uint32_t in32 = 0xA0B0C0D0;
1✔
586
         constexpr uint64_t in64 = 0xABCDEF0123456789;
1✔
587

588
         // clang-format off
589
         constexpr std::array<uint8_t, 16> cex_mem = {
1✔
590
            0x00, 0x11, 0x22, 0x33, 0x44, 0x55, 0x66, 0x77, 0x88, 0x99, 0xAA, 0xBB, 0xCC, 0xDD, 0xEE, 0xFF,
591
         };
592
         // clang-format on
593

594
         // get_byte<> w/ 16bit
595
         constexpr auto cex_byte_16_0 = Botan::get_byte<0>(in16);
1✔
596
         result.test_u8_eq(cex_byte_16_0, 0x12);
1✔
597
         constexpr auto cex_byte_16_1 = Botan::get_byte<1>(in16);
1✔
598
         result.test_u8_eq(cex_byte_16_1, 0x34);
1✔
599

600
         // get_byte<> w/ 32bit
601
         constexpr auto cex_byte_32_0 = Botan::get_byte<0>(in32);
1✔
602
         result.test_u8_eq(cex_byte_32_0, 0xA0);
1✔
603
         constexpr auto cex_byte_32_1 = Botan::get_byte<1>(in32);
1✔
604
         result.test_u8_eq(cex_byte_32_1, 0xB0);
1✔
605
         constexpr auto cex_byte_32_2 = Botan::get_byte<2>(in32);
1✔
606
         result.test_u8_eq(cex_byte_32_2, 0xC0);
1✔
607
         constexpr auto cex_byte_32_3 = Botan::get_byte<3>(in32);
1✔
608
         result.test_u8_eq(cex_byte_32_3, 0xD0);
1✔
609

610
         // get_byte<> w/ 64bit
611
         constexpr auto cex_byte_64_0 = Botan::get_byte<0>(in64);
1✔
612
         result.test_u8_eq(cex_byte_64_0, 0xAB);
1✔
613
         constexpr auto cex_byte_64_1 = Botan::get_byte<1>(in64);
1✔
614
         result.test_u8_eq(cex_byte_64_1, 0xCD);
1✔
615
         constexpr auto cex_byte_64_2 = Botan::get_byte<2>(in64);
1✔
616
         result.test_u8_eq(cex_byte_64_2, 0xEF);
1✔
617
         constexpr auto cex_byte_64_3 = Botan::get_byte<3>(in64);
1✔
618
         result.test_u8_eq(cex_byte_64_3, 0x01);
1✔
619
         constexpr auto cex_byte_64_4 = Botan::get_byte<4>(in64);
1✔
620
         result.test_u8_eq(cex_byte_64_4, 0x23);
1✔
621
         constexpr auto cex_byte_64_5 = Botan::get_byte<5>(in64);
1✔
622
         result.test_u8_eq(cex_byte_64_5, 0x45);
1✔
623
         constexpr auto cex_byte_64_6 = Botan::get_byte<6>(in64);
1✔
624
         result.test_u8_eq(cex_byte_64_6, 0x67);
1✔
625
         constexpr auto cex_byte_64_7 = Botan::get_byte<7>(in64);
1✔
626
         result.test_u8_eq(cex_byte_64_7, 0x89);
1✔
627

628
         // make_uintXX()
629
         constexpr auto cex_uint16_t = Botan::make_uint16(0x12, 0x34);
1✔
630
         result.test_u16_eq(cex_uint16_t, in16);
1✔
631
         constexpr auto cex_uint32_t = Botan::make_uint32(0xA0, 0xB0, 0xC0, 0xD0);
1✔
632
         result.test_u32_eq(cex_uint32_t, in32);
1✔
633
         constexpr auto cex_uint64_t = Botan::make_uint64(0xAB, 0xCD, 0xEF, 0x01, 0x23, 0x45, 0x67, 0x89);
1✔
634
         result.test_u64_eq(cex_uint64_t, in64);
1✔
635

636
         // store_le/be with a single integer
637
         constexpr std::array<uint8_t, 2> cex_store_le16 = Botan::store_le(in16);
1✔
638
         result.test_bin_eq("store_le(u16 arr)", cex_store_le16, "3412");
1✔
639
         constexpr std::array<uint8_t, 4> cex_store_le32 = Botan::store_le(in32);
1✔
640
         result.test_bin_eq("store_le(u32 arr)", cex_store_le32, "D0C0B0A0");
1✔
641
         constexpr std::array<uint8_t, 8> cex_store_le64 = Botan::store_le(in64);
1✔
642
         result.test_bin_eq("store_le(u32,arr)", cex_store_le64, "8967452301EFCDAB");
1✔
643

644
         constexpr std::array<uint8_t, 2> cex_store_be16 = Botan::store_be(in16);
1✔
645
         result.test_bin_eq("store_be(u16 arr)", cex_store_be16, "1234");
1✔
646
         constexpr std::array<uint8_t, 4> cex_store_be32 = Botan::store_be(in32);
1✔
647
         result.test_bin_eq("store_be(u32 arr)", cex_store_be32, "A0B0C0D0");
1✔
648
         constexpr std::array<uint8_t, 8> cex_store_be64 = Botan::store_be(in64);
1✔
649
         result.test_bin_eq("store_be(u64 arr)", cex_store_be64, "ABCDEF0123456789");
1✔
650

651
         // store_le/be with multiple integers, both as a parameter pack and a range (std::array for constexpr)
652
         constexpr std::array<uint8_t, 16> cex_store_le16s =
1✔
653
            Botan::store_le(in16, in16, in16, in16, in16, in16, in16, in16);
654
         constexpr std::array<uint8_t, 16> cex_store_le16s2 =
1✔
655
            Botan::store_le(std::array{in16, in16, in16, in16, in16, in16, in16, in16});
656
         result.test_bin_eq("store_le", cex_store_le16s, "34123412341234123412341234123412");
1✔
657
         result.test_bin_eq("cex_store_le16s", cex_store_le16s, cex_store_le16s2);
1✔
658
         constexpr std::array<uint8_t, 16> cex_store_le32s = Botan::store_le(in32, in32, in32, in32);
1✔
659
         constexpr std::array<uint8_t, 16> cex_store_le32s2 = Botan::store_le(std::array{in32, in32, in32, in32});
1✔
660
         result.test_bin_eq("cex_store_le32s", cex_store_le32s, "D0C0B0A0D0C0B0A0D0C0B0A0D0C0B0A0");
1✔
661
         result.test_bin_eq("cex_store_le32s2", cex_store_le32s, cex_store_le32s2);
1✔
662
         constexpr std::array<uint8_t, 16> cex_store_le64s = Botan::store_le(in64, in64);
1✔
663
         constexpr std::array<uint8_t, 16> cex_store_le64s2 = Botan::store_le(std::array{in64, in64});
1✔
664
         result.test_bin_eq("cex_store_le64s", cex_store_le64s, "8967452301EFCDAB8967452301EFCDAB");
1✔
665
         result.test_bin_eq("cex_store_le64s2", cex_store_le64s, cex_store_le64s2);
1✔
666

667
         constexpr std::array<uint8_t, 16> cex_store_be16s =
1✔
668
            Botan::store_be(in16, in16, in16, in16, in16, in16, in16, in16);
669
         constexpr std::array<uint8_t, 16> cex_store_be16s2 =
1✔
670
            Botan::store_be(std::array{in16, in16, in16, in16, in16, in16, in16, in16});
671
         result.test_bin_eq("cex_store_be16s", cex_store_be16s, "12341234123412341234123412341234");
1✔
672
         result.test_bin_eq("cex_store_be16s2", cex_store_be16s, cex_store_be16s2);
1✔
673
         constexpr std::array<uint8_t, 16> cex_store_be32s = Botan::store_be(in32, in32, in32, in32);
1✔
674
         constexpr std::array<uint8_t, 16> cex_store_be32s2 = Botan::store_be(std::array{in32, in32, in32, in32});
1✔
675
         result.test_bin_eq("cex_store_be32s", cex_store_be32s, "A0B0C0D0A0B0C0D0A0B0C0D0A0B0C0D0");
1✔
676
         result.test_bin_eq("cex_store_be32s2", cex_store_be32s, cex_store_be32s2);
1✔
677
         constexpr std::array<uint8_t, 16> cex_store_be64s = Botan::store_be(in64, in64);
1✔
678
         constexpr std::array<uint8_t, 16> cex_store_be64s2 = Botan::store_be(std::array{in64, in64});
1✔
679
         result.test_bin_eq("cex_store_be64s", cex_store_be64s, "ABCDEF0123456789ABCDEF0123456789");
1✔
680
         result.test_bin_eq("cex_store_be64s2", cex_store_be64s, cex_store_be64s2);
1✔
681

682
         // load_le/be a single integer
683
         constexpr uint16_t cex_load_le16 = Botan::load_le<uint16_t>(cex_store_le16);
1✔
684
         result.test_u16_eq(cex_load_le16, in16);
1✔
685
         constexpr uint32_t cex_load_le32 = Botan::load_le<uint32_t>(cex_store_le32);
1✔
686
         result.test_u32_eq(cex_load_le32, in32);
1✔
687
         constexpr uint64_t cex_load_le64 = Botan::load_le<uint64_t>(cex_store_le64);
1✔
688
         result.test_u64_eq(cex_load_le64, in64);
1✔
689

690
         constexpr uint16_t cex_load_be16 = Botan::load_be<uint16_t>(cex_store_be16);
1✔
691
         result.test_u16_eq(cex_load_be16, in16);
1✔
692
         constexpr uint32_t cex_load_be32 = Botan::load_be<uint32_t>(cex_store_be32);
1✔
693
         result.test_u32_eq(cex_load_be32, in32);
1✔
694
         constexpr uint64_t cex_load_be64 = Botan::load_be<uint64_t>(cex_store_be64);
1✔
695
         result.test_u64_eq(cex_load_be64, in64);
1✔
696

697
         // load_le/be multiple integers into a std::array for constexpr
698
         constexpr auto cex_load_le16s = Botan::load_le<std::array<uint16_t, cex_mem.size() / 2>>(cex_mem);
1✔
699
         test_arb_eq(result,
1✔
700
                     "constexpr load_le(u16)",
701
                     cex_load_le16s,
702
                     {0x1100, 0x3322, 0x5544, 0x7766, 0x9988, 0xBBAA, 0xDDCC, 0xFFEE});
703
         constexpr auto cex_load_le32s = Botan::load_le<std::array<uint32_t, cex_mem.size() / 4>>(cex_mem);
1✔
704
         test_arb_eq(
1✔
705
            result, "constexpr load_le(u32)", cex_load_le32s, {0x33221100, 0x77665544, 0xBBAA9988, 0xFFEEDDCC});
706
         constexpr auto cex_load_le64s = Botan::load_le<std::array<uint64_t, cex_mem.size() / 8>>(cex_mem);
1✔
707
         test_arb_eq(result, "constexpr load_le(u64)", cex_load_le64s, {0x7766554433221100, 0xFFEEDDCCBBAA9988});
1✔
708

709
         constexpr auto cex_load_be16s = Botan::load_be<std::array<uint16_t, cex_mem.size() / 2>>(cex_mem);
1✔
710
         test_arb_eq(result,
1✔
711
                     "constexpr load_be(u16)",
712
                     cex_load_be16s,
713
                     {0x0011, 0x2233, 0x4455, 0x6677, 0x8899, 0xAABB, 0xCCDD, 0xEEFF});
714
         constexpr auto cex_load_be32s = Botan::load_be<std::array<uint32_t, cex_mem.size() / 4>>(cex_mem);
1✔
715
         test_arb_eq(
1✔
716
            result, "constexpr load_be(u32)", cex_load_be32s, {0x00112233, 0x44556677, 0x8899AABB, 0xCCDDEEFF});
717
         constexpr auto cex_load_be64s = Botan::load_be<std::array<uint64_t, cex_mem.size() / 8>>(cex_mem);
1✔
718
         test_arb_eq(result, "constexpr load_be(u64)", cex_load_be64s, {0x0011223344556677, 0x8899AABBCCDDEEFF});
1✔
719

720
         return result;
1✔
721
      }
×
722

723
      static std::vector<Test::Result> test_copy_out_be_le() {
1✔
724
         return {
1✔
725
            CHECK("copy_out_be with 16bit input (word aligned)",
726
                  [&](auto& result) {
1✔
727
                     std::vector<uint8_t> out_vector(4);
1✔
728
                     const std::array<uint16_t, 2> in_array = {0x0A0B, 0x0C0D};
1✔
729
                     Botan::copy_out_be(out_vector, in_array);
1✔
730
                     result.test_bin_eq("copy_out_be", out_vector, "0A0B0C0D");
1✔
731
                  }),
1✔
732

733
            CHECK("copy_out_be with 16bit input (partial words)",
734
                  [&](auto& result) {
1✔
735
                     std::vector<uint8_t> out_vector(3);
1✔
736
                     const std::array<uint16_t, 2> in_array = {0x0A0B, 0x0C0D};
1✔
737
                     Botan::copy_out_be(out_vector, in_array);
1✔
738
                     result.test_bin_eq("copy_out_be(u16)", out_vector, "0A0B0C");
1✔
739
                  }),
1✔
740

741
            CHECK("copy_out_le with 16bit input (word aligned)",
742
                  [&](auto& result) {
1✔
743
                     std::vector<uint8_t> out_vector(4);
1✔
744
                     const std::array<uint16_t, 2> in_array = {0x0A0B, 0x0C0D};
1✔
745
                     Botan::copy_out_le(out_vector, in_array);
1✔
746
                     result.test_bin_eq("copy_out_le(u16)", out_vector, "0B0A0D0C");
1✔
747
                  }),
1✔
748

749
            CHECK("copy_out_le with 16bit input (partial words)",
750
                  [&](auto& result) {
1✔
751
                     std::vector<uint8_t> out_vector(3);
1✔
752
                     const std::array<uint16_t, 2> in_array = {0x0A0B, 0x0C0D};
1✔
753
                     Botan::copy_out_le(out_vector, in_array);
1✔
754
                     result.test_bin_eq("copy_out_le(u16)", out_vector, "0B0A0D");
1✔
755
                  }),
1✔
756

757
            CHECK("copy_out_be with 64bit input (word aligned)",
758
                  [&](auto& result) {
1✔
759
                     std::vector<uint8_t> out_vector(16);
1✔
760
                     const std::array<uint64_t, 2> in_array = {0x0A0B0C0D0E0F1011, 0x1213141516171819};
1✔
761
                     Botan::copy_out_be(out_vector, in_array);
1✔
762
                     result.test_bin_eq("copy_out_be(u64)", out_vector, "0A0B0C0D0E0F10111213141516171819");
1✔
763
                  }),
1✔
764

765
            CHECK("copy_out_le with 64bit input (word aligned)",
766
                  [&](auto& result) {
1✔
767
                     std::vector<uint8_t> out_vector(16);
1✔
768
                     const std::array<uint64_t, 2> in_array = {0x0A0B0C0D0E0F1011, 0x1213141516171819};
1✔
769
                     Botan::copy_out_le(out_vector, in_array);
1✔
770
                     result.test_bin_eq("copy_out_le(u64)", out_vector, "11100F0E0D0C0B0A1918171615141312");
1✔
771
                  }),
1✔
772

773
            CHECK("copy_out_be with 64bit input (partial words)",
774
                  [&](auto& result) {
1✔
775
                     std::vector<uint8_t> out_vector(15);
1✔
776
                     const std::array<uint64_t, 2> in_array = {0x0A0B0C0D0E0F1011, 0x1213141516171819};
1✔
777
                     Botan::copy_out_be(out_vector, in_array);
1✔
778
                     result.test_bin_eq("copy_out_be(u64)", out_vector, "0A0B0C0D0E0F101112131415161718");
1✔
779
                  }),
1✔
780

781
            CHECK("copy_out_le with 64bit input (partial words)",
782
                  [&](auto& result) {
1✔
783
                     std::vector<uint8_t> out_vector(15);
1✔
784
                     const std::array<uint64_t, 2> in_array = {0x0A0B0C0D0E0F1011, 0x1213141516171819};
1✔
785
                     Botan::copy_out_le(out_vector, in_array);
1✔
786
                     result.test_bin_eq("copy_out_le(u64)", out_vector, "11100F0E0D0C0B0A19181716151413");
1✔
787
                  }),
1✔
788
         };
9✔
789
      }
1✔
790
};
791

792
BOTAN_REGISTER_SMOKE_TEST("utils", "util", Utility_Function_Tests);
793

794
class BitOps_Tests final : public Test {
1✔
795
   public:
796
      std::vector<Test::Result> run() override {
1✔
797
         std::vector<Test::Result> results;
1✔
798

799
         results.push_back(test_power_of_2());
2✔
800
         results.push_back(test_ctz());
2✔
801
         results.push_back(test_sig_bytes());
2✔
802
         results.push_back(test_popcount());
2✔
803
         results.push_back(test_reverse_bits());
2✔
804

805
         return results;
1✔
806
      }
×
807

808
   private:
809
      template <typename T>
810
      void test_ctz(Test::Result& result, T val, size_t expected) {
6✔
811
         Botan::CT::poison(val);
812
         const size_t computed = Botan::ctz<T>(val);
6✔
813
         Botan::CT::unpoison_all(computed, val);
814
         result.test_sz_eq("ctz(" + std::to_string(val) + ")", computed, expected);
24✔
815
      }
6✔
816

817
      Test::Result test_ctz() {
1✔
818
         Test::Result result("ctz");
1✔
819
         test_ctz<uint32_t>(result, 0, 32);
1✔
820
         test_ctz<uint32_t>(result, 1, 0);
1✔
821
         test_ctz<uint32_t>(result, 0x80, 7);
1✔
822
         test_ctz<uint32_t>(result, 0x8000000, 27);
1✔
823
         test_ctz<uint32_t>(result, 0x8100000, 20);
1✔
824
         test_ctz<uint32_t>(result, 0x80000000, 31);
1✔
825

826
         return result;
1✔
827
      }
×
828

829
      template <typename T>
830
      void test_sig_bytes(Test::Result& result, T val, size_t expected) {
14✔
831
         Botan::CT::poison(val);
832
         const size_t computed = Botan::significant_bytes<T>(val);
14✔
833
         Botan::CT::unpoison_all(computed, val);
834
         result.test_sz_eq("significant_bytes(" + std::to_string(val) + ")", computed, expected);
56✔
835
      }
14✔
836

837
      Test::Result test_sig_bytes() {
1✔
838
         Test::Result result("significant_bytes");
1✔
839
         test_sig_bytes<uint32_t>(result, 0, 0);
1✔
840
         test_sig_bytes<uint32_t>(result, 1, 1);
1✔
841
         test_sig_bytes<uint32_t>(result, 0x80, 1);
1✔
842
         test_sig_bytes<uint32_t>(result, 255, 1);
1✔
843
         test_sig_bytes<uint32_t>(result, 256, 2);
1✔
844
         test_sig_bytes<uint32_t>(result, 65535, 2);
1✔
845
         test_sig_bytes<uint32_t>(result, 65536, 3);
1✔
846
         test_sig_bytes<uint32_t>(result, 0x80000000, 4);
1✔
847

848
         test_sig_bytes<uint64_t>(result, 0, 0);
1✔
849
         test_sig_bytes<uint64_t>(result, 1, 1);
1✔
850
         test_sig_bytes<uint64_t>(result, 0x80, 1);
1✔
851
         test_sig_bytes<uint64_t>(result, 256, 2);
1✔
852
         test_sig_bytes<uint64_t>(result, 0x80000000, 4);
1✔
853
         test_sig_bytes<uint64_t>(result, 0x100000000, 5);
1✔
854

855
         return result;
1✔
856
      }
×
857

858
      template <typename T>
859
      void test_power_of_2(Test::Result& result, T val, bool expected) {
15✔
860
         result.test_bool_eq("power_of_2(" + std::to_string(val) + ")", Botan::is_power_of_2<T>(val), expected);
75✔
861
      }
15✔
862

863
      Test::Result test_power_of_2() {
1✔
864
         Test::Result result("is_power_of_2");
1✔
865

866
         test_power_of_2<uint32_t>(result, 0, false);
1✔
867
         test_power_of_2<uint32_t>(result, 1, false);
1✔
868
         test_power_of_2<uint32_t>(result, 2, true);
1✔
869
         test_power_of_2<uint32_t>(result, 3, false);
1✔
870
         test_power_of_2<uint32_t>(result, 0x8000, true);
1✔
871
         test_power_of_2<uint32_t>(result, 0x8001, false);
1✔
872
         test_power_of_2<uint32_t>(result, 0x8000000, true);
1✔
873

874
         test_power_of_2<uint64_t>(result, 0, false);
1✔
875
         test_power_of_2<uint64_t>(result, 1, false);
1✔
876
         test_power_of_2<uint64_t>(result, 2, true);
1✔
877
         test_power_of_2<uint64_t>(result, 3, false);
1✔
878
         test_power_of_2<uint64_t>(result, 0x8000, true);
1✔
879
         test_power_of_2<uint64_t>(result, 0x8001, false);
1✔
880
         test_power_of_2<uint64_t>(result, 0x8000000, true);
1✔
881
         test_power_of_2<uint64_t>(result, 0x100000000000, true);
1✔
882

883
         return result;
1✔
884
      }
×
885

886
      template <typename T>
887
      auto pc(T val) -> decltype(Botan::ct_popcount(val)) {
2✔
888
         return Botan::ct_popcount(val);
6✔
889
      }
890

891
      template <typename T>
892
      auto random_pc(Test::Result& result) {
4✔
893
         auto n = Botan::load_le<T>(Test::rng().random_array<sizeof(T)>());
4✔
894
         result.test_sz_eq(Botan::fmt("popcount({}) == {}", n, std::popcount(n)), pc(n), std::popcount(n));
4✔
895
      }
4✔
896

897
      Test::Result test_popcount() {
1✔
898
         Test::Result result("popcount");
1✔
899

900
         result.test_u8_eq("popcount<uint8_t>(0)", pc<uint8_t>(0), 0);
1✔
901
         result.test_u8_eq("popcount<uint16_t>(0)", pc<uint16_t>(0), 0);
1✔
902
         result.test_u8_eq("popcount<uint32_t>(0)", pc<uint32_t>(0), 0);
1✔
903
         result.test_u8_eq("popcount<uint64_t>(0)", pc<uint64_t>(0), 0);
1✔
904

905
         result.test_u8_eq("popcount<uint8_t>(1)", pc<uint8_t>(1), 1);
1✔
906
         result.test_u8_eq("popcount<uint16_t>(1)", pc<uint16_t>(1), 1);
1✔
907
         result.test_u8_eq("popcount<uint32_t>(1)", pc<uint32_t>(1), 1);
1✔
908
         result.test_u8_eq("popcount<uint64_t>(1)", pc<uint64_t>(1), 1);
1✔
909

910
         result.test_u8_eq("popcount<uint8_t>(0xAA)", pc<uint8_t>(0xAA), 4);
1✔
911
         result.test_u8_eq("popcount<uint16_t>(0xAAAA)", pc<uint16_t>(0xAAAA), 8);
1✔
912
         result.test_u8_eq("popcount<uint32_t>(0xAAAA...)", pc<uint32_t>(0xAAAAAAAA), 16);
1✔
913
         result.test_u8_eq("popcount<uint64_t>(0xAAAA...)", pc<uint64_t>(0xAAAAAAAAAAAAAAAA), 32);
1✔
914

915
         result.test_u8_eq("popcount<uint8_t>(0xFF)", pc<uint8_t>(0xFF), 8);
1✔
916
         result.test_u8_eq("popcount<uint16_t>(0xFFFF)", pc<uint16_t>(0xFFFF), 16);
1✔
917
         result.test_u8_eq("popcount<uint32_t>(0xFFFF...)", pc<uint32_t>(0xFFFFFFFF), 32);
1✔
918
         result.test_u8_eq("popcount<uint64_t>(0xFFFF...)", pc<uint64_t>(0xFFFFFFFFFFFFFFFF), 64);
1✔
919

920
         random_pc<uint8_t>(result);
1✔
921
         random_pc<uint16_t>(result);
1✔
922
         random_pc<uint32_t>(result);
1✔
923
         random_pc<uint64_t>(result);
1✔
924

925
         return result;
1✔
926
      }
×
927

928
      Test::Result test_reverse_bits() {
1✔
929
         Test::Result result("reverse_bits");
1✔
930

931
         result.test_u8_eq("rev(0u8)", Botan::ct_reverse_bits<uint8_t>(0b00000000), 0b00000000);
1✔
932
         result.test_u8_eq("rev(1u8)", Botan::ct_reverse_bits<uint8_t>(0b01010101), 0b10101010);
1✔
933
         result.test_u8_eq("rev(2u8)", Botan::ct_reverse_bits<uint8_t>(0b01001011), 0b11010010);
1✔
934

935
         result.test_u16_eq("rev(0u16)", Botan::ct_reverse_bits<uint16_t>(0b0000000000000000), 0b0000000000000000);
1✔
936
         result.test_u16_eq("rev(1u16)", Botan::ct_reverse_bits<uint16_t>(0b0101010101010101), 0b1010101010101010);
1✔
937
         result.test_u16_eq("rev(2u16)", Botan::ct_reverse_bits<uint16_t>(0b0100101101011010), 0b0101101011010010);
1✔
938

939
         result.test_u32_eq("rev(0u32)", Botan::ct_reverse_bits<uint32_t>(0xFFFFFFFF), 0xFFFFFFFF);
1✔
940
         result.test_u32_eq("rev(1u32)", Botan::ct_reverse_bits<uint32_t>(0x55555555), 0xAAAAAAAA);
1✔
941
         result.test_u32_eq("rev(2u32)", Botan::ct_reverse_bits<uint32_t>(0x4B6A2C1D), 0xB83456D2);
1✔
942

943
         result.test_u64_eq("rev(0u64)", Botan::ct_reverse_bits<uint64_t>(0xF0E0D0C005040302), 0x40C020A0030B070F);
1✔
944
         result.test_u64_eq("rev(1u64)", Botan::ct_reverse_bits<uint64_t>(0x5555555555555555), 0xAAAAAAAAAAAAAAAA);
1✔
945
         result.test_u64_eq("rev(2u64)", Botan::ct_reverse_bits<uint64_t>(0x4B6A2C1D5E7F8A90), 0x951FE7AB83456D2);
1✔
946

947
         return result;
1✔
948
      }
×
949
};
950

951
BOTAN_REGISTER_TEST("utils", "bit_ops", BitOps_Tests);
952

953
#if defined(BOTAN_HAS_POLY_DBL)
954

955
class Poly_Double_Tests final : public Text_Based_Test {
×
956
   public:
957
      Poly_Double_Tests() : Text_Based_Test("poly_dbl.vec", "In,Out") {}
2✔
958

959
      Test::Result run_one_test(const std::string& /*header*/, const VarMap& vars) override {
82✔
960
         Test::Result result("Polynomial doubling");
82✔
961
         const std::vector<uint8_t> in = vars.get_req_bin("In");
82✔
962
         const std::vector<uint8_t> out = vars.get_req_bin("Out");
82✔
963

964
         std::vector<uint8_t> b = in;
82✔
965
         Botan::poly_double_n(b.data(), b.size());
82✔
966

967
         result.test_bin_eq("Expected value", b, out);
82✔
968
         return result;
82✔
969
      }
246✔
970
};
971

972
BOTAN_REGISTER_TEST("utils", "poly_dbl", Poly_Double_Tests);
973

974
#endif
975

976
class Version_Tests final : public Test {
1✔
977
   public:
978
      std::vector<Test::Result> run() override {
1✔
979
         Test::Result result("Versions");
1✔
980

981
         result.test_u32_eq("Version datestamp matches macro", Botan::version_datestamp(), BOTAN_VERSION_DATESTAMP);
1✔
982

983
         const char* version_cstr = Botan::version_cstr();
1✔
984
         const std::string version_str = Botan::version_string();
1✔
985
         result.test_str_eq("Same version string", version_str, std::string(version_cstr));
1✔
986

987
         const char* sversion_cstr = Botan::short_version_cstr();
1✔
988
         const std::string sversion_str = Botan::short_version_string();
1✔
989
         result.test_str_eq("Same short version string", sversion_str, std::string(sversion_cstr));
1✔
990

991
         const auto expected_sversion =
1✔
992
            Botan::fmt("{}.{}.{}", BOTAN_VERSION_MAJOR, BOTAN_VERSION_MINOR, BOTAN_VERSION_PATCH);
1✔
993

994
         // May have a suffix eg 4.0.0-rc2
995
         result.test_is_true("Short version string has expected format", sversion_str.starts_with(expected_sversion));
1✔
996

997
         const std::string version_check_ok =
1✔
998
            Botan::runtime_version_check(BOTAN_VERSION_MAJOR, BOTAN_VERSION_MINOR, BOTAN_VERSION_PATCH);
1✔
999

1000
         result.test_is_true("Correct version no warning", version_check_ok.empty());
1✔
1001

1002
         const std::string version_check_bad = Botan::runtime_version_check(1, 19, 42);
1✔
1003

1004
         const std::string expected_error =
1✔
1005
            "Warning: linked version (" + sversion_str + ") does not match version built against (1.19.42)\n";
2✔
1006

1007
         result.test_str_eq("Expected warning text", version_check_bad, expected_error);
1✔
1008

1009
         return {result};
3✔
1010
      }
2✔
1011
};
1012

1013
BOTAN_REGISTER_TEST("utils", "versioning", Version_Tests);
1014

1015
class Date_Format_Tests final : public Text_Based_Test {
×
1016
   public:
1017
      Date_Format_Tests() : Text_Based_Test("dates.vec", "Date") {}
2✔
1018

1019
      static std::vector<uint32_t> parse_date(const std::string& s) {
11✔
1020
         const std::vector<std::string> parts = Botan::split_on(s, ',');
11✔
1021
         if(parts.size() != 6) {
11✔
1022
            throw Test_Error("Bad date format '" + s + "'");
×
1023
         }
1024

1025
         std::vector<uint32_t> u32s;
11✔
1026
         u32s.reserve(parts.size());
11✔
1027
         for(const auto& sub : parts) {
77✔
1028
            u32s.push_back(Botan::to_u32bit(sub));
66✔
1029
         }
1030
         return u32s;
11✔
1031
      }
11✔
1032

1033
      Test::Result run_one_test(const std::string& type, const VarMap& vars) override {
11✔
1034
         const std::string date_str = vars.get_req_str("Date");
11✔
1035
         Test::Result result("Date parsing");
11✔
1036

1037
         const std::vector<uint32_t> d = parse_date(date_str);
11✔
1038

1039
         if(type == "valid" || type == "valid.not_std" || type == "valid.64_bit_time_t") {
11✔
1040
            const Botan::calendar_point c(d[0], d[1], d[2], d[3], d[4], d[5]);
11✔
1041
            result.test_u32_eq(date_str + " year", c.year(), d[0]);
11✔
1042
            result.test_u32_eq(date_str + " month", c.month(), d[1]);
11✔
1043
            result.test_u32_eq(date_str + " day", c.day(), d[2]);
11✔
1044
            result.test_u32_eq(date_str + " hour", c.hour(), d[3]);
11✔
1045
            result.test_u32_eq(date_str + " minute", c.minutes(), d[4]);
11✔
1046
            result.test_u32_eq(date_str + " second", c.seconds(), d[5]);
11✔
1047

1048
            if(type == "valid.not_std" ||
11✔
1049
               (type == "valid.64_bit_time_t" && c.year() > 2037 && sizeof(std::time_t) == 4)) {
1050
               result.test_throws("valid but out of std::timepoint range", [c]() { c.to_std_timepoint(); });
6✔
1051
            } else {
1052
               const Botan::calendar_point c2(c.to_std_timepoint());
8✔
1053
               result.test_u32_eq(date_str + " year", c2.year(), d[0]);
8✔
1054
               result.test_u32_eq(date_str + " month", c2.month(), d[1]);
8✔
1055
               result.test_u32_eq(date_str + " day", c2.day(), d[2]);
8✔
1056
               result.test_u32_eq(date_str + " hour", c2.hour(), d[3]);
8✔
1057
               result.test_u32_eq(date_str + " minute", c2.minutes(), d[4]);
8✔
1058
               result.test_u32_eq(date_str + " second", c2.seconds(), d[5]);
16✔
1059
            }
1060
         } else if(type == "invalid") {
×
1061
            result.test_throws("invalid date",
×
1062
                               [d]() { const Botan::calendar_point c(d[0], d[1], d[2], d[3], d[4], d[5]); });
×
1063
         } else {
1064
            throw Test_Error("Unexpected header '" + type + "' in date format tests");
×
1065
         }
1066

1067
         return result;
22✔
1068
      }
125✔
1069

1070
      std::vector<Test::Result> run_final_tests() override {
1✔
1071
         Test::Result result("calendar_point::to_string");
1✔
1072
         const Botan::calendar_point d(2008, 5, 15, 9, 30, 33);
1✔
1073
         // desired format: <YYYY>-<MM>-<dd>T<HH>:<mm>:<ss>
1074
         result.test_str_eq("calendar_point::to_string", d.to_string(), "2008-05-15T09:30:33");
1✔
1075
         return {result};
3✔
1076
      }
2✔
1077
};
1078

1079
BOTAN_REGISTER_TEST("utils", "util_dates", Date_Format_Tests);
1080

1081
class Charset_Tests final : public Text_Based_Test {
×
1082
   public:
1083
      Charset_Tests() : Text_Based_Test("charset.vec", "In,Out") {}
2✔
1084

1085
      Test::Result run_one_test(const std::string& type, const VarMap& vars) override {
53✔
1086
         Test::Result result("Charset");
53✔
1087

1088
         const std::vector<uint8_t> in = vars.get_req_bin("In");
53✔
1089

1090
         const auto in_sv = std::string_view(reinterpret_cast<const char*>(in.data()), in.size());
53✔
1091

1092
         if(type == "UTF8-UCS2-INVALID") {
53✔
1093
            result.test_throws<Botan::Decoding_Error>("utf8_to_ucs2 rejects invalid input",
13✔
1094
                                                      [&] { Botan::utf8_to_ucs2(in_sv); });
26✔
1095
            return result;
13✔
1096
         }
1097

1098
         if(type == "UTF8-UCS4-INVALID") {
40✔
1099
            result.test_throws<Botan::Decoding_Error>("utf8_to_ucs4 rejects invalid input",
10✔
1100
                                                      [&] { Botan::utf8_to_ucs4(in_sv); });
20✔
1101
            return result;
10✔
1102
         }
1103

1104
         const std::vector<uint8_t> expected = vars.get_req_bin("Out");
30✔
1105

1106
         std::string converted;
30✔
1107

1108
         if(type == "UCS2-UTF8") {
30✔
1109
            converted = Botan::ucs2_to_utf8(in);
4✔
1110
         } else if(type == "UCS4-UTF8") {
26✔
1111
            converted = Botan::ucs4_to_utf8(in);
1✔
1112
         } else if(type == "UTF8-UCS2") {
25✔
1113
            std::vector<uint8_t> ucs2 = Botan::utf8_to_ucs2(in_sv);
11✔
1114
            converted = std::string(ucs2.begin(), ucs2.end());
22✔
1115
         } else if(type == "UTF8-UCS4") {
25✔
1116
            std::vector<uint8_t> ucs4 = Botan::utf8_to_ucs4(in_sv);
11✔
1117
            converted = std::string(ucs4.begin(), ucs4.end());
22✔
1118
         } else if(type == "LATIN1-UTF8") {
14✔
1119
            converted = Botan::latin1_to_utf8(in);
3✔
1120
         } else {
1121
            throw Test_Error("Unexpected header '" + type + "' in charset tests");
×
1122
         }
1123

1124
         result.test_bin_eq(
30✔
1125
            "string converted successfully", std::vector<uint8_t>(converted.begin(), converted.end()), expected);
30✔
1126

1127
         return result;
30✔
1128
      }
113✔
1129
};
1130

1131
BOTAN_REGISTER_TEST("utils", "charset", Charset_Tests);
1132

1133
#if defined(BOTAN_HAS_IPV4_ADDRESS)
1134

1135
class IPv4_Parsing_Tests final : public Text_Based_Test {
×
1136
   public:
1137
      IPv4_Parsing_Tests() : Text_Based_Test("utils/ipv4.vec", "IPv4") {}
2✔
1138

1139
      Test::Result run_one_test(const std::string& status, const VarMap& vars) override {
47✔
1140
         Test::Result result("IPv4 parsing");
47✔
1141

1142
         const std::string input = vars.get_req_str("IPv4");
47✔
1143
         const bool valid = (status == "Valid");
47✔
1144

1145
         auto ipv4 = Botan::IPv4Address::from_string(input);
47✔
1146

1147
         result.test_bool_eq("IPv4Address::from_string accepts only valid", ipv4.has_value(), valid);
47✔
1148

1149
         if(ipv4) {
47✔
1150
            const std::string rt = ipv4->to_string();
13✔
1151
            result.test_str_eq("IPv4Address::from_string and IPv4Address::to_string round trip", input, rt);
13✔
1152
         }
13✔
1153

1154
         return result;
47✔
1155
      }
47✔
1156
};
1157

1158
BOTAN_REGISTER_TEST("utils", "ipv4_parse", IPv4_Parsing_Tests);
1159

1160
#endif
1161

1162
#if defined(BOTAN_HAS_IPV6_ADDRESS)
1163

1164
class IPv6_Parsing_Tests final : public Text_Based_Test {
×
1165
   public:
1166
      IPv6_Parsing_Tests() : Text_Based_Test("utils/ipv6.vec", "IPv6") {}
2✔
1167

1168
      Test::Result run_one_test(const std::string& header, const VarMap& vars) override {
39✔
1169
         Test::Result result("IPv6 parsing");
39✔
1170

1171
         const std::string input = vars.get_req_str("IPv6");
39✔
1172
         const bool valid = (header == "Valid");
39✔
1173

1174
         auto ipv6 = Botan::IPv6Address::from_string(input);
39✔
1175

1176
         result.test_bool_eq("IPv6Address::from_string accepts only valid", ipv6.has_value(), valid);
39✔
1177

1178
         if(ipv6) {
39✔
1179
            const std::string rt = ipv6->to_string();
10✔
1180
            result.test_str_eq("IPv6Address to_string and from_string round trip", input, rt);
10✔
1181
         }
10✔
1182

1183
         return result;
39✔
1184
      }
39✔
1185
};
1186

1187
BOTAN_REGISTER_TEST("utils", "ipv6_parse", IPv6_Parsing_Tests);
1188

1189
class IPv6_Noncanonical_Parsing_Tests final : public Text_Based_Test {
×
1190
   public:
1191
      IPv6_Noncanonical_Parsing_Tests() : Text_Based_Test("utils/ipv6_nc.vec", "Input,Canonical") {}
2✔
1192

1193
      Test::Result run_one_test(const std::string& /*header*/, const VarMap& vars) override {
14✔
1194
         Test::Result result("IPv6 parsing of non-canonical form");
14✔
1195

1196
         const std::string input_str = vars.get_req_str("Input");
14✔
1197
         const std::string canonical_str = vars.get_req_str("Canonical");
14✔
1198

1199
         const auto ipv6 = Botan::IPv6Address::from_string(input_str);
14✔
1200
         const auto canonical = Botan::IPv6Address::from_string(canonical_str);
14✔
1201

1202
         result.test_is_true("IPv6 non-canonical parsing worked", ipv6.has_value());
14✔
1203
         result.test_is_true("IPv6 canonical parsing worked", canonical.has_value());
14✔
1204

1205
         if(ipv6.has_value() && canonical.has_value()) {
14✔
1206
            result.test_is_true("IPv6 non-canonical decoding", *ipv6 == *canonical);
14✔
1207
         }
1208

1209
         return result;
14✔
1210
      }
14✔
1211
};
1212

1213
BOTAN_REGISTER_TEST("utils", "ipv6_parse_non_canonical", IPv6_Noncanonical_Parsing_Tests);
1214

1215
#endif
1216

1217
class ReadKV_Tests final : public Text_Based_Test {
×
1218
   public:
1219
      ReadKV_Tests() : Text_Based_Test("utils/read_kv.vec", "Input,Expected") {}
2✔
1220

1221
      Test::Result run_one_test(const std::string& status, const VarMap& vars) override {
16✔
1222
         Test::Result result("read_kv");
16✔
1223

1224
         const bool is_valid = (status == "Valid");
16✔
1225

1226
         const std::string input = vars.get_req_str("Input");
16✔
1227
         const std::string expected = vars.get_req_str("Expected");
16✔
1228

1229
         if(is_valid) {
16✔
1230
            confirm_kv(result, Botan::read_kv(input), split_group(expected));
14✔
1231
         } else {
1232
            // In this case "expected" is the expected exception message
1233
            result.test_throws("Invalid key value input throws exception", expected, [&]() { Botan::read_kv(input); });
18✔
1234
         }
1235
         return result;
16✔
1236
      }
16✔
1237

1238
   private:
1239
      static std::vector<std::string> split_group(const std::string& str) {
7✔
1240
         std::vector<std::string> elems;
7✔
1241
         if(str.empty()) {
7✔
1242
            return elems;
1243
         }
1244

1245
         std::string substr;
6✔
1246
         for(const char c : str) {
115✔
1247
            if(c == '|') {
109✔
1248
               elems.push_back(substr);
16✔
1249
               substr.clear();
16✔
1250
            } else {
1251
               substr += c;
202✔
1252
            }
1253
         }
1254

1255
         if(!substr.empty()) {
6✔
1256
            elems.push_back(substr);
6✔
1257
         }
1258

1259
         return elems;
6✔
1260
      }
6✔
1261

1262
      static void confirm_kv(Test::Result& result,
7✔
1263
                             const std::map<std::string, std::string>& kv,
1264
                             const std::vector<std::string>& expected) {
1265
         if(!result.test_sz_eq("expected size", expected.size() % 2, size_t(0))) {
7✔
1266
            return;
1267
         }
1268

1269
         for(size_t i = 0; i != expected.size(); i += 2) {
18✔
1270
            auto j = kv.find(expected[i]);
11✔
1271
            if(result.test_is_true("Found key", j != kv.end())) {
11✔
1272
               result.test_str_eq("Matching value", j->second, expected[i + 1]);
11✔
1273
            }
1274
         }
1275

1276
         result.test_sz_eq("KV has same size as expected", kv.size(), expected.size() / 2);
7✔
1277
      }
1278
};
1279

1280
BOTAN_REGISTER_TEST("utils", "util_read_kv", ReadKV_Tests);
1281

1282
#if defined(BOTAN_HAS_CPUID)
1283

1284
class CPUID_Tests final : public Test {
1✔
1285
   public:
1286
      std::vector<Test::Result> run() override {
1✔
1287
         Test::Result result("CPUID");
1✔
1288

1289
         const std::string cpuid_string = Botan::CPUID::to_string();
1✔
1290
         result.test_success("CPUID::to_string doesn't crash");
1✔
1291

1292
         for(size_t b = 0; b != 32; ++b) {
33✔
1293
            try {
32✔
1294
               const auto bit = static_cast<uint32_t>(1) << b;
32✔
1295
               // NOLINTNEXTLINE(clang-analyzer-optin.core.EnumCastOutOfRange)
1296
               const auto feat = Botan::CPUID::Feature(static_cast<Botan::CPUID::Feature::Bit>(bit));
32✔
1297

1298
               const std::string feat_str = feat.to_string();
32✔
1299

1300
               result.test_is_true("Feature string is not empty", !feat_str.empty());
20✔
1301

1302
               if(auto from_str = Botan::CPUID::Feature::from_string(feat_str)) {
20✔
1303
                  result.test_u32_eq("Feature::from_string returns expected bit", from_str->as_u32(), bit);
20✔
1304
               } else {
1305
                  result.test_failure(
×
1306
                     Botan::fmt("Feature::from_string didn't recognize its own output ({})", feat_str));
×
1307
               }
1308
            } catch(Botan::Invalid_State&) {
32✔
1309
               // This will thrown if the bit is not a valid one
1310
            }
12✔
1311
         }
1312

1313
   #if defined(BOTAN_TARGET_ARCH_IS_X86_FAMILY)
1314

1315
         const auto bit = Botan::CPUID::Feature::SSE2;
1✔
1316

1317
         if(Botan::CPUID::has(bit)) {
1✔
1318
            result.test_is_true("Output string includes sse2", cpuid_string.find("sse2") != std::string::npos);
1✔
1319

1320
            Botan::CPUID::clear_cpuid_bit(bit);
1✔
1321

1322
            result.test_is_false("After clearing cpuid bit, CPUID::has for SSE2 returns false", Botan::CPUID::has(bit));
1✔
1323

1324
            Botan::CPUID::initialize();  // reset state
1✔
1325
            result.test_is_true("After reinitializing, CPUID::has for SSE2 returns true again", Botan::CPUID::has(bit));
1✔
1326
         }
1327
   #else
1328
         BOTAN_UNUSED(cpuid_string);
1329
   #endif
1330

1331
         return {result};
3✔
1332
      }
2✔
1333
};
1334

1335
BOTAN_REGISTER_SERIALIZED_TEST("utils", "cpuid", CPUID_Tests);
1336

1337
#endif
1338

1339
#if defined(BOTAN_HAS_UUID)
1340

1341
class UUID_Tests : public Test {
1✔
1342
   public:
1343
      std::vector<Test::Result> run() override {
1✔
1344
         Test::Result result("UUID");
1✔
1345

1346
         const Botan::UUID empty_uuid;
1✔
1347
         const Botan::UUID random_uuid1(this->rng());
1✔
1348
         const Botan::UUID random_uuid2(this->rng());
1✔
1349
         const Botan::UUID loaded_uuid(std::vector<uint8_t>(16, 4));
1✔
1350

1351
         result.test_throws("Cannot load wrong number of bytes",
1✔
1352
                            []() { const Botan::UUID u(std::vector<uint8_t>(15)); });
1✔
1353

1354
         result.test_is_false("Empty UUID is empty", empty_uuid.is_valid());
1✔
1355
         result.test_is_true("Empty UUID equals another empty UUID", empty_uuid == Botan::UUID());
1✔
1356

1357
         result.test_throws("Empty UUID cannot become a string", [&]() { empty_uuid.to_string(); });
2✔
1358

1359
         result.test_is_true("Random UUID not empty", random_uuid1.is_valid());
1✔
1360
         result.test_is_true("Random UUID not empty", random_uuid2.is_valid());
1✔
1361

1362
         result.test_is_true("Random UUIDs are distinct", random_uuid1 != random_uuid2);
1✔
1363
         result.test_is_true("Random UUIDs not equal to empty", random_uuid1 != empty_uuid);
1✔
1364

1365
         const std::string uuid4_str = loaded_uuid.to_string();
1✔
1366
         result.test_str_eq("String matches expected", uuid4_str, "04040404-0404-0404-0404-040404040404");
1✔
1367

1368
         const std::string uuid_r1_str = random_uuid1.to_string();
1✔
1369
         result.test_is_true("UUID from string matches", Botan::UUID(uuid_r1_str) == random_uuid1);
1✔
1370

1371
         class AllSame_RNG : public Botan::RandomNumberGenerator {
×
1372
            public:
1373
               explicit AllSame_RNG(uint8_t b) : m_val(b) {}
2✔
1374

1375
               void fill_bytes_with_input(std::span<uint8_t> output, std::span<const uint8_t> /* ignored */) override {
2✔
1376
                  std::fill(output.begin(), output.end(), m_val);
2✔
1377
               }
2✔
1378

1379
               std::string name() const override { return "zeros"; }
×
1380

1381
               bool accepts_input() const override { return false; }
×
1382

1383
               void clear() override {}
×
1384

1385
               bool is_seeded() const override { return true; }
×
1386

1387
            private:
1388
               uint8_t m_val;
1389
         };
1390

1391
         AllSame_RNG zeros(0x00);
1✔
1392
         const Botan::UUID zero_uuid(zeros);
1✔
1393
         result.test_str_eq(
1✔
1394
            "Zero UUID matches expected", zero_uuid.to_string(), "00000000-0000-4000-8000-000000000000");
1✔
1395

1396
         AllSame_RNG ones(0xFF);
1✔
1397
         const Botan::UUID ones_uuid(ones);
1✔
1398
         result.test_str_eq(
1✔
1399
            "Ones UUID matches expected", ones_uuid.to_string(), "FFFFFFFF-FFFF-4FFF-BFFF-FFFFFFFFFFFF");
1✔
1400

1401
         return {result};
3✔
1402
      }
6✔
1403
};
1404

1405
BOTAN_REGISTER_TEST("utils", "uuid", UUID_Tests);
1406

1407
#endif
1408

1409
class Formatter_Tests : public Test {
1✔
1410
   public:
1411
      std::vector<Test::Result> run() override {
1✔
1412
         Test::Result result("Format utility");
1✔
1413

1414
         /*
1415
         In a number of these tests, we are not strictly depending on the
1416
         behavior, for instance checking `fmt("{}") == "{}"` is more about
1417
         checking that we don't crash, rather than we return that precise string.
1418
         */
1419

1420
         result.test_str_eq("test 1", Botan::fmt("hi"), "hi");
1✔
1421
         result.test_str_eq("test 2", Botan::fmt("ignored", 5), "ignored");
1✔
1422
         result.test_str_eq("test 3", Botan::fmt("answer is {}", 42), "answer is 42");
1✔
1423
         result.test_str_eq("test 4", Botan::fmt("{", 5), "{");
1✔
1424
         result.test_str_eq("test 4", Botan::fmt("{}"), "{}");
1✔
1425
         result.test_str_eq("test 5", Botan::fmt("{} == '{}'", 5, "five"), "5 == 'five'");
1✔
1426

1427
         return {result};
3✔
1428
      }
2✔
1429
};
1430

1431
BOTAN_REGISTER_TEST("utils", "fmt", Formatter_Tests);
1432

1433
class ScopedCleanup_Tests : public Test {
1✔
1434
   public:
1435
      std::vector<Test::Result> run() override {
1✔
1436
         return {
1✔
1437
            CHECK("leaving a scope results in cleanup",
1438
                  [](Test::Result& result) {
1✔
1439
                     bool ran = false;
1✔
1440
                     {
1✔
1441
                        auto clean = Botan::scoped_cleanup([&] { ran = true; });
1✔
1442
                     }
1✔
1443
                     result.test_is_true("cleanup ran", ran);
1✔
1444
                  }),
1✔
1445

1446
            CHECK("leaving a function, results in cleanup",
1447
                  [](Test::Result& result) {
1✔
1448
                     bool ran = false;
1✔
1449
                     bool fn_called = false;
1✔
1450
                     auto fn = [&] {
2✔
1451
                        auto clean = Botan::scoped_cleanup([&] { ran = true; });
1✔
1452
                        fn_called = true;
1✔
1453
                     };
2✔
1454

1455
                     result.test_is_true("cleanup not yet ran", !ran);
1✔
1456
                     fn();
1✔
1457
                     result.test_is_true("fn called", fn_called);
1✔
1458
                     result.test_is_true("cleanup ran", ran);
1✔
1459
                  }),
1✔
1460

1461
            CHECK("stack unwinding results in cleanup",
1462
                  [](Test::Result& result) {
1✔
1463
                     bool ran = false;
1✔
1464
                     bool fn_called = false;
1✔
1465
                     bool exception_caught = false;
1✔
1466
                     auto fn = [&] {
2✔
1467
                        auto clean = Botan::scoped_cleanup([&] { ran = true; });
1✔
1468
                        fn_called = true;
1✔
1469
                        throw std::runtime_error("test");
1✔
1470
                     };
2✔
1471

1472
                     result.test_is_true("cleanup not yet ran", !ran);
1✔
1473
                     try {
1✔
1474
                        fn();
1✔
1475
                     } catch(const std::exception&) {
1✔
1476
                        exception_caught = true;
1✔
1477
                     }
1✔
1478

1479
                     result.test_is_true("fn called", fn_called);
1✔
1480
                     result.test_is_true("cleanup ran", ran);
1✔
1481
                     result.test_is_true("exception caught", exception_caught);
1✔
1482
                  }),
1✔
1483

1484
            CHECK("cleanup isn't called after disengaging",
1485
                  [](Test::Result& result) {
1✔
1486
                     bool ran = false;
1✔
1487
                     {
1✔
1488
                        auto clean = Botan::scoped_cleanup([&] { ran = true; });
1✔
1489
                        clean.disengage();
1✔
1490
                     }
1✔
1491
                     result.test_is_true("cleanup not ran", !ran);
1✔
1492
                  }),
1✔
1493

1494
         };
5✔
1495
      }
1✔
1496
};
1497

1498
BOTAN_REGISTER_TEST("utils", "scoped_cleanup", ScopedCleanup_Tests);
1499

1500
}  // namespace
1501

1502
}  // namespace Botan_Tests
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc