1 // SPDX-License-Identifier: GPL-2.0
2 #include "tests.h"
3 #include "util/blake2s.h"
4 #include "util/debug.h"
5
6 #include <linux/compiler.h>
7 #include <stdlib.h>
8 #include <string2.h>
9
test_strreplace(char needle,const char * haystack,const char * replace,const char * expected)10 static int test_strreplace(char needle, const char *haystack,
11 const char *replace, const char *expected)
12 {
13 char *new = strreplace_chars(needle, haystack, replace);
14 int ret = strcmp(new, expected);
15
16 free(new);
17 return ret == 0;
18 }
19
20 /* Maximum data length tested by test_blake2s() */
21 #define MAX_DATA_LEN 512
22
23 /*
24 * Hash length tested by test_blake2s(). BLAKE2s supports variable-length
25 * hashes. However, the only user of BLAKE2s in 'perf' uses 20-byte hashes,
26 * matching the length of the ELF build ID field. So that's the length we test.
27 */
28 #define HASH_LEN 20
29
30 /* Test the implementation of the BLAKE2s hash algorithm. */
test_blake2s(void)31 static int test_blake2s(void)
32 {
33 u8 data[MAX_DATA_LEN];
34 u8 hash[HASH_LEN];
35 u8 hash2[HASH_LEN];
36 struct blake2s_ctx main_ctx;
37 /*
38 * This value was generated by the following Python code:
39 *
40 * import hashlib
41 *
42 * data = bytes(i % 256 for i in range(513))
43 * h = hashlib.blake2s(digest_size=20)
44 * for i in range(513):
45 * h.update(hashlib.blake2s(data=data[:i], digest_size=20).digest())
46 * print(h.hexdigest())
47 */
48 static const u8 expected_hash_of_hashes[20] = {
49 0xef, 0x9b, 0x13, 0x98, 0x78, 0x8e, 0x74, 0x59, 0x9c, 0xd5,
50 0x0c, 0xf0, 0x33, 0x97, 0x79, 0x3d, 0x3e, 0xd0, 0x95, 0xa6
51 };
52 size_t i;
53
54 /* Generate MAX_DATA_LEN bytes of data. */
55 for (i = 0; i < MAX_DATA_LEN; i++)
56 data[i] = i;
57
58 blake2s_init(&main_ctx, sizeof(hash));
59 for (i = 0; i <= MAX_DATA_LEN; i++) {
60 struct blake2s_ctx ctx;
61
62 /* Compute the BLAKE2s hash of 'i' data bytes. */
63 blake2s_init(&ctx, HASH_LEN);
64 blake2s_update(&ctx, data, i);
65 blake2s_final(&ctx, hash);
66
67 /* Verify that multiple updates produce the same result. */
68 blake2s_init(&ctx, HASH_LEN);
69 blake2s_update(&ctx, data, i / 2);
70 blake2s_update(&ctx, &data[i / 2], i - (i / 2));
71 blake2s_final(&ctx, hash2);
72 TEST_ASSERT_VAL("inconsistent BLAKE2s hashes",
73 memcmp(hash, hash2, HASH_LEN) == 0);
74
75 /*
76 * Pass the hash to another BLAKE2s context, so that we
77 * incrementally compute the hash of all the hashes.
78 */
79 blake2s_update(&main_ctx, hash, HASH_LEN);
80 }
81
82 /* Verify the hash of all the hashes. */
83 blake2s_final(&main_ctx, hash);
84 TEST_ASSERT_VAL("wrong BLAKE2s hashes",
85 memcmp(hash, expected_hash_of_hashes, HASH_LEN) == 0);
86 return 0;
87 }
88
test__util(struct test_suite * t __maybe_unused,int subtest __maybe_unused)89 static int test__util(struct test_suite *t __maybe_unused, int subtest __maybe_unused)
90 {
91 TEST_ASSERT_VAL("empty string", test_strreplace(' ', "", "123", ""));
92 TEST_ASSERT_VAL("no match", test_strreplace('5', "123", "4", "123"));
93 TEST_ASSERT_VAL("replace 1", test_strreplace('3', "123", "4", "124"));
94 TEST_ASSERT_VAL("replace 2", test_strreplace('a', "abcabc", "ef", "efbcefbc"));
95 TEST_ASSERT_VAL("replace long", test_strreplace('a', "abcabc", "longlong",
96 "longlongbclonglongbc"));
97
98 return test_blake2s();
99 }
100
101 DEFINE_SUITE("util", util);
102