2021-10-10 18:24:22 +02:00
|
|
|
# modifed from pmbootstrap's binfmt.py, Copyright 2018 Oliver Smith, GPL-licensed
|
|
|
|
|
|
|
|
import os
|
|
|
|
import logging
|
|
|
|
|
2022-08-26 22:54:30 +02:00
|
|
|
from constants import Arch, QEMU_ARCHES
|
2022-08-15 17:41:23 +02:00
|
|
|
from exec.cmd import run_root_cmd
|
2021-10-10 18:24:22 +02:00
|
|
|
from utils import mount
|
|
|
|
|
|
|
|
|
|
|
|
def binfmt_info():
|
|
|
|
# Parse the info file
|
|
|
|
full = {}
|
|
|
|
info = "/usr/lib/binfmt.d/qemu-static.conf"
|
|
|
|
logging.debug("parsing: " + info)
|
|
|
|
with open(info, "r") as handle:
|
|
|
|
for line in handle:
|
|
|
|
if line.startswith('#') or ":" not in line:
|
|
|
|
continue
|
|
|
|
splitted = line.split(":")
|
|
|
|
result = {
|
|
|
|
# _ = splitted[0] # empty
|
|
|
|
'name': splitted[1],
|
|
|
|
'type': splitted[2],
|
|
|
|
'offset': splitted[3],
|
|
|
|
'magic': splitted[4],
|
|
|
|
'mask': splitted[5],
|
|
|
|
'interpreter': splitted[6],
|
|
|
|
'flags': splitted[7],
|
|
|
|
'line': line,
|
|
|
|
}
|
|
|
|
if not result['name'].startswith('qemu-'):
|
|
|
|
logging.fatal(f'Unknown binfmt handler "{result["name"]}"')
|
|
|
|
logging.debug(f'binfmt line: {line}')
|
|
|
|
continue
|
|
|
|
arch = ''.join(result['name'].split('-')[1:])
|
|
|
|
full[arch] = result
|
|
|
|
|
|
|
|
return full
|
|
|
|
|
|
|
|
|
2022-08-26 22:54:30 +02:00
|
|
|
def is_registered(arch: Arch) -> bool:
|
|
|
|
qemu_arch = QEMU_ARCHES[arch]
|
|
|
|
return os.path.exists("/proc/sys/fs/binfmt_misc/qemu-" + qemu_arch)
|
2021-10-10 18:24:22 +02:00
|
|
|
|
|
|
|
|
2022-08-26 22:54:30 +02:00
|
|
|
def register(arch: Arch):
|
|
|
|
if arch not in QEMU_ARCHES:
|
|
|
|
raise Exception(f'binfmt.register(): unknown arch {arch} (not in QEMU_ARCHES)')
|
|
|
|
qemu_arch = QEMU_ARCHES[arch]
|
2021-10-10 18:24:22 +02:00
|
|
|
if is_registered(arch):
|
|
|
|
return
|
|
|
|
|
|
|
|
lines = binfmt_info()
|
|
|
|
|
|
|
|
# Build registration string
|
|
|
|
# https://en.wikipedia.org/wiki/Binfmt_misc
|
|
|
|
# :name:type:offset:magic:mask:interpreter:flags
|
2022-08-26 22:54:30 +02:00
|
|
|
info = lines[qemu_arch]
|
2021-10-10 18:24:22 +02:00
|
|
|
code = info['line']
|
|
|
|
binfmt = '/proc/sys/fs/binfmt_misc'
|
|
|
|
register = binfmt + '/register'
|
|
|
|
if not os.path.exists(register):
|
|
|
|
logging.info('mounting binfmt_misc')
|
|
|
|
result = mount('binfmt_misc', binfmt, options=[], fs_type='binfmt_misc')
|
|
|
|
if result.returncode != 0:
|
|
|
|
raise Exception(f'Failed mounting binfmt_misc to {binfmt}')
|
|
|
|
|
|
|
|
# Register in binfmt_misc
|
|
|
|
logging.info(f"Registering qemu binfmt ({arch})")
|
2022-09-18 02:29:53 +02:00
|
|
|
run_root_cmd(f'echo "{code}" > {register} 2>/dev/null')
|
2021-10-10 18:24:22 +02:00
|
|
|
if not is_registered(arch):
|
|
|
|
logging.debug(f'binfmt line: {code}')
|
|
|
|
raise Exception(f'Failed to register qemu-user for {arch} with binfmt_misc, {binfmt}/{info["name"]} not found')
|
|
|
|
|
|
|
|
|
2022-02-14 20:53:15 +01:00
|
|
|
def unregister(arch):
|
2022-08-26 22:54:30 +02:00
|
|
|
if arch not in QEMU_ARCHES:
|
|
|
|
raise Exception(f'binfmt.unregister(): unknown arch {arch} (not in QEMU_ARCHES)')
|
|
|
|
qemu_arch = QEMU_ARCHES[arch]
|
|
|
|
binfmt_file = "/proc/sys/fs/binfmt_misc/qemu-" + qemu_arch
|
2021-10-10 18:24:22 +02:00
|
|
|
if not os.path.exists(binfmt_file):
|
|
|
|
return
|
|
|
|
logging.info(f"Unregistering qemu binfmt ({arch})")
|
2022-09-18 02:29:53 +02:00
|
|
|
run_root_cmd(f"echo -1 > {binfmt_file}")
|