1#!/usr/bin/env python3 2 3# Copyright (c) 2022 Microchip Technology Inc. 4# SPDX-License-Identifier: Apache-2.0 5 6import sys 7import argparse 8import hashlib 9 10verbose_mode = False 11 12# Header parameters 13HDR_SIZE = 0x140 14HDR_VER_MEC172X = 0x03 15HDR_VER_MEC152X = 0x02 16HDR_SPI_CLK_12MHZ = 0x3 17HDR_SPI_CLK_16MHZ = 0x2 18HDR_SPI_CLK_24MHZ = 0x1 19HDR_SPI_CLK_48MHZ = 0 20HDR_SPI_DRV_STR_1X = 0 21HDR_SPI_DRV_STR_2X = 0x4 22HDR_SPI_DRV_STR_4X = 0x8 23HDR_SPI_DRV_STR_6X = 0xc 24HDR_SPI_SLEW_SLOW = 0 25HDR_SPI_SLEW_FAST = 0x10 26HDR_SPI_CPOL_LO = 0 27HDR_SPI_CPOL_HI = 0x20 28HDR_SPI_CHPHA_MOSI_EDGE_2 = 0 29HDR_SPI_CHPHA_MOSI_EDGE_1 = 0x40 30HDR_SPI_CHPHA_MISO_EDGE_1 = 0 31HDR_SPI_CHPHA_MISO_EDGE_2 = 0x80 32 33# User defined constants HDR_SPI_RD_ (0, 1, 2, 3) as per boot rom spec. 34# 1st digit - number of I/O pins used to transmit the opcode 35# 2nd digit - number of I/O pins used to transmit the SPI address 36# 3rd digit - number of pins used to read data from flash 37# 4th digit (if present) - dummy clocks between address and data phase 38HDR_SPI_RD_111 = 0 39HDR_SPI_RD_1118 = 1 40HDR_SPI_RD_1128 = 2 41HDR_SPI_RD_1148 = 3 42 43# Payload parameters 44PLD_LOAD_ADDR = 0xc0000 45PLD_LOAD_ADDR_MEC172X = 0xc0000 46PLD_LOAD_ADDR_MEC152X = 0xe0000 47PLD_ENTRY_ADDR = 0 48PLD_GRANULARITY = 128 49PLD_PAD_SIZE = 128 50PLD_PAD_BYTE = b'\xff' 51 52MCHP_CHAR_P = 0x50 53MCHP_CHAR_H = 0x48 54MCHP_CHAR_C = 0x43 55MCHP_CHAR_M = 0x4D 56 57EC_INFO_BLOCK_SIZE = 128 58ENCR_KEY_HDR_SIZE = 128 59COSIG_SIZE = 96 60TRAILER_SIZE = 160 61TRAILER_PAD_BYTE = b'\xff' 62 63TAG_SPI_LOC = 0 64HDR_SPI_LOC = 0x100 65PLD_SPI_LOC = 0x1000 66 67CRC_TABLE = [0x00, 0x07, 0x0e, 0x09, 0x1c, 0x1b, 0x12, 0x15, 68 0x38, 0x3f, 0x36, 0x31, 0x24, 0x23, 0x2a, 0x2d] 69 70CHIP_DICT = { 71 'mec152x': { 'sram_base': 0xe0000, 'sram_size': 0x40000, 'header_ver': 2 }, 72 'mec172x': { 'sram_base': 0xc0000, 'sram_size': 0x68000, 'header_ver': 3 }, 73} 74 75CHIP_DEFAULT = 'mec172x' 76SPI_READ_MODE_DEFAULT = 'fast' 77SPI_FREQ_MHZ_DEFAULT = 12 78SPI_MODE_DEFAULT = 0 79SPI_MODE_MIN = 0 80SPI_MODE_MAX = 7 81SPI_DRIVE_STRENGTH_MULT_DEFAULT = "1x" 82SPI_SLEW_RATE_DEFAULT = "slow" 83 84def print_bytes(title, b): 85 """Print bytes or bytearray as hex values""" 86 print("{0} = {{ ".format(title), end='') 87 count = 1 88 for v in b: 89 print("0x{0:02x}, ".format(v), end='') 90 if (count % 8) == 0: 91 print("") 92 count = count + 1 93 94 print("}") 95 96def crc8(crc, data): 97 """Update CRC8 value. 98 99 CRC8-ITU calculation 100 """ 101 for v in data: 102 crc = ((crc << 4) & 0xff) ^ (CRC_TABLE[(crc >> 4) ^ (v >> 4)]) 103 crc = ((crc << 4) & 0xff) ^ (CRC_TABLE[(crc >> 4) ^ (v & 0xf)]) 104 return crc ^ 0x55 105 106def build_tag(hdr_spi_loc): 107 """Build MEC172x Boot-ROM TAG 108 109 MEC172x Boot-ROM TAG is 4 bytes 110 bits[23:0] = bits[31:8] of the Header SPI address 111 Header location must be a mutliple of 256 bytes 112 bits[31:24] = CRC8-ITU of bits[23:0] 113 return immutable bytes type 114 """ 115 tag = bytearray([(hdr_spi_loc >> 8) & 0xff, 116 (hdr_spi_loc >> 16) & 0xff, 117 (hdr_spi_loc >> 24) & 0xff]) 118 tag.append(crc8(0, tag)) 119 120 return bytes(tag) 121 122def build_header(chip, spi_config, hdr_spi_loc, pld_spi_loc, pld_entry_addr, pld_len): 123 """Build MEC152x/MEC172x Boot-ROM SPI image header 124 125 Args: 126 chip: mec152x or mec172x 127 spi_config: spi configuration 128 hdr_spi_loc: Header location in SPI Image 129 pld_spi_loc: Payload(FW binary) location in SPI Image 130 pld_entry_addr: Payload load address in MEC172x SPI SRAM 131 Payload entry point address: index 0 instructs Boot-ROM to assume 132 ARM vector table at beginning of payload and reset handler 133 address is at offset 4 of payload. 134 pld_len: Payload length, must be multiple of PLD_GRANULARITY 135 136 return: immutable bytes type for built header 137 """ 138 hdr = bytearray(HDR_SIZE) 139 140 hdr[0] = MCHP_CHAR_P 141 hdr[1] = MCHP_CHAR_H 142 hdr[2] = MCHP_CHAR_C 143 hdr[3] = MCHP_CHAR_M 144 145 hdr[4] = CHIP_DICT[chip]['header_ver'] & 0xff 146 147 if spi_config['spi_freq_mhz'] == 48: 148 hdr[5] = HDR_SPI_CLK_48MHZ 149 elif spi_config['spi_freq_mhz'] == 24: 150 hdr[5] = HDR_SPI_CLK_24MHZ 151 elif spi_config['spi_freq_mhz'] == 16: 152 hdr[5] = HDR_SPI_CLK_16MHZ 153 else: 154 hdr[5] = HDR_SPI_CLK_12MHZ 155 156 if spi_config['spi_mode'] & 0x01: 157 hdr[5] |= HDR_SPI_CPOL_HI 158 if spi_config['spi_mode'] & 0x02: 159 hdr[5] |= HDR_SPI_CHPHA_MOSI_EDGE_1 160 if spi_config['spi_mode'] & 0x04: 161 hdr[5] |= HDR_SPI_CHPHA_MISO_EDGE_2 162 163 # translate 1x, 2x, 4x, 6x to 0, 1, 2, 3 164 if spi_config['spi_drive_str'] == "6x": 165 hdr[5] |= HDR_SPI_DRV_STR_6X 166 elif spi_config['spi_drive_str'] == "4x": 167 hdr[5] |= HDR_SPI_DRV_STR_4X 168 elif spi_config['spi_drive_str'] == "2x": 169 hdr[5] |= HDR_SPI_DRV_STR_2X 170 else: 171 hdr[5] |= HDR_SPI_DRV_STR_1X 172 173 # translate "slow", "fast" to 0, 1 174 if spi_config['spi_slew_rate'] == "fast": 175 hdr[5] |= HDR_SPI_SLEW_FAST 176 177 # MEC172x b[0]=0 do not allow 96MHz SPI clock 178 hdr[6] = 0 # not using authentication or encryption 179 180 if spi_config['spi_read_mode'] == 'quad': 181 hdr[7] = HDR_SPI_RD_1148 182 elif spi_config['spi_read_mode'] == 'dual': 183 hdr[7] = HDR_SPI_RD_1128 184 elif spi_config['spi_read_mode'] == 'normal': 185 hdr[7] = HDR_SPI_RD_111 186 else: 187 hdr[7] = HDR_SPI_RD_1118 188 189 # payload load address in SRAM 190 pld_load_addr = CHIP_DICT[chip]['sram_base'] 191 hdr[8] = pld_load_addr & 0xff 192 hdr[9] = (pld_load_addr >> 8) & 0xff 193 hdr[0xA] = (pld_load_addr >> 16) & 0xff 194 hdr[0xB] = (pld_load_addr >> 24) & 0xff 195 196 # payload entry point address in SRAM 197 hdr[0xC] = pld_entry_addr & 0xff 198 hdr[0xD] = (pld_entry_addr >> 8) & 0xff 199 hdr[0xE] = (pld_entry_addr >> 16) & 0xff 200 hdr[0xF] = (pld_entry_addr >> 24) & 0xff 201 202 # payload size (16-bit) in granularity units 203 pld_units = pld_len // PLD_GRANULARITY 204 hdr[0x10] = pld_units & 0xff 205 hdr[0x11] = (pld_units >> 8) & 0xff 206 # hdr[0x12:0x13] = 0 reserved 207 208 # Unsigned offset from start of Header to start of FW Binary 209 # FW binary(payload) must always be located after header 210 pld_offset = pld_spi_loc - hdr_spi_loc 211 hdr[0x14] = pld_offset & 0xff 212 hdr[0x15] = (pld_offset >> 8) & 0xff 213 hdr[0x16] = (pld_offset >> 16) & 0xff 214 hdr[0x17] = (pld_offset >> 24) & 0xff 215 216 # hdr[0x18] = 0 not using authentication 217 # hdr[0x19] = 0 not adjusting SPI flash device drive strength 218 # hdr[0x1A through 0x1F] = 0 reserved 219 # hdr[0x20 through 0x27] = 0 not adjust SPI flash device drive strength 220 # hdr[0x28 through 0x47] = 0 reserved 221 # hdr[0x48 through 0x4F] = 0 reserved 222 # hdr[0x50 through 0x7F] = ECDSA P-384 Public key x-component 223 # hdr[0x80 through 0xAF] = ECDSA P-384 Public key y-component 224 # hdr[0xB0 through 0xDF] = SHA-384 digest of hdr[0 through 0xAF] Always required 225 # hdr[0xE0 through 0x10F] = ECDSA signature R-component of hdr[0 through 0xDF] 226 # hdr[0x110 through 0x13F] = ECDSA signature S-component of hdr[0 through 0xDF] 227 228 h = hashlib.sha384() 229 h.update(hdr[0:0xB0]) 230 hdr_digest = h.digest() 231 232 if verbose_mode: 233 print_bytes("hdr_sha384_digest", hdr_digest) 234 235 hdr[0xB0:0xE0] = hdr_digest 236 237 return bytes(hdr) 238 239def parse_args(): 240 parser = argparse.ArgumentParser(allow_abbrev=False) 241 # Use a lambda to handle base 10 or base 16 (hex) input 242 parser.add_argument("-c", 243 type=str, 244 dest="chip", 245 choices = ["mec152x", "mec172x"], 246 default="mec172x", 247 help="Chip name: mec172x(default) or mec152x") 248 parser.add_argument("-i", 249 type=str, 250 dest="infilename", 251 default="zephyr.bin", 252 help="Input firmware binary file path/name (default: %(default)s)") 253 parser.add_argument("-o", 254 type=str, 255 dest="outfilename", 256 default="zephyr.mchp.bin", 257 help="Output SPI image file path/name (default: %(default)s)") 258 parser.add_argument("-s", 259 type=int, 260 dest="spi_size_kb", 261 default=256, 262 help="SPI image size in kilobytes (default: %(default)s)") 263 parser.add_argument("-e", 264 type=int, 265 dest="entry_point", 266 default=0, 267 help="FW entry point address Lookup in image (default: %(default)s)") 268 parser.add_argument("-f", 269 type=int, 270 dest="spi_freq_mhz", 271 choices = [12, 16, 24, 48], 272 default=12, 273 help="SPI frequency: 12, 16, 24, or 48 MHz") 274 parser.add_argument("-r", 275 type=str, 276 dest="spi_read_mode", 277 choices = ["normal", "fast", "dual", "quad"], 278 default="fast", 279 help="SPI read mode: normal, fast, dual or quad") 280 parser.add_argument("-m", 281 type=int, 282 dest="spi_mode", 283 choices = [0, 1, 2, 3, 4, 5, 6, 7], 284 default=0, 285 help="SPI signalling mode 3-bit field: 0-7") 286 parser.add_argument("--drvstr", 287 type=str, 288 dest="spi_drive_strength", 289 choices = ["1x", "2x", "4x", "6x"], 290 default="1x", 291 help="SPI pin driver strength multiplier encoded") 292 parser.add_argument("--slewrate", 293 type=str, 294 dest="spi_slew_rate", 295 choices = ["slow", "fast"], 296 default="slow", 297 help="SPI pins slew rate") 298 parser.add_argument("--fill", 299 dest="fill", 300 action='store_true', 301 help="Fill with 0xFF to flash size") 302 parser.add_argument("-v", 303 dest="verbose", 304 action='store_true', 305 help="Enable messages to console") 306 307 ret_args = parser.parse_args() 308 309 return ret_args 310 311def main(): 312 """MEC SPI Gen""" 313 args = parse_args() 314 315 verbose_mode = args.verbose 316 317 if verbose_mode: 318 print("Command line arguments/defaults") 319 print(" chip = {0}".format(args.chip)) 320 print(" infilename = {0}".format(args.infilename)) 321 print(" outfilename = {0}".format(args.outfilename)) 322 print(" SPI size (kilobytes) = {0}".format(args.spi_size_kb)) 323 print(" Entry point address = {0}".format(args.entry_point)) 324 print(" SPI frequency MHz = {0}".format(args.spi_freq_mhz)) 325 print(" SPI Read Mode = {0}".format(args.spi_read_mode)) 326 print(" SPI Signalling Mode = {0}".format(args.spi_mode)) 327 print(" SPI drive strength = {0}".format(args.spi_drive_strength)) 328 print(" SPI slew rate fast = {0}".format(args.spi_slew_rate)) 329 print(" Verbose = {0}".format(args.verbose)) 330 331 if args.infilename is None: 332 print("ERROR: Specify input binary file name with -i") 333 sys.exit(-1) 334 335 if args.outfilename is None: 336 print("ERROR: Specify output binary file name with -o") 337 sys.exit(-1) 338 339 chip = args.chip 340 spi_read_mode = args.spi_read_mode 341 spi_freq_mhz = args.spi_freq_mhz 342 spi_mode = args.spi_mode 343 spi_drive_str_mult = args.spi_drive_strength 344 spi_slew = args.spi_slew_rate 345 346 spi_size = args.spi_size_kb * 1024 347 348 indata = None 349 with open(args.infilename, "rb") as fin: 350 indata = fin.read() 351 352 indata_len = len(indata) 353 if verbose_mode: 354 print("Read input FW binary: length = {0}".format(indata_len)) 355 356 # if necessary pad input data to PLD_GRANULARITY required by Boot-ROM loader 357 pad_len = 0 358 if (indata_len % PLD_GRANULARITY) != 0: 359 pad_len = PLD_GRANULARITY - (indata_len % PLD_GRANULARITY) 360 # NOTE: MCHP Production SPI Image Gen. pads with 0 361 padding = PLD_PAD_BYTE * pad_len 362 indata = indata + padding 363 364 indata_len += pad_len 365 366 if verbose_mode: 367 print("Padded FW binary: length = {0}".format(indata_len)) 368 369 # Do we have enough space for 4KB block containing TAG and Header, padded FW binary, 370 # EC Info Block, Co-Sig Block, and Trailer? 371 mec_add_info_size = PLD_SPI_LOC + EC_INFO_BLOCK_SIZE + COSIG_SIZE + TRAILER_SIZE 372 if indata_len > (spi_size - mec_add_info_size): 373 print("ERROR: FW binary exceeds flash size! indata_len = {0} spi_size = {1}".format(indata_len, spi_size)) 374 sys.exit(-1) 375 376 entry_point = args.entry_point 377 if args.entry_point == 0: 378 # Look up entry point in image 379 # Assumes Cortex-M4 vector table 380 # at beginning of image and second 381 # word in table is address of reset handler 382 entry_point = int.from_bytes(indata[4:8], byteorder="little") 383 384 tag = build_tag(HDR_SPI_LOC) 385 386 if verbose_mode: 387 print_bytes("TAG", tag) 388 print("Build Header at {0}: Load Address = 0x{1:0x} Entry Point Address = 0x{2:0x}".format( 389 HDR_SPI_LOC, PLD_LOAD_ADDR, entry_point)) 390 391 spi_config_info = { 392 "spi_freq_mhz": spi_freq_mhz, 393 "spi_mode": spi_mode, 394 "spi_read_mode": spi_read_mode, 395 "spi_drive_str": spi_drive_str_mult, 396 "spi_slew_rate": spi_slew, 397 } 398 399 header = build_header(chip, spi_config_info, HDR_SPI_LOC, PLD_SPI_LOC, entry_point, indata_len) 400 401 if verbose_mode: 402 print_bytes("HEADER", header) 403 print("") 404 405 # appended to end of padded payload 406 ec_info_block = bytearray(EC_INFO_BLOCK_SIZE) 407 ec_info_loc = PLD_SPI_LOC + len(indata) 408 409 # appended to end of (padded payload + ec_info_block) 410 cosig = bytearray(b'\xff' * COSIG_SIZE) 411 cosig_loc = ec_info_loc + EC_INFO_BLOCK_SIZE 412 413 # appended to end of (padded payload + ec_info_block + cosig) 414 # trailer[0:0x30] = SHA384(indata || ec_info_block || cosig) 415 # trailer[0x30:] = 0xFF 416 trailer = bytearray(b'\xff' * TRAILER_SIZE) 417 trailer_loc = cosig_loc + COSIG_SIZE 418 419 h = hashlib.sha384() 420 h.update(indata) 421 h.update(ec_info_block) 422 h.update(cosig) 423 image_digest = h.digest() 424 trailer[0:len(image_digest)] = image_digest 425 426 if verbose_mode: 427 print("SHA-384 digest (paddedFW || ec_info_block || cosig)") 428 print_bytes("digest", image_digest) 429 430 spi_bufs = [] 431 spi_bufs.append(("TAG", TAG_SPI_LOC, tag)) 432 spi_bufs.append(("HEADER", HDR_SPI_LOC, header)) 433 spi_bufs.append(("PAYLOAD", PLD_SPI_LOC, indata)) 434 spi_bufs.append(("EC_INFO", ec_info_loc, ec_info_block)) 435 spi_bufs.append(("COSIG", cosig_loc, cosig)) 436 spi_bufs.append(("TRAILER", trailer_loc, trailer)) 437 438 spi_bufs.sort(key=lambda x: x[1]) 439 440 if verbose_mode: 441 i = 0 442 for sb in spi_bufs: 443 print("buf[{0}]: {1} location=0x{2:0x} length=0x{3:0x}".format(i, sb[0], sb[1], len(sb[2]))) 444 print("") 445 446 fill = bytes(b'\xff' * 256) 447 if verbose_mode: 448 print("len(fill) = {0}".format(len(fill))) 449 450 loc = 0 451 with open(args.outfilename, "wb") as fout: 452 for sb in spi_bufs: 453 if verbose_mode: 454 print("sb: {0} location=0x{1:0x} len=0x{2:0x}".format(sb[0], sb[1], len(sb[2]))) 455 if loc < sb[1]: 456 fill_len = sb[1] - loc 457 if verbose_mode: 458 print("loc = 0x{0:0x}: Fill with 0xFF len=0x{1:0x}".format(loc, fill_len)) 459 nfill = fill_len // 256 460 rem = fill_len % 256 461 for _ in range(nfill): 462 fout.write(fill) 463 if rem > 0: 464 fout.write(fill[0:rem]) 465 loc = loc + fill_len 466 if verbose_mode: 467 print("loc = 0x{0:0x}: write {1} len=0x{2:0x}".format(loc, sb[0], len(sb[2]))) 468 fout.write(sb[2]) 469 loc = loc + len(sb[2]) 470 if args.fill and (loc < spi_size): 471 fill_len = spi_size - loc 472 nfill = fill_len // 256 473 rem = fill_len % 256 474 for _ in range(nfill): 475 fout.write(fill) 476 if rem > 0: 477 fout.write(fill[0:rem]) 478 loc = loc + fill_len 479 if verbose_mode: 480 print("Final loc = 0x{0:0x}".format(loc)) 481 482 if verbose_mode: 483 print("MEC SPI Gen done") 484 485if __name__ == '__main__': 486 main() 487