summaryrefslogtreecommitdiff
path: root/tools/perf/tests/util.c
blob: bf2c5b1338844c2e8247e5ad5ff25f366b2dde8f (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
// SPDX-License-Identifier: GPL-2.0
#include "tests.h"
#include "util/blake2s.h"
#include "util/debug.h"

#include <linux/compiler.h>
#include <stdlib.h>
#include <string2.h>

static int test_strreplace(char needle, const char *haystack,
			   const char *replace, const char *expected)
{
	char *new = strreplace_chars(needle, haystack, replace);
	int ret = strcmp(new, expected);

	free(new);
	return ret == 0;
}

/* Maximum data length tested by test_blake2s() */
#define MAX_DATA_LEN 512

/*
 * Hash length tested by test_blake2s().  BLAKE2s supports variable-length
 * hashes.  However, the only user of BLAKE2s in 'perf' uses 20-byte hashes,
 * matching the length of the ELF build ID field.  So that's the length we test.
 */
#define HASH_LEN 20

/* Test the implementation of the BLAKE2s hash algorithm. */
static int test_blake2s(void)
{
	u8 data[MAX_DATA_LEN];
	u8 hash[HASH_LEN];
	u8 hash2[HASH_LEN];
	struct blake2s_ctx main_ctx;
	/*
	 * This value was generated by the following Python code:
	 *
	 * import hashlib
	 *
	 * data = bytes(i % 256 for i in range(513))
	 * h = hashlib.blake2s(digest_size=20)
	 * for i in range(513):
	 *     h.update(hashlib.blake2s(data=data[:i], digest_size=20).digest())
	 * print(h.hexdigest())
	 */
	static const u8 expected_hash_of_hashes[20] = {
		0xef, 0x9b, 0x13, 0x98, 0x78, 0x8e, 0x74, 0x59, 0x9c, 0xd5,
		0x0c, 0xf0, 0x33, 0x97, 0x79, 0x3d, 0x3e, 0xd0, 0x95, 0xa6
	};
	size_t i;

	/* Generate MAX_DATA_LEN bytes of data. */
	for (i = 0; i < MAX_DATA_LEN; i++)
		data[i] = i;

	blake2s_init(&main_ctx, sizeof(hash));
	for (i = 0; i <= MAX_DATA_LEN; i++) {
		struct blake2s_ctx ctx;

		/* Compute the BLAKE2s hash of 'i' data bytes. */
		blake2s_init(&ctx, HASH_LEN);
		blake2s_update(&ctx, data, i);
		blake2s_final(&ctx, hash);

		/* Verify that multiple updates produce the same result. */
		blake2s_init(&ctx, HASH_LEN);
		blake2s_update(&ctx, data, i / 2);
		blake2s_update(&ctx, &data[i / 2], i - (i / 2));
		blake2s_final(&ctx, hash2);
		TEST_ASSERT_VAL("inconsistent BLAKE2s hashes",
				memcmp(hash, hash2, HASH_LEN) == 0);

		/*
		 * Pass the hash to another BLAKE2s context, so that we
		 * incrementally compute the hash of all the hashes.
		 */
		blake2s_update(&main_ctx, hash, HASH_LEN);
	}

	/* Verify the hash of all the hashes. */
	blake2s_final(&main_ctx, hash);
	TEST_ASSERT_VAL("wrong BLAKE2s hashes",
			memcmp(hash, expected_hash_of_hashes, HASH_LEN) == 0);
	return 0;
}

static int test__util(struct test_suite *t __maybe_unused, int subtest __maybe_unused)
{
	TEST_ASSERT_VAL("empty string", test_strreplace(' ', "", "123", ""));
	TEST_ASSERT_VAL("no match", test_strreplace('5', "123", "4", "123"));
	TEST_ASSERT_VAL("replace 1", test_strreplace('3', "123", "4", "124"));
	TEST_ASSERT_VAL("replace 2", test_strreplace('a', "abcabc", "ef", "efbcefbc"));
	TEST_ASSERT_VAL("replace long", test_strreplace('a', "abcabc", "longlong",
							"longlongbclonglongbc"));

	return test_blake2s();
}

DEFINE_SUITE("util", util);