Phase 4.4 : binaire prêt pour test sur image bootée

Modifications du test display-backend pour le runtime VT-handler réel :
- DebugSink wrapper qui mirroir stdout vers /scheme/debug (serial host stdio)
  → la sortie est visible côté host même si on tourne sur un VT non actif
- Lecture env var VT (mise par le init Redox)
- Appel `inputd -A <vt>` après open_display, comme Orbital le fait
  (cf orbital/src/main.rs ligne ~43)
- Sleep 500 ms en fin de main pour laisser le serial flush

README enrichi avec les étapes Voie B précises (commande redoxfs locale,
clavier FR via QEMU_USER_FLAGS, switch VT Ctrl+Alt+F2, login root/password).

Note locale (non versionnée) : ajout d'un hook QEMU_USER_FLAGS dans
~/Projets/Redox/redox-src/mk/qemu.mk pour passer des args qemu user-supplied.

Leyoda 2026 – GPLv3
This commit is contained in:
Votre Nom 2026-05-08 20:04:16 +02:00
parent c6ad583a72
commit 9ceb9a04fc
2 changed files with 120 additions and 50 deletions

View file

@ -76,21 +76,40 @@ Ouvre une fenêtre QEMU, boote une mini-VM Redox avec framebuffer, lance le bina
À fermer manuellement quand le test est fini.
### Voie B — Image complète + `make qemu`
Pousser le binaire dans une image Redox bootable via redoxfs-fuse :
Pousser le binaire dans une image Redox bootable via redoxfs-fuse, puis booter
en mode interactif. Étapes :
```bash
# Monter l'image
sudo modprobe fuse
# 1. Monter l'image (redoxfs reste en foreground tant que monté)
mkdir -p /tmp/redox-mnt
redoxfs ~/Projets/Redox/redox-src/build/x86_64/desktop/harddrive.img /tmp/redox-mnt
# Copier le binaire
sudo cp target/x86_64-unknown-redox/release/redox-wl-test-display-backend /tmp/redox-mnt/usr/bin/
# Démonter
~/Projets/Redox/redox-src/build/fstools/bin/redoxfs \
~/Projets/Redox/redox-src/build/x86_64/desktop/harddrive.img \
/tmp/redox-mnt &
sleep 2
# 2. Copier le binaire dans /usr/bin de l'image
cp ~/Projets/Redox/redox-wayland-compositor/crates/redox-wl-test-display-backend/target/x86_64-unknown-redox/release/redox-wl-test-display-backend \
/tmp/redox-mnt/usr/bin/
# 3. Démonter
fusermount -u /tmp/redox-mnt
# Booter
cd ~/Projets/Redox/redox-src && make qemu audio=no
# Dans Redox : login user / password root, puis lancer le binaire
rmdir /tmp/redox-mnt
# 4. Booter (avec clavier FR si besoin, voir hook QEMU_USER_FLAGS dans mk/qemu.mk)
cd ~/Projets/Redox/redox-src
make qemu audio=no QEMU_USER_FLAGS="-k fr"
```
Une fois Redox bootée :
- **Ctrl+Alt+F2** pour basculer sur le **VT 2 (console texte)**, sans Orbital
- Login : `root` / mot de passe `password`
- Lancer : `redox-wl-test-display-backend`
> Le binaire écrit aussi sur `/scheme/debug` (serial), donc même si tu lances
> depuis Orbital (VT 3) la sortie sera visible côté host dans le terminal qui
> a lancé `make qemu`.
## Roadmap
Voir `REDOX_COSMIC_XWAYLAND_RS_PLAN.md` pour le plan complet 14 phases / 5 ans.

View file

