1*592f4aedSMax Krummenacher /*
2*592f4aedSMax Krummenacher * Copyright (C) 2014-2016, Toradex AG
3*592f4aedSMax Krummenacher *
4*592f4aedSMax Krummenacher * SPDX-License-Identifier: GPL-2.0+
5*592f4aedSMax Krummenacher */
6*592f4aedSMax Krummenacher
7*592f4aedSMax Krummenacher /*
8*592f4aedSMax Krummenacher * Helpers for i.MX OTP fusing during module production
9*592f4aedSMax Krummenacher */
10*592f4aedSMax Krummenacher
11*592f4aedSMax Krummenacher #include <common.h>
12*592f4aedSMax Krummenacher #ifndef CONFIG_SPL_BUILD
13*592f4aedSMax Krummenacher #include <console.h>
14*592f4aedSMax Krummenacher #include <fuse.h>
15*592f4aedSMax Krummenacher
mfgr_fuse(void)16*592f4aedSMax Krummenacher static int mfgr_fuse(void)
17*592f4aedSMax Krummenacher {
18*592f4aedSMax Krummenacher unsigned val, val6;
19*592f4aedSMax Krummenacher
20*592f4aedSMax Krummenacher fuse_sense(0, 5, &val);
21*592f4aedSMax Krummenacher printf("Fuse 0, 5: %8x\n", val);
22*592f4aedSMax Krummenacher fuse_sense(0, 6, &val6);
23*592f4aedSMax Krummenacher printf("Fuse 0, 6: %8x\n", val6);
24*592f4aedSMax Krummenacher fuse_sense(4, 3, &val);
25*592f4aedSMax Krummenacher printf("Fuse 4, 3: %8x\n", val);
26*592f4aedSMax Krummenacher fuse_sense(4, 2, &val);
27*592f4aedSMax Krummenacher printf("Fuse 4, 2: %8x\n", val);
28*592f4aedSMax Krummenacher if (val6 & 0x10) {
29*592f4aedSMax Krummenacher puts("BT_FUSE_SEL already fused, will do nothing\n");
30*592f4aedSMax Krummenacher return CMD_RET_FAILURE;
31*592f4aedSMax Krummenacher }
32*592f4aedSMax Krummenacher /* boot cfg */
33*592f4aedSMax Krummenacher fuse_prog(0, 5, 0x00005072);
34*592f4aedSMax Krummenacher /* BT_FUSE_SEL */
35*592f4aedSMax Krummenacher fuse_prog(0, 6, 0x00000010);
36*592f4aedSMax Krummenacher return CMD_RET_SUCCESS;
37*592f4aedSMax Krummenacher }
38*592f4aedSMax Krummenacher
do_mfgr_fuse(cmd_tbl_t * cmdtp,int flag,int argc,char * const argv[])39*592f4aedSMax Krummenacher int do_mfgr_fuse(cmd_tbl_t *cmdtp, int flag, int argc,
40*592f4aedSMax Krummenacher char * const argv[])
41*592f4aedSMax Krummenacher {
42*592f4aedSMax Krummenacher int ret;
43*592f4aedSMax Krummenacher puts("Fusing...\n");
44*592f4aedSMax Krummenacher ret = mfgr_fuse();
45*592f4aedSMax Krummenacher if (ret == CMD_RET_SUCCESS)
46*592f4aedSMax Krummenacher puts("done.\n");
47*592f4aedSMax Krummenacher else
48*592f4aedSMax Krummenacher puts("failed.\n");
49*592f4aedSMax Krummenacher return ret;
50*592f4aedSMax Krummenacher }
51*592f4aedSMax Krummenacher
do_updt_fuse(cmd_tbl_t * cmdtp,int flag,int argc,char * const argv[])52*592f4aedSMax Krummenacher int do_updt_fuse(cmd_tbl_t *cmdtp, int flag, int argc,
53*592f4aedSMax Krummenacher char * const argv[])
54*592f4aedSMax Krummenacher {
55*592f4aedSMax Krummenacher unsigned val;
56*592f4aedSMax Krummenacher int ret;
57*592f4aedSMax Krummenacher int confirmed = argc >= 1 && !strcmp(argv[1], "-y");
58*592f4aedSMax Krummenacher
59*592f4aedSMax Krummenacher /* can be used in scripts for command availability check */
60*592f4aedSMax Krummenacher if (argc >= 1 && !strcmp(argv[1], "-n"))
61*592f4aedSMax Krummenacher return CMD_RET_SUCCESS;
62*592f4aedSMax Krummenacher
63*592f4aedSMax Krummenacher /* boot cfg */
64*592f4aedSMax Krummenacher fuse_sense(0, 5, &val);
65*592f4aedSMax Krummenacher printf("Fuse 0, 5: %8x\n", val);
66*592f4aedSMax Krummenacher if (val & 0x10) {
67*592f4aedSMax Krummenacher puts("Fast boot mode already fused, no need to fuse\n");
68*592f4aedSMax Krummenacher return CMD_RET_SUCCESS;
69*592f4aedSMax Krummenacher }
70*592f4aedSMax Krummenacher if (!confirmed) {
71*592f4aedSMax Krummenacher puts("Warning: Programming fuses is an irreversible operation!\n"
72*592f4aedSMax Krummenacher " Updating to fast boot mode prevents easy\n"
73*592f4aedSMax Krummenacher " downgrading to previous BSP versions.\n"
74*592f4aedSMax Krummenacher "\nReally perform this fuse programming? <y/N>\n");
75*592f4aedSMax Krummenacher if (!confirm_yesno())
76*592f4aedSMax Krummenacher return CMD_RET_FAILURE;
77*592f4aedSMax Krummenacher }
78*592f4aedSMax Krummenacher puts("Fusing fast boot mode...\n");
79*592f4aedSMax Krummenacher ret = fuse_prog(0, 5, 0x00005072);
80*592f4aedSMax Krummenacher if (ret == CMD_RET_SUCCESS)
81*592f4aedSMax Krummenacher puts("done.\n");
82*592f4aedSMax Krummenacher else
83*592f4aedSMax Krummenacher puts("failed.\n");
84*592f4aedSMax Krummenacher return ret;
85*592f4aedSMax Krummenacher }
86*592f4aedSMax Krummenacher
87*592f4aedSMax Krummenacher U_BOOT_CMD(
88*592f4aedSMax Krummenacher mfgr_fuse, 1, 0, do_mfgr_fuse,
89*592f4aedSMax Krummenacher "OTP fusing during module production",
90*592f4aedSMax Krummenacher ""
91*592f4aedSMax Krummenacher );
92*592f4aedSMax Krummenacher
93*592f4aedSMax Krummenacher U_BOOT_CMD(
94*592f4aedSMax Krummenacher updt_fuse, 2, 0, do_updt_fuse,
95*592f4aedSMax Krummenacher "OTP fusing during module update",
96*592f4aedSMax Krummenacher "updt_fuse [-n] [-y] - boot cfg fast boot mode fusing"
97*592f4aedSMax Krummenacher );
98*592f4aedSMax Krummenacher #endif /* CONFIG_SPL_BUILD */
99