mirror of
				https://github.com/smaeul/u-boot.git
				synced 2025-10-25 18:18:19 +01:00 
			
		
		
		
	As part of bringing the master branch back in to next, we need to allow for all of these changes to exist here. Reported-by: Jonas Karlman <jonas@kwiboo.se> Signed-off-by: Tom Rini <trini@konsulko.com>
		
			
				
	
	
		
			192 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			192 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0+
 | |
| /*
 | |
|  * Copyright (c) 2016 Beniamino Galvani
 | |
|  *
 | |
|  * Author: Beniamino Galvani <b.galvani@gmail.com>
 | |
|  * Author: Vyacheslav Bocharov <adeep@lexina.in>
 | |
|  * Author: Neil Armstrong <narmstrong@baylibre.com>
 | |
|  * Author: Alexey Romanov <avromanov@sberdevices.ru>
 | |
|  */
 | |
| 
 | |
| #include <command.h>
 | |
| #include <env.h>
 | |
| #include <asm/arch/sm.h>
 | |
| #include <stdlib.h>
 | |
| #include <display_options.h>
 | |
| #include <vsprintf.h>
 | |
| 
 | |
| static int do_sm_serial(struct cmd_tbl *cmdtp, int flag, int argc,
 | |
| 			char *const argv[])
 | |
| {
 | |
| 	ulong address;
 | |
| 	int ret;
 | |
| 
 | |
| 	if (argc < 2)
 | |
| 		return CMD_RET_USAGE;
 | |
| 
 | |
| 	address = simple_strtoul(argv[1], NULL, 0);
 | |
| 
 | |
| 	ret = meson_sm_get_serial((void *)address, SM_SERIAL_SIZE);
 | |
| 	if (ret)
 | |
| 		return CMD_RET_FAILURE;
 | |
| 
 | |
| 	return CMD_RET_SUCCESS;
 | |
| }
 | |
| 
 | |
| #define MAX_REBOOT_REASONS 14
 | |
| 
 | |
| static const char *reboot_reasons[MAX_REBOOT_REASONS] = {
 | |
| 	[REBOOT_REASON_COLD] = "cold_boot",
 | |
| 	[REBOOT_REASON_NORMAL] = "normal",
 | |
| 	[REBOOT_REASON_RECOVERY] = "recovery",
 | |
| 	[REBOOT_REASON_UPDATE] = "update",
 | |
| 	[REBOOT_REASON_FASTBOOT] = "fastboot",
 | |
| 	[REBOOT_REASON_SUSPEND_OFF] = "suspend_off",
 | |
| 	[REBOOT_REASON_HIBERNATE] = "hibernate",
 | |
| 	[REBOOT_REASON_BOOTLOADER] = "bootloader",
 | |
| 	[REBOOT_REASON_SHUTDOWN_REBOOT] = "shutdown_reboot",
 | |
| 	[REBOOT_REASON_RPMBP] = "rpmbp",
 | |
| 	[REBOOT_REASON_CRASH_DUMP] = "crash_dump",
 | |
| 	[REBOOT_REASON_KERNEL_PANIC] = "kernel_panic",
 | |
| 	[REBOOT_REASON_WATCHDOG_REBOOT] = "watchdog_reboot",
 | |
| };
 | |
| 
 | |
| static int do_sm_reboot_reason(struct cmd_tbl *cmdtp, int flag, int argc,
 | |
| 			       char *const argv[])
 | |
| {
 | |
| 	const char *reason_str;
 | |
| 	char *destarg = NULL;
 | |
| 	int reason;
 | |
| 
 | |
| 	if (argc > 1)
 | |
| 		destarg = argv[1];
 | |
| 
 | |
| 	reason = meson_sm_get_reboot_reason();
 | |
| 	if (reason < 0)
 | |
| 		return CMD_RET_FAILURE;
 | |
| 
 | |
| 	if (reason >= MAX_REBOOT_REASONS ||
 | |
| 	    !reboot_reasons[reason])
 | |
| 		reason_str = "unknown";
 | |
| 	else
 | |
| 		reason_str = reboot_reasons[reason];
 | |
| 
 | |
| 	if (destarg)
 | |
| 		env_set(destarg, reason_str);
 | |
| 	else
 | |
| 		printf("reboot reason: %s (%x)\n", reason_str, reason);
 | |
| 
 | |
| 	return CMD_RET_SUCCESS;
 | |
| }
 | |
| 
 | |
| static int do_efuse_read(struct cmd_tbl *cmdtp, int flag, int argc,
 | |
| 			char *const argv[])
 | |
