| #! /usr/bin/env python3 |
| # |
| # Copyright 2017 Linaro Limited |
| # Copyright (c) 2017-2018, Arm Limited. |
| # |
| # Licensed under the Apache License, Version 2.0 (the "License"); |
| # you may not use this file except in compliance with the License. |
| # You may obtain a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| |
| """ |
| Assemble multiple images into a single image that can be flashed on the device. |
| """ |
| |
| import argparse |
| import errno |
| import io |
| import re |
| import os |
| import shutil |
| |
| offset_re = re.compile(r"^#define ([0-9A-Z_]+)_IMAGE_OFFSET\s+((0x)?[0-9a-fA-F]+)") |
| size_re = re.compile(r"^#define ([0-9A-Z_]+)_IMAGE_MAX_SIZE\s+((0x)?[0-9a-fA-F]+)") |
| |
| class Assembly(): |
| def __init__(self, layout_path, output): |
| self.output = output |
| self.layout_path = layout_path |
| self.find_slots() |
| try: |
| os.unlink(output) |
| except OSError as e: |
| if e.errno != errno.ENOENT: |
| raise |
| |
| def find_slots(self): |
| offsets = {} |
| sizes = {} |
| |
| if os.path.isabs(self.layout_path): |
| configFile = self.layout_path |
| else: |
| scriptsDir = os.path.dirname(os.path.abspath(__file__)) |
| configFile = os.path.join(scriptsDir, self.layout_path) |
| |
| with open(configFile, 'r') as fd: |
| for line in fd: |
| m = offset_re.match(line) |
| if m is not None: |
| offsets[m.group(1)] = int(m.group(2), 0) |
| m = size_re.match(line) |
| if m is not None: |
| sizes[m.group(1)] = int(m.group(2), 0) |
| |
| if 'SECURE' not in offsets: |
| raise Exception("Image config does not have secure partition") |
| |
| if 'NON_SECURE' not in offsets: |
| raise Exception("Image config does not have non-secure partition") |
| |
| self.offsets = offsets |
| self.sizes = sizes |
| |
| def add_image(self, source, partition): |
| with open(self.output, 'ab') as ofd: |
| ofd.seek(0, os.SEEK_END) |
| pos = ofd.tell() |
| if pos > self.offsets[partition]: |
| raise Exception("Partitions not in order, unsupported") |
| if pos < self.offsets[partition]: |
| ofd.write(b'\xFF' * (self.offsets[partition] - pos)) |
| statinfo = os.stat(source) |
| if statinfo.st_size > self.sizes[partition]: |
| raise Exception("Image {} is too large for partition".format(source)) |
| with open(source, 'rb') as rfd: |
| shutil.copyfileobj(rfd, ofd, 0x10000) |
| |
| def main(): |
| parser = argparse.ArgumentParser() |
| |
| parser.add_argument('-l', '--layout', required=True, |
| help='Location of the memory layout file') |
| parser.add_argument('-s', '--secure', required=True, |
| help='Unsigned secure image') |
| parser.add_argument('-n', '--non_secure', |
| help='Unsigned non-secure image') |
| parser.add_argument('-o', '--output', required=True, |
| help='Filename to write full image to') |
| |
| args = parser.parse_args() |
| output = Assembly(args.layout, args.output) |
| |
| |
| output.add_image(args.secure, "SECURE") |
| output.add_image(args.non_secure, "NON_SECURE") |
| |
| if __name__ == '__main__': |
| main() |