blob: 86807f4084fce81cfd3573df94419b4ac5188b26 [file] [log] [blame]
#!/usr/bin/env python3
# -*- coding: utf-8 -*-"
# Copyright 2019 The ChromiumOS Authors
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
# A script to pack EC binary with manifest header according to
# Based on 607297_Host_ISH_Firmware_Load_Chrome_OS_SAS_Rev0p5.pdf,
# https://chrome-internal.googlesource.com/chromeos/intel-ish/+/refs/heads/upstream/master/modules/api/ish_api/include/loader_common.h#211,
# and b/124788278#comment10
"""Script to pack EC binary with manifest header.
Package ecos main FW binary (kernel) and AON task binary into final EC binary
image with a manifest header, ISH shim loader will parse this header and load
each binaries into right memory location.
"""
import argparse
import struct
MANIFEST_ENTRY_SIZE = 0x80
HEADER_SIZE = 0x1000
PAGE_SIZE = 0x1000
def parseargs():
"""Parse command line arguments."""
parser = argparse.ArgumentParser()
parser.add_argument(
"-k",
"--kernel",
help="EC kernel binary to pack, usually ec.RW.bin or ec.RW.flat.",
required=True,
)
parser.add_argument(
"--kernel-size", type=int, help="Size of EC kernel image", required=True
)
parser.add_argument(
"-a",
"--aon",
help="EC aontask binary to pack, usually ish_aontask.bin.",
required=False,
)
parser.add_argument(
"--aon-size", type=int, help="Size of EC aontask image", required=False
)
parser.add_argument("-o", "--output", help="Output flash binary file")
return parser.parse_args()
def gen_manifest(ext_id, comp_app_name, code_offset, module_size):
"""Returns a binary blob that represents a manifest entry"""
manifest = bytearray(MANIFEST_ENTRY_SIZE)
# 4 bytes of ASCII encode ID (little endian)
struct.pack_into("<4s", manifest, 0, ext_id)
# 8 bytes of ASCII encode ID (little endian)
struct.pack_into("<8s", manifest, 32, comp_app_name)
# 4 bytes of code offset (little endian)
struct.pack_into("<I", manifest, 96, code_offset)
# 2 bytes of module in page size increments (little endian)
struct.pack_into("<H", manifest, 100, module_size)
return manifest
def roundup_page(size):
"""Returns roundup-ed page size from size of bytes"""
return int(size / PAGE_SIZE) + (size % PAGE_SIZE > 0)
def main():
"""Main function."""
args = parseargs()
print(" Packing EC image file for ISH")
with open(args.output, "wb") as output_file:
print(" kernel binary size:", args.kernel_size)
kern_rdup_pg_size = roundup_page(args.kernel_size)
# Add manifest for main ISH binary
output_file.write(
gen_manifest(b"ISHM", b"ISH_KERN", HEADER_SIZE, kern_rdup_pg_size)
)
if args.aon is not None:
print(" AON binary size: ", args.aon_size)
aon_rdup_pg_size = roundup_page(args.aon_size)
# Add manifest for aontask binary
output_file.write(
gen_manifest(
b"ISHM",
b"AON_TASK",
(
HEADER_SIZE
+ kern_rdup_pg_size * PAGE_SIZE
- MANIFEST_ENTRY_SIZE
),
aon_rdup_pg_size,
)
)
# Add manifest that signals end of manifests
output_file.write(gen_manifest(b"ISHE", b"", 0, 0))
# Pad the remaining HEADER with 0s
if args.aon is not None:
output_file.write(
b"\x00" * (HEADER_SIZE - (MANIFEST_ENTRY_SIZE * 3))
)
else:
output_file.write(
b"\x00" * (HEADER_SIZE - (MANIFEST_ENTRY_SIZE * 2))
)
# Append original kernel image
with open(args.kernel, "rb") as in_file:
output_file.write(in_file.read())
# Filling padings due to size round up as pages
output_file.write(
b"\x00" * (kern_rdup_pg_size * PAGE_SIZE - args.kernel_size)
)
if args.aon is not None:
# Append original aon image
with open(args.aon, "rb") as in_file:
output_file.write(in_file.read())
# Filling padings due to size round up as pages
output_file.write(
b"\x00" * (aon_rdup_pg_size * PAGE_SIZE - args.aon_size)
)
if __name__ == "__main__":
main()