blob: e2206cdf0d54a5422a9c5e02eb06f6e746c79f97 [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +00002/*
3 * (C) Copyright 2009-2013 ADVANSEE
4 * Benoît Thébaudeau <benoit.thebaudeau@advansee.com>
5 *
6 * Based on the mpc512x iim code:
7 * Copyright 2008 Silicon Turnkey Express, Inc.
8 * Martha Marx <mmarx@silicontkx.com>
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +00009 */
10
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +000011#include <command.h>
Simon Glassa73bda42015-11-08 23:47:45 -070012#include <console.h>
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +000013#include <fuse.h>
Angus Ainslief08743f2021-11-28 08:02:53 -080014#include <mapmem.h>
Tom Rinidec7ea02024-05-20 13:35:03 -060015#include <vsprintf.h>
Masahiro Yamada56a931c2016-09-21 11:28:55 +090016#include <linux/errno.h>
Tom Riniee8ed542025-05-14 16:46:00 -060017#include <linux/string.h>
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +000018
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +000019static int confirm_prog(void)
20{
21 puts("Warning: Programming fuses is an irreversible operation!\n"
22 " This may brick your system.\n"
23 " Use this command only if you are sure of "
24 "what you are doing!\n"
25 "\nReally perform this fuse programming? <y/N>\n");
26
Pierre Aubert5fef9d32014-04-24 10:30:07 +020027 if (confirm_yesno())
28 return 1;
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +000029
30 puts("Fuse programming aborted\n");
31 return 0;
32}
33
Simon Glassed38aef2020-05-10 11:40:03 -060034static int do_fuse(struct cmd_tbl *cmdtp, int flag, int argc,
35 char *const argv[])
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +000036{
Simon Glass793a98e2023-11-18 14:05:20 -070037 const char *op = cmd_arg1(argc, argv);
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +000038 int confirmed = argc >= 3 && !strcmp(argv[2], "-y");
Angus Ainslie5149bca2021-11-28 08:02:52 -080039 u32 bank, word, cnt, val, cmp;
Angus Ainslief08743f2021-11-28 08:02:53 -080040 ulong addr;
41 void *buf, *start;
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +000042 int ret, i;
43
44 argc -= 2 + confirmed;
45 argv += 2 + confirmed;
46
Harsha Vardhan V M80f12282025-03-19 14:17:12 +053047 if (IS_ENABLED(CONFIG_CMD_FUSE_WRITEBUFF) && !strcmp(op, "writebuff")) {
48 if (argc == 1)
49 addr = simple_strtoul(argv[0], NULL, 16);
50 else
51 return CMD_RET_USAGE;
52 } else {
53 if (argc < 2)
54 return CMD_RET_USAGE;
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +000055
Harsha Vardhan V M80f12282025-03-19 14:17:12 +053056 bank = simple_strtoul(argv[0], NULL, 0);
57 word = simple_strtoul(argv[1], NULL, 0);
58 }
Harsha Vardhan V M56e9d2b2025-03-19 14:17:10 +053059
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +000060 if (!strcmp(op, "read")) {
61 if (argc == 2)
62 cnt = 1;
Harsha Vardhan V M56e9d2b2025-03-19 14:17:10 +053063 else if (argc == 3)
64 cnt = simple_strtoul(argv[2], NULL, 0);
65 else
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +000066 return CMD_RET_USAGE;
67
68 printf("Reading bank %u:\n", bank);
69 for (i = 0; i < cnt; i++, word++) {
70 if (!(i % 4))
71 printf("\nWord 0x%.8x:", word);
72
73 ret = fuse_read(bank, word, &val);
74 if (ret)
75 goto err;
76
77 printf(" %.8x", val);
78 }
79 putc('\n');
Angus Ainslief08743f2021-11-28 08:02:53 -080080 } else if (!strcmp(op, "readm")) {
81 if (argc == 3)
82 cnt = 1;
Harsha Vardhan V M56e9d2b2025-03-19 14:17:10 +053083 else if (argc == 4)
84 cnt = simple_strtoul(argv[3], NULL, 0);
85 else
Angus Ainslief08743f2021-11-28 08:02:53 -080086 return CMD_RET_USAGE;
87
88 addr = simple_strtoul(argv[2], NULL, 16);
89
90 start = map_sysmem(addr, 4);
91 buf = start;
92
93 printf("Reading bank %u len %u to 0x%lx\n", bank, cnt, addr);
94 for (i = 0; i < cnt; i++, word++) {
95 ret = fuse_read(bank, word, &val);
96 if (ret)
97 goto err;
98
99 *((u32 *)buf) = val;
100 buf += 4;
101 }
102
103 unmap_sysmem(start);
Angus Ainslie5149bca2021-11-28 08:02:52 -0800104 } else if (!strcmp(op, "cmp")) {
Harsha Vardhan V M56e9d2b2025-03-19 14:17:10 +0530105 if (argc == 3)
106 cmp = simple_strtoul(argv[2], NULL, 0);
107 else
Angus Ainslie5149bca2021-11-28 08:02:52 -0800108 return CMD_RET_USAGE;
109
110 printf("Comparing bank %u:\n", bank);
111 printf("\nWord 0x%.8x:", word);
112 printf("\nValue 0x%.8x:", cmp);
113
114 ret = fuse_read(bank, word, &val);
115 if (ret)
116 goto err;
117
118 printf("0x%.8x\n", val);
119 if (val != cmp) {
120 printf("failed\n");
121 return CMD_RET_FAILURE;
122 }
123 printf("passed\n");
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +0000124 } else if (!strcmp(op, "sense")) {
125 if (argc == 2)
126 cnt = 1;
Harsha Vardhan V M56e9d2b2025-03-19 14:17:10 +0530127 else if (argc == 3)
128 cnt = simple_strtoul(argv[2], NULL, 0);
129 else
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +0000130 return CMD_RET_USAGE;
131
132 printf("Sensing bank %u:\n", bank);
133 for (i = 0; i < cnt; i++, word++) {
134 if (!(i % 4))
135 printf("\nWord 0x%.8x:", word);
136
137 ret = fuse_sense(bank, word, &val);
138 if (ret)
139 goto err;
140
141 printf(" %.8x", val);
142 }
143 putc('\n');
144 } else if (!strcmp(op, "prog")) {
145 if (argc < 3)
146 return CMD_RET_USAGE;
147
148 for (i = 2; i < argc; i++, word++) {
Harsha Vardhan V M56e9d2b2025-03-19 14:17:10 +0530149 val = simple_strtoul(argv[i], NULL, 16);
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +0000150
151 printf("Programming bank %u word 0x%.8x to 0x%.8x...\n",
152 bank, word, val);
153 if (!confirmed && !confirm_prog())
154 return CMD_RET_FAILURE;
155 ret = fuse_prog(bank, word, val);
156 if (ret)
157 goto err;
158 }
159 } else if (!strcmp(op, "override")) {
160 if (argc < 3)
161 return CMD_RET_USAGE;
162
163 for (i = 2; i < argc; i++, word++) {
Harsha Vardhan V M56e9d2b2025-03-19 14:17:10 +0530164 val = simple_strtoul(argv[i], NULL, 16);
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +0000165
166 printf("Overriding bank %u word 0x%.8x with "
167 "0x%.8x...\n", bank, word, val);
168 ret = fuse_override(bank, word, val);
169 if (ret)
170 goto err;
171 }
Harsha Vardhan V M80f12282025-03-19 14:17:12 +0530172 } else if (IS_ENABLED(CONFIG_CMD_FUSE_WRITEBUFF) && !strcmp(op, "writebuff")) {
173 printf("Programming fuses using a structured buffer in memory "
174 "starting at addr 0x%lx\n", addr);
175 if (!confirmed && !confirm_prog())
176 return CMD_RET_FAILURE;
177
178 ret = fuse_writebuff(addr);
179 if (ret)
180 goto err;
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +0000181 } else {
182 return CMD_RET_USAGE;
183 }
184
185 return 0;
186
187err:
188 puts("ERROR\n");
Hector Palacios17e99cc2014-11-20 09:27:42 +0100189 return CMD_RET_FAILURE;
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +0000190}
191
192U_BOOT_CMD(
193 fuse, CONFIG_SYS_MAXARGS, 0, do_fuse,
194 "Fuse sub-system",
195 "read <bank> <word> [<cnt>] - read 1 or 'cnt' fuse words,\n"
196 " starting at 'word'\n"
Angus Ainslie5149bca2021-11-28 08:02:52 -0800197 "fuse cmp <bank> <word> <hexval> - compare 'hexval' to fuse\n"
198 " at 'word'\n"
Angus Ainslief08743f2021-11-28 08:02:53 -0800199 "fuse readm <bank> <word> <addr> [<cnt>] - read 1 or 'cnt' fuse words,\n"
200 " starting at 'word' into memory at 'addr'\n"
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +0000201 "fuse sense <bank> <word> [<cnt>] - sense 1 or 'cnt' fuse words,\n"
202 " starting at 'word'\n"
203 "fuse prog [-y] <bank> <word> <hexval> [<hexval>...] - program 1 or\n"
204 " several fuse words, starting at 'word' (PERMANENT)\n"
205 "fuse override <bank> <word> <hexval> [<hexval>...] - override 1 or\n"
Harsha Vardhan V M80f12282025-03-19 14:17:12 +0530206 " several fuse words, starting at 'word'\n"
207#ifdef CONFIG_CMD_FUSE_WRITEBUFF
208 "fuse writebuff [-y] <addr> - program fuse data\n"
209 " using a structured buffer in memory starting at 'addr'\n"
210#endif /* CONFIG_CMD_FUSE_WRITEBUFF */
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +0000211);