# SPDX-License-Identifier: MIT # # X server lifecycle manager # # Handles starting and stopping Xvfb, Xwayland, and Xorg servers, # optionally under valgrind or with AddressSanitizer (ASAN) support, # with automatic display number allocation via the -displayfd mechanism. from typing import Iterator import os import select import shutil import subprocess import tempfile import time import warnings from pathlib import Path from asan import AsanError from valgrind import ValgrindError class XServerProcess: """ Manages an X server subprocess for testing. Supports Xvfb, Xwayland, and Xorg. Uses the -displayfd pipe mechanism for automatic display number allocation. Optionally wraps the server in valgrind for memory error detection, or detects AddressSanitizer (ASAN) errors when running an ASAN-instrumented binary. """ VALGRIND_ERROR_EXIT = 99 def __init__( self, server_type="xvfb", valgrind=False, valgrind_suppressions=None, asan=False, server_path=None, extra_args=None, log_file=None, ): self.server_type = server_type self.asan = asan if asan and valgrind: warnings.warn( "ASAN and valgrind are incompatible; disabling valgrind " "in favour of ASAN.", stacklevel=2, ) valgrind = False self.valgrind = valgrind if valgrind_suppressions is None: default_supp = Path(__file__).resolve().parent / "valgrind.suppressions" self.valgrind_suppressions = ( default_supp if default_supp.is_file() else None ) else: self.valgrind_suppressions = valgrind_suppressions self.server_path = server_path or self._find_server() self.extra_args = extra_args or [] self.log_file = log_file self._process = None self._display_num = None self._valgrind_xml_file = None self._asan_log_path = None self._stderr_file = None self._weston_process = None def _find_server(self): """Discover the server binary from environment or build directory.""" env_map = { "xvfb": "XVFB_PATH", "xwayland": "XWAYLAND_PATH", "xorg": "XORG_PATH", } env_var = env_map.get(self.server_type) if env_var and os.environ.get(env_var): path = Path(os.environ[env_var]) if path.is_file() and os.access(path, os.X_OK): return path def find_meson_builddir(source_root: Path) -> Iterator[Path]: for d in source_root.iterdir(): if d.is_dir() and (d / "meson-private").exists(): yield d # Try XSERVER_BUILDDIR env var or fall back to the first # build directory relative to this file's location # test/pyxtest/xserver.py -> ../../build/hw/... builddir = os.environ.get("XSERVER_BUILDDIR") if builddir is None: try: builddir = next( find_meson_builddir(Path(__file__).resolve().parent.parent.parent) ) except StopIteration: pass if builddir: build_paths = { "xvfb": Path(builddir, "hw", "vfb", "Xvfb"), "xwayland": Path(builddir, "hw", "xwayland", "Xwayland"), "xorg": Path(builddir, "hw", "xfree86", "Xorg"), } path = build_paths.get(self.server_type) if path and path.is_file() and os.access(path, os.X_OK): return path # Fall back to system PATH binary_names = { "xvfb": "Xvfb", "xwayland": "Xwayland", "xorg": "Xorg", } name = binary_names.get(self.server_type) server_in_path = shutil.which(name) if server_in_path: warnings.warn( f"Using system {self.server_type} server from PATH: {server_in_path}. " f"Set {env_var} or XSERVER_BUILDDIR to use a specific build.", stacklevel=2, ) return server_in_path raise FileNotFoundError( f"Failed to find {self.server_type} server binary. " f"Set {env_var} environment variable or build the server first." ) @property def display_num(self): return self._display_num @property def display(self): if self._display_num is not None: return f":{self._display_num}" return None def start(self, timeout=10): """Start the X server and wait for it to be ready.""" read_fd, write_fd = os.pipe() cmd = self._build_command(write_fd) self._stderr_file = tempfile.NamedTemporaryFile( prefix=f"xserver-{self.server_type}-stderr-", suffix=".log", delete=False, mode="w", ) if self.server_type == "xwayland": self._start_wayland_compositor() env = os.environ.copy() if self._weston_process: env.setdefault("WAYLAND_DISPLAY", "wayland-security-test") if self.asan: self._setup_asan_env(env) try: self._process = subprocess.Popen( cmd, stderr=self._stderr_file, pass_fds=(write_fd,), env=env, ) except Exception: os.close(write_fd) os.close(read_fd) raise os.close(write_fd) try: display_bytes = b"" deadline = time.monotonic() + timeout while time.monotonic() < deadline: if self._process.poll() is not None: display_bytes = None break ready, _, _ = select.select([read_fd], [], [], 1.0) if ready: chunk = os.read(read_fd, 64) if not chunk: break display_bytes += chunk if b"\n" in display_bytes: break if not display_bytes: if self._process.poll() is not None: stderr_content = self._read_stderr() raise RuntimeError( f"X server exited with code {self._process.returncode} " f"before sending display number (waited {timeout}s).\n" f"Command: {' '.join(str(s) for s in cmd)}\n" f"Stderr:\n{stderr_content}" ) self.stop() raise RuntimeError( f"Timed out waiting for X server display number (waited {timeout}s)" ) self._display_num = int(display_bytes.strip()) finally: os.close(read_fd) return self._display_num def _build_command(self, displayfd): """Build the full command line including optional valgrind wrapper.""" server_args = [self.server_path] if self.server_type == "xvfb": server_args.extend( [ "-screen", "scrn", "1280x1024x24", ] ) elif self.server_type == "xwayland": server_args.extend(["-nokeymap"]) elif self.server_type == "xorg": # -logfile is only permitted if we're running as root if self.log_file and os.geteuid() == 0: server_args.extend(["-logfile", str(self.log_file)]) server_args.extend(["-noreset", "+byteswappedclients"]) # Auto-detect xkb directory if the compiled-in default doesn't exist xkb_dir = os.environ.get("XKB_CONFIG_ROOT") if not xkb_dir: for candidate in ["/usr/share/X11/xkb", "/usr/local/share/X11/xkb"]: if os.path.isdir(candidate): xkb_dir = candidate break if xkb_dir: server_args.extend(["-xkbdir", xkb_dir]) server_args.extend(["-displayfd", str(displayfd)]) server_args.extend(self.extra_args) if not self.valgrind: return server_args self._valgrind_xml_file = tempfile.NamedTemporaryFile( prefix=f"xserver-{self.server_type}-valgrind-", suffix=".xml", delete=False ) self._valgrind_xml_file.close() valgrind_cmd = [ "valgrind", "--tool=memcheck", "--leak-check=full", "--track-origins=yes", "--show-reachable=no", "--gen-suppressions=all", f"--error-exitcode={self.VALGRIND_ERROR_EXIT}", "--xml=yes", f"--xml-file={self._valgrind_xml_file.name}", ] if self.valgrind_suppressions: valgrind_cmd.append(f"--suppressions={self.valgrind_suppressions}") return valgrind_cmd + server_args def _setup_asan_env(self, env): """Configure ASAN_OPTIONS for the server subprocess. Sets up a log file for ASAN output and configures ASAN to not detect leaks (too noisy for these tests). Any existing ASAN_OPTIONS from the environment are preserved. """ asan_log_file = tempfile.NamedTemporaryFile( prefix=f"xserver-{self.server_type}-asan-", suffix=".log", delete=False, ) self._asan_log_path = Path(asan_log_file.name) asan_log_file.close() asan_opts = { "log_path": str(self._asan_log_path), "detect_leaks": "0", } # Merge with any existing ASAN_OPTIONS from the environment existing = env.get("ASAN_OPTIONS", "") for part in existing.split(":"): part = part.strip() if "=" in part: key, val = part.split("=", 1) # Don't override our log_path if key not in asan_opts: asan_opts[key] = val env["ASAN_OPTIONS"] = ":".join(f"{k}={v}" for k, v in asan_opts.items()) def _start_wayland_compositor(self): """Start weston as a headless compositor for Xwayland testing.""" if not shutil.which("weston"): raise FileNotFoundError( "weston is required for Xwayland testing but was not found" ) self._weston_process = subprocess.Popen( ["weston", "--no-config", "--backend=headless-backend.so"], stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL, ) time.sleep(1) if self._weston_process.poll() is not None: raise RuntimeError( f"weston exited with code {self._weston_process.returncode}" ) @property def is_alive(self) -> bool: """Check if the server process is still running.""" if self._process is None: return False return self._process.poll() is None @property def crash_signal(self) -> int | None: """If the server crashed, return the signal number. Otherwise None.""" if self._process is None: return None ret = self._process.poll() if ret is not None and ret < 0: return -ret return None def stop(self): """Stop the server and return any valgrind/ASAN errors.""" errors = [] if self._process and self._process.poll() is None: self._process.terminate() try: self._process.wait(timeout=10) except subprocess.TimeoutExpired: self._process.kill() self._process.wait(timeout=5) if self._weston_process and self._weston_process.poll() is None: self._weston_process.terminate() try: self._weston_process.wait(timeout=5) except subprocess.TimeoutExpired: self._weston_process.kill() self._weston_process.wait() if self.valgrind and self._valgrind_xml_file: errors = ValgrindError.from_xml(Path(self._valgrind_xml_file.name)) if self._stderr_file: self._stderr_file.close() return errors def _read_stderr(self): """Read the captured stderr content.""" if self._stderr_file: self._stderr_file.flush() try: return Path(self._stderr_file.name).read_text() except OSError: pass return "" def check_valgrind_errors(self): """Parse and assert no valgrind errors occurred.""" if self._valgrind_xml_file is None: return [] errors = ValgrindError.from_xml(Path(self._valgrind_xml_file.name)) if errors: msg = f"Valgrind found {len(errors)} error(s):\n\n" msg += "\n\n".join(str(e) for e in errors) raise AssertionError(msg) return errors def get_asan_errors(self) -> list[AsanError]: """Parse ASAN errors from log file and/or stderr.""" errors: list[AsanError] = [] # Check the ASAN log file(s) first (ASAN appends . to log_path) if self._asan_log_path: errors.extend(AsanError.from_log(self._asan_log_path)) # Also check stderr in case ASAN wrote there stderr_text = self._read_stderr() if stderr_text and "AddressSanitizer" in stderr_text: stderr_errors = AsanError.from_text(stderr_text) # Avoid duplicates: only add stderr errors if log file had none if not errors: errors.extend(stderr_errors) return errors def __enter__(self): self.start() return self def __exit__(self, exc_type, exc_val, exc_tb): self.stop() return False def __repr__(self): state = "running" if self.is_alive else "stopped" display = self.display or "N/A" flags = "" if self.valgrind: flags += " +valgrind" if self.asan: flags += " +asan" return f""