2020-01-24 03:00:42 +00:00
|
|
|
#!/usr/bin/env python
|
|
|
|
import sys, os
|
|
|
|
from struct import pack as pk, unpack as up
|
|
|
|
|
2020-07-14 02:50:37 +01:00
|
|
|
def atmosphere_target_firmware(major, minor, micro, rev = 0):
|
|
|
|
return (major << 24) | (minor << 16) | (micro << 8) | rev
|
2020-04-19 06:19:09 +01:00
|
|
|
|
2020-01-24 03:00:42 +00:00
|
|
|
def align_up(val, algn):
|
|
|
|
val += algn - 1
|
|
|
|
return val - (val % algn)
|
|
|
|
|
|
|
|
def main(argc, argv):
|
2020-10-27 20:55:19 +00:00
|
|
|
if argc != 4:
|
|
|
|
print('Usage: %s kernel_ldr.bin kernel.bin output.bin' % argv[0])
|
2020-01-24 03:00:42 +00:00
|
|
|
return 1
|
2020-10-27 20:55:19 +00:00
|
|
|
with open(argv[1], 'rb') as f:
|
2020-01-24 03:00:42 +00:00
|
|
|
kernel_ldr = f.read()
|
2020-10-27 20:55:19 +00:00
|
|
|
with open(argv[2], 'rb') as f:
|
2020-01-24 03:00:42 +00:00
|
|
|
kernel = f.read()
|
2020-01-24 05:12:36 +00:00
|
|
|
kernel_metadata_offset = 4
|
2020-01-24 03:00:42 +00:00
|
|
|
assert (kernel_metadata_offset <= len(kernel) - 0x40)
|
|
|
|
assert (kernel[kernel_metadata_offset:kernel_metadata_offset + 4] == b'MSS0')
|
2021-10-08 20:29:53 +01:00
|
|
|
|
|
|
|
bss_start, bss_end, kernel_end = up('<III', kernel[kernel_metadata_offset + 0x30:kernel_metadata_offset + 0x3C])
|
|
|
|
assert (bss_end >= bss_start)
|
|
|
|
assert (bss_end == kernel_end)
|
|
|
|
|
|
|
|
assert (len(kernel) <= kernel_end)
|
|
|
|
if len(kernel) < kernel_end:
|
|
|
|
kernel += b'\x00' * (kernel_end - len(kernel))
|
|
|
|
assert (kernel_end == len(kernel))
|
2020-01-24 03:00:42 +00:00
|
|
|
|
|
|
|
embedded_ini = b''
|
2020-02-14 06:05:20 +00:00
|
|
|
try:
|
|
|
|
with open('ini.bin', 'rb') as f:
|
|
|
|
embedded_ini = f.read()
|
|
|
|
except:
|
|
|
|
pass
|
2021-10-08 20:32:23 +01:00
|
|
|
embedded_ini_offset = align_up(kernel_end, 0x1000)
|
2020-02-14 06:05:20 +00:00
|
|
|
embedded_ini_end = embedded_ini_offset + len(embedded_ini) # TODO: Create and embed an INI, eventually.
|
2020-01-24 03:00:42 +00:00
|
|
|
|
2021-10-08 20:32:23 +01:00
|
|
|
kernel_ldr_offset = align_up(embedded_ini_end, 0x1000) + (0x1000 if len(embedded_ini) == 0 else 0)
|
2020-01-24 03:00:42 +00:00
|
|
|
kernel_ldr_end = kernel_ldr_offset + len(kernel_ldr)
|
2020-02-14 06:05:20 +00:00
|
|
|
mesosphere_end = align_up(kernel_ldr_end, 0x1000)
|
2020-01-24 03:00:42 +00:00
|
|
|
|
2020-10-27 20:55:19 +00:00
|
|
|
with open(argv[3], 'wb') as f:
|
2020-01-24 05:12:36 +00:00
|
|
|
f.write(kernel[:kernel_metadata_offset + 4])
|
2021-10-08 20:32:23 +01:00
|
|
|
f.write(pk('<QQI', embedded_ini_offset, kernel_ldr_offset, atmosphere_target_firmware(13, 0, 0)))
|
2020-04-19 06:19:09 +01:00
|
|
|
f.write(kernel[kernel_metadata_offset + 0x18:])
|
2020-01-24 03:00:42 +00:00
|
|
|
f.seek(embedded_ini_offset)
|
|
|
|
f.write(embedded_ini)
|
|
|
|
f.seek(embedded_ini_end)
|
|
|
|
f.seek(kernel_ldr_offset)
|
|
|
|
f.write(kernel_ldr)
|
2020-02-14 06:05:20 +00:00
|
|
|
f.seek(mesosphere_end)
|
|
|
|
f.write(b'\x00'*0x1000)
|
2020-01-24 03:00:42 +00:00
|
|
|
return 0
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
sys.exit(main(len(sys.argv), sys.argv))
|