summaryrefslogtreecommitdiff
path: root/tools/testing/selftests/exec/load_address.c
blob: 8257fddba8c8db21ed1ea49aee37c15d9bdb6e03 (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
102
103
104
105
// SPDX-License-Identifier: GPL-2.0-only
#ifndef _GNU_SOURCE
#define _GNU_SOURCE
#endif
#include <link.h>
#include <stdio.h>
#include <stdlib.h>
#include <stdbool.h>
#include "../kselftest.h"

struct Statistics {
	unsigned long long load_address;
	unsigned long long alignment;
	bool interp;
};

int ExtractStatistics(struct dl_phdr_info *info, size_t size, void *data)
{
	struct Statistics *stats = (struct Statistics *) data;
	int i;

	if (info->dlpi_name != NULL && info->dlpi_name[0] != '\0') {
		// Ignore headers from other than the executable.
		return 2;
	}

	stats->load_address = (unsigned long long) info->dlpi_addr;
	stats->alignment = 0;

	for (i = 0; i < info->dlpi_phnum; i++) {
		unsigned long long align;

		if (info->dlpi_phdr[i].p_type == PT_INTERP) {
			stats->interp = true;
			continue;
		}

		if (info->dlpi_phdr[i].p_type != PT_LOAD)
			continue;

		align = info->dlpi_phdr[i].p_align;

		if (align > stats->alignment)
			stats->alignment = align;
	}

	return 1;  // Terminate dl_iterate_phdr.
}

int main(int argc, char **argv)
{
	struct Statistics extracted = { };
	unsigned long long misalign, pow2;
	bool interp_needed;
	char buf[1024];
	FILE *maps;
	int ret;

	ksft_print_header();
	ksft_set_plan(4);

	/* Dump maps file for debugging reference. */
	maps = fopen("/proc/self/maps", "r");
	if (!maps)
		ksft_exit_fail_msg("FAILED: /proc/self/maps: %s\n", strerror(errno));
	while (fgets(buf, sizeof(buf), maps)) {
		ksft_print_msg("%s", buf);
	}
	fclose(maps);

	/* Walk the program headers. */
	ret = dl_iterate_phdr(ExtractStatistics, &extracted);
	if (ret != 1)
		ksft_exit_fail_msg("FAILED: dl_iterate_phdr\n");

	/* Report our findings. */
	ksft_print_msg("load_address=%#llx alignment=%#llx\n",
		       extracted.load_address, extracted.alignment);

	/* If we're named with ".static." we expect no INTERP. */
	interp_needed = strstr(argv[0], ".static.") == NULL;

	/* Were we built as expected? */
	ksft_test_result(interp_needed == extracted.interp,
			 "%s INTERP program header %s\n",
			 interp_needed ? "Wanted" : "Unwanted",
			 extracted.interp ? "seen" : "missing");

	/* Did we find an alignment? */
	ksft_test_result(extracted.alignment != 0,
			 "Alignment%s found\n", extracted.alignment ? "" : " NOT");

	/* Is the alignment sane? */
	pow2 = extracted.alignment & (extracted.alignment - 1);
	ksft_test_result(pow2 == 0,
			 "Alignment is%s a power of 2: %#llx\n",
			 pow2 == 0 ? "" : " NOT", extracted.alignment);

	/* Is the load address aligned? */
	misalign = extracted.load_address & (extracted.alignment - 1);
	ksft_test_result(misalign == 0, "Load Address is %saligned (%#llx)\n",
			 misalign ? "MIS" : "", misalign);

	ksft_finished();
}