extract_kernel: update from AOSP (with fix)

pull/53/head
cfig 5 years ago
parent df147f9325
commit 0dbf161ce8
No known key found for this signature in database
GPG Key ID: B104C307F0FDABB7

@ -39,12 +39,12 @@ COMPRESSION_ALGO = (
# "Linux version " UTS_RELEASE " (" LINUX_COMPILE_BY "@" # "Linux version " UTS_RELEASE " (" LINUX_COMPILE_BY "@"
# LINUX_COMPILE_HOST ") (" LINUX_COMPILER ") " UTS_VERSION "\n"; # LINUX_COMPILE_HOST ") (" LINUX_COMPILER ") " UTS_VERSION "\n";
LINUX_BANNER_PREFIX = b'Linux version ' LINUX_BANNER_PREFIX = b'Linux version '
LINUX_BANNER_REGEX = LINUX_BANNER_PREFIX + \ LINUX_BANNER_REGEX = LINUX_BANNER_PREFIX.decode() + \
r'([0-9]+[.][0-9]+[.][0-9]+).* \(.*@.*\) \(.*\) .*\n' r'(?P<release>(?P<version>[0-9]+[.][0-9]+[.][0-9]+).*) \(.*@.*\) \((?P<compiler>.*)\) .*\n'
def get_version(input_bytes, start_idx): def get_from_release(input_bytes, start_idx, key):
null_idx = input_bytes.find('\x00', start_idx) null_idx = input_bytes.find(b'\x00', start_idx)
if null_idx < 0: if null_idx < 0:
return None return None
try: try:
@ -53,24 +53,51 @@ def get_version(input_bytes, start_idx):
return None return None
mo = re.match(LINUX_BANNER_REGEX, linux_banner) mo = re.match(LINUX_BANNER_REGEX, linux_banner)
if mo: if mo:
return mo.group(1) return mo.group(key)
return None return None
def dump_version(input_bytes): def dump_from_release(input_bytes, key):
"""
Helper of dump_version and dump_release
"""
idx = 0 idx = 0
while True: while True:
idx = input_bytes.find(LINUX_BANNER_PREFIX, idx) idx = input_bytes.find(LINUX_BANNER_PREFIX, idx)
if idx < 0: if idx < 0:
return None return None
version = get_version(input_bytes, idx) value = get_from_release(input_bytes, idx, key)
if version: if value:
return version return value
idx += len(LINUX_BANNER_PREFIX) idx += len(LINUX_BANNER_PREFIX)
def dump_version(input_bytes):
"""
Dump kernel version, w.x.y, from input_bytes. Search for the string
"Linux version " and do pattern matching after it. See LINUX_BANNER_REGEX.
"""
return dump_from_release(input_bytes, "version")
def dump_compiler(input_bytes):
"""
Dump kernel version, w.x.y, from input_bytes. Search for the string
"Linux version " and do pattern matching after it. See LINUX_BANNER_REGEX.
"""
return dump_from_release(input_bytes, "compiler")
def dump_release(input_bytes):
"""
Dump kernel release, w.x.y-..., from input_bytes. Search for the string
"Linux version " and do pattern matching after it. See LINUX_BANNER_REGEX.
"""
return dump_from_release(input_bytes, "release")
def dump_configs(input_bytes): def dump_configs(input_bytes):
""" """
Dump kernel configuration from input_bytes. This can be done when Dump kernel configuration from input_bytes. This can be done when
@ -100,19 +127,25 @@ def dump_configs(input_bytes):
return o return o
def try_decompress(cmd, search_bytes, input_bytes): def try_decompress_bytes(cmd, input_bytes):
idx = input_bytes.find(search_bytes)
if idx < 0:
return None
idx = 0
sp = subprocess.Popen(cmd, stdin=subprocess.PIPE, stdout=subprocess.PIPE, sp = subprocess.Popen(cmd, stdin=subprocess.PIPE, stdout=subprocess.PIPE,
stderr=subprocess.PIPE) stderr=subprocess.PIPE)
o, _ = sp.communicate(input=input_bytes[idx:]) o, _ = sp.communicate(input=input_bytes)
# ignore errors # ignore errors
return o return o
def try_decompress(cmd, search_bytes, input_bytes):
idx = 0
while True:
idx = input_bytes.find(search_bytes, idx)
if idx < 0:
return
yield try_decompress_bytes(cmd, input_bytes[idx:])
idx += 1
def decompress_dump(func, input_bytes): def decompress_dump(func, input_bytes):
""" """
Run func(input_bytes) first; and if that fails (returns value evaluates to Run func(input_bytes) first; and if that fails (returns value evaluates to
@ -122,18 +155,38 @@ def decompress_dump(func, input_bytes):
if o: if o:
return o return o
for cmd, search_bytes in COMPRESSION_ALGO: for cmd, search_bytes in COMPRESSION_ALGO:
decompressed = try_decompress(cmd, search_bytes, input_bytes) for decompressed in try_decompress(cmd, search_bytes, input_bytes):
if decompressed: if decompressed:
o = func(decompressed) o = decompress_dump(func, decompressed)
if o: if o:
return o return o
# Force decompress the whole file even if header doesn't match # Force decompress the whole file even if header doesn't match
decompressed = try_decompress(cmd, b"", input_bytes) decompressed = try_decompress_bytes(cmd, input_bytes)
if decompressed: if decompressed:
o = func(decompressed) o = decompress_dump(func, decompressed)
if o: if o:
return o return o
def dump_to_file(f, dump_fn, input_bytes, desc):
"""
Call decompress_dump(dump_fn, input_bytes) and write to f. If it fails, return
False; otherwise return True.
"""
if f is not None:
o = decompress_dump(dump_fn, input_bytes)
if o:
if isinstance(o, str):
f.write(o.encode())
else:
f.write(o)
else:
sys.stderr.write(
"Cannot extract kernel {}".format(desc))
return False
return True
def main(): def main():
parser = argparse.ArgumentParser( parser = argparse.ArgumentParser(
formatter_class=argparse.RawTextHelpFormatter, formatter_class=argparse.RawTextHelpFormatter,
@ -159,6 +212,20 @@ def main():
nargs='?', nargs='?',
type=argparse.FileType('wb'), type=argparse.FileType('wb'),
const=sys.stdout) const=sys.stdout)
parser.add_argument('--output-release',
help='If specified, write kernel release. Use stdout if '
'no file is specified.',
metavar='FILE',
nargs='?',
type=argparse.FileType('wb'),
const=sys.stdout)
parser.add_argument('--output-compiler',
help='If specified, write the compiler information. Use stdout if no file '
'is specified.',
metavar='FILE',
nargs='?',
type=argparse.FileType('wb'),
const=sys.stdout)
parser.add_argument('--tools', parser.add_argument('--tools',
help='Decompression tools to use. If not specified, PATH ' help='Decompression tools to use. If not specified, PATH '
'is searched.', 'is searched.',
@ -175,25 +242,22 @@ def main():
input_bytes = args.input.read() input_bytes = args.input.read()
ret = 0 ret = 0
if args.output_configs is not None: if not dump_to_file(args.output_configs, dump_configs, input_bytes,
o = decompress_dump(dump_configs, input_bytes) "configs in {}".format(args.input.name)):
if o:
args.output_configs.write(o)
else:
sys.stderr.write(
"Cannot extract kernel configs in {}".format(args.input.name))
ret = 1 ret = 1
if args.output_version is not None: if not dump_to_file(args.output_version, dump_version, input_bytes,
o = decompress_dump(dump_version, input_bytes) "version in {}".format(args.input.name)):
if o: ret = 1
args.output_version.write(o) if not dump_to_file(args.output_release, dump_release, input_bytes,
else: "kernel release in {}".format(args.input.name)):
sys.stderr.write( ret = 1
"Cannot extract kernel versions in {}".format(args.input.name))
if not dump_to_file(args.output_compiler, dump_compiler, input_bytes,
"kernel compiler in {}".format(args.input.name)):
ret = 1 ret = 1
return ret return ret
if __name__ == '__main__': if __name__ == '__main__':
exit(main()) sys.exit(main())

@ -0,0 +1,39 @@
#
# release.mk
# yu, 2020-12-20 00:19
#
define gw
#!/usr/bin/env sh\n
if [ "x$$1" = "xassemble" ]; then\n
echo "already assembled"\n
exit\n
fi\n
if [ "x$$1" = "xcheck" ]; then\n
echo "no check is needed"\n
exit 0\n
fi\n
if [ "x$$1" = "xclean" ]; then\n
echo "no cleaning is needed"\n
exit 0\n
fi\n
java -jar bbootimg/bbootimg.jar $$*
endef
export gw
all:
cd ../bbootimg && gradle build
cp ../bbootimg/build/libs/bbootimg.jar .
cd ../aosp/boot_signer && gradle build
cp ../aosp/boot_signer/build/libs/boot_signer.jar .
cd .. && rm -fr avbImpl bbootimg build build.gradle.kts gradle gradlew gradlew.bat settings.gradle.kts
cd ../aosp && rm -r libavb1.1 libavb1.2 mkbootfs.10 mkbootfs.11
rm -r ../aosp/boot_signer
mkdir -p ../aosp/boot_signer/build/libs/ && mv boot_signer.jar ../aosp/boot_signer/build/libs/
mkdir ../bbootimg && mv bbootimg.jar ../bbootimg/
echo $$gw > gradlew
chmod 755 gradlew
mv gradlew ../
# vim:ft=make
#
Loading…
Cancel
Save