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 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 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 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