| {
 | |
| 	ulong address, offset, size;
 | |
| 	int ret;
 | |
| 
 | |
| 	if (argc < 4)
 | |
| 		return CMD_RET_USAGE;
 | |
| 
 | |
| 	offset = simple_strtoul(argv[1], NULL, 0);
 | |
| 	size = simple_strtoul(argv[2], NULL, 0);
 | |
| 
 | |
| 	address = simple_strtoul(argv[3], NULL, 0);
 | |
| 
 | |
| 	ret = meson_sm_read_efuse(offset, (void *)address, size);
 | |
| 	if (ret != size)
 | |
| 		return CMD_RET_FAILURE;
 | |
| 
 | |
| 	return CMD_RET_SUCCESS;
 | |
| }
 | |
| 
 | |
| static int do_efuse_write(struct cmd_tbl *cmdtp, int flag, int argc,
 | |
| 			char *const argv[])
 | |
| {
 | |
| 	ulong address, offset, size;
 | |
| 	int ret;
 | |
| 
 | |
| 	if (argc < 4)
 | |
| 		return CMD_RET_USAGE;
 | |
| 
 | |
| 	offset = simple_strtoul(argv[1], NULL, 0);
 | |
| 	size = simple_strtoul(argv[2], NULL, 0);
 | |
| 
 | |
| 	address = simple_strtoul(argv[3], NULL, 0);
 | |
| 
 | |
| 	ret = meson_sm_write_efuse(offset, (void *)address, size);
 | |
| 	if (ret != size)
 | |
| 		return CMD_RET_FAILURE;
 | |
| 
 | |
| 	return CMD_RET_SUCCESS;
 | |
| }
 | |
| 
 | |
| static int do_efuse_dump(struct cmd_tbl *cmdtp, int flag, int argc,
 | |
| 			char *const argv[])
 | |
| {
 | |
| 	ulong offset, size;
 | |
| 	u8 *buffer;
 | |
| 	int ret;
 | |
| 
 | |
| 	if (argc != 3)
 | |
| 		return CMD_RET_USAGE;
 | |
| 
 | |
| 	offset = simple_strtoul(argv[1], NULL, 0);
 | |
| 	size = simple_strtoul(argv[2], NULL, 0);
 | |
| 	buffer = malloc(size);
 | |
| 	if (!buffer) {
 | |
| 		pr_err("Failed to allocate %lu bytes\n", size);
 | |
| 		return CMD_RET_FAILURE;
 | |
| 	}
 | |
| 
 | |
| 	ret = meson_sm_read_efuse(offset, (void *)buffer, size);
 | |
| 	if (ret != size) {
 | |
| 		ret = CMD_RET_FAILURE;
 | |
| 		goto free_buffer;
 | |
| 	}
 | |
| 
 | |
| 	print_buffer(0, buffer, 1, size, 0);
 | |
| 
 | |
| free_buffer:
 | |
| 	free(buffer);
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static struct cmd_tbl cmd_sm_sub[] = {
 | |
| 	U_BOOT_CMD_MKENT(serial, 2, 1, do_sm_serial, "", ""),
 | |
| 	U_BOOT_CMD_MKENT(reboot_reason, 1, 1, do_sm_reboot_reason, "", ""),
 | |
| 	U_BOOT_CMD_MKENT(efuseread, 4, 1, do_efuse_read, "", ""),
 | |
| 	U_BOOT_CMD_MKENT(efusewrite, 4, 0, do_efuse_write, "", ""),
 | |
| 	U_BOOT_CMD_MKENT(efusedump, 3, 1, do_efuse_dump, "", ""),
 | |
| };
 | |
| 
 | |
| static int do_sm(struct cmd_tbl *cmdtp, int flag, int argc,
 | |
| 		 char *const argv[])
 | |
| {
 | |
| 	struct cmd_tbl *c;
 | |
| 
 | |
| 	if (argc < 2)
 | |
| 		return CMD_RET_USAGE;
 | |
| 
 | |
| 	/* Strip off leading 'sm' command argument */
 | |
| 	argc--;
 | |
| 	argv++;
 | |
| 
 | |
| 	c = find_cmd_tbl(argv[0], &cmd_sm_sub[0], ARRAY_SIZE(cmd_sm_sub));
 | |
| 
 | |
| 	if (c)
 | |
| 		return c->cmd(cmdtp, flag, argc, argv);
 | |
| 	else
 | |
| 		return CMD_RET_USAGE;
 | |
| }
 | |
| 
 | |
| U_BOOT_CMD(
 | |
| 	sm, 5, 0, do_sm,
 | |
| 	"Secure Monitor Control",
 | |
| 	"serial <address> - read chip unique id to memory address\n"
 | |
| 	"sm reboot_reason [name] - get reboot reason and store to environment\n"
 | |
| 	"sm efuseread <offset> <size> <address> - read efuse to memory address\n"
 | |
| 	"sm efusewrite <offset> <size> <address> - write into efuse from memory address\n"
 | |
| 	"sm efusedump <offset> <size> - dump efuse data range to console"
 | |
| );
 |