Files
cleonos/wine/cleonos_wine_lib/runner.py

2284 lines
73 KiB
Python
Raw Normal View History

2026-04-13 18:32:22 +08:00
from __future__ import annotations
import os
import struct
import sys
2026-04-13 19:08:35 +08:00
import time
2026-04-21 22:22:31 +08:00
from dataclasses import dataclass, field
2026-04-13 18:32:22 +08:00
from pathlib import Path
2026-04-19 16:07:44 +08:00
from typing import Dict, List, Optional, Tuple
2026-04-13 18:32:22 +08:00
from .constants import (
DEFAULT_MAX_EXEC_DEPTH,
2026-04-19 16:07:44 +08:00
FD_INHERIT,
2026-04-13 18:32:22 +08:00
FS_NAME_MAX,
MAX_CSTR,
MAX_IO_READ,
2026-04-19 16:07:44 +08:00
O_APPEND,
O_CREAT,
O_RDONLY,
O_RDWR,
O_TRUNC,
O_WRONLY,
2026-04-13 18:32:22 +08:00
PAGE_SIZE,
2026-04-19 16:07:44 +08:00
PROC_STATE_EXITED,
PROC_STATE_PENDING,
PROC_STATE_RUNNING,
PROC_STATE_STOPPED,
SIGCONT,
SIGKILL,
SIGSTOP,
SIGTERM,
2026-04-16 22:29:08 +08:00
SYS_AUDIO_AVAILABLE,
SYS_AUDIO_PLAY_TONE,
SYS_AUDIO_STOP,
2026-04-13 18:32:22 +08:00
SYS_CONTEXT_SWITCHES,
SYS_CUR_TASK,
2026-04-19 16:07:44 +08:00
SYS_DL_CLOSE,
SYS_DL_OPEN,
SYS_DL_SYM,
2026-04-13 18:32:22 +08:00
SYS_EXEC_PATH,
2026-04-16 22:29:08 +08:00
SYS_EXEC_PATHV,
2026-04-19 16:07:44 +08:00
SYS_EXEC_PATHV_IO,
2026-04-13 18:32:22 +08:00
SYS_EXEC_REQUESTS,
SYS_EXEC_SUCCESS,
2026-04-13 19:08:35 +08:00
SYS_EXIT,
2026-04-19 16:07:44 +08:00
SYS_FD_CLOSE,
SYS_FD_DUP,
SYS_FD_OPEN,
SYS_FD_READ,
SYS_FD_WRITE,
2026-04-21 22:22:31 +08:00
SYS_FB_BLIT,
SYS_FB_CLEAR,
SYS_FB_INFO,
2026-04-13 18:32:22 +08:00
SYS_FS_APPEND,
SYS_FS_CHILD_COUNT,
SYS_FS_GET_CHILD_NAME,
SYS_FS_MKDIR,
SYS_FS_NODE_COUNT,
SYS_FS_READ,
SYS_FS_REMOVE,
SYS_FS_STAT_SIZE,
SYS_FS_STAT_TYPE,
SYS_FS_WRITE,
2026-04-19 16:07:44 +08:00
SYS_GETPID,
2026-04-13 18:32:22 +08:00
SYS_KBD_BUFFERED,
SYS_KBD_DROPPED,
SYS_KBD_GET_CHAR,
SYS_KBD_HOTKEY_SWITCHES,
SYS_KBD_POPPED,
SYS_KBD_PUSHED,
SYS_KELF_COUNT,
SYS_KELF_RUNS,
SYS_LOG_JOURNAL_COUNT,
SYS_LOG_JOURNAL_READ,
SYS_LOG_WRITE,
2026-04-19 16:07:44 +08:00
SYS_PROC_ARGC,
SYS_PROC_ARGV,
SYS_PROC_COUNT,
SYS_PROC_ENVC,
SYS_PROC_ENV,
SYS_PROC_FAULT_ERROR,
SYS_PROC_FAULT_RIP,
SYS_PROC_FAULT_VECTOR,
SYS_PROC_KILL,
SYS_PROC_LAST_SIGNAL,
SYS_PROC_PID_AT,
SYS_PROC_SNAPSHOT,
SYS_RESTART,
2026-04-13 18:32:22 +08:00
SYS_SERVICE_COUNT,
SYS_SERVICE_READY_COUNT,
2026-04-19 16:07:44 +08:00
SYS_SHUTDOWN,
SYS_SLEEP_TICKS,
SYS_SPAWN_PATH,
SYS_SPAWN_PATHV,
SYS_STATS_ID_COUNT,
SYS_STATS_RECENT_ID,
SYS_STATS_RECENT_WINDOW,
SYS_STATS_TOTAL,
2026-04-13 18:32:22 +08:00
SYS_TASK_COUNT,
SYS_TIMER_TICKS,
SYS_TTY_ACTIVE,
SYS_TTY_COUNT,
SYS_TTY_SWITCH,
SYS_TTY_WRITE,
SYS_TTY_WRITE_CHAR,
SYS_USER_EXEC_REQUESTED,
SYS_USER_LAUNCH_FAIL,
SYS_USER_LAUNCH_OK,
SYS_USER_LAUNCH_TRIES,
SYS_USER_SHELL_READY,
2026-04-19 16:07:44 +08:00
SYS_WAITPID,
SYS_YIELD,
2026-04-13 18:32:22 +08:00
page_ceil,
page_floor,
u64,
u64_neg1,
)
2026-04-21 22:22:31 +08:00
from .fb_window import FBWindow
2026-04-13 18:32:22 +08:00
from .input_pump import InputPump
from .platform import (
Uc,
UcError,
UC_ARCH_X86,
2026-04-16 22:29:08 +08:00
UC_ERR_FETCH_PROT,
UC_ERR_FETCH_UNMAPPED,
UC_ERR_INSN_INVALID,
UC_ERR_READ_PROT,
UC_ERR_READ_UNMAPPED,
UC_ERR_WRITE_PROT,
UC_ERR_WRITE_UNMAPPED,
2026-04-13 18:32:22 +08:00
UC_HOOK_CODE,
UC_HOOK_INTR,
UC_MODE_64,
UC_PROT_ALL,
UC_PROT_EXEC,
UC_PROT_READ,
UC_PROT_WRITE,
UC_X86_REG_RAX,
UC_X86_REG_RBP,
UC_X86_REG_RBX,
UC_X86_REG_RCX,
UC_X86_REG_RDX,
UC_X86_REG_RSP,
2026-04-16 22:29:08 +08:00
UC_X86_REG_RIP,
2026-04-13 18:32:22 +08:00
)
from .state import SharedKernelState
@dataclass
class ELFSegment:
vaddr: int
memsz: int
flags: int
data: bytes
@dataclass
class ELFImage:
entry: int
segments: List[ELFSegment]
2026-04-19 16:07:44 +08:00
@dataclass
class FDEntry:
kind: str
flags: int
offset: int = 0
path: str = ""
tty_index: int = 0
2026-04-21 22:22:31 +08:00
@dataclass
class DLImage:
handle: int
guest_path: str
host_path: str
owner_pid: int
ref_count: int
map_start: int
map_end: int
load_bias: int
symbols: Dict[str, int] = field(default_factory=dict)
2026-04-16 22:29:08 +08:00
EXEC_PATH_MAX = 192
EXEC_ARG_LINE_MAX = 256
EXEC_ENV_LINE_MAX = 512
EXEC_MAX_ARGS = 24
EXEC_MAX_ENVS = 24
EXEC_ITEM_MAX = 128
EXEC_STATUS_SIGNAL_FLAG = 1 << 63
2026-04-19 16:07:44 +08:00
PROC_PATH_MAX = 192
FD_MAX = 64
2026-04-21 22:22:31 +08:00
DL_MAX_NAME = 192
DL_MAX_SYMBOL = 128
DL_BASE_START = 0x0000000100000000
DL_BASE_GAP = 0x0000000000100000
FB_DEFAULT_WIDTH = 1280
FB_DEFAULT_HEIGHT = 800
FB_MAX_DIM = 4096
FB_MAX_UPLOAD_BYTES = 64 * 1024 * 1024
2026-04-16 22:29:08 +08:00
2026-04-13 18:32:22 +08:00
class CLeonOSWineNative:
def __init__(
self,
elf_path: Path,
rootfs: Path,
guest_path_hint: str,
*,
state: Optional[SharedKernelState] = None,
depth: int = 0,
max_exec_depth: int = DEFAULT_MAX_EXEC_DEPTH,
no_kbd: bool = False,
verbose: bool = False,
top_level: bool = True,
2026-04-13 19:08:35 +08:00
pid: int = 0,
ppid: int = 0,
2026-04-16 22:29:08 +08:00
argv_items: Optional[List[str]] = None,
env_items: Optional[List[str]] = None,
2026-04-19 16:07:44 +08:00
inherited_fds: Optional[Dict[int, FDEntry]] = None,
2026-04-21 22:22:31 +08:00
fb_window: bool = False,
fb_scale: int = 2,
fb_max_fps: int = 60,
fb_hold_ms: int = 2500,
2026-04-13 18:32:22 +08:00
) -> None:
self.elf_path = elf_path
self.rootfs = rootfs
self.guest_path_hint = guest_path_hint
self.state = state if state is not None else SharedKernelState()
self.depth = depth
self.max_exec_depth = max_exec_depth
self.no_kbd = no_kbd
self.verbose = verbose
self.top_level = top_level
2026-04-13 19:08:35 +08:00
self.pid = int(pid)
self.ppid = int(ppid)
2026-04-21 22:22:31 +08:00
self.fb_window = bool(fb_window)
self.fb_scale = max(1, int(fb_scale))
self.fb_max_fps = max(1, int(fb_max_fps))
self.fb_hold_ms = max(0, int(fb_hold_ms))
2026-04-16 22:29:08 +08:00
self.argv_items = list(argv_items) if argv_items is not None else []
self.env_items = list(env_items) if env_items is not None else []
2026-04-13 19:08:35 +08:00
self._exit_requested = False
self._exit_status = 0
2026-04-13 18:32:22 +08:00
self.image = self._parse_elf(self.elf_path)
self.exit_code: Optional[int] = None
self._input_pump: Optional[InputPump] = None
self._stack_base = 0x00007FFF00000000
self._stack_size = 0x0000000000020000
self._ret_sentinel = 0x00007FFF10000000
self._mapped_ranges: List[Tuple[int, int]] = []
2026-04-19 16:07:44 +08:00
self._tty_index = int(self.state.tty_active)
self._fds: Dict[int, FDEntry] = {}
self._fd_inherited = inherited_fds if inherited_fds is not None else {}
2026-04-21 22:22:31 +08:00
self._dl_images: Dict[int, DLImage] = {}
self._dl_path_to_handle: Dict[str, int] = {}
self._dl_next_handle = 1
self._dl_next_base = DL_BASE_START
self._fb_width = self._bounded_env_int("CLEONOS_WINE_FB_WIDTH", FB_DEFAULT_WIDTH, 64, FB_MAX_DIM)
self._fb_height = self._bounded_env_int("CLEONOS_WINE_FB_HEIGHT", FB_DEFAULT_HEIGHT, 64, FB_MAX_DIM)
self._fb_bpp = 32
self._fb_pitch = self._fb_width * 4
self._fb_pixels = bytearray(self._fb_pitch * self._fb_height)
self._fb_window: Optional[FBWindow] = None
self._fb_window_failed = False
self._fb_dirty = False
self._fb_presented_once = False
2026-04-13 18:32:22 +08:00
2026-04-16 22:29:08 +08:00
default_path = self._normalize_guest_path(self.guest_path_hint or f"/{self.elf_path.name}")
self.argv_items, self.env_items = self._prepare_exec_items(default_path, self.argv_items, self.env_items)
2026-04-19 16:07:44 +08:00
self._init_default_fds()
@staticmethod
def _clone_fd_entry(entry: FDEntry) -> FDEntry:
return FDEntry(kind=entry.kind, flags=int(entry.flags), offset=int(entry.offset), path=str(entry.path), tty_index=int(entry.tty_index))
@staticmethod
def _fd_access_mode(flags: int) -> int:
return int(flags) & 0x3
@classmethod
def _fd_access_mode_valid(cls, flags: int) -> bool:
mode = cls._fd_access_mode(flags)
return mode in (O_RDONLY, O_WRONLY, O_RDWR)
@classmethod
def _fd_can_read(cls, flags: int) -> bool:
mode = cls._fd_access_mode(flags)
return mode in (O_RDONLY, O_RDWR)
@classmethod
def _fd_can_write(cls, flags: int) -> bool:
mode = cls._fd_access_mode(flags)
return mode in (O_WRONLY, O_RDWR)
2026-04-21 22:22:31 +08:00
@staticmethod
def _bounded_env_int(name: str, default: int, min_value: int, max_value: int) -> int:
raw = os.environ.get(name)
value = default
if raw is not None:
try:
value = int(raw.strip(), 10)
except Exception:
value = default
if value < min_value:
value = min_value
if value > max_value:
value = max_value
return value
def _ensure_fb_window(self) -> None:
if not self.fb_window:
return
if self._fb_window is not None or self._fb_window_failed:
return
self._fb_window = FBWindow.create(
self._fb_width,
self._fb_height,
self.fb_scale,
self.fb_max_fps,
verbose=self.verbose,
)
if self._fb_window is None:
self._fb_window_failed = True
def _fb_poll_window(self) -> None:
self._ensure_fb_window()
if self._fb_window is not None:
self._fb_window.pump_input(self.state)
if self._fb_window.is_closed():
self._fb_window = None
def _fb_present(self, *, force: bool = False) -> None:
self._ensure_fb_window()
if self._fb_window is None:
return
self._fb_window.pump_input(self.state)
did_present = self._fb_window.present(self._fb_pixels, force=force)
if did_present:
self._fb_dirty = False
self._fb_presented_once = True
if self._fb_window.is_closed():
self._fb_window = None
def _fb_mark_dirty(self) -> None:
self._fb_dirty = True
self._fb_present(force=False)
def _fb_hold_after_exit(self) -> None:
end_ns: int
if self._fb_window is None:
return
if self.fb_hold_ms <= 0 or self._fb_presented_once is False:
return
end_ns = time.monotonic_ns() + (self.fb_hold_ms * 1_000_000)
while time.monotonic_ns() < end_ns:
if self._fb_window is None:
return
self._fb_window.pump_input(self.state)
if self._fb_window.is_closed():
self._fb_window = None
return
self._fb_window.present(self._fb_pixels, force=True)
time.sleep(0.016)
2026-04-19 16:07:44 +08:00
def _init_default_fds(self) -> None:
self._fds = {
0: FDEntry(kind="tty", flags=O_RDONLY, offset=0, tty_index=self._tty_index),
1: FDEntry(kind="tty", flags=O_WRONLY, offset=0, tty_index=self._tty_index),
2: FDEntry(kind="tty", flags=O_WRONLY, offset=0, tty_index=self._tty_index),
}
for target in (0, 1, 2):
inherited = self._fd_inherited.get(target)
if inherited is not None:
self._fds[target] = self._clone_fd_entry(inherited)
for target in (0, 1, 2):
entry = self._fds.get(target)
if entry is not None and entry.kind == "tty":
self._tty_index = int(entry.tty_index)
break
def _fd_lookup(self, fd: int) -> Optional[FDEntry]:
if fd < 0 or fd >= FD_MAX:
return None
return self._fds.get(int(fd))
def _fd_find_free(self) -> int:
for fd in range(FD_MAX):
if fd not in self._fds:
return fd
return -1
def _stdio_entry_for_child(self, target_fd: int, override_fd: int, require_read: bool, require_write: bool) -> Optional[FDEntry]:
if override_fd == FD_INHERIT:
src = self._fd_lookup(target_fd)
else:
src = self._fd_lookup(override_fd)
if src is None:
return None
if require_read and not self._fd_can_read(src.flags):
return None
if require_write and not self._fd_can_write(src.flags):
return None
return self._clone_fd_entry(src)
def _build_child_stdio_map(self, stdin_fd: int, stdout_fd: int, stderr_fd: int) -> Optional[Dict[int, FDEntry]]:
child_map: Dict[int, FDEntry] = {}
in_entry = self._stdio_entry_for_child(0, stdin_fd, require_read=True, require_write=False)
out_entry = self._stdio_entry_for_child(1, stdout_fd, require_read=False, require_write=True)
err_entry = self._stdio_entry_for_child(2, stderr_fd, require_read=False, require_write=True)
if in_entry is None or out_entry is None or err_entry is None:
return None
child_map[0] = in_entry
child_map[1] = out_entry
child_map[2] = err_entry
return child_map
2026-04-16 22:29:08 +08:00
2026-04-13 18:32:22 +08:00
def run(self) -> Optional[int]:
2026-04-13 19:08:35 +08:00
if self.pid == 0:
self.pid = self.state.alloc_pid(self.ppid)
prev_pid = self.state.get_current_pid()
self.state.set_current_pid(self.pid)
2026-04-16 22:29:08 +08:00
self.state.set_proc_cmdline(self.pid, self.argv_items, self.env_items)
self.state.set_proc_fault(self.pid, 0, 0, 0, 0)
2026-04-19 16:07:44 +08:00
self.state.set_proc_running(self.pid, self.argv_items[0] if self.argv_items else self.guest_path_hint, self._tty_index)
2026-04-13 19:08:35 +08:00
2026-04-13 18:32:22 +08:00
uc = Uc(UC_ARCH_X86, UC_MODE_64)
self._install_hooks(uc)
self._load_segments(uc)
self._prepare_stack_and_return(uc)
2026-04-21 22:22:31 +08:00
self._ensure_fb_window()
self._fb_present(force=True)
2026-04-13 18:32:22 +08:00
if self.top_level and not self.no_kbd:
self._input_pump = InputPump(self.state)
self._input_pump.start()
2026-04-16 22:29:08 +08:00
interrupted = False
runtime_fault_status: Optional[int] = None
2026-04-13 19:08:35 +08:00
2026-04-13 18:32:22 +08:00
try:
uc.emu_start(self.image.entry, 0)
except KeyboardInterrupt:
2026-04-16 22:29:08 +08:00
interrupted = True
2026-04-13 18:32:22 +08:00
if self.top_level:
print("\n[WINE] interrupted by user", file=sys.stderr)
except UcError as exc:
2026-04-16 22:29:08 +08:00
runtime_fault_status = self._status_from_uc_error(uc, exc)
2026-04-13 18:32:22 +08:00
if self.verbose or self.top_level:
print(f"[WINE][ERROR] runtime crashed: {exc}", file=sys.stderr)
finally:
if self.top_level and self._input_pump is not None:
self._input_pump.stop()
2026-04-21 22:22:31 +08:00
if self._fb_window is not None:
self._fb_hold_after_exit()
if self._fb_window is not None:
self._fb_window.close()
self._fb_window = None
2026-04-13 18:32:22 +08:00
2026-04-16 22:29:08 +08:00
if interrupted:
2026-04-13 19:08:35 +08:00
self.state.mark_exited(self.pid, u64_neg1())
self.state.set_current_pid(prev_pid)
return None
2026-04-16 22:29:08 +08:00
if runtime_fault_status is not None:
self.exit_code = runtime_fault_status
2026-04-13 18:32:22 +08:00
if self.exit_code is None:
self.exit_code = self._reg_read(uc, UC_X86_REG_RAX)
2026-04-13 19:08:35 +08:00
if self._exit_requested:
self.exit_code = self._exit_status
self.exit_code = u64(self.exit_code)
self.state.mark_exited(self.pid, self.exit_code)
self.state.set_current_pid(prev_pid)
return self.exit_code
2026-04-13 18:32:22 +08:00
def _install_hooks(self, uc: Uc) -> None:
uc.hook_add(UC_HOOK_INTR, self._hook_intr)
uc.hook_add(UC_HOOK_CODE, self._hook_code, begin=self._ret_sentinel, end=self._ret_sentinel)
def _hook_code(self, uc: Uc, address: int, size: int, _user_data) -> None:
_ = size
if address == self._ret_sentinel:
self.exit_code = self._reg_read(uc, UC_X86_REG_RAX)
uc.emu_stop()
def _hook_intr(self, uc: Uc, intno: int, _user_data) -> None:
if intno != 0x80:
raise UcError(1)
syscall_id = self._reg_read(uc, UC_X86_REG_RAX)
arg0 = self._reg_read(uc, UC_X86_REG_RBX)
arg1 = self._reg_read(uc, UC_X86_REG_RCX)
arg2 = self._reg_read(uc, UC_X86_REG_RDX)
2026-04-19 16:07:44 +08:00
self.state.record_syscall(syscall_id)
2026-04-13 18:32:22 +08:00
self.state.context_switches = u64(self.state.context_switches + 1)
ret = self._dispatch_syscall(uc, syscall_id, arg0, arg1, arg2)
self._reg_write(uc, UC_X86_REG_RAX, u64(ret))
def _dispatch_syscall(self, uc: Uc, sid: int, arg0: int, arg1: int, arg2: int) -> int:
2026-04-21 22:22:31 +08:00
self._fb_poll_window()
2026-04-13 18:32:22 +08:00
if sid == SYS_LOG_WRITE:
data = self._read_guest_bytes(uc, arg0, arg1)
text = data.decode("utf-8", errors="replace")
self._host_write(text)
self.state.log_journal_push(text)
return len(data)
if sid == SYS_TIMER_TICKS:
return self.state.timer_ticks()
if sid == SYS_TASK_COUNT:
return self.state.task_count
if sid == SYS_CUR_TASK:
return self.state.current_task
if sid == SYS_SERVICE_COUNT:
return self.state.service_count
if sid == SYS_SERVICE_READY_COUNT:
return self.state.service_ready
if sid == SYS_CONTEXT_SWITCHES:
return self.state.context_switches
if sid == SYS_KELF_COUNT:
return self.state.kelf_count
if sid == SYS_KELF_RUNS:
return self.state.kelf_runs
if sid == SYS_FS_NODE_COUNT:
return self._fs_node_count()
if sid == SYS_FS_CHILD_COUNT:
return self._fs_child_count(uc, arg0)
if sid == SYS_FS_GET_CHILD_NAME:
return self._fs_get_child_name(uc, arg0, arg1, arg2)
if sid == SYS_FS_READ:
return self._fs_read(uc, arg0, arg1, arg2)
if sid == SYS_EXEC_PATH:
return self._exec_path(uc, arg0)
2026-04-16 22:29:08 +08:00
if sid == SYS_EXEC_PATHV:
return self._exec_pathv(uc, arg0, arg1, arg2)
2026-04-19 16:07:44 +08:00
if sid == SYS_EXEC_PATHV_IO:
return self._exec_pathv_io(uc, arg0, arg1, arg2)
2026-04-13 19:08:35 +08:00
if sid == SYS_SPAWN_PATH:
return self._spawn_path(uc, arg0)
2026-04-16 22:29:08 +08:00
if sid == SYS_SPAWN_PATHV:
return self._spawn_pathv(uc, arg0, arg1, arg2)
2026-04-13 19:08:35 +08:00
if sid == SYS_WAITPID:
return self._wait_pid(uc, arg0, arg1)
if sid == SYS_GETPID:
return self.state.get_current_pid()
2026-04-16 22:29:08 +08:00
if sid == SYS_PROC_ARGC:
return self._proc_argc()
if sid == SYS_PROC_ARGV:
return self._proc_argv(uc, arg0, arg1, arg2)
if sid == SYS_PROC_ENVC:
return self._proc_envc()
if sid == SYS_PROC_ENV:
return self._proc_env(uc, arg0, arg1, arg2)
if sid == SYS_PROC_LAST_SIGNAL:
return self._proc_last_signal()
if sid == SYS_PROC_FAULT_VECTOR:
return self._proc_fault_vector()
if sid == SYS_PROC_FAULT_ERROR:
return self._proc_fault_error()
if sid == SYS_PROC_FAULT_RIP:
return self._proc_fault_rip()
2026-04-19 16:07:44 +08:00
if sid == SYS_PROC_COUNT:
return self._proc_count()
if sid == SYS_PROC_PID_AT:
return self._proc_pid_at(uc, arg0, arg1)
if sid == SYS_PROC_SNAPSHOT:
return self._proc_snapshot(uc, arg0, arg1, arg2)
if sid == SYS_PROC_KILL:
return self._proc_kill(uc, arg0, arg1)
2026-04-13 19:08:35 +08:00
if sid == SYS_EXIT:
return self._request_exit(uc, arg0)
if sid == SYS_SLEEP_TICKS:
return self._sleep_ticks(arg0)
if sid == SYS_YIELD:
return self._yield_once()
2026-04-16 22:29:08 +08:00
if sid == SYS_AUDIO_AVAILABLE:
return 0
if sid == SYS_AUDIO_PLAY_TONE:
return 0
if sid == SYS_AUDIO_STOP:
return 1
2026-04-14 19:13:55 +08:00
if sid == SYS_SHUTDOWN:
self._host_write("\n[WINE] shutdown requested by guest\n")
self._exit_requested = True
self._exit_status = 0
return 1
if sid == SYS_RESTART:
self._host_write("\n[WINE] restart requested by guest\n")
self._exit_requested = True
self._exit_status = 0
return 1
2026-04-13 18:32:22 +08:00
if sid == SYS_EXEC_REQUESTS:
return self.state.exec_requests
if sid == SYS_EXEC_SUCCESS:
return self.state.exec_success
if sid == SYS_USER_SHELL_READY:
return self.state.user_shell_ready
if sid == SYS_USER_EXEC_REQUESTED:
return self.state.user_exec_requested
if sid == SYS_USER_LAUNCH_TRIES:
return self.state.user_launch_tries
if sid == SYS_USER_LAUNCH_OK:
return self.state.user_launch_ok
if sid == SYS_USER_LAUNCH_FAIL:
return self.state.user_launch_fail
if sid == SYS_TTY_COUNT:
return self.state.tty_count
if sid == SYS_TTY_ACTIVE:
return self.state.tty_active
if sid == SYS_TTY_SWITCH:
if arg0 >= self.state.tty_count:
return u64_neg1()
self.state.tty_active = int(arg0)
return self.state.tty_active
if sid == SYS_TTY_WRITE:
data = self._read_guest_bytes(uc, arg0, arg1)
self._host_write(data.decode("utf-8", errors="replace"))
return len(data)
if sid == SYS_TTY_WRITE_CHAR:
ch = chr(arg0 & 0xFF)
if ch in ("\b", "\x7f"):
self._host_write("\b \b")
else:
self._host_write(ch)
return 1
if sid == SYS_KBD_GET_CHAR:
key = self.state.pop_key()
return u64_neg1() if key is None else key
if sid == SYS_FS_STAT_TYPE:
return self._fs_stat_type(uc, arg0)
if sid == SYS_FS_STAT_SIZE:
return self._fs_stat_size(uc, arg0)
if sid == SYS_FS_MKDIR:
return self._fs_mkdir(uc, arg0)
if sid == SYS_FS_WRITE:
return self._fs_write(uc, arg0, arg1, arg2)
if sid == SYS_FS_APPEND:
return self._fs_append(uc, arg0, arg1, arg2)
if sid == SYS_FS_REMOVE:
return self._fs_remove(uc, arg0)
if sid == SYS_LOG_JOURNAL_COUNT:
return self.state.log_journal_count()
if sid == SYS_LOG_JOURNAL_READ:
return self._log_journal_read(uc, arg0, arg1, arg2)
if sid == SYS_KBD_BUFFERED:
return self.state.buffered_count()
if sid == SYS_KBD_PUSHED:
return self.state.kbd_push_count
if sid == SYS_KBD_POPPED:
return self.state.kbd_pop_count
if sid == SYS_KBD_DROPPED:
return self.state.kbd_drop_count
if sid == SYS_KBD_HOTKEY_SWITCHES:
return self.state.kbd_hotkey_switches
2026-04-19 16:07:44 +08:00
if sid == SYS_STATS_TOTAL:
return self.state.stats_total()
if sid == SYS_STATS_ID_COUNT:
return self.state.stats_id_count(arg0)
if sid == SYS_STATS_RECENT_WINDOW:
return self.state.stats_recent_window()
if sid == SYS_STATS_RECENT_ID:
return self.state.stats_recent_id_count(arg0)
if sid == SYS_FD_OPEN:
return self._fd_open(uc, arg0, arg1, arg2)
if sid == SYS_FD_READ:
return self._fd_read(uc, arg0, arg1, arg2)
if sid == SYS_FD_WRITE:
return self._fd_write(uc, arg0, arg1, arg2)
if sid == SYS_FD_CLOSE:
return self._fd_close(arg0)
if sid == SYS_FD_DUP:
return self._fd_dup(arg0)
if sid == SYS_DL_OPEN:
2026-04-21 22:22:31 +08:00
return self._dl_open(uc, arg0)
2026-04-19 16:07:44 +08:00
if sid == SYS_DL_CLOSE:
2026-04-21 22:22:31 +08:00
return self._dl_close(arg0)
2026-04-19 16:07:44 +08:00
if sid == SYS_DL_SYM:
2026-04-21 22:22:31 +08:00
return self._dl_sym(uc, arg0, arg1)
if sid == SYS_FB_INFO:
return self._fb_info(uc, arg0)
if sid == SYS_FB_BLIT:
return self._fb_blit(uc, arg0)
if sid == SYS_FB_CLEAR:
return self._fb_clear(arg0)
2026-04-13 18:32:22 +08:00
return u64_neg1()
def _host_write(self, text: str) -> None:
if not text:
return
sys.stdout.write(text)
sys.stdout.flush()
2026-04-19 16:07:44 +08:00
def _host_write_bytes(self, data: bytes) -> None:
if not data:
return
if hasattr(sys.stdout, "buffer"):
sys.stdout.buffer.write(data)
sys.stdout.flush()
return
self._host_write(data.decode("utf-8", errors="replace"))
2026-04-13 18:32:22 +08:00
def _load_segments(self, uc: Uc) -> None:
for seg in self.image.segments:
start = page_floor(seg.vaddr)
end = page_ceil(seg.vaddr + seg.memsz)
self._map_region(uc, start, end - start, UC_PROT_ALL)
for seg in self.image.segments:
if seg.data:
self._mem_write(uc, seg.vaddr, seg.data)
for seg in self.image.segments:
start = page_floor(seg.vaddr)
end = page_ceil(seg.vaddr + seg.memsz)
size = end - start
perms = 0
if seg.flags & 0x4:
perms |= UC_PROT_READ
if seg.flags & 0x2:
perms |= UC_PROT_WRITE
if seg.flags & 0x1:
perms |= UC_PROT_EXEC
if perms == 0:
perms = UC_PROT_READ
try:
uc.mem_protect(start, size, perms)
except Exception:
pass
def _prepare_stack_and_return(self, uc: Uc) -> None:
self._map_region(uc, self._stack_base, self._stack_size, UC_PROT_READ | UC_PROT_WRITE)
self._map_region(uc, self._ret_sentinel, PAGE_SIZE, UC_PROT_READ | UC_PROT_EXEC)
self._mem_write(uc, self._ret_sentinel, b"\x90")
rsp = self._stack_base + self._stack_size - 8
self._mem_write(uc, rsp, struct.pack("<Q", self._ret_sentinel))
self._reg_write(uc, UC_X86_REG_RSP, rsp)
self._reg_write(uc, UC_X86_REG_RBP, rsp)
def _map_region(self, uc: Uc, addr: int, size: int, perms: int) -> None:
if size <= 0:
return
start = page_floor(addr)
end = page_ceil(addr + size)
if self._is_range_mapped(start, end):
return
uc.mem_map(start, end - start, perms)
self._mapped_ranges.append((start, end))
def _is_range_mapped(self, start: int, end: int) -> bool:
for ms, me in self._mapped_ranges:
if start >= ms and end <= me:
return True
return False
2026-04-21 22:22:31 +08:00
def _range_overlaps_mapped(self, start: int, end: int) -> bool:
for ms, me in self._mapped_ranges:
if start < me and end > ms:
return True
return False
2026-04-13 18:32:22 +08:00
@staticmethod
def _reg_read(uc: Uc, reg: int) -> int:
return int(uc.reg_read(reg))
@staticmethod
def _reg_write(uc: Uc, reg: int, value: int) -> None:
uc.reg_write(reg, u64(value))
@staticmethod
def _mem_write(uc: Uc, addr: int, data: bytes) -> None:
if addr == 0 or not data:
return
uc.mem_write(addr, data)
def _read_guest_cstring(self, uc: Uc, addr: int, max_len: int = MAX_CSTR) -> str:
if addr == 0:
return ""
out = bytearray()
for i in range(max_len):
try:
ch = uc.mem_read(addr + i, 1)
except UcError:
break
if not ch or ch[0] == 0:
break
out.append(ch[0])
return out.decode("utf-8", errors="replace")
def _read_guest_bytes(self, uc: Uc, addr: int, size: int) -> bytes:
if addr == 0 or size == 0:
return b""
safe_size = int(min(size, MAX_IO_READ))
try:
return bytes(uc.mem_read(addr, safe_size))
except UcError:
return b""
2026-04-19 16:07:44 +08:00
def _read_guest_bytes_exact(self, uc: Uc, addr: int, size: int) -> Optional[bytes]:
if size < 0 or addr == 0:
return None
if size == 0:
return b""
out = bytearray()
cursor = int(addr)
left = int(size)
while left > 0:
chunk = min(left, MAX_IO_READ)
try:
data = uc.mem_read(cursor, chunk)
except UcError:
return None
if len(data) != chunk:
return None
out.extend(data)
cursor += chunk
left -= chunk
return bytes(out)
2026-04-13 18:32:22 +08:00
def _write_guest_bytes(self, uc: Uc, addr: int, data: bytes) -> bool:
if addr == 0:
return False
try:
uc.mem_write(addr, data)
return True
except UcError:
return False
@staticmethod
2026-04-21 22:22:31 +08:00
def _parse_elf_image_from_blob(data: bytes, *, require_entry: bool) -> ELFImage:
2026-04-13 18:32:22 +08:00
if len(data) < 64:
2026-04-21 22:22:31 +08:00
raise RuntimeError("ELF too small")
2026-04-13 18:32:22 +08:00
if data[0:4] != b"\x7fELF":
2026-04-21 22:22:31 +08:00
raise RuntimeError("invalid ELF magic")
2026-04-13 18:32:22 +08:00
if data[4] != 2 or data[5] != 1:
2026-04-21 22:22:31 +08:00
raise RuntimeError("unsupported ELF class/endianness")
2026-04-13 18:32:22 +08:00
entry = struct.unpack_from("<Q", data, 0x18)[0]
phoff = struct.unpack_from("<Q", data, 0x20)[0]
phentsize = struct.unpack_from("<H", data, 0x36)[0]
phnum = struct.unpack_from("<H", data, 0x38)[0]
2026-04-21 22:22:31 +08:00
if require_entry and entry == 0:
raise RuntimeError("ELF entry is 0")
2026-04-13 18:32:22 +08:00
if phentsize == 0 or phnum == 0:
2026-04-21 22:22:31 +08:00
raise RuntimeError("ELF has no program headers")
2026-04-13 18:32:22 +08:00
segments: List[ELFSegment] = []
for i in range(phnum):
off = phoff + i * phentsize
if off + 56 > len(data):
break
p_type, p_flags, p_offset, p_vaddr, _p_paddr, p_filesz, p_memsz, _p_align = struct.unpack_from(
"<IIQQQQQQ", data, off
)
if p_type != 1 or p_memsz == 0:
continue
fs = int(p_filesz)
fo = int(p_offset)
if fs > 0:
if fo >= len(data):
seg_data = b""
else:
seg_data = data[fo : min(len(data), fo + fs)]
else:
seg_data = b""
segments.append(ELFSegment(vaddr=int(p_vaddr), memsz=int(p_memsz), flags=int(p_flags), data=seg_data))
if not segments:
2026-04-21 22:22:31 +08:00
raise RuntimeError("ELF has no PT_LOAD segments")
2026-04-13 18:32:22 +08:00
return ELFImage(entry=int(entry), segments=segments)
2026-04-21 22:22:31 +08:00
@staticmethod
def _parse_elf(path: Path) -> ELFImage:
data = path.read_bytes()
try:
return CLeonOSWineNative._parse_elf_image_from_blob(data, require_entry=True)
except RuntimeError as exc:
raise RuntimeError(f"{exc}: {path}") from exc
2026-04-13 18:32:22 +08:00
def _fs_node_count(self) -> int:
count = 1
for _root, dirs, files in os.walk(self.rootfs):
dirs[:] = [d for d in dirs if not d.startswith(".")]
files = [f for f in files if not f.startswith(".")]
count += len(dirs) + len(files)
return count
def _fs_child_count(self, uc: Uc, dir_ptr: int) -> int:
path = self._read_guest_cstring(uc, dir_ptr)
host_dir = self._guest_to_host(path, must_exist=True)
if host_dir is None or not host_dir.is_dir():
return u64_neg1()
return len(self._list_children(host_dir))
def _fs_get_child_name(self, uc: Uc, dir_ptr: int, index: int, out_ptr: int) -> int:
if out_ptr == 0:
return 0
path = self._read_guest_cstring(uc, dir_ptr)
host_dir = self._guest_to_host(path, must_exist=True)
if host_dir is None or not host_dir.is_dir():
return 0
children = self._list_children(host_dir)
if index >= len(children):
return 0
name = children[int(index)]
encoded = name.encode("utf-8", errors="replace")
if len(encoded) >= FS_NAME_MAX:
encoded = encoded[: FS_NAME_MAX - 1]
return 1 if self._write_guest_bytes(uc, out_ptr, encoded + b"\x00") else 0
def _fs_read(self, uc: Uc, path_ptr: int, out_ptr: int, buf_size: int) -> int:
if out_ptr == 0 or buf_size == 0:
return 0
path = self._read_guest_cstring(uc, path_ptr)
host_path = self._guest_to_host(path, must_exist=True)
if host_path is None or not host_path.is_file():
return 0
read_size = int(min(buf_size, MAX_IO_READ))
try:
data = host_path.read_bytes()[:read_size]
except Exception:
return 0
if not data:
return 0
return len(data) if self._write_guest_bytes(uc, out_ptr, data) else 0
def _fs_stat_type(self, uc: Uc, path_ptr: int) -> int:
path = self._read_guest_cstring(uc, path_ptr)
host_path = self._guest_to_host(path, must_exist=True)
if host_path is None:
return u64_neg1()
if host_path.is_dir():
return 2
if host_path.is_file():
return 1
return u64_neg1()
def _fs_stat_size(self, uc: Uc, path_ptr: int) -> int:
path = self._read_guest_cstring(uc, path_ptr)
host_path = self._guest_to_host(path, must_exist=True)
if host_path is None:
return u64_neg1()
if host_path.is_dir():
return 0
if host_path.is_file():
try:
return host_path.stat().st_size
except Exception:
return u64_neg1()
return u64_neg1()
@staticmethod
def _guest_path_is_under_temp(path: str) -> bool:
return path == "/temp" or path.startswith("/temp/")
def _fs_mkdir(self, uc: Uc, path_ptr: int) -> int:
path = self._normalize_guest_path(self._read_guest_cstring(uc, path_ptr))
if not self._guest_path_is_under_temp(path):
return 0
host_path = self._guest_to_host(path, must_exist=False)
if host_path is None:
return 0
if host_path.exists() and host_path.is_file():
return 0
try:
host_path.mkdir(parents=True, exist_ok=True)
return 1
except Exception:
return 0
def _fs_write_common(self, uc: Uc, path_ptr: int, data_ptr: int, size: int, append_mode: bool) -> int:
path = self._normalize_guest_path(self._read_guest_cstring(uc, path_ptr))
if not self._guest_path_is_under_temp(path) or path == "/temp":
return 0
if size < 0 or size > self.state.fs_write_max:
return 0
host_path = self._guest_to_host(path, must_exist=False)
if host_path is None:
return 0
if host_path.exists() and host_path.is_dir():
return 0
data = b""
if size > 0:
if data_ptr == 0:
return 0
data = self._read_guest_bytes(uc, data_ptr, size)
if len(data) != int(size):
return 0
try:
host_path.parent.mkdir(parents=True, exist_ok=True)
mode = "ab" if append_mode else "wb"
with host_path.open(mode) as fh:
if data:
fh.write(data)
return 1
except Exception:
return 0
def _fs_write(self, uc: Uc, path_ptr: int, data_ptr: int, size: int) -> int:
return self._fs_write_common(uc, path_ptr, data_ptr, size, append_mode=False)
def _fs_append(self, uc: Uc, path_ptr: int, data_ptr: int, size: int) -> int:
return self._fs_write_common(uc, path_ptr, data_ptr, size, append_mode=True)
def _fs_remove(self, uc: Uc, path_ptr: int) -> int:
path = self._normalize_guest_path(self._read_guest_cstring(uc, path_ptr))
if not self._guest_path_is_under_temp(path) or path == "/temp":
return 0
host_path = self._guest_to_host(path, must_exist=True)
if host_path is None:
return 0
try:
if host_path.is_dir():
if any(host_path.iterdir()):
return 0
host_path.rmdir()
else:
host_path.unlink()
return 1
except Exception:
return 0
def _log_journal_read(self, uc: Uc, index_from_oldest: int, out_ptr: int, out_size: int) -> int:
if out_ptr == 0 or out_size == 0:
return 0
line = self.state.log_journal_read(int(index_from_oldest))
if line is None:
return 0
encoded = line.encode("utf-8", errors="replace")
max_payload = int(out_size) - 1
if max_payload < 0:
return 0
if len(encoded) > max_payload:
encoded = encoded[:max_payload]
return 1 if self._write_guest_bytes(uc, out_ptr, encoded + b"\x00") else 0
def _exec_path(self, uc: Uc, path_ptr: int) -> int:
2026-04-19 16:07:44 +08:00
return self._spawn_path_common(
uc,
path_ptr,
0,
0,
return_pid=False,
env_line_override=None,
stdin_fd=FD_INHERIT,
stdout_fd=FD_INHERIT,
stderr_fd=FD_INHERIT,
)
2026-04-16 22:29:08 +08:00
def _exec_pathv(self, uc: Uc, path_ptr: int, argv_ptr: int, env_ptr: int) -> int:
2026-04-19 16:07:44 +08:00
return self._spawn_path_common(
uc,
path_ptr,
argv_ptr,
env_ptr,
return_pid=False,
env_line_override=None,
stdin_fd=FD_INHERIT,
stdout_fd=FD_INHERIT,
stderr_fd=FD_INHERIT,
)
def _exec_pathv_io(self, uc: Uc, path_ptr: int, argv_ptr: int, req_ptr: int) -> int:
req_data: Optional[bytes]
env_ptr: int
stdin_fd: int
stdout_fd: int
stderr_fd: int
env_line: str
if req_ptr == 0:
return u64_neg1()
req_data = self._read_guest_bytes_exact(uc, req_ptr, 32)
if req_data is None or len(req_data) != 32:
return u64_neg1()
env_ptr, stdin_fd, stdout_fd, stderr_fd = struct.unpack("<QQQQ", req_data)
env_line = self._read_guest_cstring(uc, env_ptr, EXEC_ENV_LINE_MAX) if env_ptr != 0 else ""
return self._spawn_path_common(
uc,
path_ptr,
argv_ptr,
0,
return_pid=False,
env_line_override=env_line,
stdin_fd=stdin_fd,
stdout_fd=stdout_fd,
stderr_fd=stderr_fd,
)
2026-04-13 19:08:35 +08:00
def _spawn_path(self, uc: Uc, path_ptr: int) -> int:
2026-04-19 16:07:44 +08:00
return self._spawn_path_common(
uc,
path_ptr,
0,
0,
return_pid=True,
env_line_override=None,
stdin_fd=FD_INHERIT,
stdout_fd=FD_INHERIT,
stderr_fd=FD_INHERIT,
)
2026-04-13 19:08:35 +08:00
2026-04-16 22:29:08 +08:00
def _spawn_pathv(self, uc: Uc, path_ptr: int, argv_ptr: int, env_ptr: int) -> int:
2026-04-19 16:07:44 +08:00
return self._spawn_path_common(
uc,
path_ptr,
argv_ptr,
env_ptr,
return_pid=True,
env_line_override=None,
stdin_fd=FD_INHERIT,
stdout_fd=FD_INHERIT,
stderr_fd=FD_INHERIT,
)
2026-04-16 22:29:08 +08:00
def _spawn_path_common(
self,
uc: Uc,
path_ptr: int,
argv_ptr: int,
env_ptr: int,
*,
return_pid: bool,
2026-04-19 16:07:44 +08:00
env_line_override: Optional[str],
stdin_fd: int,
stdout_fd: int,
stderr_fd: int,
2026-04-16 22:29:08 +08:00
) -> int:
path = self._read_guest_cstring(uc, path_ptr, EXEC_PATH_MAX)
2026-04-13 18:32:22 +08:00
guest_path = self._normalize_guest_path(path)
2026-04-16 22:29:08 +08:00
argv_line = self._read_guest_cstring(uc, argv_ptr, EXEC_ARG_LINE_MAX) if argv_ptr != 0 else ""
2026-04-19 16:07:44 +08:00
env_line = (
env_line_override
if env_line_override is not None
else (self._read_guest_cstring(uc, env_ptr, EXEC_ENV_LINE_MAX) if env_ptr != 0 else "")
)
2026-04-13 18:32:22 +08:00
host_path = self._guest_to_host(guest_path, must_exist=True)
2026-04-19 16:07:44 +08:00
child_stdio = self._build_child_stdio_map(int(stdin_fd), int(stdout_fd), int(stderr_fd))
2026-04-13 18:32:22 +08:00
self.state.exec_requests = u64(self.state.exec_requests + 1)
self.state.user_exec_requested = 1
self.state.user_launch_tries = u64(self.state.user_launch_tries + 1)
2026-04-19 16:07:44 +08:00
if child_stdio is None:
self.state.user_launch_fail = u64(self.state.user_launch_fail + 1)
return u64_neg1()
2026-04-13 18:32:22 +08:00
if host_path is None or not host_path.is_file():
self.state.user_launch_fail = u64(self.state.user_launch_fail + 1)
return u64_neg1()
if self.depth >= self.max_exec_depth:
print(f"[WINE][WARN] exec depth exceeded: {guest_path}", file=sys.stderr)
self.state.user_launch_fail = u64(self.state.user_launch_fail + 1)
return u64_neg1()
2026-04-13 19:08:35 +08:00
parent_pid = self.state.get_current_pid()
child_pid = self.state.alloc_pid(parent_pid)
2026-04-16 22:29:08 +08:00
argv_items, env_items = self._build_child_exec_items(guest_path, argv_line, env_line)
self.state.set_proc_cmdline(child_pid, argv_items, env_items)
self.state.set_proc_fault(child_pid, 0, 0, 0, 0)
2026-04-13 19:08:35 +08:00
2026-04-13 18:32:22 +08:00
child = CLeonOSWineNative(
elf_path=host_path,
rootfs=self.rootfs,
guest_path_hint=guest_path,
state=self.state,
depth=self.depth + 1,
max_exec_depth=self.max_exec_depth,
no_kbd=True,
verbose=self.verbose,
top_level=False,
2026-04-13 19:08:35 +08:00
pid=child_pid,
ppid=parent_pid,
2026-04-16 22:29:08 +08:00
argv_items=argv_items,
env_items=env_items,
2026-04-19 16:07:44 +08:00
inherited_fds=child_stdio,
2026-04-21 22:22:31 +08:00
fb_window=self.fb_window,
fb_scale=self.fb_scale,
fb_max_fps=self.fb_max_fps,
fb_hold_ms=self.fb_hold_ms,
2026-04-13 18:32:22 +08:00
)
child_ret = child.run()
2026-04-13 19:08:35 +08:00
2026-04-13 18:32:22 +08:00
if child_ret is None:
self.state.user_launch_fail = u64(self.state.user_launch_fail + 1)
return u64_neg1()
self.state.exec_success = u64(self.state.exec_success + 1)
self.state.user_launch_ok = u64(self.state.user_launch_ok + 1)
2026-04-13 19:08:35 +08:00
2026-04-13 18:32:22 +08:00
if guest_path.lower().startswith("/system/"):
self.state.kelf_runs = u64(self.state.kelf_runs + 1)
2026-04-13 19:08:35 +08:00
if return_pid:
return child_pid
2026-04-16 22:29:08 +08:00
return u64(child_ret)
def _proc_argc(self) -> int:
return self.state.proc_argc(self.state.get_current_pid())
def _proc_argv(self, uc: Uc, index: int, out_ptr: int, out_size: int) -> int:
return self._copy_proc_item_to_guest(
uc,
self.state.proc_argv_item(self.state.get_current_pid(), int(index)),
out_ptr,
out_size,
)
def _proc_envc(self) -> int:
return self.state.proc_envc(self.state.get_current_pid())
def _proc_env(self, uc: Uc, index: int, out_ptr: int, out_size: int) -> int:
return self._copy_proc_item_to_guest(
uc,
self.state.proc_env_item(self.state.get_current_pid(), int(index)),
out_ptr,
out_size,
)
def _proc_last_signal(self) -> int:
return self.state.proc_signal(self.state.get_current_pid())
def _proc_fault_vector(self) -> int:
return self.state.proc_fault_vector_value(self.state.get_current_pid())
def _proc_fault_error(self) -> int:
return self.state.proc_fault_error_value(self.state.get_current_pid())
def _proc_fault_rip(self) -> int:
return self.state.proc_fault_rip_value(self.state.get_current_pid())
2026-04-13 18:32:22 +08:00
2026-04-19 16:07:44 +08:00
def _proc_count(self) -> int:
return self.state.proc_count()
def _proc_pid_at(self, uc: Uc, index: int, out_ptr: int) -> int:
if out_ptr == 0:
return 0
pid = self.state.proc_pid_at(int(index))
if pid is None:
return 0
return 1 if self._write_guest_bytes(uc, out_ptr, struct.pack("<Q", u64(pid))) else 0
def _proc_snapshot(self, uc: Uc, pid: int, out_ptr: int, out_size: int) -> int:
if out_ptr == 0 or out_size < (13 * 8 + PROC_PATH_MAX):
return 0
target = int(pid)
state_value = self.state.proc_state_value(target)
if state_value == 0:
return 0
path = self.state.proc_path_value(target)
encoded_path = path.encode("utf-8", errors="replace")
if len(encoded_path) >= PROC_PATH_MAX:
encoded_path = encoded_path[: PROC_PATH_MAX - 1]
path_buf = encoded_path + b"\x00" + (b"\x00" * (PROC_PATH_MAX - len(encoded_path) - 1))
blob = struct.pack(
"<13Q",
u64(target),
u64(self.state.proc_ppid(target)),
u64(state_value),
u64(self.state.proc_started_tick_value(target)),
u64(self.state.proc_exited_tick_value(target)),
u64(self.state.proc_exit_status_value(target)),
u64(self.state.proc_runtime_ticks(target)),
u64(self.state.proc_mem_bytes_value(target)),
u64(self.state.proc_tty_index_value(target)),
u64(self.state.proc_signal(target)),
u64(self.state.proc_fault_vector_value(target)),
u64(self.state.proc_fault_error_value(target)),
u64(self.state.proc_fault_rip_value(target)),
) + path_buf
return 1 if self._write_guest_bytes(uc, out_ptr, blob) else 0
def _proc_kill(self, uc: Uc, pid: int, signal: int) -> int:
target = int(pid)
if target <= 0:
return u64_neg1()
current_state = self.state.proc_state_value(target)
if current_state == 0:
return u64_neg1()
effective_signal = int(signal) & 0xFF
if effective_signal == 0:
effective_signal = SIGTERM
self.state.set_proc_fault(target, effective_signal, 0, 0, 0)
if current_state == PROC_STATE_EXITED:
return 1
if effective_signal == SIGCONT:
if current_state == PROC_STATE_STOPPED:
self.state.set_proc_pending(target)
return 1
if effective_signal == SIGSTOP and current_state in (PROC_STATE_PENDING, PROC_STATE_STOPPED):
self.state.set_proc_stopped(target)
return 1
status = self._encode_signal_status(effective_signal, 0, 0)
if target == self.state.get_current_pid():
self._exit_requested = True
self._exit_status = u64(status)
if effective_signal == SIGSTOP:
self.state.set_proc_stopped(target)
uc.emu_stop()
return 1
if effective_signal == SIGSTOP:
self.state.set_proc_stopped(target)
return 1
self.state.mark_exited(target, status)
return 1
2026-04-13 19:08:35 +08:00
def _wait_pid(self, uc: Uc, pid: int, out_ptr: int) -> int:
wait_ret, status = self.state.wait_pid(int(pid))
if wait_ret == 1 and out_ptr != 0:
self._write_guest_bytes(uc, out_ptr, struct.pack("<Q", u64(status)))
return int(wait_ret)
def _request_exit(self, uc: Uc, status: int) -> int:
self._exit_requested = True
self._exit_status = u64(status)
uc.emu_stop()
return 1
def _sleep_ticks(self, ticks: int) -> int:
ticks = int(u64(ticks))
if ticks == 0:
return 0
start = self.state.timer_ticks()
while (self.state.timer_ticks() - start) < ticks:
time.sleep(0.001)
return self.state.timer_ticks() - start
def _yield_once(self) -> int:
time.sleep(0)
return self.state.timer_ticks()
2026-04-19 16:07:44 +08:00
def _fd_open(self, uc: Uc, path_ptr: int, flags: int, mode: int) -> int:
_ = mode
guest_path = self._normalize_guest_path(self._read_guest_cstring(uc, path_ptr, EXEC_PATH_MAX))
open_flags = int(u64(flags))
lower_path = guest_path.lower()
fd_slot: int
entry: FDEntry
if not guest_path.startswith("/"):
return u64_neg1()
if not self._fd_access_mode_valid(open_flags):
return u64_neg1()
if ((open_flags & O_TRUNC) != 0 or (open_flags & O_APPEND) != 0) and not self._fd_can_write(open_flags):
return u64_neg1()
fd_slot = self._fd_find_free()
if fd_slot < 0:
return u64_neg1()
if lower_path == "/dev/tty":
entry = FDEntry(kind="tty", flags=open_flags, offset=0, tty_index=self._tty_index)
self._fds[fd_slot] = entry
return fd_slot
if lower_path == "/dev/null":
entry = FDEntry(kind="dev_null", flags=open_flags, offset=0, tty_index=self._tty_index)
self._fds[fd_slot] = entry
return fd_slot
if lower_path == "/dev/zero":
entry = FDEntry(kind="dev_zero", flags=open_flags, offset=0, tty_index=self._tty_index)
self._fds[fd_slot] = entry
return fd_slot
if lower_path == "/dev/random":
entry = FDEntry(kind="dev_random", flags=open_flags, offset=0, tty_index=self._tty_index)
self._fds[fd_slot] = entry
return fd_slot
host_path = self._guest_to_host(guest_path, must_exist=False)
if host_path is None:
return u64_neg1()
try:
if not host_path.exists():
if (open_flags & O_CREAT) == 0 or not self._fd_can_write(open_flags):
return u64_neg1()
host_path.parent.mkdir(parents=True, exist_ok=True)
host_path.write_bytes(b"")
if host_path.is_dir():
return u64_neg1()
if (open_flags & O_TRUNC) != 0:
host_path.write_bytes(b"")
offset = int(host_path.stat().st_size) if (open_flags & O_APPEND) != 0 else 0
except Exception:
return u64_neg1()
entry = FDEntry(kind="file", flags=open_flags, offset=offset, path=str(host_path), tty_index=self._tty_index)
self._fds[fd_slot] = entry
return fd_slot
def _fd_read(self, uc: Uc, fd: int, out_ptr: int, size: int) -> int:
req = int(u64(size))
entry = self._fd_lookup(int(fd))
data: bytes
if req == 0:
return 0
if out_ptr == 0:
return u64_neg1()
if entry is None or not self._fd_can_read(entry.flags):
return u64_neg1()
req = min(req, MAX_IO_READ)
if entry.kind == "tty":
out = bytearray()
while len(out) < req:
key = self.state.pop_key()
if key is None:
break
out.append(key & 0xFF)
data = bytes(out)
elif entry.kind == "dev_null":
return 0
elif entry.kind == "dev_zero":
data = b"\x00" * req
elif entry.kind == "dev_random":
data = os.urandom(req)
elif entry.kind == "file":
try:
with open(entry.path, "rb") as fh:
fh.seek(entry.offset)
data = fh.read(req)
except Exception:
return u64_neg1()
else:
return u64_neg1()
if len(data) == 0:
return 0
if not self._write_guest_bytes(uc, int(out_ptr), data):
return u64_neg1()
entry.offset += len(data)
return len(data)
def _fd_write(self, uc: Uc, fd: int, buf_ptr: int, size: int) -> int:
req = int(u64(size))
entry = self._fd_lookup(int(fd))
data: Optional[bytes]
write_pos: int
if req == 0:
return 0
if buf_ptr == 0:
return u64_neg1()
if entry is None or not self._fd_can_write(entry.flags):
return u64_neg1()
req = min(req, MAX_IO_READ)
data = self._read_guest_bytes_exact(uc, int(buf_ptr), req)
if data is None:
return u64_neg1()
if entry.kind == "tty":
self._host_write_bytes(data)
entry.offset += len(data)
return len(data)
if entry.kind in ("dev_null", "dev_zero", "dev_random"):
entry.offset += len(data)
return len(data)
if entry.kind != "file":
return u64_neg1()
try:
host_path = Path(entry.path)
if not host_path.exists():
if (entry.flags & O_CREAT) == 0 or not self._fd_can_write(entry.flags):
return u64_neg1()
host_path.parent.mkdir(parents=True, exist_ok=True)
host_path.write_bytes(b"")
with open(host_path, "r+b") as fh:
write_pos = int(entry.offset)
fh.seek(write_pos)
fh.write(data)
except Exception:
return u64_neg1()
entry.offset += len(data)
return len(data)
def _fd_close(self, fd: int) -> int:
key = int(fd)
if key not in self._fds:
return u64_neg1()
del self._fds[key]
return 0
def _fd_dup(self, fd: int) -> int:
src = self._fd_lookup(int(fd))
if src is None:
return u64_neg1()
slot = self._fd_find_free()
if slot < 0:
return u64_neg1()
self._fds[slot] = self._clone_fd_entry(src)
return slot
2026-04-21 22:22:31 +08:00
def _dl_alloc_handle(self) -> int:
handle = int(u64(self._dl_next_handle))
if handle == 0:
handle = 1
while handle in self._dl_images or handle == 0:
handle = int(u64(handle + 1))
if handle == 0:
handle = 1
self._dl_next_handle = int(u64(handle + 1))
if self._dl_next_handle == 0:
self._dl_next_handle = 1
return handle
def _dl_pick_base(self, min_vaddr: int, max_vaddr: int) -> Tuple[int, int, int]:
old_base = page_floor(min_vaddr)
span = page_ceil(max_vaddr - old_base)
candidate = page_ceil(self._dl_next_base)
retries = 0
if span <= 0:
return 0, 0, 0
while retries < 1024:
start = candidate
end = start + span
if not self._range_overlaps_mapped(start, end):
self._dl_next_base = end + DL_BASE_GAP
return start, end, start - old_base
candidate = end + DL_BASE_GAP
retries += 1
return 0, 0, 0
@staticmethod
def _dl_rebase_non_exec_segment(data: bytes, old_base: int, old_end: int, delta: int) -> bytes:
if not data or delta == 0:
return data
patched = bytearray(data)
limit = len(patched) - (len(patched) % 8)
for off in range(0, limit, 8):
value = struct.unpack_from("<Q", patched, off)[0]
if old_base <= value < old_end:
struct.pack_into("<Q", patched, off, u64(value + delta))
return bytes(patched)
@staticmethod
def _read_elf_cstring(blob: bytes, start: int, end: int) -> str:
if start < 0 or end <= start or start >= len(blob):
return ""
limit = min(end, len(blob))
cur = start
while cur < limit and blob[cur] != 0:
cur += 1
if cur <= start:
return ""
return blob[start:cur].decode("utf-8", errors="ignore")
@classmethod
def _dl_extract_symbols(cls, blob: bytes, load_bias: int) -> Dict[str, int]:
symbols: Dict[str, int] = {}
sections: List[Tuple[int, int, int, int, int, int, int, int, int, int]] = []
if len(blob) < 0x40:
return symbols
try:
shoff = struct.unpack_from("<Q", blob, 0x28)[0]
shentsize = struct.unpack_from("<H", blob, 0x3A)[0]
shnum = struct.unpack_from("<H", blob, 0x3C)[0]
except struct.error:
return symbols
if shoff == 0 or shentsize < 64 or shnum == 0:
return symbols
if shoff + (shentsize * shnum) > len(blob):
return symbols
for idx in range(shnum):
off = shoff + (idx * shentsize)
try:
sh = struct.unpack_from("<IIQQQQIIQQ", blob, off)
except struct.error:
return symbols
sections.append(sh)
for sh in sections:
sh_type = int(sh[1])
sh_offset = int(sh[4])
sh_size = int(sh[5])
sh_link = int(sh[6])
sh_entsize = int(sh[9])
if sh_type not in (2, 11):
continue
if sh_link < 0 or sh_link >= len(sections):
continue
if sh_entsize < 24:
sh_entsize = 24
if sh_offset < 0 or sh_size <= 0 or sh_offset >= len(blob):
continue
sym_end = min(len(blob), sh_offset + sh_size)
if sym_end <= sh_offset:
continue
strtab = sections[sh_link]
str_off = int(strtab[4])
str_size = int(strtab[5])
if str_size <= 0 or str_off < 0 or str_off >= len(blob):
continue
str_end = min(len(blob), str_off + str_size)
count = (sym_end - sh_offset) // sh_entsize
for i in range(count):
ent_off = sh_offset + (i * sh_entsize)
if ent_off + 24 > len(blob):
break
try:
st_name, _st_info, _st_other, st_shndx, st_value, _st_size = struct.unpack_from(
"<IBBHQQ", blob, ent_off
)
except struct.error:
break
if st_name == 0 or st_shndx == 0 or st_value == 0:
continue
name = cls._read_elf_cstring(blob, str_off + st_name, str_end)
if not name:
continue
addr = int(u64(st_value + load_bias))
if addr == 0:
continue
if name not in symbols:
symbols[name] = addr
return symbols
def _dl_open(self, uc: Uc, path_ptr: int) -> int:
guest_path = self._normalize_guest_path(self._read_guest_cstring(uc, path_ptr, DL_MAX_NAME))
cached_handle = self._dl_path_to_handle.get(guest_path)
host_path: Optional[Path]
file_blob: bytes
image: ELFImage
e_type: int
min_vaddr: int
max_vaddr: int
map_start: int
map_end: int
load_bias: int
handle: int
owner_pid: int
if not guest_path.startswith("/") or guest_path == "/":
return u64_neg1()
if len(guest_path.encode("utf-8", errors="replace")) >= DL_MAX_NAME:
return u64_neg1()
if cached_handle is not None:
cached = self._dl_images.get(cached_handle)
if cached is not None:
cached.ref_count += 1
return cached.handle
host_path = self._guest_to_host(guest_path, must_exist=True)
if host_path is None or not host_path.is_file():
return u64_neg1()
try:
file_blob = host_path.read_bytes()
image = self._parse_elf_image_from_blob(file_blob, require_entry=False)
e_type = struct.unpack_from("<H", file_blob, 0x10)[0]
except Exception:
return u64_neg1()
min_vaddr = min(seg.vaddr for seg in image.segments)
max_vaddr = max(seg.vaddr + seg.memsz for seg in image.segments)
if max_vaddr <= min_vaddr:
return u64_neg1()
map_start, map_end, load_bias = self._dl_pick_base(min_vaddr, max_vaddr)
if map_start == 0 or map_end <= map_start:
return u64_neg1()
try:
for seg in image.segments:
seg_start = page_floor(seg.vaddr + load_bias)
seg_end = page_ceil(seg.vaddr + load_bias + seg.memsz)
self._map_region(uc, seg_start, seg_end - seg_start, UC_PROT_ALL)
for seg in image.segments:
payload = seg.data
if e_type == 2 and (seg.flags & 0x1) == 0 and load_bias != 0 and payload:
payload = self._dl_rebase_non_exec_segment(payload, min_vaddr, max_vaddr, load_bias)
if payload:
self._mem_write(uc, seg.vaddr + load_bias, payload)
for seg in image.segments:
seg_start = page_floor(seg.vaddr + load_bias)
seg_end = page_ceil(seg.vaddr + load_bias + seg.memsz)
perms = 0
if seg.flags & 0x4:
perms |= UC_PROT_READ
if seg.flags & 0x2:
perms |= UC_PROT_WRITE
if seg.flags & 0x1:
perms |= UC_PROT_EXEC
if perms == 0:
perms = UC_PROT_READ
try:
uc.mem_protect(seg_start, seg_end - seg_start, perms)
except Exception:
pass
except Exception:
return u64_neg1()
handle = self._dl_alloc_handle()
owner_pid = self.state.get_current_pid()
self._dl_images[handle] = DLImage(
handle=handle,
guest_path=guest_path,
host_path=str(host_path),
owner_pid=owner_pid,
ref_count=1,
map_start=map_start,
map_end=map_end,
load_bias=load_bias,
symbols=self._dl_extract_symbols(file_blob, load_bias),
)
self._dl_path_to_handle[guest_path] = handle
return handle
def _dl_close(self, handle: int) -> int:
key = int(handle)
image = self._dl_images.get(key)
if key == 0 or image is None:
return u64_neg1()
if image.ref_count > 1:
image.ref_count -= 1
return 0
del self._dl_images[key]
if self._dl_path_to_handle.get(image.guest_path) == key:
del self._dl_path_to_handle[image.guest_path]
return 0
def _dl_sym(self, uc: Uc, handle: int, symbol_ptr: int) -> int:
symbol = self._read_guest_cstring(uc, symbol_ptr, DL_MAX_SYMBOL)
image = self._dl_images.get(int(handle))
addr: Optional[int]
if image is None or not symbol:
return 0
addr = image.symbols.get(symbol)
if addr is None and not symbol.startswith("_"):
addr = image.symbols.get(f"_{symbol}")
return int(u64(addr)) if addr is not None else 0
def _fb_info(self, uc: Uc, out_ptr: int) -> int:
if out_ptr == 0:
return 0
self._ensure_fb_window()
payload = struct.pack(
"<QQQQ",
u64(self._fb_width),
u64(self._fb_height),
u64(self._fb_pitch),
u64(self._fb_bpp),
)
ok = 1 if self._write_guest_bytes(uc, out_ptr, payload) else 0
if ok == 1:
self._fb_present(force=True)
return ok
def _fb_clear(self, rgb: int) -> int:
if len(self._fb_pixels) == 0:
return 0
pixel = struct.pack("<I", int(u64(rgb) & 0xFFFFFFFF))
self._fb_pixels[:] = pixel * (len(self._fb_pixels) // 4)
self._fb_mark_dirty()
return 1
def _fb_blit(self, uc: Uc, req_ptr: int) -> int:
req_blob = self._read_guest_bytes_exact(uc, int(req_ptr), 56)
pixels_ptr: int
src_width: int
src_height: int
src_pitch_bytes: int
dst_x: int
dst_y: int
scale: int
total_src_bytes: int
src_blob: Optional[bytes]
max_src_w: int
max_src_h: int
copy_w: int
copy_h: int
if req_ptr == 0 or req_blob is None or len(req_blob) != 56:
return 0
pixels_ptr, src_width, src_height, src_pitch_bytes, dst_x, dst_y, scale = struct.unpack("<QQQQQQQ", req_blob)
if pixels_ptr == 0 or src_width == 0 or src_height == 0 or scale == 0:
return 0
if src_width > 4096 or src_height > 4096 or scale > 8:
return 0
if src_pitch_bytes == 0:
src_pitch_bytes = src_width * 4
if src_pitch_bytes < (src_width * 4):
return 0
if src_height > (u64_neg1() // src_pitch_bytes):
return 0
if dst_x >= self._fb_width or dst_y >= self._fb_height:
return 0
total_src_bytes = src_pitch_bytes * src_height
if total_src_bytes == 0 or total_src_bytes > FB_MAX_UPLOAD_BYTES:
return 0
src_blob = self._read_guest_bytes_exact(uc, pixels_ptr, total_src_bytes)
if src_blob is None:
return 0
max_src_w = (self._fb_width - dst_x + scale - 1) // scale
max_src_h = (self._fb_height - dst_y + scale - 1) // scale
copy_w = min(src_width, max_src_w)
copy_h = min(src_height, max_src_h)
if copy_w <= 0 or copy_h <= 0:
return 0
if scale == 1:
row_bytes = copy_w * 4
for y in range(copy_h):
src_off = y * src_pitch_bytes
dst_off = ((dst_y + y) * self._fb_width + dst_x) * 4
self._fb_pixels[dst_off : dst_off + row_bytes] = src_blob[src_off : src_off + row_bytes]
else:
draw_row_pixels = min(self._fb_width - dst_x, copy_w * scale)
draw_row_bytes = draw_row_pixels * 4
for y in range(copy_h):
src_row_off = y * src_pitch_bytes
src_row = memoryview(src_blob)[src_row_off : src_row_off + (copy_w * 4)]
expanded = bytearray(copy_w * scale * 4)
write_off = 0
for x in range(copy_w):
pixel = bytes(src_row[x * 4 : (x + 1) * 4])
expanded[write_off : write_off + (scale * 4)] = pixel * scale
write_off += scale * 4
draw_y = dst_y + (y * scale)
repeat_h = min(scale, self._fb_height - draw_y)
row_data = expanded[:draw_row_bytes]
for sy in range(repeat_h):
dst_off = ((draw_y + sy) * self._fb_width + dst_x) * 4
self._fb_pixels[dst_off : dst_off + draw_row_bytes] = row_data
self._fb_mark_dirty()
return 1
2026-04-16 22:29:08 +08:00
@staticmethod
def _truncate_item_text(text: str, max_bytes: int = EXEC_ITEM_MAX) -> str:
if max_bytes <= 1:
return ""
encoded = (text or "").encode("utf-8", errors="replace")
if len(encoded) >= max_bytes:
encoded = encoded[: max_bytes - 1]
return encoded.decode("utf-8", errors="ignore")
@staticmethod
def _parse_whitespace_items(line: str, max_count: int) -> List[str]:
out: List[str] = []
i = 0
text = line or ""
length = len(text)
while i < length:
while i < length and text[i] in (" ", "\t", "\r", "\n"):
i += 1
if i >= length:
break
start = i
while i < length and text[i] not in (" ", "\t", "\r", "\n"):
i += 1
out.append(text[start:i])
if len(out) >= max_count:
break
return out
@staticmethod
def _parse_env_items(line: str, max_count: int) -> List[str]:
out: List[str] = []
i = 0
text = line or ""
length = len(text)
while i < length:
while i < length and text[i] in (" ", "\t", ";", "\r", "\n"):
i += 1
if i >= length:
break
start = i
while i < length and text[i] not in (";", "\r", "\n"):
i += 1
value = text[start:i].rstrip(" \t")
if value:
out.append(value)
if len(out) >= max_count:
break
return out
@classmethod
def _prepare_exec_items(cls, path: str, argv_items: List[str], env_items: List[str]) -> Tuple[List[str], List[str]]:
normalized_path = cls._normalize_guest_path(path)
prepared_argv: List[str] = []
for item in argv_items[:EXEC_MAX_ARGS]:
value = cls._truncate_item_text(item)
if value:
prepared_argv.append(value)
if not prepared_argv:
prepared_argv = [cls._truncate_item_text(normalized_path)]
elif prepared_argv[0] == "":
prepared_argv[0] = cls._truncate_item_text(normalized_path)
prepared_env: List[str] = []
for item in env_items[:EXEC_MAX_ENVS]:
value = cls._truncate_item_text(item)
if value:
prepared_env.append(value)
return prepared_argv[:EXEC_MAX_ARGS], prepared_env[:EXEC_MAX_ENVS]
@classmethod
def _build_child_exec_items(cls, guest_path: str, argv_line: str, env_line: str) -> Tuple[List[str], List[str]]:
argv_items = [guest_path]
argv_items.extend(cls._parse_whitespace_items(argv_line or "", EXEC_MAX_ARGS - 1))
env_items = cls._parse_env_items(env_line or "", EXEC_MAX_ENVS)
return cls._prepare_exec_items(guest_path, argv_items, env_items)
def _copy_proc_item_to_guest(self, uc: Uc, item: Optional[str], out_ptr: int, out_size: int) -> int:
if out_ptr == 0 or out_size == 0 or item is None:
return 0
max_out = int(min(int(out_size), EXEC_ITEM_MAX))
if max_out <= 0:
return 0
encoded = self._truncate_item_text(item, max_out + 1).encode("utf-8", errors="replace")
if len(encoded) >= max_out:
encoded = encoded[: max_out - 1]
return 1 if self._write_guest_bytes(uc, out_ptr, encoded + b"\x00") else 0
@staticmethod
def _signal_from_vector(vector: int) -> int:
if vector in (0, 16, 19):
return 8
if vector == 6:
return 4
if vector == 3:
return 5
if vector in (10, 11, 12, 13, 14, 17):
return 11
return 6
@staticmethod
def _encode_signal_status(signal: int, vector: int, error_code: int) -> int:
return u64(
EXEC_STATUS_SIGNAL_FLAG
| (int(signal) & 0xFF)
| ((int(vector) & 0xFF) << 8)
| ((int(error_code) & 0xFFFF) << 16)
)
def _status_from_uc_error(self, uc: Uc, exc: UcError) -> int:
errno = int(getattr(exc, "errno", 0))
vector = 13
error_code = 0
if errno in (UC_ERR_READ_UNMAPPED, UC_ERR_WRITE_UNMAPPED, UC_ERR_FETCH_UNMAPPED):
vector = 14
if errno == UC_ERR_WRITE_UNMAPPED:
error_code = 0x2
elif errno == UC_ERR_FETCH_UNMAPPED:
error_code = 0x10
elif errno in (UC_ERR_READ_PROT, UC_ERR_WRITE_PROT, UC_ERR_FETCH_PROT):
vector = 14
error_code = 0x1
if errno == UC_ERR_WRITE_PROT:
error_code |= 0x2
elif errno == UC_ERR_FETCH_PROT:
error_code |= 0x10
elif errno == UC_ERR_INSN_INVALID:
vector = 6
error_code = 0
rip = 0
try:
rip = self._reg_read(uc, UC_X86_REG_RIP)
except Exception:
rip = 0
signal = self._signal_from_vector(vector)
self.state.set_proc_fault(self.pid, signal, vector, error_code, rip)
return self._encode_signal_status(signal, vector, error_code)
2026-04-13 18:32:22 +08:00
def _guest_to_host(self, guest_path: str, *, must_exist: bool) -> Optional[Path]:
norm = self._normalize_guest_path(guest_path)
if norm == "/":
return self.rootfs if (not must_exist or self.rootfs.exists()) else None
current = self.rootfs
for part in [p for p in norm.split("/") if p]:
candidate = current / part
if candidate.exists():
current = candidate
continue
if current.exists() and current.is_dir():
match = self._find_case_insensitive(current, part)
if match is not None:
current = match
continue
current = candidate
if must_exist and not current.exists():
return None
return current
@staticmethod
def _find_case_insensitive(parent: Path, name: str) -> Optional[Path]:
target = name.lower()
try:
for entry in parent.iterdir():
if entry.name.lower() == target:
return entry
except Exception:
return None
return None
@staticmethod
def _normalize_guest_path(path: str) -> str:
p = (path or "").replace("\\", "/").strip()
if not p:
return "/"
if not p.startswith("/"):
p = "/" + p
parts = []
for token in p.split("/"):
if token in ("", "."):
continue
if token == "..":
if parts:
parts.pop()
continue
parts.append(token)
return "/" + "/".join(parts)
@staticmethod
def _list_children(dir_path: Path) -> List[str]:
try:
names = [entry.name for entry in dir_path.iterdir() if not entry.name.startswith(".")]
except Exception:
return []
names.sort(key=lambda x: x.lower())
return names
def resolve_rootfs(path_arg: Optional[str]) -> Path:
if path_arg:
root = Path(path_arg).expanduser().resolve()
if not root.exists() or not root.is_dir():
raise FileNotFoundError(f"rootfs not found: {root}")
return root
candidates = [
Path("build/x86_64/ramdisk_root"),
Path("ramdisk"),
]
for candidate in candidates:
if candidate.exists() and candidate.is_dir():
return candidate.resolve()
raise FileNotFoundError("rootfs not found; pass --rootfs")
def _guest_to_host_for_resolve(rootfs: Path, guest_path: str) -> Optional[Path]:
norm = CLeonOSWineNative._normalize_guest_path(guest_path)
if norm == "/":
return rootfs
current = rootfs
for part in [p for p in norm.split("/") if p]:
candidate = current / part
if candidate.exists():
current = candidate
continue
if current.exists() and current.is_dir():
match = None
for entry in current.iterdir():
if entry.name.lower() == part.lower():
match = entry
break
if match is not None:
current = match
continue
current = candidate
return current if current.exists() else None
def resolve_elf_target(elf_arg: str, rootfs: Path) -> Tuple[Path, str]:
host_candidate = Path(elf_arg).expanduser()
if host_candidate.exists():
host_path = host_candidate.resolve()
try:
rel = host_path.relative_to(rootfs)
guest_path = "/" + rel.as_posix()
except ValueError:
guest_path = "/" + host_path.name
return host_path, guest_path
guest_path = CLeonOSWineNative._normalize_guest_path(elf_arg)
host_path = _guest_to_host_for_resolve(rootfs, guest_path)
if host_path is None:
raise FileNotFoundError(f"ELF not found as host path or guest path: {elf_arg}")
2026-04-14 19:13:55 +08:00
return host_path.resolve(), guest_path