@ -21,43 +21,91 @@
//! parce qu'Orbital tient déjà le display. Ce test reste lecture seule à ce
//! stade — phase 4 vraie consiste à *prendre la place* d'Orbital.
use std::io;
use std::process::ExitCode;
use std::env;
use std::fs::OpenOptions;
use std::io::{self, Write};
use std::process::{Command, ExitCode};
use std::sync::OnceLock;
use drm::Device as _;
use drm::control::{Device as _, connector::State};
use graphics_ipc::V2GraphicsHandle;
use inputd::ConsumerHandle;
/// Sink that mirrors stdout to /scheme/debug so the host serial console
/// captures every diagnostic line without having to be on the active VT.
struct DebugSink(std::sync::Mutex<Option<std::fs::File>>);
impl DebugSink {
fn new() -> Self {
let f = OpenOptions::new().write(true).open("/scheme/debug").ok();
DebugSink(std::sync::Mutex::new(f))
}
fn writeln(&self, s: &str) {
println!("{s}");
if let Ok(mut g) = self.0.lock() {
if let Some(f) = g.as_mut() {
let _ = writeln!(f, "{s}");
}
}
}
}
fn dlog(s: &str) {
static SINK: OnceLock<DebugSink> = OnceLock::new();
SINK.get_or_init(DebugSink::new).writeln(s);
}
fn run() -> Result<(), Box<dyn std::error::Error>> {
println!("[disp] Phase 4 display backend test on Redox");
dlog("[disp] Phase 4 display backend test on Redox");
// VT is set by init when we're started as a VT handler (cf Orbital's main.rs).
let vt = env::var("VT").ok();
dlog(&format!("[disp] VT env = {:?}", vt));
// Step 1: open consumer handle (talks to inputd)
let consumer = ConsumerHandle::new_vt().map_err(|e| {
format!(
"ConsumerHandle::new_vt failed (expected in headless redoxer): {e} \
(errno {})",
"ConsumerHandle::new_vt failed: {e} (errno {})",
io::Error::last_os_error().raw_os_error().unwrap_or(0)
)
})?;
println!("[disp] inputd consumer handle opened");
dlog("[disp] inputd consumer handle opened");
// Step 2: open display via inputd's fpath redirection
let display_file = consumer.open_display_v2().map_err(|e| {
format!(
"open_display_v2 failed (expected in headless redoxer): {e}"
)
})?;
println!("[disp] display file opened from inputd path");
let display_file = consumer
.open_display_v2()
.map_err(|e| format!("open_display_v2 failed: {e}"))?;
dlog("[disp] display file opened from inputd path");
// Step 3: wrap as V2GraphicsHandle (subset DRM Linux)
// Step 3: register as VT handler with inputd, exactly like Orbital does
// (cf orbital/src/main.rs ~line 43). This must happen AFTER the display is
// opened so inputd considers us authoritative for that VT.
if let Some(v) = &vt {
match Command::new("inputd").arg("-A").arg(v).status() {
Ok(status) if status.success() => {
dlog(&format!("[disp] registered with inputd -A {v}"));
}
Ok(status) => {
dlog(&format!("[disp] inputd -A {v} exit {:?}", status));
}
Err(e) => {
dlog(&format!("[disp] inputd -A {v} spawn err: {e}"));
}
}
}
// Step 4: wrap as V2GraphicsHandle (subset DRM Linux)
let handle = V2GraphicsHandle::from_file(display_file)?;
println!("[disp] V2GraphicsHandle created");
dlog("[disp] V2GraphicsHandle created");
// Step 4: enumerate connectors
// Step 5: enumerate connectors
let resources = handle.resource_handles()?;
let connectors = resources.connectors();
println!("[disp] {} connector(s) reported by KMS subset", connectors.len());
dlog(&format!(
"[disp] {} connector(s) reported by KMS subset",
connectors.len()
));
let mut connected_count = 0;
for (i, &conn) in connectors.iter().enumerate() {
@ -65,22 +113,25 @@ fn run() -> Result<(), Box<dyn std::error::Error>> {
Ok(info) => {
let state = info.state();
let modes = info.modes();
println!(
dlog(&format!(
"[disp] #{i} connector {:?}: state={:?}, {} mode(s)",
conn,
state,
modes.len()
);
));
if state == State::Connected {
connected_count += 1;
if let Some(mode) = modes.first() {
let (w, h) = mode.size();
println!("[disp] first mode: {w}x{h}");
dlog(&format!("[disp] first mode: {w}x{h}"));
}
}
}
Err(e) => {
eprintln!("[disp] #{i} connector {:?}: get_connector failed: {e}", conn);
dlog(&format!(
"[disp] #{i} connector {:?}: get_connector failed: {e}",
conn
));
}
}
}
@ -88,54 +139,54 @@ fn run() -> Result<(), Box<dyn std::error::Error>> {
if connected_count == 0 {
return Err("no connected display found".into());
}
println!("[disp] {connected_count} connected display(s)");
dlog(&format!("[disp] {connected_count} connected display(s)"));
// Step 5: capabilities probe
// Step 6: capabilities probe
use drm::DriverCapability;
let dumb = handle.get_driver_capability(DriverCapability::DumbBuffer);
let cursor_w = handle.get_driver_capability(DriverCapability::CursorWidth);
let cursor_h = handle.get_driver_capability(DriverCapability::CursorHeight);
println!(
dlog(&format!(
"[disp] driver caps: dumb_buffer={:?} cursor={}x{}",
dumb,
cursor_w.map(|v| v.to_string()).unwrap_or_else(|_| "?".into()),
cursor_h.map(|v| v.to_string()).unwrap_or_else(|_| "?".into()),
);
));
// Step 6: try allocating a CpuBackedBuffer of a small size, paint a pattern,
// and destroy it. This validates the buffer pipeline of graphics-ipc without
// setting it as scanout (which would require a CRTC takeover from Orbital).
// Step 7: allocate a small CpuBackedBuffer, paint a pattern, destroy.
use drm::buffer::DrmFourcc;
use graphics_ipc::CpuBackedBuffer;
let (test_w, test_h) = (64u32, 64u32);
let mut buf = CpuBackedBuffer::new(&handle, (test_w, test_h), DrmFourcc::Argb8888, 32)?;
println!(
dlog(&format!(
"[disp] CpuBackedBuffer allocated {}x{} ARGB8888 (shadow={})",
test_w,
test_h,
buf.has_shadow_buf()
);
));
let bytes = buf.shadow_buf();
for (i, b) in bytes.iter_mut().enumerate() {
*b = (i & 0xFF) as u8;
}
buf.sync_rect(0, 0, test_w, test_h);
println!("[disp] painted test pattern + sync_rect");
dlog("[disp] painted test pattern + sync_rect");
buf.destroy(&handle)?;
println!("[disp] CpuBackedBuffer destroyed");
dlog("[disp] CpuBackedBuffer destroyed");
Ok(())
}
fn main() -> ExitCode {
match run() {
Ok(()) => {
println!("[disp] PASS: display backend pipeline reachable");
ExitCode::SUCCESS
}
Err(e) => {
eprintln!("[disp] FAIL: {e}");
ExitCode::FAILURE
}
let res = run();
match &res {
Ok(()) => dlog("[disp] PASS: display backend pipeline reachable"),
Err(e) => dlog(&format!("[disp] FAIL: {e}")),
}
// sleep briefly so the serial gets a chance to flush before the OS may shut down
std::thread::sleep(std::time::Duration::from_millis(500));
if res.is_ok() {
ExitCode::SUCCESS
} else {
ExitCode::FAILURE
}
}