12874c5fdSThomas Gleixner // SPDX-License-Identifier: GPL-2.0-or-later
2c853d945SWu Zhangjin /*
3c853d945SWu Zhangjin  * Copyright (C) 2010 "Wu Zhangjin" <wuzhangjin@gmail.com>
4c853d945SWu Zhangjin  */
5c853d945SWu Zhangjin 
6c853d945SWu Zhangjin #include <sys/types.h>
7c853d945SWu Zhangjin #include <sys/stat.h>
8c853d945SWu Zhangjin #include <errno.h>
9c853d945SWu Zhangjin #include <stdint.h>
10c853d945SWu Zhangjin #include <stdio.h>
11c853d945SWu Zhangjin #include <stdlib.h>
121196364fSKevin Darbyshire-Bryant #include <linux/sizes.h>
13c853d945SWu Zhangjin 
main(int argc,char * argv[])14c853d945SWu Zhangjin int main(int argc, char *argv[])
15c853d945SWu Zhangjin {
16893d20fbSRalf Baechle 	unsigned long long vmlinux_size, vmlinux_load_addr, vmlinuz_load_addr;
17c853d945SWu Zhangjin 	struct stat sb;
18c853d945SWu Zhangjin 
19c853d945SWu Zhangjin 	if (argc != 3) {
20c853d945SWu Zhangjin 		fprintf(stderr, "Usage: %s <pathname> <vmlinux_load_addr>\n",
21c853d945SWu Zhangjin 				argv[0]);
22c853d945SWu Zhangjin 		return EXIT_FAILURE;
23c853d945SWu Zhangjin 	}
24c853d945SWu Zhangjin 
25c853d945SWu Zhangjin 	if (stat(argv[1], &sb) == -1) {
26c853d945SWu Zhangjin 		perror("stat");
27c853d945SWu Zhangjin 		return EXIT_FAILURE;
28c853d945SWu Zhangjin 	}
29c853d945SWu Zhangjin 
30c853d945SWu Zhangjin 	/* Convert hex characters to dec number */
31c853d945SWu Zhangjin 	errno = 0;
32c853d945SWu Zhangjin 	if (sscanf(argv[2], "%llx", &vmlinux_load_addr) != 1) {
33c853d945SWu Zhangjin 		if (errno != 0)
34c853d945SWu Zhangjin 			perror("sscanf");
35c853d945SWu Zhangjin 		else
36c853d945SWu Zhangjin 			fprintf(stderr, "No matching characters\n");
37c853d945SWu Zhangjin 
38c853d945SWu Zhangjin 		return EXIT_FAILURE;
39c853d945SWu Zhangjin 	}
40c853d945SWu Zhangjin 
41c853d945SWu Zhangjin 	vmlinux_size = (uint64_t)sb.st_size;
42c853d945SWu Zhangjin 	vmlinuz_load_addr = vmlinux_load_addr + vmlinux_size;
43c853d945SWu Zhangjin 
44c853d945SWu Zhangjin 	/*
45bec0de4cSHuacai Chen 	 * Align with 64KB: KEXEC needs load sections to be aligned to PAGE_SIZE,
46bec0de4cSHuacai Chen 	 * which may be as large as 64KB depending on the kernel configuration.
47c853d945SWu Zhangjin 	 */
48c853d945SWu Zhangjin 
49bec0de4cSHuacai Chen 	vmlinuz_load_addr += (SZ_64K - vmlinux_size % SZ_64K);
50c853d945SWu Zhangjin 
51c853d945SWu Zhangjin 	printf("0x%llx\n", vmlinuz_load_addr);
52c853d945SWu Zhangjin 
53c853d945SWu Zhangjin 	return EXIT_SUCCESS;
54c853d945SWu Zhangjin }
55