blob: 6c42c096809332acc911886fe23cef323ab7f7e7 [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>
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +000017
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +000018static int confirm_prog(void)
19{
20 puts("Warning: Programming fuses is an irreversible operation!\n"
21 " This may brick your system.\n"
22 " Use this command only if you are sure of "
23 "what you are doing!\n"
24 "\nReally perform this fuse programming? <y/N>\n");
25
Pierre Aubert5fef9d32014-04-24 10:30:07 +020026 if (confirm_yesno())
27 return 1;
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +000028
29 puts("Fuse programming aborted\n");
30 return 0;
31}
32
Simon Glassed38aef2020-05-10 11:40:03 -060033static int do_fuse(struct cmd_tbl *cmdtp, int flag, int argc,
34 char *const argv[])
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +000035{
Simon Glass793a98e2023-11-18 14:05:20 -070036 const char *op = cmd_arg1(argc, argv);
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +000037 int confirmed = argc >= 3 && !strcmp(argv[2], "-y");
Angus Ainslie5149bca2021-11-28 08:02:52 -080038 u32 bank, word, cnt, val, cmp;
Angus Ainslief08743f2021-11-28 08:02:53 -080039 ulong addr;
40 void *buf, *start;
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +000041 int ret, i;
42
43 argc -= 2 + confirmed;
44 argv += 2 + confirmed;
45
Harsha Vardhan V M80f12282025-03-19 14:17:12 +053046 if (IS_ENABLED(CONFIG_CMD_FUSE_WRITEBUFF) && !strcmp(op, "writebuff")) {
47 if (argc == 1)
48 addr = simple_strtoul(argv[0], NULL, 16);
49 else
50 return CMD_RET_USAGE;
51 } else {
52 if (argc < 2)
53 return CMD_RET_USAGE;
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +000054
Harsha Vardhan V M80f12282025-03-19 14:17:12 +053055 bank = simple_strtoul(argv[0], NULL, 0);
56 word = simple_strtoul(argv[1], NULL, 0);
57 }
Harsha Vardhan V M56e9d2b2025-03-19 14:17:10 +053058
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +000059 if (!strcmp(op, "read")) {
60 if (argc == 2)
61 cnt = 1;
Harsha Vardhan V M56e9d2b2025-03-19 14:17:10 +053062 else if (argc == 3)
63 cnt = simple_strtoul(argv[2], NULL, 0);
64 else
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +000065 return CMD_RET_USAGE;
66
67 printf("Reading bank %u:\n", bank);
68 for (i = 0; i < cnt; i++, word++) {
69 if (!(i % 4))
70 printf("\nWord 0x%.8x:", word);
71
72 ret = fuse_read(bank, word, &val);
73 if (ret)
74 goto err;
75
76 printf(" %.8x", val);
77 }
78 putc('\n');
Angus Ainslief08743f2021-11-28 08:02:53 -080079 } else if (!strcmp(op, "readm")) {
80 if (argc == 3)
81 cnt = 1;
Harsha Vardhan V M56e9d2b2025-03-19 14:17:10 +053082 else if (argc == 4)
83 cnt = simple_strtoul(argv[3], NULL, 0);
84 else
Angus Ainslief08743f2021-11-28 08:02:53 -080085 return CMD_RET_USAGE;
86
87 addr = simple_strtoul(argv[2], NULL, 16);
88
89 start = map_sysmem(addr, 4);
90 buf = start;
91
92 printf("Reading bank %u len %u to 0x%lx\n", bank, cnt, addr);
93 for (i = 0; i < cnt; i++, word++) {
94 ret = fuse_read(bank, word, &val);
95 if (ret)
96 goto err;
97
98 *((u32 *)buf) = val;
99 buf += 4;
100 }
101
102 unmap_sysmem(start);
Angus Ainslie5149bca2021-11-28 08:02:52 -0800103 } else if (!strcmp(op, "cmp")) {
Harsha Vardhan V M56e9d2b2025-03-19 14:17:10 +0530104 if (argc == 3)
105 cmp = simple_strtoul(argv[2], NULL, 0);
106 else
Angus Ainslie5149bca2021-11-28 08:02:52 -0800107 return CMD_RET_USAGE;
108
109 printf("Comparing bank %u:\n", bank);
110 printf("\nWord 0x%.8x:", word);
111 printf("\nValue 0x%.8x:", cmp);
112
113 ret = fuse_read(bank, word, &val);
114 if (ret)
115 goto err;
116
117 printf("0x%.8x\n", val);
118 if (val != cmp) {
119 printf("failed\n");
120 return CMD_RET_FAILURE;
121 }
122 printf("passed\n");
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +0000123 } else if (!strcmp(op, "sense")) {
124 if (argc == 2)
125 cnt = 1;
Harsha Vardhan V M56e9d2b2025-03-19 14:17:10 +0530126 else if (argc == 3)
127 cnt = simple_strtoul(argv[2], NULL, 0);
128 else
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +0000129 return CMD_RET_USAGE;
130
131 printf("Sensing bank %u:\n", bank);
132 for (i = 0; i < cnt; i++, word++) {
133 if (!(i % 4))
134 printf("\nWord 0x%.8x:", word);
135
136 ret = fuse_sense(bank, word, &val);
137 if (ret)
138 goto err;
139
140 printf(" %.8x", val);
141 }
142 putc('\n');
143 } else if (!strcmp(op, "prog")) {
144 if (argc < 3)
145 return CMD_RET_USAGE;
146
147 for (i = 2; i < argc; i++, word++) {
Harsha Vardhan V M56e9d2b2025-03-19 14:17:10 +0530148 val = simple_strtoul(argv[i], NULL, 16);
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +0000149
150 printf("Programming bank %u word 0x%.8x to 0x%.8x...\n",
151 bank, word, val);
152 if (!confirmed && !confirm_prog())
153 return CMD_RET_FAILURE;
154 ret = fuse_prog(bank, word, val);
155 if (ret)
156 goto err;
157 }
158 } else if (!strcmp(op, "override")) {
159 if (argc < 3)
160 return CMD_RET_USAGE;
161
162 for (i = 2; i < argc; i++, word++) {
Harsha Vardhan V M56e9d2b2025-03-19 14:17:10 +0530163 val = simple_strtoul(argv[i], NULL, 16);
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +0000164
165 printf("Overriding bank %u word 0x%.8x with "
166 "0x%.8x...\n", bank, word, val);
167 ret = fuse_override(bank, word, val);
168 if (ret)
169 goto err;
170 }
Harsha Vardhan V M80f12282025-03-19 14:17:12 +0530171 } else if (IS_ENABLED(CONFIG_CMD_FUSE_WRITEBUFF) && !strcmp(op, "writebuff")) {
172 printf("Programming fuses using a structured buffer in memory "
173 "starting at addr 0x%lx\n", addr);
174 if (!confirmed && !confirm_prog())
175 return CMD_RET_FAILURE;
176
177 ret = fuse_writebuff(addr);
178 if (ret)
179 goto err;
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +0000180 } else {
181 return CMD_RET_USAGE;
182 }
183
184 return 0;
185
186err:
187 puts("ERROR\n");
Hector Palacios17e99cc2014-11-20 09:27:42 +0100188 return CMD_RET_FAILURE;
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +0000189}
190
191U_BOOT_CMD(
192 fuse, CONFIG_SYS_MAXARGS, 0, do_fuse,
193 "Fuse sub-system",
194 "read <bank> <word> [<cnt>] - read 1 or 'cnt' fuse words,\n"
195 " starting at 'word'\n"
Angus Ainslie5149bca2021-11-28 08:02:52 -0800196 "fuse cmp <bank> <word> <hexval> - compare 'hexval' to fuse\n"
197 " at 'word'\n"
Angus Ainslief08743f2021-11-28 08:02:53 -0800198 "fuse readm <bank> <word> <addr> [<cnt>] - read 1 or 'cnt' fuse words,\n"
199 " starting at 'word' into memory at 'addr'\n"
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +0000200 "fuse sense <bank> <word> [<cnt>] - sense 1 or 'cnt' fuse words,\n"
201 " starting at 'word'\n"
202 "fuse prog [-y] <bank> <word> <hexval> [<hexval>...] - program 1 or\n"
203 " several fuse words, starting at 'word' (PERMANENT)\n"
204 "fuse override <bank> <word> <hexval> [<hexval>...] - override 1 or\n"
Harsha Vardhan V M80f12282025-03-19 14:17:12 +0530205 " several fuse words, starting at 'word'\n"
206#ifdef CONFIG_CMD_FUSE_WRITEBUFF
207 "fuse writebuff [-y] <addr> - program fuse data\n"
208 " using a structured buffer in memory starting at 'addr'\n"
209#endif /* CONFIG_CMD_FUSE_WRITEBUFF */
Benoît Thébaudeaua8e88552013-04-23 10:17:40 +0000210);