xref: /rk3399_ARM-atf/tools/nxp/create_pbl/create_pbl.c (revision 326694760f270861631aee5bec748572751cf50c)
1*32669476SPankaj Gupta /*
2*32669476SPankaj Gupta  * Copyright 2021 NXP
3*32669476SPankaj Gupta  *
4*32669476SPankaj Gupta  * SPDX-License-Identifier: BSD-3-Clause
5*32669476SPankaj Gupta  *
6*32669476SPankaj Gupta  */
7*32669476SPankaj Gupta 
8*32669476SPankaj Gupta #include <stdbool.h>
9*32669476SPankaj Gupta #include <stdint.h>
10*32669476SPankaj Gupta #include <stdio.h>
11*32669476SPankaj Gupta #include <stdlib.h>
12*32669476SPankaj Gupta #include <string.h>
13*32669476SPankaj Gupta 
14*32669476SPankaj Gupta #include <getopt.h>
15*32669476SPankaj Gupta #include <unistd.h>
16*32669476SPankaj Gupta 
17*32669476SPankaj Gupta #define NUM_MEM_BLOCK		1
18*32669476SPankaj Gupta #define FOUR_BYTE_ALIGN		4
19*32669476SPankaj Gupta #define EIGHT_BYTE_ALIGN	8
20*32669476SPankaj Gupta #define SIZE_TWO_PBL_CMD	24
21*32669476SPankaj Gupta 
22*32669476SPankaj Gupta /* Define for add_boot_ptr_cmd() */
23*32669476SPankaj Gupta #define BOOTPTR_ADDR 0x09570604
24*32669476SPankaj Gupta #define CSF_ADDR_SB 0x09ee0200
25*32669476SPankaj Gupta /* CCSR write command to address 0x1e00400 i.e BOOTLOCPTR */
26*32669476SPankaj Gupta #define BOOTPTR_ADDR_CH3 0x31e00400
27*32669476SPankaj Gupta /* Load CSF header command */
28*32669476SPankaj Gupta #define CSF_ADDR_SB_CH3 0x80220000
29*32669476SPankaj Gupta 
30*32669476SPankaj Gupta #define	MAND_ARG_MASK				0xFFF3
31*32669476SPankaj Gupta #define	ARG_INIT_MASK				0xFF00
32*32669476SPankaj Gupta #define RCW_FILE_NAME_ARG_MASK			0x0080
33*32669476SPankaj Gupta #define IN_FILE_NAME_ARG_MASK			0x0040
34*32669476SPankaj Gupta #define CHASSIS_ARG_MASK			0x0020
35*32669476SPankaj Gupta #define BOOT_SRC_ARG_MASK			0x0010
36*32669476SPankaj Gupta #define ENTRY_POINT_ADDR_ARG_MASK		0x0008
37*32669476SPankaj Gupta #define BL2_BIN_STRG_LOC_BOOT_SRC_ARG_MASK	0x0004
38*32669476SPankaj Gupta #define BL2_BIN_CPY_DEST_ADDR_ARG_MASK		0x0002
39*32669476SPankaj Gupta #define OP_FILE_NAME_ARG_MASK			0x0001
40*32669476SPankaj Gupta 
41*32669476SPankaj Gupta /* Define for add_cpy_cmd() */
42*32669476SPankaj Gupta #define OFFSET_MASK		        0x00ffffff
43*32669476SPankaj Gupta #define WRITE_CMD_BASE		    0x81000000
44*32669476SPankaj Gupta #define MAX_PBI_DATA_LEN_BYTE	64
45*32669476SPankaj Gupta 
46*32669476SPankaj Gupta /* 140 Bytes = Preamble + LOAD RCW command + RCW (128 bytes) + Checksum */
47*32669476SPankaj Gupta #define CHS3_CRC_PAYLOAD_START_OFFSET 140
48*32669476SPankaj Gupta 
49*32669476SPankaj Gupta #define PBI_CRC_POLYNOMIAL	0x04c11db7
50*32669476SPankaj Gupta 
51*32669476SPankaj Gupta typedef enum {
52*32669476SPankaj Gupta 	CHASSIS_UNKNOWN,
53*32669476SPankaj Gupta 	CHASSIS_2,
54*32669476SPankaj Gupta 	CHASSIS_3,
55*32669476SPankaj Gupta 	CHASSIS_3_2,
56*32669476SPankaj Gupta 	CHASSIS_MAX    /* must be last item in list */
57*32669476SPankaj Gupta } chassis_t;
58*32669476SPankaj Gupta 
59*32669476SPankaj Gupta typedef enum {
60*32669476SPankaj Gupta 	UNKNOWN_BOOT = 0,
61*32669476SPankaj Gupta 	IFC_NOR_BOOT,
62*32669476SPankaj Gupta 	IFC_NAND_BOOT,
63*32669476SPankaj Gupta 	QSPI_BOOT,
64*32669476SPankaj Gupta 	SD_BOOT,
65*32669476SPankaj Gupta 	EMMC_BOOT,
66*32669476SPankaj Gupta 	FLXSPI_NOR_BOOT,
67*32669476SPankaj Gupta 	FLXSPI_NAND_BOOT,
68*32669476SPankaj Gupta 	FLXSPI_NAND4K_BOOT,
69*32669476SPankaj Gupta 	MAX_BOOT    /* must be last item in list */
70*32669476SPankaj Gupta } boot_src_t;
71*32669476SPankaj Gupta 
72*32669476SPankaj Gupta /* Base Addresses where PBL image is copied depending on the boot source.
73*32669476SPankaj Gupta  * Boot address map varies as per Chassis architecture.
74*32669476SPankaj Gupta  */
75*32669476SPankaj Gupta #define BASE_ADDR_UNDEFINED  0xFFFFFFFF
76*32669476SPankaj Gupta #define BASE_ADDR_QSPI       0x20000000
77*32669476SPankaj Gupta #define BASE_ADDR_SD         0x00001000
78*32669476SPankaj Gupta #define BASE_ADDR_IFC_NOR    0x30000000
79*32669476SPankaj Gupta #define BASE_ADDR_EMMC       0x00001000
80*32669476SPankaj Gupta #define BASE_ADDR_FLX_NOR    0x20000000
81*32669476SPankaj Gupta #define BASE_ADDR_NAND       0x20000000
82*32669476SPankaj Gupta 
83*32669476SPankaj Gupta uint32_t base_addr_ch3[MAX_BOOT] = {
84*32669476SPankaj Gupta 	BASE_ADDR_UNDEFINED,
85*32669476SPankaj Gupta 	BASE_ADDR_IFC_NOR,
86*32669476SPankaj Gupta 	BASE_ADDR_UNDEFINED,	/*IFC NAND */
87*32669476SPankaj Gupta 	BASE_ADDR_QSPI,
88*32669476SPankaj Gupta 	BASE_ADDR_SD,
89*32669476SPankaj Gupta 	BASE_ADDR_EMMC,
90*32669476SPankaj Gupta 	BASE_ADDR_UNDEFINED,	/*FLXSPI NOR */
91*32669476SPankaj Gupta 	BASE_ADDR_UNDEFINED,	/*FLXSPI NAND 2K */
92*32669476SPankaj Gupta 	BASE_ADDR_UNDEFINED	/*FLXSPI NAND 4K */
93*32669476SPankaj Gupta };
94*32669476SPankaj Gupta 
95*32669476SPankaj Gupta uint32_t base_addr_ch32[MAX_BOOT] = {
96*32669476SPankaj Gupta 	BASE_ADDR_UNDEFINED,
97*32669476SPankaj Gupta 	BASE_ADDR_UNDEFINED,	/* IFC NOR */
98*32669476SPankaj Gupta 	BASE_ADDR_UNDEFINED,	/* IFC NAND */
99*32669476SPankaj Gupta 	BASE_ADDR_UNDEFINED,	/* QSPI */
100*32669476SPankaj Gupta 	BASE_ADDR_SD,
101*32669476SPankaj Gupta 	BASE_ADDR_EMMC,
102*32669476SPankaj Gupta 	BASE_ADDR_FLX_NOR,
103*32669476SPankaj Gupta 	BASE_ADDR_UNDEFINED,	/*FLXSPI NAND 2K */
104*32669476SPankaj Gupta 	BASE_ADDR_UNDEFINED	/*FLXSPI NAND 4K */
105*32669476SPankaj Gupta };
106*32669476SPankaj Gupta 
107*32669476SPankaj Gupta /* for Chassis 3 */
108*32669476SPankaj Gupta uint32_t blk_cpy_hdr_map_ch3[] = {
109*32669476SPankaj Gupta 
110*32669476SPankaj Gupta 	0,		    /* Unknown Boot Source */
111*32669476SPankaj Gupta 	0x80000020,	/* NOR_BOOT */
112*32669476SPankaj Gupta 	0x0,		/* NAND_BOOT */
113*32669476SPankaj Gupta 	0x80000062,	/* QSPI_BOOT */
114*32669476SPankaj Gupta 	0x80000040,	/* SD_BOOT */
115*32669476SPankaj Gupta 	0x80000041,	/* EMMC_BOOT */
116*32669476SPankaj Gupta 	0x0,		/* FLEXSPI NOR_BOOT */
117*32669476SPankaj Gupta 	0x0,	/* FLEX SPI NAND2K BOOT */
118*32669476SPankaj Gupta 	0x0,	/* CHASIS3_2_NAND4K_BOOT */
119*32669476SPankaj Gupta };
120*32669476SPankaj Gupta 
121*32669476SPankaj Gupta uint32_t blk_cpy_hdr_map_ch32[] = {
122*32669476SPankaj Gupta 	0,		    /* Unknown Boot Source */
123*32669476SPankaj Gupta 	0x0,		/* NOR_BOOT */
124*32669476SPankaj Gupta 	0x0,		/* NAND_BOOT */
125*32669476SPankaj Gupta 	0x0,		/* QSPI_BOOT */
126*32669476SPankaj Gupta 	0x80000008,	/* SD_BOOT */
127*32669476SPankaj Gupta 	0x80000009,	/* EMMC_BOOT */
128*32669476SPankaj Gupta 	0x8000000F,	/* FLEXSPI NOR_BOOT */
129*32669476SPankaj Gupta 	0x8000000C,	/* FLEX SPI NAND2K BOOT */
130*32669476SPankaj Gupta 	0x8000000D,	/* CHASIS3_2_NAND4K_BOOT */
131*32669476SPankaj Gupta };
132*32669476SPankaj Gupta 
133*32669476SPankaj Gupta char *boot_src_string[] = {
134*32669476SPankaj Gupta 	"UNKNOWN_BOOT",
135*32669476SPankaj Gupta 	"IFC_NOR_BOOT",
136*32669476SPankaj Gupta 	"IFC_NAND_BOOT",
137*32669476SPankaj Gupta 	"QSPI_BOOT",
138*32669476SPankaj Gupta 	"SD_BOOT",
139*32669476SPankaj Gupta 	"EMMC_BOOT",
140*32669476SPankaj Gupta 	"FLXSPI_NOR_BOOT",
141*32669476SPankaj Gupta 	"FLXSPI_NAND_BOOT",
142*32669476SPankaj Gupta 	"FLXSPI_NAND4K_BOOT",
143*32669476SPankaj Gupta };
144*32669476SPankaj Gupta 
145*32669476SPankaj Gupta enum stop_command {
146*32669476SPankaj Gupta 	STOP_COMMAND = 0,
147*32669476SPankaj Gupta 	CRC_STOP_COMMAND
148*32669476SPankaj Gupta };
149*32669476SPankaj Gupta 
150*32669476SPankaj Gupta /* Structure will get populated in the main function
151*32669476SPankaj Gupta  * as part of parsing the command line arguments.
152*32669476SPankaj Gupta  * All member parameters are mandatory except:
153*32669476SPankaj Gupta  *	-ep
154*32669476SPankaj Gupta  *	-src_addr
155*32669476SPankaj Gupta  */
156*32669476SPankaj Gupta struct pbl_image {
157*32669476SPankaj Gupta 	char *rcw_nm;		/* Input RCW File */
158*32669476SPankaj Gupta 	char *sec_imgnm;	/* Input BL2 binary */
159*32669476SPankaj Gupta 	char *imagefile;	/* Generated output file */
160*32669476SPankaj Gupta 	boot_src_t boot_src;	/* Boot Source - QSPI, SD, NOR, NAND etc */
161*32669476SPankaj Gupta 	uint32_t src_addr;	/* Source Address */
162*32669476SPankaj Gupta 	uint32_t addr;		/* Load address */
163*32669476SPankaj Gupta 	uint32_t ep;		/* Entry point <opt> default is load address */
164*32669476SPankaj Gupta 	chassis_t chassis;	/* Chassis type */
165*32669476SPankaj Gupta } pblimg;
166*32669476SPankaj Gupta 
167*32669476SPankaj Gupta #define SUCCESS			 0
168*32669476SPankaj Gupta #define FAILURE			-1
169*32669476SPankaj Gupta #define CRC_STOP_CMD_ARM	0x08610040
170*32669476SPankaj Gupta #define CRC_STOP_CMD_ARM_CH3	0x808f0000
171*32669476SPankaj Gupta #define STOP_CMD_ARM_CH3	0x80ff0000
172*32669476SPankaj Gupta #define BYTE_SWAP_32(word)	((((word) & 0xff000000) >> 24)|	\
173*32669476SPankaj Gupta 				(((word) & 0x00ff0000) >>  8) |	\
174*32669476SPankaj Gupta 				(((word) & 0x0000ff00) <<  8) |	\
175*32669476SPankaj Gupta 				(((word) & 0x000000ff) << 24))
176*32669476SPankaj Gupta 
177*32669476SPankaj Gupta #define PBI_LEN_MASK	0xFFF00000
178*32669476SPankaj Gupta #define PBI_LEN_SHIFT	20
179*32669476SPankaj Gupta #define NUM_RCW_WORD	35
180*32669476SPankaj Gupta #define PBI_LEN_ADD		6
181*32669476SPankaj Gupta 
182*32669476SPankaj Gupta #define MAX_CRC_ENTRIES 256
183*32669476SPankaj Gupta 
184*32669476SPankaj Gupta /* SoC numeric identifier */
185*32669476SPankaj Gupta #define SOC_LS1012 1012
186*32669476SPankaj Gupta #define SOC_LS1023 1023
187*32669476SPankaj Gupta #define SOC_LS1026 1026
188*32669476SPankaj Gupta #define SOC_LS1028 1028
189*32669476SPankaj Gupta #define SOC_LS1043 1043
190*32669476SPankaj Gupta #define SOC_LS1046 1046
191*32669476SPankaj Gupta #define SOC_LS1088 1088
192*32669476SPankaj Gupta #define SOC_LS2080 2080
193*32669476SPankaj Gupta #define SOC_LS2088 2088
194*32669476SPankaj Gupta #define SOC_LX2160 2160
195*32669476SPankaj Gupta 
196*32669476SPankaj Gupta static uint32_t pbl_size;
197*32669476SPankaj Gupta bool sb_flag;
198*32669476SPankaj Gupta 
199*32669476SPankaj Gupta /***************************************************************************
200*32669476SPankaj Gupta  * Description	:	CRC32 Lookup Table
201*32669476SPankaj Gupta  ***************************************************************************/
202*32669476SPankaj Gupta static uint32_t crc32_lookup[] = {
203*32669476SPankaj Gupta 	 0x00000000, 0x77073096, 0xEE0E612C, 0x990951BA,
204*32669476SPankaj Gupta 	 0x076DC419, 0x706AF48F, 0xE963A535, 0x9E6495A3,
205*32669476SPankaj Gupta 	 0x0EDB8832, 0x79DCB8A4, 0xE0D5E91E, 0x97D2D988,
206*32669476SPankaj Gupta 	 0x09B64C2B, 0x7EB17CBD, 0xE7B82D07, 0x90BF1D91,
207*32669476SPankaj Gupta 	 0x1DB71064, 0x6AB020F2, 0xF3B97148, 0x84BE41DE,
208*32669476SPankaj Gupta 	 0x1ADAD47D, 0x6DDDE4EB, 0xF4D4B551, 0x83D385C7,
209*32669476SPankaj Gupta 	 0x136C9856, 0x646BA8C0, 0xFD62F97A, 0x8A65C9EC,
210*32669476SPankaj Gupta 	 0x14015C4F, 0x63066CD9, 0xFA0F3D63, 0x8D080DF5,
211*32669476SPankaj Gupta 	 0x3B6E20C8, 0x4C69105E, 0xD56041E4, 0xA2677172,
212*32669476SPankaj Gupta 	 0x3C03E4D1, 0x4B04D447, 0xD20D85FD, 0xA50AB56B,
213*32669476SPankaj Gupta 	 0x35B5A8FA, 0x42B2986C, 0xDBBBC9D6, 0xACBCF940,
214*32669476SPankaj Gupta 	 0x32D86CE3, 0x45DF5C75, 0xDCD60DCF, 0xABD13D59,
215*32669476SPankaj Gupta 	 0x26D930AC, 0x51DE003A, 0xC8D75180, 0xBFD06116,
216*32669476SPankaj Gupta 	 0x21B4F4B5, 0x56B3C423, 0xCFBA9599, 0xB8BDA50F,
217*32669476SPankaj Gupta 	 0x2802B89E, 0x5F058808, 0xC60CD9B2, 0xB10BE924,
218*32669476SPankaj Gupta 	 0x2F6F7C87, 0x58684C11, 0xC1611DAB, 0xB6662D3D,
219*32669476SPankaj Gupta 	 0x76DC4190, 0x01DB7106, 0x98D220BC, 0xEFD5102A,
220*32669476SPankaj Gupta 	 0x71B18589, 0x06B6B51F, 0x9FBFE4A5, 0xE8B8D433,
221*32669476SPankaj Gupta 	 0x7807C9A2, 0x0F00F934, 0x9609A88E, 0xE10E9818,
222*32669476SPankaj Gupta 	 0x7F6A0DBB, 0x086D3D2D, 0x91646C97, 0xE6635C01,
223*32669476SPankaj Gupta 	 0x6B6B51F4, 0x1C6C6162, 0x856530D8, 0xF262004E,
224*32669476SPankaj Gupta 	 0x6C0695ED, 0x1B01A57B, 0x8208F4C1, 0xF50FC457,
225*32669476SPankaj Gupta 	 0x65B0D9C6, 0x12B7E950, 0x8BBEB8EA, 0xFCB9887C,
226*32669476SPankaj Gupta 	 0x62DD1DDF, 0x15DA2D49, 0x8CD37CF3, 0xFBD44C65,
227*32669476SPankaj Gupta 	 0x4DB26158, 0x3AB551CE, 0xA3BC0074, 0xD4BB30E2,
228*32669476SPankaj Gupta 	 0x4ADFA541, 0x3DD895D7, 0xA4D1C46D, 0xD3D6F4FB,
229*32669476SPankaj Gupta 	 0x4369E96A, 0x346ED9FC, 0xAD678846, 0xDA60B8D0,
230*32669476SPankaj Gupta 	 0x44042D73, 0x33031DE5, 0xAA0A4C5F, 0xDD0D7CC9,
231*32669476SPankaj Gupta 	 0x5005713C, 0x270241AA, 0xBE0B1010, 0xC90C2086,
232*32669476SPankaj Gupta 	 0x5768B525, 0x206F85B3, 0xB966D409, 0xCE61E49F,
233*32669476SPankaj Gupta 	 0x5EDEF90E, 0x29D9C998, 0xB0D09822, 0xC7D7A8B4,
234*32669476SPankaj Gupta 	 0x59B33D17, 0x2EB40D81, 0xB7BD5C3B, 0xC0BA6CAD,
235*32669476SPankaj Gupta 	 0xEDB88320, 0x9ABFB3B6, 0x03B6E20C, 0x74B1D29A,
236*32669476SPankaj Gupta 	 0xEAD54739, 0x9DD277AF, 0x04DB2615, 0x73DC1683,
237*32669476SPankaj Gupta 	 0xE3630B12, 0x94643B84, 0x0D6D6A3E, 0x7A6A5AA8,
238*32669476SPankaj Gupta 	 0xE40ECF0B, 0x9309FF9D, 0x0A00AE27, 0x7D079EB1,
239*32669476SPankaj Gupta 	 0xF00F9344, 0x8708A3D2, 0x1E01F268, 0x6906C2FE,
240*32669476SPankaj Gupta 	 0xF762575D, 0x806567CB, 0x196C3671, 0x6E6B06E7,
241*32669476SPankaj Gupta 	 0xFED41B76, 0x89D32BE0, 0x10DA7A5A, 0x67DD4ACC,
242*32669476SPankaj Gupta 	 0xF9B9DF6F, 0x8EBEEFF9, 0x17B7BE43, 0x60B08ED5,
243*32669476SPankaj Gupta 	 0xD6D6A3E8, 0xA1D1937E, 0x38D8C2C4, 0x4FDFF252,
244*32669476SPankaj Gupta 	 0xD1BB67F1, 0xA6BC5767, 0x3FB506DD, 0x48B2364B,
245*32669476SPankaj Gupta 	 0xD80D2BDA, 0xAF0A1B4C, 0x36034AF6, 0x41047A60,
246*32669476SPankaj Gupta 	 0xDF60EFC3, 0xA867DF55, 0x316E8EEF, 0x4669BE79,
247*32669476SPankaj Gupta 	 0xCB61B38C, 0xBC66831A, 0x256FD2A0, 0x5268E236,
248*32669476SPankaj Gupta 	 0xCC0C7795, 0xBB0B4703, 0x220216B9, 0x5505262F,
249*32669476SPankaj Gupta 	 0xC5BA3BBE, 0xB2BD0B28, 0x2BB45A92, 0x5CB36A04,
250*32669476SPankaj Gupta 	 0xC2D7FFA7, 0xB5D0CF31, 0x2CD99E8B, 0x5BDEAE1D,
251*32669476SPankaj Gupta 	 0x9B64C2B0, 0xEC63F226, 0x756AA39C, 0x026D930A,
252*32669476SPankaj Gupta 	 0x9C0906A9, 0xEB0E363F, 0x72076785, 0x05005713,
253*32669476SPankaj Gupta 	 0x95BF4A82, 0xE2B87A14, 0x7BB12BAE, 0x0CB61B38,
254*32669476SPankaj Gupta 	 0x92D28E9B, 0xE5D5BE0D, 0x7CDCEFB7, 0x0BDBDF21,
255*32669476SPankaj Gupta 	 0x86D3D2D4, 0xF1D4E242, 0x68DDB3F8, 0x1FDA836E,
256*32669476SPankaj Gupta 	 0x81BE16CD, 0xF6B9265B, 0x6FB077E1, 0x18B74777,
257*32669476SPankaj Gupta 	 0x88085AE6, 0xFF0F6A70, 0x66063BCA, 0x11010B5C,
258*32669476SPankaj Gupta 	 0x8F659EFF, 0xF862AE69, 0x616BFFD3, 0x166CCF45,
259*32669476SPankaj Gupta 	 0xA00AE278, 0xD70DD2EE, 0x4E048354, 0x3903B3C2,
260*32669476SPankaj Gupta 	 0xA7672661, 0xD06016F7, 0x4969474D, 0x3E6E77DB,
261*32669476SPankaj Gupta 	 0xAED16A4A, 0xD9D65ADC, 0x40DF0B66, 0x37D83BF0,
262*32669476SPankaj Gupta 	 0xA9BCAE53, 0xDEBB9EC5, 0x47B2CF7F, 0x30B5FFE9,
263*32669476SPankaj Gupta 	 0xBDBDF21C, 0xCABAC28A, 0x53B39330, 0x24B4A3A6,
264*32669476SPankaj Gupta 	 0xBAD03605, 0xCDD70693, 0x54DE5729, 0x23D967BF,
265*32669476SPankaj Gupta 	 0xB3667A2E, 0xC4614AB8, 0x5D681B02, 0x2A6F2B94,
266*32669476SPankaj Gupta 	 0xB40BBE37, 0xC30C8EA1, 0x5A05DF1B, 0x2D02EF8D
267*32669476SPankaj Gupta 	};
268*32669476SPankaj Gupta 
269*32669476SPankaj Gupta 
270*32669476SPankaj Gupta static void print_usage(void)
271*32669476SPankaj Gupta {
272*32669476SPankaj Gupta 	printf("\nCorrect Usage of Tool is:\n");
273*32669476SPankaj Gupta 	printf("\n ./create_pbl [options] (mentioned below):\n\n");
274*32669476SPankaj Gupta 	printf("\t-r  <RCW file-name>     - name of RCW binary file.\n");
275*32669476SPankaj Gupta 	printf("\t-i  <BL2 Bin file-name> - file to be added to rcw file.\n");
276*32669476SPankaj Gupta 	printf("\t-c  <Number>            - Chassis Architecture (=2 or =3\n");
277*32669476SPankaj Gupta 	printf("\t                          or =4 for 3.2).\n");
278*32669476SPankaj Gupta 	printf("\t-b  <qspi/nor/nand/sd>  - Boot source.\n");
279*32669476SPankaj Gupta 	printf("\t-d  <Address>           - Destination address where BL2\n");
280*32669476SPankaj Gupta 	printf("\t                          image is to be copied\n");
281*32669476SPankaj Gupta 	printf("\t-o  <output filename>	  - Name of PBL image generated\n");
282*32669476SPankaj Gupta 	printf("\t                          as an output of the tool.\n");
283*32669476SPankaj Gupta 	printf("\t-f  <Address>           - BL2 image Src Offset\n");
284*32669476SPankaj Gupta 	printf("\t                          on Boot Source for block copy.\n");
285*32669476SPankaj Gupta 	printf("\t                          command for chassis >=3.)\n");
286*32669476SPankaj Gupta 	printf("\t-e  <Address>           - [Optional] Entry Point Address\n");
287*32669476SPankaj Gupta 	printf("\t                          of the BL2.bin\n");
288*32669476SPankaj Gupta 	printf("\t-s  Secure Boot.\n");
289*32669476SPankaj Gupta 	printf("\t-h  Help.\n");
290*32669476SPankaj Gupta 	printf("\n\n");
291*32669476SPankaj Gupta 	exit(0);
292*32669476SPankaj Gupta 
293*32669476SPankaj Gupta }
294*32669476SPankaj Gupta 
295*32669476SPankaj Gupta /***************************************************************************
296*32669476SPankaj Gupta  * Function	:	crypto_calculate_checksum()
297*32669476SPankaj Gupta  * Arguments	:	data - Pointer to FILE
298*32669476SPankaj Gupta  *			num - Number of 32 bit words for checksum
299*32669476SPankaj Gupta  * Return	:	Checksum Value
300*32669476SPankaj Gupta  * Description	:	Calculate Checksum over the data
301*32669476SPankaj Gupta  ***************************************************************************/
302*32669476SPankaj Gupta uint32_t crypto_calculate_checksum(FILE *fp_rcw_pbi_op, uint32_t num)
303*32669476SPankaj Gupta {
304*32669476SPankaj Gupta 	uint32_t i;
305*32669476SPankaj Gupta 	uint64_t sum = 0;
306*32669476SPankaj Gupta 	uint32_t word;
307*32669476SPankaj Gupta 
308*32669476SPankaj Gupta 	fseek(fp_rcw_pbi_op, 0L, SEEK_SET);
309*32669476SPankaj Gupta 	for (i = 0; i < num ; i++) {
310*32669476SPankaj Gupta 		if ((fread(&word, sizeof(word), NUM_MEM_BLOCK, fp_rcw_pbi_op))
311*32669476SPankaj Gupta 			< NUM_MEM_BLOCK) {
312*32669476SPankaj Gupta 			printf("%s: Error reading word.\n", __func__);
313*32669476SPankaj Gupta 			return FAILURE;
314*32669476SPankaj Gupta 		}
315*32669476SPankaj Gupta 		sum = sum + word;
316*32669476SPankaj Gupta 		sum = sum & 0xFFFFFFFF;
317*32669476SPankaj Gupta 	}
318*32669476SPankaj Gupta 	return (uint32_t)sum;
319*32669476SPankaj Gupta }
320*32669476SPankaj Gupta 
321*32669476SPankaj Gupta /***************************************************************************
322*32669476SPankaj Gupta  * Function	:	add_pbi_stop_cmd
323*32669476SPankaj Gupta  * Arguments	:	fp_rcw_pbi_op - output rcw_pbi file pointer
324*32669476SPankaj Gupta  * Return	:	SUCCESS or FAILURE
325*32669476SPankaj Gupta  * Description	:	This function insert pbi stop command.
326*32669476SPankaj Gupta  ***************************************************************************/
327*32669476SPankaj Gupta int add_pbi_stop_cmd(FILE *fp_rcw_pbi_op, enum stop_command flag)
328*32669476SPankaj Gupta {
329*32669476SPankaj Gupta 	int ret = FAILURE;
330*32669476SPankaj Gupta 	int32_t pbi_stop_cmd;
331*32669476SPankaj Gupta 	uint32_t pbi_crc = 0xffffffff, i, j, c;
332*32669476SPankaj Gupta 	uint32_t crc_table[MAX_CRC_ENTRIES];
333*32669476SPankaj Gupta 	uint8_t data;
334*32669476SPankaj Gupta 
335*32669476SPankaj Gupta 	switch (pblimg.chassis) {
336*32669476SPankaj Gupta 	case CHASSIS_2:
337*32669476SPankaj Gupta 		pbi_stop_cmd = BYTE_SWAP_32(CRC_STOP_CMD_ARM);
338*32669476SPankaj Gupta 		break;
339*32669476SPankaj Gupta 	case CHASSIS_3:
340*32669476SPankaj Gupta 	case CHASSIS_3_2:
341*32669476SPankaj Gupta 		/* Based on flag add the corresponsding cmd
342*32669476SPankaj Gupta 		 * -- stop cmd or stop with CRC cmd
343*32669476SPankaj Gupta 		 */
344*32669476SPankaj Gupta 		if (flag == CRC_STOP_COMMAND) {
345*32669476SPankaj Gupta 			pbi_stop_cmd = CRC_STOP_CMD_ARM_CH3;
346*32669476SPankaj Gupta 		} else {
347*32669476SPankaj Gupta 			pbi_stop_cmd = STOP_CMD_ARM_CH3;
348*32669476SPankaj Gupta 		}
349*32669476SPankaj Gupta 		break;
350*32669476SPankaj Gupta 	case CHASSIS_UNKNOWN:
351*32669476SPankaj Gupta 	case CHASSIS_MAX:
352*32669476SPankaj Gupta 	default:
353*32669476SPankaj Gupta 		printf("Internal Error: Invalid Chassis val = %d.\n",
354*32669476SPankaj Gupta 			pblimg.chassis);
355*32669476SPankaj Gupta 		goto pbi_stop_err;
356*32669476SPankaj Gupta 	}
357*32669476SPankaj Gupta 
358*32669476SPankaj Gupta 	if (fwrite(&pbi_stop_cmd, sizeof(pbi_stop_cmd), NUM_MEM_BLOCK,
359*32669476SPankaj Gupta 			fp_rcw_pbi_op) != NUM_MEM_BLOCK) {
360*32669476SPankaj Gupta 		printf("%s: Error in Writing PBI STOP CMD\n", __func__);
361*32669476SPankaj Gupta 		goto pbi_stop_err;
362*32669476SPankaj Gupta 	}
363*32669476SPankaj Gupta 
364*32669476SPankaj Gupta 	if (flag == CRC_STOP_COMMAND) {
365*32669476SPankaj Gupta 		for (i = 0; i < MAX_CRC_ENTRIES; i++) {
366*32669476SPankaj Gupta 			c = i << 24;
367*32669476SPankaj Gupta 			for (j = 0; j < 8; j++) {
368*32669476SPankaj Gupta 				c = (c & 0x80000000) ?
369*32669476SPankaj Gupta 					PBI_CRC_POLYNOMIAL ^ (c << 1) : c << 1;
370*32669476SPankaj Gupta 			}
371*32669476SPankaj Gupta 
372*32669476SPankaj Gupta 			crc_table[i] = c;
373*32669476SPankaj Gupta 		}
374*32669476SPankaj Gupta 	}
375*32669476SPankaj Gupta 
376*32669476SPankaj Gupta 	switch (pblimg.chassis) {
377*32669476SPankaj Gupta 	case CHASSIS_2:
378*32669476SPankaj Gupta 		/* Chassis 2: CRC is calculated on  RCW + PBL cmd.*/
379*32669476SPankaj Gupta 		fseek(fp_rcw_pbi_op, 0L, SEEK_SET);
380*32669476SPankaj Gupta 		break;
381*32669476SPankaj Gupta 	case CHASSIS_3:
382*32669476SPankaj Gupta 	case CHASSIS_3_2:
383*32669476SPankaj Gupta 		/* Chassis 3: CRC is calculated on  PBL cmd only. */
384*32669476SPankaj Gupta 		fseek(fp_rcw_pbi_op, CHS3_CRC_PAYLOAD_START_OFFSET, SEEK_SET);
385*32669476SPankaj Gupta 		break;
386*32669476SPankaj Gupta 	case CHASSIS_UNKNOWN:
387*32669476SPankaj Gupta 	case CHASSIS_MAX:
388*32669476SPankaj Gupta 		printf("%s: Unknown Chassis.\n", __func__);
389*32669476SPankaj Gupta 		goto pbi_stop_err;
390*32669476SPankaj Gupta 	}
391*32669476SPankaj Gupta 
392*32669476SPankaj Gupta 	while ((fread(&data, sizeof(data), NUM_MEM_BLOCK, fp_rcw_pbi_op))
393*32669476SPankaj Gupta 		== NUM_MEM_BLOCK) {
394*32669476SPankaj Gupta 		if (flag == CRC_STOP_COMMAND) {
395*32669476SPankaj Gupta 			if (pblimg.chassis == CHASSIS_2) {
396*32669476SPankaj Gupta 				pbi_crc = crc_table
397*32669476SPankaj Gupta 					  [((pbi_crc >> 24) ^ (data)) & 0xff] ^
398*32669476SPankaj Gupta 					  (pbi_crc << 8);
399*32669476SPankaj Gupta 			} else {
400*32669476SPankaj Gupta 				pbi_crc =  (pbi_crc >> 8) ^
401*32669476SPankaj Gupta 					   crc32_lookup[((pbi_crc) ^
402*32669476SPankaj Gupta 							   (data)) & 0xff];
403*32669476SPankaj Gupta 			}
404*32669476SPankaj Gupta 		}
405*32669476SPankaj Gupta 	}
406*32669476SPankaj Gupta 
407*32669476SPankaj Gupta 	switch (pblimg.chassis) {
408*32669476SPankaj Gupta 	case CHASSIS_2:
409*32669476SPankaj Gupta 		pbi_crc = BYTE_SWAP_32(pbi_crc);
410*32669476SPankaj Gupta 		break;
411*32669476SPankaj Gupta 	case CHASSIS_3:
412*32669476SPankaj Gupta 	case CHASSIS_3_2:
413*32669476SPankaj Gupta 		if (flag == CRC_STOP_COMMAND) {
414*32669476SPankaj Gupta 			pbi_crc = pbi_crc ^ 0xFFFFFFFF;
415*32669476SPankaj Gupta 		} else {
416*32669476SPankaj Gupta 			pbi_crc = 0x00000000;
417*32669476SPankaj Gupta 		}
418*32669476SPankaj Gupta 		break;
419*32669476SPankaj Gupta 	case CHASSIS_UNKNOWN:
420*32669476SPankaj Gupta 	case CHASSIS_MAX:
421*32669476SPankaj Gupta 		printf("%s: Unknown Chassis.\n", __func__);
422*32669476SPankaj Gupta 		goto pbi_stop_err;
423*32669476SPankaj Gupta 	}
424*32669476SPankaj Gupta 
425*32669476SPankaj Gupta 	if (fwrite(&pbi_crc, sizeof(pbi_crc), NUM_MEM_BLOCK, fp_rcw_pbi_op)
426*32669476SPankaj Gupta 		!= NUM_MEM_BLOCK) {
427*32669476SPankaj Gupta 		printf("%s: Error in Writing PBI PBI CRC\n", __func__);
428*32669476SPankaj Gupta 		goto pbi_stop_err;
429*32669476SPankaj Gupta 	}
430*32669476SPankaj Gupta 	ret = SUCCESS;
431*32669476SPankaj Gupta 
432*32669476SPankaj Gupta pbi_stop_err:
433*32669476SPankaj Gupta 	return ret;
434*32669476SPankaj Gupta }
435*32669476SPankaj Gupta 
436*32669476SPankaj Gupta /*
437*32669476SPankaj Gupta  * Returns:
438*32669476SPankaj Gupta  *     File size in bytes, on Success.
439*32669476SPankaj Gupta  *     FAILURE, on failure.
440*32669476SPankaj Gupta  */
441*32669476SPankaj Gupta int get_filesize(const char *c)
442*32669476SPankaj Gupta {
443*32669476SPankaj Gupta 	FILE *fp;
444*32669476SPankaj Gupta 	int ret = FAILURE;
445*32669476SPankaj Gupta 
446*32669476SPankaj Gupta 	fp = fopen(c, "rb");
447*32669476SPankaj Gupta 	if (fp == NULL) {
448*32669476SPankaj Gupta 		fprintf(stderr, "%s: Error in opening the file: %s\n",
449*32669476SPankaj Gupta 			__func__, c);
450*32669476SPankaj Gupta 		goto filesize_err;
451*32669476SPankaj Gupta 	}
452*32669476SPankaj Gupta 
453*32669476SPankaj Gupta 	fseek(fp, 0L, SEEK_END);
454*32669476SPankaj Gupta 	ret = ftell(fp);
455*32669476SPankaj Gupta 	fclose(fp);
456*32669476SPankaj Gupta 
457*32669476SPankaj Gupta filesize_err:
458*32669476SPankaj Gupta 	return ret;
459*32669476SPankaj Gupta }
460*32669476SPankaj Gupta 
461*32669476SPankaj Gupta /***************************************************************************
462*32669476SPankaj Gupta  * Function	:	get_bootptr
463*32669476SPankaj Gupta  * Arguments	:	fp_rcw_pbi_op - Pointer to output file
464*32669476SPankaj Gupta  * Return	:	SUCCESS or FAILURE
465*32669476SPankaj Gupta  * Description	:	Add bootptr pbi command to output file
466*32669476SPankaj Gupta  ***************************************************************************/
467*32669476SPankaj Gupta int add_boot_ptr_cmd(FILE *fp_rcw_pbi_op)
468*32669476SPankaj Gupta {
469*32669476SPankaj Gupta 	uint32_t bootptr_addr;
470*32669476SPankaj Gupta 	int ret = FAILURE;
471*32669476SPankaj Gupta 
472*32669476SPankaj Gupta 	switch (pblimg.chassis) {
473*32669476SPankaj Gupta 	case CHASSIS_2:
474*32669476SPankaj Gupta 		if (sb_flag == true)
475*32669476SPankaj Gupta 			bootptr_addr = BYTE_SWAP_32(CSF_ADDR_SB);
476*32669476SPankaj Gupta 		else
477*32669476SPankaj Gupta 			bootptr_addr = BYTE_SWAP_32(BOOTPTR_ADDR);
478*32669476SPankaj Gupta 		pblimg.ep    = BYTE_SWAP_32(pblimg.ep);
479*32669476SPankaj Gupta 		break;
480*32669476SPankaj Gupta 	case CHASSIS_3:
481*32669476SPankaj Gupta 	case CHASSIS_3_2:
482*32669476SPankaj Gupta 		if (sb_flag == true)
483*32669476SPankaj Gupta 			bootptr_addr = CSF_ADDR_SB_CH3;
484*32669476SPankaj Gupta 		else
485*32669476SPankaj Gupta 			bootptr_addr = BOOTPTR_ADDR_CH3;
486*32669476SPankaj Gupta 		break;
487*32669476SPankaj Gupta 	case CHASSIS_UNKNOWN:
488*32669476SPankaj Gupta 	case CHASSIS_MAX:
489*32669476SPankaj Gupta 	default:
490*32669476SPankaj Gupta 		printf("Internal Error: Invalid Chassis val = %d.\n",
491*32669476SPankaj Gupta 			pblimg.chassis);
492*32669476SPankaj Gupta 		goto bootptr_err;
493*32669476SPankaj Gupta 	}
494*32669476SPankaj Gupta 
495*32669476SPankaj Gupta 	if (fwrite(&bootptr_addr, sizeof(bootptr_addr), NUM_MEM_BLOCK,
496*32669476SPankaj Gupta 		fp_rcw_pbi_op) != NUM_MEM_BLOCK) {
497*32669476SPankaj Gupta 		printf("%s: Error in Writing PBI Words:[%d].\n",
498*32669476SPankaj Gupta 			 __func__, ret);
499*32669476SPankaj Gupta 		goto bootptr_err;
500*32669476SPankaj Gupta 	}
501*32669476SPankaj Gupta 
502*32669476SPankaj Gupta 	if (pblimg.ep != 0) {
503*32669476SPankaj Gupta 		if (fwrite(&pblimg.ep, sizeof(pblimg.ep), NUM_MEM_BLOCK,
504*32669476SPankaj Gupta 			fp_rcw_pbi_op) != NUM_MEM_BLOCK) {
505*32669476SPankaj Gupta 			printf("%s: Error in Writing PBI Words\n", __func__);
506*32669476SPankaj Gupta 			goto bootptr_err;
507*32669476SPankaj Gupta 		}
508*32669476SPankaj Gupta 	}
509*32669476SPankaj Gupta 
510*32669476SPankaj Gupta 	printf("\nBoot Location Pointer= %x\n", BYTE_SWAP_32(pblimg.ep));
511*32669476SPankaj Gupta 	ret = SUCCESS;
512*32669476SPankaj Gupta 
513*32669476SPankaj Gupta bootptr_err:
514*32669476SPankaj Gupta 	return ret;
515*32669476SPankaj Gupta }
516*32669476SPankaj Gupta 
517*32669476SPankaj Gupta /***************************************************************************
518*32669476SPankaj Gupta  * Function	:	add_blk_cpy_cmd
519*32669476SPankaj Gupta  * Arguments	:	pbi_word - pointer to pbi commands
520*32669476SPankaj Gupta  *			args - Command  line args flag.
521*32669476SPankaj Gupta  * Return	:	SUCCESS or FAILURE
522*32669476SPankaj Gupta  * Description	:	Add pbi commands for block copy cmd in pbi_words
523*32669476SPankaj Gupta  ***************************************************************************/
524*32669476SPankaj Gupta int add_blk_cpy_cmd(FILE *fp_rcw_pbi_op, uint16_t args)
525*32669476SPankaj Gupta {
526*32669476SPankaj Gupta 	uint32_t blk_cpy_hdr;
527*32669476SPankaj Gupta 	uint32_t file_size, new_file_size;
528*32669476SPankaj Gupta 	uint32_t align = 4;
529*32669476SPankaj Gupta 	int ret = FAILURE;
530*32669476SPankaj Gupta 	int num_pad_bytes = 0;
531*32669476SPankaj Gupta 
532*32669476SPankaj Gupta 	if ((args & BL2_BIN_STRG_LOC_BOOT_SRC_ARG_MASK) == 0) {
533*32669476SPankaj Gupta 		printf("ERROR: Offset not specified for Block Copy Cmd.\n");
534*32669476SPankaj Gupta 		printf("\tSee Usage and use -f option\n");
535*32669476SPankaj Gupta 		goto blk_copy_err;
536*32669476SPankaj Gupta 	}
537*32669476SPankaj Gupta 
538*32669476SPankaj Gupta 	switch (pblimg.chassis) {
539*32669476SPankaj Gupta 	case CHASSIS_3:
540*32669476SPankaj Gupta 		/* Block copy command */
541*32669476SPankaj Gupta 		blk_cpy_hdr = blk_cpy_hdr_map_ch3[pblimg.boot_src];
542*32669476SPankaj Gupta 		pblimg.src_addr += base_addr_ch3[pblimg.boot_src];
543*32669476SPankaj Gupta 		break;
544*32669476SPankaj Gupta 	case CHASSIS_3_2:
545*32669476SPankaj Gupta 		/* Block copy command */
546*32669476SPankaj Gupta 		blk_cpy_hdr = blk_cpy_hdr_map_ch32[pblimg.boot_src];
547*32669476SPankaj Gupta 		pblimg.src_addr += base_addr_ch32[pblimg.boot_src];
548*32669476SPankaj Gupta 		break;
549*32669476SPankaj Gupta 	default:
550*32669476SPankaj Gupta 		printf("%s: Error invalid chassis type for this command.\n",
551*32669476SPankaj Gupta 				__func__);
552*32669476SPankaj Gupta 		goto blk_copy_err;
553*32669476SPankaj Gupta 	}
554*32669476SPankaj Gupta 
555*32669476SPankaj Gupta 	file_size = get_filesize(pblimg.sec_imgnm);
556*32669476SPankaj Gupta 	if (file_size > 0) {
557*32669476SPankaj Gupta 		new_file_size = (file_size + (file_size % align));
558*32669476SPankaj Gupta 
559*32669476SPankaj Gupta 		/* Add Block copy command */
560*32669476SPankaj Gupta 		if (fwrite(&blk_cpy_hdr, sizeof(blk_cpy_hdr), NUM_MEM_BLOCK,
561*32669476SPankaj Gupta 			fp_rcw_pbi_op) != NUM_MEM_BLOCK) {
562*32669476SPankaj Gupta 			printf("%s: Error writing blk_cpy_hdr to the file.\n",
563*32669476SPankaj Gupta 				 __func__);
564*32669476SPankaj Gupta 			goto blk_copy_err;
565*32669476SPankaj Gupta 		}
566*32669476SPankaj Gupta 
567*32669476SPankaj Gupta 		if ((args & BL2_BIN_STRG_LOC_BOOT_SRC_ARG_MASK) == 0)
568*32669476SPankaj Gupta 			num_pad_bytes = pblimg.src_addr % 4;
569*32669476SPankaj Gupta 
570*32669476SPankaj Gupta 		/* Add Src address word */
571*32669476SPankaj Gupta 		if (fwrite(&pblimg.src_addr + num_pad_bytes,
572*32669476SPankaj Gupta 			   sizeof(pblimg.src_addr), NUM_MEM_BLOCK,
573*32669476SPankaj Gupta 			   fp_rcw_pbi_op) != NUM_MEM_BLOCK) {
574*32669476SPankaj Gupta 			printf("%s: Error writing BLK SRC Addr to the file.\n",
575*32669476SPankaj Gupta 				 __func__);
576*32669476SPankaj Gupta 			goto blk_copy_err;
577*32669476SPankaj Gupta 		}
578*32669476SPankaj Gupta 
579*32669476SPankaj Gupta 		/* Add Dest address word */
580*32669476SPankaj Gupta 		if (fwrite(&pblimg.addr, sizeof(pblimg.addr),
581*32669476SPankaj Gupta 			NUM_MEM_BLOCK, fp_rcw_pbi_op) != NUM_MEM_BLOCK) {
582*32669476SPankaj Gupta 			printf("%s: Error writing DST Addr to the file.\n",
583*32669476SPankaj Gupta 			__func__);
584*32669476SPankaj Gupta 			goto blk_copy_err;
585*32669476SPankaj Gupta 		}
586*32669476SPankaj Gupta 
587*32669476SPankaj Gupta 		/* Add size */
588*32669476SPankaj Gupta 		if (fwrite(&new_file_size, sizeof(new_file_size),
589*32669476SPankaj Gupta 			NUM_MEM_BLOCK, fp_rcw_pbi_op) != NUM_MEM_BLOCK) {
590*32669476SPankaj Gupta 			printf("%s: Error writing size to the file.\n",
591*32669476SPankaj Gupta 				__func__);
592*32669476SPankaj Gupta 			goto blk_copy_err;
593*32669476SPankaj Gupta 		}
594*32669476SPankaj Gupta 	}
595*32669476SPankaj Gupta 
596*32669476SPankaj Gupta 	ret = SUCCESS;
597*32669476SPankaj Gupta 
598*32669476SPankaj Gupta blk_copy_err:
599*32669476SPankaj Gupta 	return ret;
600*32669476SPankaj Gupta }
601*32669476SPankaj Gupta 
602*32669476SPankaj Gupta /***************************************************************************
603*32669476SPankaj Gupta  * Function	:	add_cpy_cmd
604*32669476SPankaj Gupta  * Arguments	:	pbi_word - pointer to pbi commands
605*32669476SPankaj Gupta  * Return	:	SUCCESS or FAILURE
606*32669476SPankaj Gupta  * Description	:	Append pbi commands for copying BL2 image to the
607*32669476SPankaj Gupta  *			load address stored in pbl_image.addr
608*32669476SPankaj Gupta  ***************************************************************************/
609*32669476SPankaj Gupta int add_cpy_cmd(FILE *fp_rcw_pbi_op)
610*32669476SPankaj Gupta {
611*32669476SPankaj Gupta 	uint32_t ALTCBAR_ADDRESS = BYTE_SWAP_32(0x09570158);
612*32669476SPankaj Gupta 	uint32_t WAIT_CMD_WRITE_ADDRESS = BYTE_SWAP_32(0x096100c0);
613*32669476SPankaj Gupta 	uint32_t WAIT_CMD = BYTE_SWAP_32(0x000FFFFF);
614*32669476SPankaj Gupta 	int file_size;
615*32669476SPankaj Gupta 	uint32_t pbi_cmd, altcbar;
616*32669476SPankaj Gupta 	uint8_t pbi_data[MAX_PBI_DATA_LEN_BYTE];
617*32669476SPankaj Gupta 	uint32_t dst_offset;
618*32669476SPankaj Gupta 	FILE *fp_img = NULL;
619*32669476SPankaj Gupta 	int ret = FAILURE;
620*32669476SPankaj Gupta 
621*32669476SPankaj Gupta 	altcbar = pblimg.addr;
622*32669476SPankaj Gupta 	dst_offset = pblimg.addr;
623*32669476SPankaj Gupta 	fp_img = fopen(pblimg.sec_imgnm, "rb");
624*32669476SPankaj Gupta 	if (fp_img == NULL) {
625*32669476SPankaj Gupta 		printf("%s: Error in opening the file: %s\n", __func__,
626*32669476SPankaj Gupta 		      pblimg.sec_imgnm);
627*32669476SPankaj Gupta 		goto add_cpy_err;
628*32669476SPankaj Gupta 	}
629*32669476SPankaj Gupta 	file_size = get_filesize(pblimg.sec_imgnm);
630*32669476SPankaj Gupta 	altcbar = 0xfff00000 & altcbar;
631*32669476SPankaj Gupta 	altcbar = BYTE_SWAP_32(altcbar >> 16);
632*32669476SPankaj Gupta 	if (fwrite(&ALTCBAR_ADDRESS, sizeof(ALTCBAR_ADDRESS), NUM_MEM_BLOCK,
633*32669476SPankaj Gupta 		fp_rcw_pbi_op) != NUM_MEM_BLOCK) {
634*32669476SPankaj Gupta 		printf("%s: Error in writing address of ALTCFG CMD.\n",
635*32669476SPankaj Gupta 			 __func__);
636*32669476SPankaj Gupta 		goto add_cpy_err;
637*32669476SPankaj Gupta 	}
638*32669476SPankaj Gupta 	if (fwrite(&altcbar, sizeof(altcbar), NUM_MEM_BLOCK, fp_rcw_pbi_op)
639*32669476SPankaj Gupta 		!= NUM_MEM_BLOCK) {
640*32669476SPankaj Gupta 		printf("%s: Error in writing ALTCFG CMD.\n", __func__);
641*32669476SPankaj Gupta 		goto add_cpy_err;
642*32669476SPankaj Gupta 	}
643*32669476SPankaj Gupta 	if (fwrite(&WAIT_CMD_WRITE_ADDRESS, sizeof(WAIT_CMD_WRITE_ADDRESS),
644*32669476SPankaj Gupta 		NUM_MEM_BLOCK, fp_rcw_pbi_op) != NUM_MEM_BLOCK) {
645*32669476SPankaj Gupta 		printf("%s: Error in writing address of WAIT_CMD.\n",
646*32669476SPankaj Gupta 			__func__);
647*32669476SPankaj Gupta 		goto add_cpy_err;
648*32669476SPankaj Gupta 	}
649*32669476SPankaj Gupta 	if (fwrite(&WAIT_CMD, sizeof(WAIT_CMD), NUM_MEM_BLOCK, fp_rcw_pbi_op)
650*32669476SPankaj Gupta 		!= NUM_MEM_BLOCK) {
651*32669476SPankaj Gupta 		printf("%s: Error in writing WAIT_CMD.\n", __func__);
652*32669476SPankaj Gupta 		goto add_cpy_err;
653*32669476SPankaj Gupta 	}
654*32669476SPankaj Gupta 	do {
655*32669476SPankaj Gupta 		memset(pbi_data, 0, MAX_PBI_DATA_LEN_BYTE);
656*32669476SPankaj Gupta 
657*32669476SPankaj Gupta 		ret = fread(&pbi_data, MAX_PBI_DATA_LEN_BYTE,
658*32669476SPankaj Gupta 				NUM_MEM_BLOCK, fp_img);
659*32669476SPankaj Gupta 		if ((ret != NUM_MEM_BLOCK) && (!feof(fp_img))) {
660*32669476SPankaj Gupta 			printf("%s: Error writing ALTCFG Word: [%d].\n",
661*32669476SPankaj Gupta 				__func__, ret);
662*32669476SPankaj Gupta 			goto add_cpy_err;
663*32669476SPankaj Gupta 		}
664*32669476SPankaj Gupta 
665*32669476SPankaj Gupta 		dst_offset &= OFFSET_MASK;
666*32669476SPankaj Gupta 		pbi_cmd = WRITE_CMD_BASE | dst_offset;
667*32669476SPankaj Gupta 		pbi_cmd = BYTE_SWAP_32(pbi_cmd);
668*32669476SPankaj Gupta 		if (fwrite(&pbi_cmd, sizeof(pbi_cmd), NUM_MEM_BLOCK,
669*32669476SPankaj Gupta 			fp_rcw_pbi_op) != NUM_MEM_BLOCK) {
670*32669476SPankaj Gupta 			printf("%s: Error writing ALTCFG Word write cmd.\n",
671*32669476SPankaj Gupta 				 __func__);
672*32669476SPankaj Gupta 			goto add_cpy_err;
673*32669476SPankaj Gupta 		}
674*32669476SPankaj Gupta 		if (fwrite(&pbi_data,  MAX_PBI_DATA_LEN_BYTE, NUM_MEM_BLOCK,
675*32669476SPankaj Gupta 			fp_rcw_pbi_op) != NUM_MEM_BLOCK) {
676*32669476SPankaj Gupta 			printf("%s: Error writing ALTCFG_Word.\n", __func__);
677*32669476SPankaj Gupta 			goto add_cpy_err;
678*32669476SPankaj Gupta 		}
679*32669476SPankaj Gupta 		dst_offset += MAX_PBI_DATA_LEN_BYTE;
680*32669476SPankaj Gupta 		file_size -= MAX_PBI_DATA_LEN_BYTE;
681*32669476SPankaj Gupta 	} while (!feof(fp_img));
682*32669476SPankaj Gupta 
683*32669476SPankaj Gupta 	ret = SUCCESS;
684*32669476SPankaj Gupta 
685*32669476SPankaj Gupta add_cpy_err:
686*32669476SPankaj Gupta 	if (fp_img != NULL) {
687*32669476SPankaj Gupta 		fclose(fp_img);
688*32669476SPankaj Gupta 	}
689*32669476SPankaj Gupta 	return ret;
690*32669476SPankaj Gupta }
691*32669476SPankaj Gupta 
692*32669476SPankaj Gupta int main(int argc, char **argv)
693*32669476SPankaj Gupta {
694*32669476SPankaj Gupta 	FILE *file = NULL;
695*32669476SPankaj Gupta 	char *ptr;
696*32669476SPankaj Gupta 	int opt;
697*32669476SPankaj Gupta 	int tmp;
698*32669476SPankaj Gupta 	uint16_t args = ARG_INIT_MASK;
699*32669476SPankaj Gupta 	FILE *fp_rcw_pbi_ip = NULL, *fp_rcw_pbi_op = NULL;
700*32669476SPankaj Gupta 	uint32_t word, word_1;
701*32669476SPankaj Gupta 	int ret = FAILURE;
702*32669476SPankaj Gupta 	bool bootptr_flag = false;
703*32669476SPankaj Gupta 	enum stop_command flag_stop_cmd = CRC_STOP_COMMAND;
704*32669476SPankaj Gupta 
705*32669476SPankaj Gupta 	/* Initializing the global structure to zero. */
706*32669476SPankaj Gupta 	memset(&pblimg, 0x0, sizeof(struct pbl_image));
707*32669476SPankaj Gupta 
708*32669476SPankaj Gupta 	while ((opt = getopt(argc, argv,
709*32669476SPankaj Gupta 			     ":b:f:r:i:e:d:c:o:h:s")) != -1) {
710*32669476SPankaj Gupta 		switch (opt) {
711*32669476SPankaj Gupta 		case 'd':
712*32669476SPankaj Gupta 			pblimg.addr = strtoull(optarg, &ptr, 16);
713*32669476SPankaj Gupta 			if (*ptr != 0) {
714*32669476SPankaj Gupta 				fprintf(stderr, "CMD Error: invalid load or destination address %s\n", optarg);
715*32669476SPankaj Gupta 				goto exit_main;
716*32669476SPankaj Gupta 			}
717*32669476SPankaj Gupta 			args |= BL2_BIN_CPY_DEST_ADDR_ARG_MASK;
718*32669476SPankaj Gupta 			break;
719*32669476SPankaj Gupta 		case 'r':
720*32669476SPankaj Gupta 			pblimg.rcw_nm = optarg;
721*32669476SPankaj Gupta 			file = fopen(pblimg.rcw_nm, "r");
722*32669476SPankaj Gupta 			if (file == NULL) {
723*32669476SPankaj Gupta 				printf("CMD Error: Opening the RCW File.\n");
724*32669476SPankaj Gupta 				goto exit_main;
725*32669476SPankaj Gupta 			} else {
726*32669476SPankaj Gupta 				args |= RCW_FILE_NAME_ARG_MASK;
727*32669476SPankaj Gupta 				fclose(file);
728*32669476SPankaj Gupta 			}
729*32669476SPankaj Gupta 			break;
730*32669476SPankaj Gupta 		case 'e':
731*32669476SPankaj Gupta 			bootptr_flag = true;
732*32669476SPankaj Gupta 			pblimg.ep = strtoull(optarg, &ptr, 16);
733*32669476SPankaj Gupta 			if (*ptr != 0) {
734*32669476SPankaj Gupta 				fprintf(stderr,
735*32669476SPankaj Gupta 				"CMD Error: Invalid entry point %s\n", optarg);
736*32669476SPankaj Gupta 				goto exit_main;
737*32669476SPankaj Gupta 			}
738*32669476SPankaj Gupta 			break;
739*32669476SPankaj Gupta 		case 'h':
740*32669476SPankaj Gupta 			print_usage();
741*32669476SPankaj Gupta 			break;
742*32669476SPankaj Gupta 		case 'i':
743*32669476SPankaj Gupta 			pblimg.sec_imgnm = optarg;
744*32669476SPankaj Gupta 			file = fopen(pblimg.sec_imgnm, "r");
745*32669476SPankaj Gupta 			if (file == NULL) {
746*32669476SPankaj Gupta 				printf("CMD Error: Opening Input file.\n");
747*32669476SPankaj Gupta 				goto exit_main;
748*32669476SPankaj Gupta 			} else {
749*32669476SPankaj Gupta 				args |= IN_FILE_NAME_ARG_MASK;
750*32669476SPankaj Gupta 				fclose(file);
751*32669476SPankaj Gupta 			}
752*32669476SPankaj Gupta 			break;
753*32669476SPankaj Gupta 		case 'c':
754*32669476SPankaj Gupta 			tmp = atoi(optarg);
755*32669476SPankaj Gupta 			switch (tmp) {
756*32669476SPankaj Gupta 			case SOC_LS1012:
757*32669476SPankaj Gupta 			case SOC_LS1023:
758*32669476SPankaj Gupta 			case SOC_LS1026:
759*32669476SPankaj Gupta 			case SOC_LS1043:
760*32669476SPankaj Gupta 			case SOC_LS1046:
761*32669476SPankaj Gupta 				pblimg.chassis = CHASSIS_2;
762*32669476SPankaj Gupta 				break;
763*32669476SPankaj Gupta 			case SOC_LS1088:
764*32669476SPankaj Gupta 			case SOC_LS2080:
765*32669476SPankaj Gupta 			case SOC_LS2088:
766*32669476SPankaj Gupta 				pblimg.chassis = CHASSIS_3;
767*32669476SPankaj Gupta 				break;
768*32669476SPankaj Gupta 			case SOC_LS1028:
769*32669476SPankaj Gupta 			case SOC_LX2160:
770*32669476SPankaj Gupta 				pblimg.chassis = CHASSIS_3_2;
771*32669476SPankaj Gupta 				break;
772*32669476SPankaj Gupta 			default:
773*32669476SPankaj Gupta 			printf("CMD Error: Invalid SoC Val = %d.\n", tmp);
774*32669476SPankaj Gupta 				goto exit_main;
775*32669476SPankaj Gupta 			}
776*32669476SPankaj Gupta 
777*32669476SPankaj Gupta 			args |= CHASSIS_ARG_MASK;
778*32669476SPankaj Gupta 			break;
779*32669476SPankaj Gupta 		case 'o':
780*32669476SPankaj Gupta 			pblimg.imagefile = optarg;
781*32669476SPankaj Gupta 			args |= OP_FILE_NAME_ARG_MASK;
782*32669476SPankaj Gupta 			break;
783*32669476SPankaj Gupta 		case 's':
784*32669476SPankaj Gupta 			sb_flag = true;
785*32669476SPankaj Gupta 			break;
786*32669476SPankaj Gupta 		case 'b':
787*32669476SPankaj Gupta 			if (strcmp(optarg, "qspi") == 0) {
788*32669476SPankaj Gupta 				pblimg.boot_src = QSPI_BOOT;
789*32669476SPankaj Gupta 			} else if (strcmp(optarg, "nor") == 0) {
790*32669476SPankaj Gupta 				pblimg.boot_src = IFC_NOR_BOOT;
791*32669476SPankaj Gupta 			} else if (strcmp(optarg, "nand") == 0) {
792*32669476SPankaj Gupta 				pblimg.boot_src = IFC_NAND_BOOT;
793*32669476SPankaj Gupta 			} else if (strcmp(optarg, "sd") == 0) {
794*32669476SPankaj Gupta 				pblimg.boot_src = SD_BOOT;
795*32669476SPankaj Gupta 			} else if (strcmp(optarg, "emmc") == 0) {
796*32669476SPankaj Gupta 				pblimg.boot_src = EMMC_BOOT;
797*32669476SPankaj Gupta 			} else if (strcmp(optarg, "flexspi_nor") == 0) {
798*32669476SPankaj Gupta 				pblimg.boot_src = FLXSPI_NOR_BOOT;
799*32669476SPankaj Gupta 			} else if (strcmp(optarg, "flexspi_nand") == 0) {
800*32669476SPankaj Gupta 				pblimg.boot_src = FLXSPI_NAND_BOOT;
801*32669476SPankaj Gupta 			} else if (strcmp(optarg, "flexspi_nand2k") == 0) {
802*32669476SPankaj Gupta 				pblimg.boot_src = FLXSPI_NAND4K_BOOT;
803*32669476SPankaj Gupta 			} else {
804*32669476SPankaj Gupta 				printf("CMD Error: Invalid boot source.\n");
805*32669476SPankaj Gupta 				goto exit_main;
806*32669476SPankaj Gupta 			}
807*32669476SPankaj Gupta 			args |= BOOT_SRC_ARG_MASK;
808*32669476SPankaj Gupta 			break;
809*32669476SPankaj Gupta 		case 'f':
810*32669476SPankaj Gupta 			pblimg.src_addr = strtoull(optarg, &ptr, 16);
811*32669476SPankaj Gupta 			if (*ptr != 0) {
812*32669476SPankaj Gupta 				fprintf(stderr,
813*32669476SPankaj Gupta 				"CMD Error: Invalid src offset %s\n", optarg);
814*32669476SPankaj Gupta 				goto exit_main;
815*32669476SPankaj Gupta 			}
816*32669476SPankaj Gupta 			args |= BL2_BIN_STRG_LOC_BOOT_SRC_ARG_MASK;
817*32669476SPankaj Gupta 			break;
818*32669476SPankaj Gupta 		default:
819*32669476SPankaj Gupta 			/* issue a warning and skip the unknown arg */
820*32669476SPankaj Gupta 			printf("Cmd Warning: Invalid Arg = %c.\n", opt);
821*32669476SPankaj Gupta 		}
822*32669476SPankaj Gupta 	}
823*32669476SPankaj Gupta 
824*32669476SPankaj Gupta 	if ((args & MAND_ARG_MASK) != MAND_ARG_MASK) {
825*32669476SPankaj Gupta 		print_usage();
826*32669476SPankaj Gupta 	}
827*32669476SPankaj Gupta 
828*32669476SPankaj Gupta 	fp_rcw_pbi_ip = fopen(pblimg.rcw_nm, "rb");
829*32669476SPankaj Gupta 	if (fp_rcw_pbi_ip == NULL) {
830*32669476SPankaj Gupta 		printf("%s: Error in opening the rcw file: %s\n",
831*32669476SPankaj Gupta 			__func__, pblimg.rcw_nm);
832*32669476SPankaj Gupta 		goto exit_main;
833*32669476SPankaj Gupta 	}
834*32669476SPankaj Gupta 
835*32669476SPankaj Gupta 	fp_rcw_pbi_op = fopen(pblimg.imagefile, "wb+");
836*32669476SPankaj Gupta 	if (fp_rcw_pbi_op == NULL) {
837*32669476SPankaj Gupta 		printf("%s: Error opening the input file: %s\n",
838*32669476SPankaj Gupta 			__func__, pblimg.imagefile);
839*32669476SPankaj Gupta 		goto exit_main;
840*32669476SPankaj Gupta 	}
841*32669476SPankaj Gupta 
842*32669476SPankaj Gupta 	printf("\nInput Boot Source: %s\n", boot_src_string[pblimg.boot_src]);
843*32669476SPankaj Gupta 	printf("Input RCW File: %s\n", pblimg.rcw_nm);
844*32669476SPankaj Gupta 	printf("Input BL2 Binary File: %s\n", pblimg.sec_imgnm);
845*32669476SPankaj Gupta 	printf("Input load address for BL2 Binary File: 0x%x\n", pblimg.addr);
846*32669476SPankaj Gupta 
847*32669476SPankaj Gupta 	printf("Chassis Type: %d\n", pblimg.chassis);
848*32669476SPankaj Gupta 	switch (pblimg.chassis) {
849*32669476SPankaj Gupta 	case CHASSIS_2:
850*32669476SPankaj Gupta 		if (fread(&word, sizeof(word), NUM_MEM_BLOCK, fp_rcw_pbi_ip)
851*32669476SPankaj Gupta 			!= NUM_MEM_BLOCK) {
852*32669476SPankaj Gupta 			printf("%s: Error in reading word from the rcw file.\n",
853*32669476SPankaj Gupta 				__func__);
854*32669476SPankaj Gupta 			goto exit_main;
855*32669476SPankaj Gupta 		}
856*32669476SPankaj Gupta 		while (BYTE_SWAP_32(word) != 0x08610040) {
857*32669476SPankaj Gupta 			if (BYTE_SWAP_32(word) == 0x09550000
858*32669476SPankaj Gupta 				|| BYTE_SWAP_32(word) == 0x000f400c) {
859*32669476SPankaj Gupta 				break;
860*32669476SPankaj Gupta 			}
861*32669476SPankaj Gupta 			if (fwrite(&word, sizeof(word), NUM_MEM_BLOCK,
862*32669476SPankaj Gupta 				fp_rcw_pbi_op) != NUM_MEM_BLOCK) {
863*32669476SPankaj Gupta 				printf("%s: [CH2] Error in Writing PBI Words\n",
864*32669476SPankaj Gupta 				__func__);
865*32669476SPankaj Gupta 				goto exit_main;
866*32669476SPankaj Gupta 			}
867*32669476SPankaj Gupta 			if (fread(&word, sizeof(word), NUM_MEM_BLOCK,
868*32669476SPankaj Gupta 				fp_rcw_pbi_ip) != NUM_MEM_BLOCK) {
869*32669476SPankaj Gupta 				printf("%s: [CH2] Error in Reading PBI Words\n",
870*32669476SPankaj Gupta 					__func__);
871*32669476SPankaj Gupta 				goto exit_main;
872*32669476SPankaj Gupta 			}
873*32669476SPankaj Gupta 		}
874*32669476SPankaj Gupta 
875*32669476SPankaj Gupta 		if (bootptr_flag == true) {
876*32669476SPankaj Gupta 			/* Add command to set boot_loc ptr */
877*32669476SPankaj Gupta 			ret = add_boot_ptr_cmd(fp_rcw_pbi_op);
878*32669476SPankaj Gupta 			if (ret != SUCCESS) {
879*32669476SPankaj Gupta 				goto exit_main;
880*32669476SPankaj Gupta 			}
881*32669476SPankaj Gupta 		}
882*32669476SPankaj Gupta 
883*32669476SPankaj Gupta 		/* Write acs write commands to output file */
884*32669476SPankaj Gupta 		ret = add_cpy_cmd(fp_rcw_pbi_op);
885*32669476SPankaj Gupta 		if (ret != SUCCESS) {
886*32669476SPankaj Gupta 			goto exit_main;
887*32669476SPankaj Gupta 		}
888*32669476SPankaj Gupta 
889*32669476SPankaj Gupta 		/* Add stop command after adding pbi commands
890*32669476SPankaj Gupta 		 * For Chasis 2.0 platforms it is always CRC &
891*32669476SPankaj Gupta 		 * Stop command
892*32669476SPankaj Gupta 		 */
893*32669476SPankaj Gupta 		flag_stop_cmd = CRC_STOP_COMMAND;
894*32669476SPankaj Gupta 		ret = add_pbi_stop_cmd(fp_rcw_pbi_op, flag_stop_cmd);
895*32669476SPankaj Gupta 		if (ret != SUCCESS) {
896*32669476SPankaj Gupta 			goto exit_main;
897*32669476SPankaj Gupta 		}
898*32669476SPankaj Gupta 
899*32669476SPankaj Gupta 	break;
900*32669476SPankaj Gupta 
901*32669476SPankaj Gupta 	case CHASSIS_3:
902*32669476SPankaj Gupta 	case CHASSIS_3_2:
903*32669476SPankaj Gupta 		if (fread(&word, sizeof(word), NUM_MEM_BLOCK, fp_rcw_pbi_ip)
904*32669476SPankaj Gupta 			!= NUM_MEM_BLOCK) {
905*32669476SPankaj Gupta 			printf("%s: Error reading PBI Cmd.\n", __func__);
906*32669476SPankaj Gupta 			goto exit_main;
907*32669476SPankaj Gupta 		}
908*32669476SPankaj Gupta 		while (word != 0x808f0000 && word != 0x80ff0000) {
909*32669476SPankaj Gupta 			pbl_size++;
910*32669476SPankaj Gupta 			/* 11th words in RCW has PBL length. Update it
911*32669476SPankaj Gupta 			 * with new length. 2 comamnds get added
912*32669476SPankaj Gupta 			 * Block copy + CCSR Write/CSF header write
913*32669476SPankaj Gupta 			 */
914*32669476SPankaj Gupta 			if (pbl_size == 11) {
915*32669476SPankaj Gupta 				word_1 = (word & PBI_LEN_MASK)
916*32669476SPankaj Gupta 					+ (PBI_LEN_ADD << 20);
917*32669476SPankaj Gupta 				word = word & ~PBI_LEN_MASK;
918*32669476SPankaj Gupta 				word = word | word_1;
919*32669476SPankaj Gupta 			}
920*32669476SPankaj Gupta 			/* Update the CRC command */
921*32669476SPankaj Gupta 			/* Check load command..
922*32669476SPankaj Gupta 			 * add a check if command is Stop with CRC
923*32669476SPankaj Gupta 			 * or stop without checksum
924*32669476SPankaj Gupta 			 */
925*32669476SPankaj Gupta 			if (pbl_size == 35) {
926*32669476SPankaj Gupta 				word = crypto_calculate_checksum(fp_rcw_pbi_op,
927*32669476SPankaj Gupta 						NUM_RCW_WORD - 1);
928*32669476SPankaj Gupta 				if (word == FAILURE) {
929*32669476SPankaj Gupta 					goto exit_main;
930*32669476SPankaj Gupta 				}
931*32669476SPankaj Gupta 			}
932*32669476SPankaj Gupta 			if (fwrite(&word, sizeof(word),	NUM_MEM_BLOCK,
933*32669476SPankaj Gupta 				fp_rcw_pbi_op) != NUM_MEM_BLOCK) {
934*32669476SPankaj Gupta 				printf("%s: [CH3] Error in Writing PBI Words\n",
935*32669476SPankaj Gupta 					__func__);
936*32669476SPankaj Gupta 				goto exit_main;
937*32669476SPankaj Gupta 			}
938*32669476SPankaj Gupta 			if (fread(&word, sizeof(word), NUM_MEM_BLOCK,
939*32669476SPankaj Gupta 				fp_rcw_pbi_ip) != NUM_MEM_BLOCK) {
940*32669476SPankaj Gupta 				printf("%s: [CH3] Error in Reading PBI Words\n",
941*32669476SPankaj Gupta 					 __func__);
942*32669476SPankaj Gupta 				goto exit_main;
943*32669476SPankaj Gupta 			}
944*32669476SPankaj Gupta 
945*32669476SPankaj Gupta 			if (word == CRC_STOP_CMD_ARM_CH3) {
946*32669476SPankaj Gupta 				flag_stop_cmd = CRC_STOP_COMMAND;
947*32669476SPankaj Gupta 			} else if (word == STOP_CMD_ARM_CH3) {
948*32669476SPankaj Gupta 				flag_stop_cmd = STOP_COMMAND;
949*32669476SPankaj Gupta 			}
950*32669476SPankaj Gupta 		}
951*32669476SPankaj Gupta 		if (bootptr_flag == true) {
952*32669476SPankaj Gupta 			/* Add command to set boot_loc ptr */
953*32669476SPankaj Gupta 			ret = add_boot_ptr_cmd(fp_rcw_pbi_op);
954*32669476SPankaj Gupta 			if (ret != SUCCESS) {
955*32669476SPankaj Gupta 				printf("%s: add_boot_ptr_cmd return failure.\n",
956*32669476SPankaj Gupta 					__func__);
957*32669476SPankaj Gupta 				goto exit_main;
958*32669476SPankaj Gupta 			}
959*32669476SPankaj Gupta 		}
960*32669476SPankaj Gupta 
961*32669476SPankaj Gupta 		/* Write acs write commands to output file */
962*32669476SPankaj Gupta 		ret = add_blk_cpy_cmd(fp_rcw_pbi_op, args);
963*32669476SPankaj Gupta 		if (ret != SUCCESS) {
964*32669476SPankaj Gupta 			printf("%s: Function add_blk_cpy_cmd return failure.\n",
965*32669476SPankaj Gupta 				 __func__);
966*32669476SPankaj Gupta 			goto exit_main;
967*32669476SPankaj Gupta 		}
968*32669476SPankaj Gupta 
969*32669476SPankaj Gupta 		/* Add stop command after adding pbi commands */
970*32669476SPankaj Gupta 		ret = add_pbi_stop_cmd(fp_rcw_pbi_op, flag_stop_cmd);
971*32669476SPankaj Gupta 		if (ret != SUCCESS) {
972*32669476SPankaj Gupta 			goto exit_main;
973*32669476SPankaj Gupta 		}
974*32669476SPankaj Gupta 
975*32669476SPankaj Gupta 	break;
976*32669476SPankaj Gupta 
977*32669476SPankaj Gupta 	default:
978*32669476SPankaj Gupta 		printf("%s: Unknown chassis type.\n",
979*32669476SPankaj Gupta 				__func__);
980*32669476SPankaj Gupta 	}
981*32669476SPankaj Gupta 
982*32669476SPankaj Gupta 	if (ret == SUCCESS) {
983*32669476SPankaj Gupta 		printf("Output file successfully created with name: %s\n\n",
984*32669476SPankaj Gupta 			   pblimg.imagefile);
985*32669476SPankaj Gupta 	}
986*32669476SPankaj Gupta 
987*32669476SPankaj Gupta exit_main:
988*32669476SPankaj Gupta 	if (fp_rcw_pbi_op != NULL) {
989*32669476SPankaj Gupta 		fclose(fp_rcw_pbi_op);
990*32669476SPankaj Gupta 	}
991*32669476SPankaj Gupta 	if (fp_rcw_pbi_ip != NULL) {
992*32669476SPankaj Gupta 		fclose(fp_rcw_pbi_ip);
993*32669476SPankaj Gupta 	}
994*32669476SPankaj Gupta 
995*32669476SPankaj Gupta 	return ret;
996*32669476SPankaj Gupta }
997