7051707129
Commands used: perl -i -p0e 's|\/\*[*\s]*Permission[*\s]*to[*\s]*use,[*\s]*copy,[*\s]*modify,[*\s]*and.or[*\s]*distribute[*\s]*this[*\s]*software[*\s]*for[*\s]*any[*\s]*purpose[*\s]*with[*\s]*or[*\s]*without[*\s]*fee[*\s]*is[*\s]*hereby[*\s]*granted,[*\s]*provided[*\s]*that[*\s]*the[*\s]*above[*\s]*copyright[*\s]*notice[*\s]*and[*\s]*this[*\s]*permission[*\s]*notice[*\s]*appear[*\s]*in[*\s]*all[*\s]*copies.[*\s]*THE[*\s]*SOFTWARE[*\s]*IS[*\s]*PROVIDED[*\s]*.*AS[*\s]*IS.*[*\s]*AND[*\s]*THE[*\s]*AUTHOR[*\s]*DISCLAIMS[*\s]*ALL[*\s]*WARRANTIES[*\s]*WITH[*\s]*REGARD[*\s]*TO[*\s]*THIS[*\s]*SOFTWARE[*\s]*INCLUDING[*\s]*ALL[*\s]*IMPLIED[*\s]*WARRANTIES[*\s]*OF[*\s]*MERCHANTABILITY[*\s]*AND[*\s]*FITNESS.[*\s]*IN[*\s]*NO[*\s]*EVENT[*\s]*SHALL[*\s]*THE[*\s]*AUTHOR[*\s]*BE[*\s]*LIABLE[*\s]*FOR[*\s]*ANY[*\s]*SPECIAL,[*\s]*DIRECT,[*\s]*INDIRECT,[*\s]*OR[*\s]*CONSEQUENTIAL[*\s]*DAMAGES[*\s]*OR[*\s]*ANY[*\s]*DAMAGES[*\s]*WHATSOEVER[*\s]*RESULTING[*\s]*FROM[*\s]*LOSS[*\s]*OF[*\s]*USE,[*\s]*DATA[*\s]*OR[*\s]*PROFITS,[*\s]*WHETHER[*\s]*IN[*\s]*AN[*\s]*ACTION[*\s]*OF[*\s]*CONTRACT,[*\s]*NEGLIGENCE[*\s]*OR[*\s]*OTHER[*\s]*TORTIOUS[*\s]*ACTION,[*\s]*ARISING[*\s]*OUT[*\s]*OF[*\s]*OR[*\s]*IN[*\s]*CONNECTION[*\s]*WITH[*\s]*THE[*\s]*USE[*\s]*OR[*\s]*PERFORMANCE[*\s]*OF[*\s]*THIS[*\s]*SOFTWARE.[*\s]*\*\/|/* SPDX-License-Identifier: ISC */|s' $(cat filelist) perl -i -p0e 's|(\#\#*)\s*Permission[\#\s]*to[\#\s]*use,[\#\s]*copy,[\#\s]*modify,[\#\s]*and.or[\#\s]*distribute[\#\s]*this[\#\s]*software[\#\s]*for[\#\s]*any[\#\s]*purpose[\#\s]*with[\#\s]*or[\#\s]*without[\#\s]*fee[\#\s]*is[\#\s]*hereby[\#\s]*granted,[\#\s]*provided[\#\s]*that[\#\s]*the[\#\s]*above[\#\s]*copyright[\#\s]*notice[\#\s]*and[\#\s]*this[\#\s]*permission[\#\s]*notice[\#\s]*appear[\#\s]*in[\#\s]*all[\#\s]*copies.[\#\s]*THE[\#\s]*SOFTWARE[\#\s]*IS[\#\s]*PROVIDED[\#\s]*.*AS[\#\s]*IS.*[\#\s]*AND[\#\s]*THE[\#\s]*AUTHOR[\#\s]*DISCLAIMS[\#\s]*ALL[\#\s]*WARRANTIES[\#\s]*WITH[\#\s]*REGARD[\#\s]*TO[\#\s]*THIS[\#\s]*SOFTWARE[\#\s]*INCLUDING[\#\s]*ALL[\#\s]*IMPLIED[\#\s]*WARRANTIES[\#\s]*OF[\#\s]*MERCHANTABILITY[\#\s]*AND[\#\s]*FITNESS.[\#\s]*IN[\#\s]*NO[\#\s]*EVENT[\#\s]*SHALL[\#\s]*THE[\#\s]*AUTHOR[\#\s]*BE[\#\s]*LIABLE[\#\s]*FOR[\#\s]*ANY[\#\s]*SPECIAL,[\#\s]*DIRECT,[\#\s]*INDIRECT,[\#\s]*OR[\#\s]*CONSEQUENTIAL[\#\s]*DAMAGES[\#\s]*OR[\#\s]*ANY[\#\s]*DAMAGES[\#\s]*WHATSOEVER[\#\s]*RESULTING[\#\s]*FROM[\#\s]*LOSS[\#\s]*OF[\#\s]*USE,[\#\s]*DATA[\#\s]*OR[\#\s]*PROFITS,[\#\s]*WHETHER[\#\s]*IN[\#\s]*AN[\#\s]*ACTION[\#\s]*OF[\#\s]*CONTRACT,[\#\s]*NEGLIGENCE[\#\s]*OR[\#\s]*OTHER[\#\s]*TORTIOUS[\#\s]*ACTION,[\#\s]*ARISING[\#\s]*OUT[\#\s]*OF[\#\s]*OR[\#\s]*IN[\#\s]*CONNECTION[\#\s]*WITH[\#\s]*THE[\#\s]*USE[\#\s]*OR[\#\s]*PERFORMANCE[\#\s]*OF[\#\s]*THIS[\#\s]*SOFTWARE.\s(\#* *\n)*|\1 SPDX-License-Identifier: ISC\n\n|s' $(cat filelist) perl -i -p0e 's|\/\*[*\s]*Redistribution[*\s]*and[*\s]*use[*\s]*in[*\s]*source[*\s]*and[*\s]*binary[*\s]*forms,[*\s]*with[*\s]*or[*\s]*without[*\s]*modification,[*\s]*are[*\s]*permitted[*\s]*provided[*\s]*that[*\s]*the[*\s]*following[*\s]*conditions[*\s]*are[*\s]*met:[*\s]*[1. ]*Redistributions[*\s]*of[*\s]*source[*\s]*code[*\s]*must[*\s]*retain[*\s]*the[*\s]*above[*\s]*copyright[*\s]*notice,[*\s]*this[*\s]*list[*\s]*of[*\s]*conditions[*\s]*and[*\s]*the[*\s]*following[*\s]*disclaimer.[*\s]*[*\s]*[2. ]*Redistributions[*\s]*in[*\s]*binary[*\s]*form[*\s]*must[*\s]*reproduce[*\s]*the[*\s]*above[*\s]*copyright[*\s]*notice,[*\s]*this[*\s]*list[*\s]*of[*\s]*conditions[*\s]*and[*\s]*the[*\s]*following[*\s]*disclaimer[*\s]*in[*\s]*the[*\s]*documentation[*\s]*and.or[*\s]*other[*\s]*materials[*\s]*provided[*\s]*with[*\s]*the[*\s]*distribution.[*\s]*[3. ]*.*used[*\s]*to[*\s]*endorse[*\s]*or[*\s]*promote[*\s]*products[*\s]*derived[*\s]*from[*\s]*this[*\s]*software[*\s]*without[*\s]*specific[*\s]*prior[*\s]*written[*\s]*permission.[*\s]*THIS[*\s]*SOFTWARE[*\s]*IS[*\s]*PROVIDED.*AS[*\s]*IS.*[*\s]*AND[*\s]*ANY[*\s]*EXPRESS[*\s]*OR[*\s]*IMPLIED[*\s]*WARRANTIES,[*\s]*INCLUDING,[*\s]*BUT[*\s]*NOT[*\s]*LIMITED[*\s]*TO,[*\s]*THE[*\s]*IMPLIED[*\s]*WARRANTIES[*\s]*OF[*\s]*MERCHANTABILITY.*FITNESS[*\s]*FOR[*\s]*A[*\s]*PARTICULAR[*\s]*PURPOSE.*ARE[*\s]*DISCLAIMED.[*\s]*IN[*\s]*NO[*\s]*EVENT[*\s]*SHALL.*LIABLE[*\s]*FOR[*\s]*ANY[*\s]*DIRECT,[*\s]*INDIRECT,[*\s]*INCIDENTAL,[*\s]*SPECIAL,[*\s]*EXEMPLARY,[*\s]*OR[*\s]*CONSEQUENTIAL[*\s]*DAMAGES[*\s]*.INCLUDING,[*\s]*BUT[*\s]*NOT[*\s]*LIMITED[*\s]*TO,[*\s]*PROCUREMENT[*\s]*OF[*\s]*SUBSTITUTE[*\s]*GOODS[*\s]*OR[*\s]*SERVICES;[*\s]*LOSS[*\s]*OF[*\s]*USE,[*\s]*DATA,[*\s]*OR[*\s]*PROFITS;[*\s]*OR[*\s]*BUSINESS[*\s]*INTERRUPTION.[*\s]*HOWEVER[*\s]*CAUSED[*\s]*AND[*\s]*ON[*\s]*ANY[*\s]*THEORY[*\s]*OF[*\s]*LIABILITY,[*\s]*WHETHER[*\s]*IN[*\s]*CONTRACT,[*\s]*STRICT[*\s]*LIABILITY,[*\s]*OR[*\s]*TORT[*\s]*.INCLUDING[*\s]*NEGLIGENCE[*\s]*OR[*\s]*OTHERWISE.[*\s]*ARISING[*\s]*IN[*\s]*ANY[*\s]*WAY[*\s]*OUT[*\s]*OF[*\s]*THE[*\s]*USE[*\s]*OF[*\s]*THIS[*\s]*SOFTWARE,[*\s]*EVEN[*\s]*IF[*\s]*ADVISED[*\s]*OF[*\s]*THE[*\s]*POSSIBILITY[*\s]*OF[*\s]*SUCH[*\s]*DAMAGE.[*\s]*\*\/|/* SPDX-License-Identifier: BSD-3-Clause */|s' $(cat filelist) $1 perl -i -p0e 's|(\#\#*) *Redistribution[\#\s]*and[\#\s]*use[\#\s]*in[\#\s]*source[\#\s]*and[\#\s]*binary[\#\s]*forms,[\#\s]*with[\#\s]*or[\#\s]*without[\#\s]*modification,[\#\s]*are[\#\s]*permitted[\#\s]*provided[\#\s]*that[\#\s]*the[\#\s]*following[\#\s]*conditions[\#\s]*are[\#\s]*met:[\#\s]*[*1. ]*Redistributions[\#\s]*of[\#\s]*source[\#\s]*code[\#\s]*must[\#\s]*retain[\#\s]*the[\#\s]*above[\#\s]*copyright[\#\s]*notice,[\#\s]*this[\#\s]*list[\#\s]*of[\#\s]*conditions[\#\s]*and[\#\s]*the[\#\s]*following[\#\s]*disclaimer.[\#\s]*[*2. ]*Redistributions[\#\s]*in[\#\s]*binary[\#\s]*form[\#\s]*must[\#\s]*reproduce[\#\s]*the[\#\s]*above[\#\s]*copyright[\#\s]*notice,[\#\s]*this[\#\s]*list[\#\s]*of[\#\s]*conditions[\#\s]*and[\#\s]*the[\#\s]*following[\#\s]*disclaimer[\#\s]*in[\#\s]*the[\#\s]*documentation[\#\s]*and.or[\#\s]*other[\#\s]*materials[\#\s]*provided[\#\s]*with[\#\s]*the[\#\s]*distribution.[\#\s]*[\#\s]*[*3. ]*.*used[\#\s]*to[\#\s]*endorse[\#\s]*or[\#\s]*promote[\#\s]*products[\#\s]*derived[\#\s]*from[\#\s]*this[\#\s]*software[\#\s]*without[\#\s]*specific[\#\s]*prior[\#\s]*written[\#\s]*permission.[\#\s]*THIS[\#\s]*SOFTWARE[\#\s]*IS[\#\s]*PROVIDED.*AS[\#\s]*IS.*[\#\s]*AND[\#\s]*ANY[\#\s]*EXPRESS[\#\s]*OR[\#\s]*IMPLIED[\#\s]*WARRANTIES,[\#\s]*INCLUDING,[\#\s]*BUT[\#\s]*NOT[\#\s]*LIMITED[\#\s]*TO,[\#\s]*THE[\#\s]*IMPLIED[\#\s]*WARRANTIES[\#\s]*OF[\#\s]*MERCHANTABILITY.*FITNESS[\#\s]*FOR[\#\s]*A[\#\s]*PARTICULAR[\#\s]*PURPOSE.*ARE[\#\s]*DISCLAIMED.[\#\s]*IN[\#\s]*NO[\#\s]*EVENT[\#\s]*SHALL.*LIABLE[\#\s]*FOR[\#\s]*ANY[\#\s]*DIRECT,[\#\s]*INDIRECT,[\#\s]*INCIDENTAL,[\#\s]*SPECIAL,[\#\s]*EXEMPLARY,[\#\s]*OR[\#\s]*CONSEQUENTIAL[\#\s]*DAMAGES[\#\s]*.INCLUDING,[\#\s]*BUT[\#\s]*NOT[\#\s]*LIMITED[\#\s]*TO,[\#\s]*PROCUREMENT[\#\s]*OF[\#\s]*SUBSTITUTE[\#\s]*GOODS[\#\s]*OR[\#\s]*SERVICES;[\#\s]*LOSS[\#\s]*OF[\#\s]*USE,[\#\s]*DATA,[\#\s]*OR[\#\s]*PROFITS;[\#\s]*OR[\#\s]*BUSINESS[\#\s]*INTERRUPTION.[\#\s]*HOWEVER[\#\s]*CAUSED[\#\s]*AND[\#\s]*ON[\#\s]*ANY[\#\s]*THEORY[\#\s]*OF[\#\s]*LIABILITY,[\#\s]*WHETHER[\#\s]*IN[\#\s]*CONTRACT,[\#\s]*STRICT[\#\s]*LIABILITY,[\#\s]*OR[\#\s]*TORT[\#\s]*.INCLUDING[\#\s]*NEGLIGENCE[\#\s]*OR[\#\s]*OTHERWISE.[\#\s]*ARISING[\#\s]*IN[\#\s]*ANY[\#\s]*WAY[\#\s]*OUT[\#\s]*OF[\#\s]*THE[\#\s]*USE[\#\s]*OF[\#\s]*THIS[\#\s]*SOFTWARE,[\#\s]*EVEN[\#\s]*IF[\#\s]*ADVISED[\#\s]*OF[\#\s]*THE[\#\s]*POSSIBILITY[\#\s]*OF[\#\s]*SUCH[\#\s]*DAMAGE.\s(\#* *\n)*|\1 SPDX-License-Identifier: BSD-3-Clause\n\n|s' $(cat filelist) Change-Id: I7ff9c503a2efe1017a4666baf0b1a758a04f5634 Signed-off-by: Patrick Georgi <pgeorgi@google.com> Reviewed-on: https://review.coreboot.org/c/coreboot/+/41204 Tested-by: build bot (Jenkins) <no-reply@coreboot.org> Reviewed-by: HAOUAS Elyes <ehaouas@noos.fr>
717 lines
25 KiB
Python
Executable file
717 lines
25 KiB
Python
Executable file
#!/usr/bin/env python3
|
|
#============================================================================
|
|
#
|
|
#/** @file createxbl.py
|
|
#
|
|
# GENERAL DESCRIPTION
|
|
# Concatentates XBL segments into one ELF image
|
|
#
|
|
# SPDX-License-Identifier: BSD-3-Clause
|
|
|
|
#**/
|
|
#
|
|
#----------------------------------------------------------------------------
|
|
#
|
|
# EDIT HISTORY FOR FILE
|
|
#
|
|
# This section contains comments describing changes made to the module.
|
|
# Notice that changes are listed in reverse chronological order.
|
|
#
|
|
# when who what, where, why
|
|
# -------- --- ------------------------------------------------------
|
|
# 05/21/19 rissha Added --mbn_version to add MBN header accordingly
|
|
# 03/26/18 tv Added -e to enable extended MBNV5 support
|
|
# 09/04/15 et Added -x and -d to embed xbl_sec ELF
|
|
# 02/11/15 ck Fixed missing elf type check in ZI OOB feature
|
|
# 11/04/14 ck Updated calls to mbn_tools functions
|
|
# 10/22/14 ck Added -z option to remove out of bounds ZI segments when
|
|
# converting from 64 to 32
|
|
# 10/10/14 ck Added -c option and logic to enable elf type swapping
|
|
# 09/12/14 ck Added single file logic
|
|
# 08/29/14 ck Added no_hash option
|
|
# 08/29/14 ck Refactored to use proper python arguments and cleaned code
|
|
# 06/16/14 niting xbl.mbn to xbl.elf
|
|
# 05/28/14 niting Initial revision
|
|
#
|
|
#============================================================================
|
|
from optparse import OptionParser
|
|
import os
|
|
import sys
|
|
import shutil
|
|
import mbn_tools
|
|
|
|
PAGE_SIZE = 4096
|
|
SEGMENT_ALIGN = 16
|
|
ELF32_HDR_SIZE = 52
|
|
ELF32_PHDR_SIZE = 32
|
|
ELF64_HDR_SIZE = 64
|
|
ELF64_PHDR_SIZE = 56
|
|
|
|
|
|
##############################################################################
|
|
# main
|
|
##############################################################################
|
|
def main():
|
|
parser = OptionParser(usage='usage: %prog [options] arguments')
|
|
|
|
parser.add_option("-f", "--first_filepath",
|
|
action="store", type="string", dest="elf_inp_file1",
|
|
help="First ELF file to merge.")
|
|
|
|
parser.add_option("-s", "--second_filepath",
|
|
action="store", type="string", dest="elf_inp_file2",
|
|
help="Second ELF file to merge.")
|
|
|
|
parser.add_option("-x", "--xbl_sec_filepath",
|
|
action="store", type="string", dest="elf_inp_xbl_sec",
|
|
help="Second ELF file to merge.")
|
|
|
|
parser.add_option("-o", "--output_filepath",
|
|
action="store", type="string", dest="binary_out",
|
|
help="Merged filename and path.")
|
|
|
|
parser.add_option("-a", "--first_elf_arch",
|
|
action="store", type="string", dest="elf_1_arch",
|
|
help="First (and output) ELF file architecture. '32' or '64'")
|
|
|
|
parser.add_option("-b", "--second_elf_arch",
|
|
action="store", type="string", dest="elf_2_arch",
|
|
help="Second ELF file architecture. '32' or '64'")
|
|
|
|
parser.add_option("-d", "--xbl_sec_elf_arch",
|
|
action="store", type="string", dest="elf_xbl_sec_arch",
|
|
help="xbl_sec file architecture. '32' or '64'")
|
|
|
|
parser.add_option("-c", "--output_elf_arch",
|
|
action="store", type="string", dest="elf_out_arch",
|
|
help="Output ELF file architecture. '32' or '64'" + \
|
|
" If not given defaults to first file arch.")
|
|
|
|
parser.add_option("-n", "--no_hash",
|
|
action="store_true", dest="hash_image",
|
|
help="Disables hashing of image after merging.")
|
|
|
|
parser.add_option("-z", "--zi_out_of_bounds",
|
|
action="store_true", dest="zi_oob",
|
|
help="Removes ZI segments that have addresses greater" + \
|
|
" than 32 bits when converting from a 64 to 32 bit ELF")
|
|
|
|
parser.add_option("--mbn_version",
|
|
action="store", type="int", dest="mbn_version",
|
|
help="Add mbn header in elf image. '3', '5' or '6'")
|
|
|
|
|
|
(options, args) = parser.parse_args()
|
|
if not options.elf_inp_file1:
|
|
parser.error('First ELF filename not given')
|
|
|
|
if not options.binary_out:
|
|
parser.error('Output filename not given')
|
|
|
|
if not options.elf_1_arch:
|
|
parser.error('First ELF architecture not given')
|
|
|
|
if (not options.elf_1_arch == '64') and (not options.elf_1_arch == '32'):
|
|
parser.error('Invalid First ELF architecture given')
|
|
|
|
# Only evaluate elf_2_arch if two files are given for merging
|
|
if options.elf_inp_file2:
|
|
if (not options.elf_2_arch == '64') and (not options.elf_2_arch == '32'):
|
|
parser.error('Invalid Second ELF architecture given')
|
|
|
|
# Only evaluate elf_xbl_sec_arch if file is given
|
|
if options.elf_inp_xbl_sec:
|
|
if (not options.elf_xbl_sec_arch == '64') and (not options.elf_xbl_sec_arch == '32'):
|
|
parser.error('Invalid xbl_sec ELF architecture given')
|
|
|
|
# If output file architecture is given ensure it is either '32' or '64'
|
|
if options.elf_out_arch:
|
|
if (not options.elf_out_arch == '64') and (not options.elf_out_arch == '32'):
|
|
parser.error('Invalid Output ELF architecture given')
|
|
|
|
|
|
gen_dict = {}
|
|
|
|
elf_inp_file1 = options.elf_inp_file1
|
|
|
|
# It is valid for only one file to be "merged". This essentially just
|
|
# strips off the section names. If second file name is not given then
|
|
# set elf_inp_file2 to ""
|
|
if options.elf_inp_file2:
|
|
elf_inp_file2 = options.elf_inp_file2
|
|
else:
|
|
elf_inp_file2 = ""
|
|
|
|
# Do same for xbl_sec
|
|
elf_inp_xbl_sec = options.elf_inp_xbl_sec if options.elf_inp_xbl_sec else ""
|
|
|
|
binary_out = options.binary_out
|
|
|
|
if options.elf_1_arch == '64':
|
|
is_elf1_64_bit = True
|
|
else:
|
|
is_elf1_64_bit = False
|
|
|
|
# If second filename is not given then set is_elf2_64_bit to false so it
|
|
# can be passed even though it is not used.
|
|
if options.elf_inp_file2:
|
|
if options.elf_2_arch == '64':
|
|
is_elf2_64_bit = True
|
|
else:
|
|
is_elf2_64_bit = False
|
|
else:
|
|
is_elf2_64_bit = False
|
|
|
|
if options.elf_inp_xbl_sec:
|
|
if options.elf_xbl_sec_arch == '64':
|
|
is_elf_xbl_sec_64_bit = True
|
|
else:
|
|
is_elf_xbl_sec_64_bit = False
|
|
else:
|
|
is_elf_xbl_sec_64_bit = False
|
|
|
|
# If output ELF arch is given then set is_out_elf_64_bit accordingly.
|
|
# If not then default to be input1's setting
|
|
if options.elf_out_arch:
|
|
if options.elf_out_arch == '64':
|
|
is_out_elf_64_bit = True
|
|
else:
|
|
is_out_elf_64_bit = False
|
|
else:
|
|
is_out_elf_64_bit = is_elf1_64_bit
|
|
|
|
|
|
# Store ZI Out of Bounds value
|
|
if not options.zi_oob:
|
|
zi_oob_enabled = False
|
|
else:
|
|
zi_oob_enabled = True
|
|
|
|
header_version = 3
|
|
|
|
if options.elf_inp_xbl_sec:
|
|
header_version = 5
|
|
|
|
if options.mbn_version:
|
|
header_version = options.mbn_version
|
|
|
|
mbn_type = 'elf'
|
|
header_format = 'reg'
|
|
gen_dict['IMAGE_KEY_IMAGE_ID'] = mbn_tools.ImageType.APPSBL_IMG
|
|
#gen_dict['IMAGE_KEY_IMAGE_SOURCE'] = 0
|
|
#gen_dict['IMAGE_KEY_IMAGE_DEST'] = 0
|
|
gen_dict['IMAGE_KEY_MBN_TYPE'] = mbn_type
|
|
image_header_secflag = 'non_secure'
|
|
|
|
source_base = os.path.splitext(str(binary_out))[0]
|
|
target_base = os.path.splitext(str(binary_out))[0]
|
|
merged_elf = source_base + "_merged.elf"
|
|
source_elf = source_base + "_nohash.elf"
|
|
target_hash = target_base + ".hash"
|
|
target_hash_hd = target_base + "_hash.hd"
|
|
target_phdr_elf = target_base + "_phdr.pbn"
|
|
target_nonsec = target_base + "_combined_hash.mbn"
|
|
|
|
|
|
#print("Input file 1:", elf_inp_file1)
|
|
#print("Input file 2:", elf_inp_file2)
|
|
#print("Output file:", binary_out)
|
|
|
|
merge_elfs([],
|
|
elf_inp_file1,
|
|
elf_inp_file2,
|
|
elf_inp_xbl_sec,
|
|
merged_elf,
|
|
is_elf1_64_bit,
|
|
is_elf2_64_bit,
|
|
is_elf_xbl_sec_64_bit,
|
|
is_out_elf_64_bit,
|
|
zi_oob_enabled,
|
|
header_version)
|
|
|
|
|
|
# Hash the image if user did not explicitly say not to
|
|
if options.hash_image:
|
|
# Just copy the merged elf to the final output name
|
|
shutil.move(merged_elf, binary_out)
|
|
else:
|
|
shutil.copy(merged_elf, source_elf)
|
|
|
|
# Create hash table
|
|
rv = mbn_tools.pboot_gen_elf([],
|
|
source_elf,
|
|
target_hash,
|
|
elf_out_file_name = target_phdr_elf,
|
|
secure_type = image_header_secflag,
|
|
header_version = header_version )
|
|
if rv:
|
|
raise RuntimeError("Failed to run pboot_gen_elf")
|
|
|
|
# Create hash table header
|
|
rv = mbn_tools.image_header([],
|
|
gen_dict,
|
|
target_hash,
|
|
target_hash_hd,
|
|
image_header_secflag,
|
|
elf_file_name = source_elf,
|
|
header_version = header_version)
|
|
if rv:
|
|
raise RuntimeError("Failed to create image header for hash segment")
|
|
|
|
files_to_cat_in_order = [target_hash_hd, target_hash]
|
|
mbn_tools.concat_files (target_nonsec, files_to_cat_in_order)
|
|
|
|
# Add the hash segment into the ELF
|
|
mbn_tools.pboot_add_hash([],
|
|
target_phdr_elf,
|
|
target_nonsec,
|
|
binary_out)
|
|
|
|
return
|
|
|
|
|
|
##############################################################################
|
|
# roundup
|
|
##############################################################################
|
|
def roundup(x, precision):
|
|
return x if x % precision == 0 else (x + precision - (x % precision))
|
|
|
|
##############################################################################
|
|
# merge_elfs
|
|
##############################################################################
|
|
def merge_elfs(env,
|
|
elf_in_file_name1,
|
|
elf_in_file_name2,
|
|
elf_in_file_xbl_sec,
|
|
elf_out_file_name,
|
|
is_elf1_64_bit,
|
|
is_elf2_64_bit,
|
|
is_elf_xbl_sec_64_bit,
|
|
is_out_elf_64_bit,
|
|
zi_oob_enabled,
|
|
header_version):
|
|
|
|
[elf_header1, phdr_table1] = \
|
|
mbn_tools.preprocess_elf_file(elf_in_file_name1)
|
|
|
|
# Check to make sure second file path exists before using
|
|
if elf_in_file_name2 != "":
|
|
[elf_header2, phdr_table2] = \
|
|
mbn_tools.preprocess_elf_file(elf_in_file_name2)
|
|
|
|
# Check to make sure xbl_sec file path exists before using
|
|
if elf_in_file_xbl_sec != "":
|
|
[elf_headerxblsec, phdr_tablexblsec] = \
|
|
mbn_tools.preprocess_elf_file(elf_in_file_xbl_sec)
|
|
|
|
# Open Files
|
|
elf_in_fp1 = mbn_tools.OPEN(elf_in_file_name1, "rb")
|
|
if elf_in_file_name2 != "":
|
|
elf_in_fp2 = mbn_tools.OPEN(elf_in_file_name2, "rb")
|
|
if elf_in_file_xbl_sec != "":
|
|
elf_in_fpxblsec = mbn_tools.OPEN(elf_in_file_xbl_sec, "rb")
|
|
|
|
if elf_out_file_name is not None:
|
|
elf_out_fp = mbn_tools.OPEN(elf_out_file_name, "wb+")
|
|
|
|
|
|
# Calculate the new program header size. This is dependant on the output
|
|
# ELF type and number of program headers going into output.
|
|
if is_out_elf_64_bit:
|
|
phdr_total_size = elf_header1.e_phnum * ELF64_PHDR_SIZE
|
|
phdr_total_count = elf_header1.e_phnum
|
|
else:
|
|
phdr_total_size = elf_header1.e_phnum * ELF32_PHDR_SIZE
|
|
phdr_total_count = elf_header1.e_phnum
|
|
|
|
|
|
# This logic only applies if two files are to be merged
|
|
if elf_in_file_name2 != "":
|
|
if is_out_elf_64_bit:
|
|
phdr_total_size += elf_header2.e_phnum * ELF64_PHDR_SIZE
|
|
phdr_total_count += elf_header2.e_phnum
|
|
else:
|
|
phdr_total_size += elf_header2.e_phnum * ELF32_PHDR_SIZE
|
|
phdr_total_count += elf_header2.e_phnum
|
|
|
|
# Account for xbl_sec header if included
|
|
if elf_in_file_xbl_sec != "":
|
|
phdr_total_count += 1
|
|
if is_out_elf_64_bit:
|
|
phdr_total_size += ELF64_PHDR_SIZE
|
|
else:
|
|
phdr_total_size += ELF32_PHDR_SIZE
|
|
|
|
# Create a new ELF header for the output file
|
|
if is_out_elf_64_bit:
|
|
out_elf_header = mbn_tools.Elf64_Ehdr(b'\0' * ELF64_HDR_SIZE)
|
|
out_elf_header.e_phoff = ELF64_HDR_SIZE
|
|
out_elf_header.e_ehsize = ELF64_HDR_SIZE
|
|
out_elf_header.e_phentsize = ELF64_PHDR_SIZE
|
|
out_elf_header.e_machine = 183
|
|
out_elf_header.e_ident = str('\x7f' + 'E' + 'L' + 'F' + \
|
|
'\x02' + \
|
|
'\x01' + \
|
|
'\x01' + \
|
|
'\x00' + \
|
|
'\x00' + \
|
|
('\x00' * 7))
|
|
|
|
out_elf_header.e_entry = elf_header1.e_entry
|
|
else:
|
|
out_elf_header = mbn_tools.Elf32_Ehdr(b'\0' * ELF32_HDR_SIZE)
|
|
out_elf_header.e_phoff = ELF32_HDR_SIZE
|
|
out_elf_header.e_ehsize = ELF32_HDR_SIZE
|
|
out_elf_header.e_phentsize = ELF32_PHDR_SIZE
|
|
out_elf_header.e_machine = 40
|
|
out_elf_header.e_entry = elf_header1.e_entry
|
|
out_elf_header.e_ident = str('\x7f' + 'E' + 'L' + 'F' + \
|
|
'\x01' + \
|
|
'\x01' + \
|
|
'\x01' + \
|
|
'\x00' + \
|
|
'\x00' + \
|
|
('\x00' * 7))
|
|
|
|
# Address needs to be verified that it is not greater than 32 bits
|
|
# as it is possible to go from a 64 bit elf to 32.
|
|
if (elf_header1.e_entry > 0xFFFFFFFF):
|
|
print("ERROR: File 1's entry point is too large to convert.")
|
|
exit()
|
|
out_elf_header.e_entry = elf_header1.e_entry
|
|
|
|
# Common header entries
|
|
out_elf_header.e_type = 2
|
|
out_elf_header.e_version = 1
|
|
out_elf_header.e_shoff = 0
|
|
out_elf_header.e_flags = 0
|
|
out_elf_header.e_shentsize = 0
|
|
out_elf_header.e_shnum = 0
|
|
out_elf_header.e_shstrndx = 0
|
|
|
|
|
|
# If ZI OOB is enabled then it is possible that a segment could be discarded
|
|
# Scan for that instance and handle before setting e_phnum and writing header
|
|
# Ensure ELF output is 32 bit
|
|
if zi_oob_enabled == True and is_out_elf_64_bit == False:
|
|
for i in range(len(phdr_table1)):
|
|
if (phdr_table1[i].p_vaddr > 0xFFFFFFFF) or \
|
|
(phdr_table1[i].p_paddr > 0xFFFFFFFF):
|
|
if phdr_table1[i].p_filesz == 0:
|
|
phdr_total_count = phdr_total_count - 1
|
|
|
|
if elf_in_file_name2 != "":
|
|
for i in range(len(phdr_table2)):
|
|
if (phdr_table2[i].p_vaddr > 0xFFFFFFFF) or \
|
|
(phdr_table2[i].p_paddr > 0xFFFFFFFF):
|
|
if phdr_table2[i].p_filesz == 0:
|
|
phdr_total_count = phdr_total_count - 1
|
|
# Do not include xbl_sec in above calculation
|
|
# xbl_sec is to be treated as a single blob
|
|
|
|
|
|
# Now it is ok to populate the ELF header and write it out
|
|
out_elf_header.e_phnum = phdr_total_count
|
|
|
|
# write elf header
|
|
if is_out_elf_64_bit == False:
|
|
elf_out_fp.write(mbn_tools.Elf32_Ehdr.getPackedData(out_elf_header))
|
|
else:
|
|
elf_out_fp.write(mbn_tools.Elf64_Ehdr.getPackedData(out_elf_header))
|
|
|
|
phdr_offset = out_elf_header.e_phoff # offset of where to put next phdr
|
|
|
|
# offset the start of the segments just after the program headers
|
|
segment_offset = roundup(out_elf_header.e_phoff + phdr_total_size, PAGE_SIZE)
|
|
|
|
|
|
# Output first elf data
|
|
for i in range(elf_header1.e_phnum):
|
|
curr_phdr = phdr_table1[i]
|
|
|
|
# Copy program header piece by piece to ensure possible conversion success
|
|
if is_out_elf_64_bit == True:
|
|
# Converting from 32 to 64 elf requires no data size validation
|
|
new_phdr = mbn_tools.Elf64_Phdr(b'\0' * ELF64_PHDR_SIZE)
|
|
new_phdr.p_type = curr_phdr.p_type
|
|
new_phdr.p_offset = segment_offset
|
|
new_phdr.p_vaddr = curr_phdr.p_vaddr
|
|
new_phdr.p_paddr = curr_phdr.p_paddr
|
|
new_phdr.p_filesz = curr_phdr.p_filesz
|
|
new_phdr.p_memsz = curr_phdr.p_memsz
|
|
new_phdr.p_flags = curr_phdr.p_flags
|
|
new_phdr.p_align = curr_phdr.p_align
|
|
else:
|
|
# Converting from 64 to 32 elf requires data size validation
|
|
# Note that there is an option to discard a segment if it is only ZI
|
|
# and its address is greater than 32 bits
|
|
new_phdr = mbn_tools.Elf32_Phdr(b'\0' * ELF32_PHDR_SIZE)
|
|
new_phdr.p_type = curr_phdr.p_type
|
|
new_phdr.p_offset = segment_offset
|
|
|
|
if curr_phdr.p_vaddr > 0xFFFFFFFF:
|
|
if (zi_oob_enabled == True) and (curr_phdr.p_filesz == 0):
|
|
continue
|
|
else:
|
|
print("ERROR: File 1 VAddr is too large for conversion.")
|
|
exit()
|
|
new_phdr.p_vaddr = curr_phdr.p_vaddr
|
|
|
|
if curr_phdr.p_paddr > 0xFFFFFFFF:
|
|
if (zi_oob_enabled == True) and (curr_phdr.p_filesz == 0):
|
|
continue
|
|
else:
|
|
print("ERROR: File 1 PAddr is too large for conversion.")
|
|
exit()
|
|
new_phdr.p_paddr = curr_phdr.p_paddr
|
|
|
|
if curr_phdr.p_filesz > 0xFFFFFFFF:
|
|
print("ERROR: File 1 Filesz is too large for conversion.")
|
|
exit()
|
|
new_phdr.p_filesz = curr_phdr.p_filesz
|
|
|
|
if curr_phdr.p_memsz > 0xFFFFFFFF:
|
|
print("ERROR: File 1 Memsz is too large for conversion.")
|
|
exit()
|
|
new_phdr.p_memsz = curr_phdr.p_memsz
|
|
|
|
if curr_phdr.p_flags > 0xFFFFFFFF:
|
|
print("ERROR: File 1 Flags is too large for conversion.")
|
|
exit()
|
|
new_phdr.p_flags = curr_phdr.p_flags
|
|
|
|
if curr_phdr.p_align > 0xFFFFFFFF:
|
|
print("ERROR: File 1 Align is too large for conversion.")
|
|
exit()
|
|
new_phdr.p_align = curr_phdr.p_align
|
|
|
|
|
|
#print("i=",i)
|
|
#print("phdr_offset=", phdr_offset)
|
|
|
|
# update output file location to next phdr location
|
|
elf_out_fp.seek(phdr_offset)
|
|
# increment phdr_offset to next location
|
|
phdr_offset += out_elf_header.e_phentsize
|
|
|
|
inp_data_offset = curr_phdr.p_offset # used to read data from input file
|
|
|
|
# print("inp_data_offset=")
|
|
# print(inp_data_offset)
|
|
#
|
|
# print("curr_phdr.p_offset=")
|
|
# print(curr_phdr.p_offset)
|
|
#
|
|
# print("curr_phdr.p_filesz=")
|
|
# print(curr_phdr.p_filesz)
|
|
|
|
# output current phdr
|
|
if is_out_elf_64_bit == False:
|
|
elf_out_fp.write(mbn_tools.Elf32_Phdr.getPackedData(new_phdr))
|
|
else:
|
|
elf_out_fp.write(mbn_tools.Elf64_Phdr.getPackedData(new_phdr))
|
|
|
|
# Copy the ELF segment
|
|
bytes_written = mbn_tools.file_copy_offset(elf_in_fp1,
|
|
inp_data_offset,
|
|
elf_out_fp,
|
|
new_phdr.p_offset,
|
|
new_phdr.p_filesz)
|
|
|
|
# update data segment offset to be aligned after previous segment
|
|
segment_offset += roundup(new_phdr.p_filesz, SEGMENT_ALIGN);
|
|
elf_in_fp1.close()
|
|
|
|
# Output second elf data if applicable
|
|
if elf_in_file_name2 != "":
|
|
for i in range(elf_header2.e_phnum):
|
|
curr_phdr = phdr_table2[i]
|
|
|
|
# Copy program header piece by piece to ensure possible conversion success
|
|
if is_out_elf_64_bit == True:
|
|
# Converting from 32 to 64 elf requires no data size validation
|
|
new_phdr = mbn_tools.Elf64_Phdr(b'\0' * ELF64_PHDR_SIZE)
|
|
new_phdr.p_type = curr_phdr.p_type
|
|
new_phdr.p_offset = segment_offset
|
|
new_phdr.p_vaddr = curr_phdr.p_vaddr
|
|
new_phdr.p_paddr = curr_phdr.p_paddr
|
|
new_phdr.p_filesz = curr_phdr.p_filesz
|
|
new_phdr.p_memsz = curr_phdr.p_memsz
|
|
new_phdr.p_flags = curr_phdr.p_flags
|
|
new_phdr.p_align = curr_phdr.p_align
|
|
else:
|
|
# Converting from 64 to 32 elf requires data size validation
|
|
# Note that there is an option to discard a segment if it is only ZI
|
|
# and its address is greater than 32 bits
|
|
new_phdr = mbn_tools.Elf32_Phdr(b'\0' * ELF32_PHDR_SIZE)
|
|
new_phdr.p_type = curr_phdr.p_type
|
|
new_phdr.p_offset = segment_offset
|
|
|
|
if curr_phdr.p_vaddr > 0xFFFFFFFF:
|
|
if (zi_oob_enabled == True) and (curr_phdr.p_filesz == 0):
|
|
continue
|
|
else:
|
|
print("ERROR: File 2 VAddr is too large for conversion.")
|
|
exit()
|
|
new_phdr.p_vaddr = curr_phdr.p_vaddr
|
|
|
|
if curr_phdr.p_paddr > 0xFFFFFFFF:
|
|
if (zi_oob_enabled == True) and (curr_phdr.p_filesz == 0):
|
|
continue
|
|
else:
|
|
print("ERROR: File 2 PAddr is too large for conversion.")
|
|
exit()
|
|
new_phdr.p_paddr = curr_phdr.p_paddr
|
|
|
|
if curr_phdr.p_filesz > 0xFFFFFFFF:
|
|
print("ERROR: File 2 Filesz is too large for conversion.")
|
|
exit()
|
|
new_phdr.p_filesz = curr_phdr.p_filesz
|
|
|
|
if curr_phdr.p_memsz > 0xFFFFFFFF:
|
|
print("ERROR: File 2 Memsz is too large for conversion.")
|
|
exit()
|
|
new_phdr.p_memsz = curr_phdr.p_memsz
|
|
|
|
if curr_phdr.p_flags > 0xFFFFFFFF:
|
|
print("ERROR: File 2 Flags is too large for conversion.")
|
|
exit()
|
|
new_phdr.p_flags = curr_phdr.p_flags
|
|
|
|
if curr_phdr.p_align > 0xFFFFFFFF:
|
|
print("ERROR: File 2 Align is too large for conversion.")
|
|
exit()
|
|
new_phdr.p_align = curr_phdr.p_align
|
|
|
|
|
|
# print("i=",i)
|
|
# print("phdr_offset=", phdr_offset)
|
|
|
|
# update output file location to next phdr location
|
|
elf_out_fp.seek(phdr_offset)
|
|
# increment phdr_offset to next location
|
|
phdr_offset += out_elf_header.e_phentsize
|
|
|
|
inp_data_offset = curr_phdr.p_offset # used to read data from input file
|
|
|
|
# print("inp_data_offset=")
|
|
# print(inp_data_offset)
|
|
#
|
|
# print("curr_phdr.p_offset=")
|
|
# print(curr_phdr.p_offset)
|
|
#
|
|
# print("curr_phdr.p_filesz=")
|
|
# print(curr_phdr.p_filesz)
|
|
|
|
# output current phdr
|
|
if is_out_elf_64_bit == False:
|
|
elf_out_fp.write(mbn_tools.Elf32_Phdr.getPackedData(new_phdr))
|
|
else:
|
|
elf_out_fp.write(mbn_tools.Elf64_Phdr.getPackedData(new_phdr))
|
|
|
|
# Copy the ELF segment
|
|
bytes_written = mbn_tools.file_copy_offset(elf_in_fp2,
|
|
inp_data_offset,
|
|
elf_out_fp,
|
|
new_phdr.p_offset,
|
|
new_phdr.p_filesz)
|
|
|
|
# update data segment offset to be aligned after previous segment
|
|
segment_offset += roundup(new_phdr.p_filesz, SEGMENT_ALIGN);
|
|
elf_in_fp2.close()
|
|
|
|
# Embed xbl_sec image if provided
|
|
if elf_in_file_xbl_sec != "":
|
|
|
|
# Scan pheaders in xbl_sec for segment that contains entry point address
|
|
entry_seg_offset = -1
|
|
entry_addr = elf_headerxblsec.e_entry
|
|
for i in range(elf_headerxblsec.e_phnum):
|
|
phdr = phdr_tablexblsec[i]
|
|
max_addr = phdr.p_vaddr + phdr.p_memsz
|
|
if phdr.p_vaddr <= entry_addr <= max_addr:
|
|
entry_seg_offset = phdr.p_offset
|
|
break
|
|
if entry_seg_offset == -1:
|
|
print("Error: Failed to find entry point in any segment!")
|
|
exit()
|
|
# magical equation for program header's phys and virt addr
|
|
phys_virt_addr = entry_addr - entry_seg_offset
|
|
|
|
if is_out_elf_64_bit:
|
|
# Converting from 32 to 64 elf requires no data size validation
|
|
new_phdr = mbn_tools.Elf64_Phdr(b'\0' * ELF64_PHDR_SIZE)
|
|
new_phdr.p_type = 0x1
|
|
new_phdr.p_offset = segment_offset
|
|
new_phdr.p_vaddr = phys_virt_addr
|
|
new_phdr.p_paddr = phys_virt_addr
|
|
new_phdr.p_filesz = os.path.getsize(elf_in_file_xbl_sec)
|
|
new_phdr.p_memsz = new_phdr.p_filesz
|
|
if header_version >= 5:
|
|
new_phdr.p_flags = (0x5 |
|
|
(mbn_tools.MI_PBT_XBL_SEC_SEGMENT <<
|
|
mbn_tools.MI_PBT_FLAG_SEGMENT_TYPE_SHIFT));
|
|
else:
|
|
new_phdr.p_flags = 0x5
|
|
new_phdr.p_align = 0x1000
|
|
else:
|
|
# Converting from 64 to 32 elf requires data size validation
|
|
# Don't discard the segment containing xbl_sec, simply error out
|
|
# if the address is greater than 32 bits
|
|
new_phdr = mbn_tools.Elf32_Phdr(b'\0' * ELF32_PHDR_SIZE)
|
|
new_phdr.p_type = 0x1 #
|
|
new_phdr.p_offset = segment_offset
|
|
if header_version >= 5:
|
|
new_phdr.p_flags = (0x5 |
|
|
(mbn_tools.MI_PBT_XBL_SEC_SEGMENT <<
|
|
mbn_tools.MI_PBT_FLAG_SEGMENT_TYPE_SHIFT));
|
|
else:
|
|
new_phdr.p_flags = 0x5
|
|
new_phdr.p_align = 0x1000
|
|
|
|
if phys_virt_addr > 0xFFFFFFFF:
|
|
if zi_oob_enabled == False or curr_phdr.p_filesz != 0:
|
|
print("ERROR: File xbl_sec VAddr or PAddr is too big for conversion.")
|
|
exit()
|
|
new_phdr.p_vaddr = phys_virt_addr
|
|
new_phdr.p_paddr = phys_virt_addr
|
|
|
|
if os.path.getsize(elf_in_file_xbl_sec) > 0xFFFFFFFF:
|
|
print("ERROR: File xbl_sec Filesz is too big for conversion.")
|
|
exit()
|
|
new_phdr.p_filesz = os.path.getsize(elf_in_file_xbl_sec)
|
|
new_phdr.p_memsz = new_phdr.p_filesz
|
|
|
|
|
|
# update output file location to next phdr location
|
|
elf_out_fp.seek(phdr_offset)
|
|
# increment phdr_offset to next location
|
|
phdr_offset += out_elf_header.e_phentsize
|
|
# Copy entire xbl_sec file, so start from byte 0
|
|
inp_data_offset = 0
|
|
|
|
# Output xbl_sec's phdr
|
|
elf_in_file_xbl_sec
|
|
if is_out_elf_64_bit == False:
|
|
elf_out_fp.write(mbn_tools.Elf32_Phdr.getPackedData(new_phdr))
|
|
else:
|
|
elf_out_fp.write(mbn_tools.Elf64_Phdr.getPackedData(new_phdr))
|
|
|
|
# Copy the ENTIRE xbl_sec image
|
|
bytes_written = mbn_tools.file_copy_offset(elf_in_fpxblsec,
|
|
inp_data_offset,
|
|
elf_out_fp,
|
|
new_phdr.p_offset,
|
|
new_phdr.p_filesz)
|
|
# update data segment offset to be aligned after previous segment
|
|
# Not necessary, unless appending more pheaders after this point
|
|
segment_offset += roundup(new_phdr.p_filesz, SEGMENT_ALIGN);
|
|
|
|
elf_in_fpxblsec.close()
|
|
|
|
elf_out_fp.close()
|
|
|
|
return 0
|
|
|
|
|
|
main()
|