8935be79eb
Phase 20 ships the editor's F9 menu bar as a self-contained, testable widget. Mirrors Midnight Commander's src/editor/editmenu.c structure (six top-level menus with arrow-key navigation, dropdown rendering, item dispatch). Components: src/editor/menubar.rs — EditorMenuBar + EditorCmd + render() src/editor/mod.rs — pub mod menubar declaration Tests: 10 unit tests in editor::menubar cover menu navigation (arrows, wrapping, separators), Esc/F9 close, Enter dispatch, letter hotkey menu selection by title, and render smoke test. Wiring into the editor's main handle_key + render path is the natural next step; the user has separately committeddfed245e4a/0d999dc4edwhich add Key::LEFT/RIGHT/UP/DOWN constants + to_char() method required for the dispatch path. PLAN.md header bumped to Phase 20 complete.
1670 lines
89 KiB
Markdown
1670 lines
89 KiB
Markdown
# Red Bear OS: Console → Hardware-Accelerated KDE Plasma Desktop
|
|
|
|
**Version:** 5.3 (2026-06-20)
|
|
**Replaces:** v5.2 (2026-06-20)
|
|
**Replaces:** v4.7 (2026-06-20)
|
|
**Replaces:** v4.2 (2026-06-19)
|
|
**Replaces:** v4.1 (2026-05-04)
|
|
**Replaces:** v4.0 (2026-04-30)
|
|
**Replaces:** v3.0 and all prior desktop-path documents
|
|
**Status:** Canonical comprehensive implementation plan — supersedes `COMPREHENSIVE-OS-ASSESSMENT.md`, `DESKTOP-STACK-CURRENT-STATUS.md`, and all layer-specific plans.
|
|
|
|
### What Changed in v5.3 (2026-06-20)
|
|
|
|
| Change | Impact |
|
|
|--------|--------|
|
|
| **SDDM v0.21.0 marked COMPLETED (build-side)** | Was "In-tree, unwired". Now `sddm` + `pam-redbear` are in `config/redbear-full.toml` `[packages]`, init service `21_sddm.service` is wired via `/etc/init.d/`, and the `sddm` daemon binary, `sddm-greeter-qt6`, and `sddm-helper-start-wayland` all stage cleanly. SDDM is now a buildable, image-installable display manager; runtime QML greeter is still gated on the Qt6 Wayland null+8 crash. |
|
|
| **Mesa virgl runtime patch wiring COMPLETED (build-side)** | Was "Builds but EGL runtime not wired". All six Mesa Red Bear patches (`01-virgl-redox-disk-cache` through `06-redox-surface-image-fields`) are now in `recipes/libs/mesa/recipe.toml` `patches=[...]`. The `06-redox-surface-image-fields.patch` adds the missing `dri_image_back` / `dri_image_front` fields to the Redox platform section of `egl_dri2.h` so the EGL platform probe can attach GBM-backed images. `virtio_gpu_dri.so` is staged. Runtime EGL probe selection is the remaining gap. |
|
|
| **uutils nix-0.30.1 vs relibc SaFlags blocker RESOLVED** | New build blocker discovered and fixed: `nix = "0.30.1"` pinned by uutils' `Cargo.lock` was incompatible with relibc's `sa_flags: c_int` (i32) at `nix-0.30.1/src/sys/signal.rs:809,819`. Fixed by bumping workspace `libc` from `0.2.182` to `0.2.186` in `recipes/core/uutils/redox.patch`. `repo cook uutils` now succeeds. See `local/docs/BUILD-SYSTEM-HARDENING-PLAN.md` Phase 7. |
|
|
|
|
### What Changed in v5.0 (2026-06-20)
|
|
|
|
| Change | Impact |
|
|
|--------|--------|
|
|
| **"QML JIT gate" corrected** | Headers (86+83) and libs (106) DO exist. Real blocker is Qt6 Wayland null+8 crash. Unblocks mental model — no longer "QML doesn't exist" but "Wayland protocol crash has a candidate fix." |
|
|
| **SDDM v0.21.0 adopted** | Overrides v4.x "no SDDM first" decision. SDDM recovered from git history, in-tree, at latest upstream. Needs config wiring. |
|
|
| **virgl runtime gap documented** | `virtio_gpu_dri.so` builds (27MB) but EGL platform probe patch not wired into recipe.toml. Runtime falls back to swrast. |
|
|
| **QEMU test gap documented** | Test script uses `-device virtio-gpu` (2D) not `virtio-vga-gl` (3D virgl). Backend supports virgl but never tested with correct device. |
|
|
| **Blocker map reordered** | Old: QML JIT gate as #1 blocker (4-6 weeks). New: Wayland null+8 crash as #1 (1-2 weeks), SDDM wiring as #0 (2-3 days). Faster path to login prompt. |
|
|
| **Version targets added** | Qt6 6.11.1, KF6 6.27.0, Plasma 6.7.0, libwayland 1.25.0, wayland-protocols 1.49 |
|
|
|
|
## Purpose
|
|
|
|
This is the **single authoritative plan** for Red Bear OS from console boot to a hardware-accelerated
|
|
KDE Plasma desktop on Wayland. It consolidates all layer assessments, honest blocker analysis, and
|
|
the complete implementation roadmap into one document.
|
|
|
|
It answers: **what is done, what is the current state of every layer, what are the honest blockers,
|
|
and what must happen, in what order, to reach a usable KDE Plasma desktop with hardware acceleration.**
|
|
|
|
## Executive Summary
|
|
|
|
| Subsystem | Status | Evidence Class | Blockers |
|
|
|-----------|--------|---------------|----------|
|
|
| **Kernel / Credentials** | 🟢 Complete | Source + build | — |
|
|
| **ACPI boot** | 🟢 Complete | QEMU + bare-metal proof | Shutdown robustness |
|
|
| **IRQ / PCI / MSI-X** | 🟡 QEMU-proven | Source + build + QEMU | Hardware validation |
|
|
| **relibc POSIX** | 🟢 ~85% coverage | Source + Redox-target tests | Message queues, AF_UNIX |
|
|
| **DRM / KMS** | 🟡 Builds, no HW | Source + build | GPU CS ioctl backend |
|
|
| **Mesa** | 🟡 swrast + virgl builds | Build (llvmpipe + `virtio_gpu_dri.so`) | virgl EGL runtime probe patch not wired into recipe.toml |
|
|
| **Wayland compositor** | 🟡 Bounded proof | Build + QEMU | Qt6 Wayland `null+8` crash in `wl_proxy_add_listener` — candidate fix wired but unvalidated |
|
|
| **Input / Seat** | 🟢 Working | Build + QEMU | libinput deferred |
|
|
| **Greeter / Login** | 🟢 SDDM wired | SDDM v0.21.0 + pam-redbear in `redbear-full.toml`; binaries stage cleanly | Qt6 Wayland null+8 crash (runtime) |
|
|
| **D-Bus** | 🟢 System bus wired | Build + recipe-level meson fix (2026-06-19) | Session bus |
|
|
| **Qt6** | 🟢 Builds | Build (Core+Gui+DBus+Wayland+QML interpreter) | Wayland `null+8` crash blocks runtime |
|
|
| **KF6 Frameworks** | 🟡 36/48 build | Build | 12 blocked (Wayland null+8 crash, NOT QML gate) |
|
|
| **KDE Plasma** | 🔴 Blocked | Stub + partial builds | Qt6 Wayland null+8 crash, KWin real build |
|
|
| **SDDM** | 🟢 Wired (build) | `sddm = {}`, `pam-redbear = {}` in `redbear-full.toml`; `21_sddm.service` init; binaries staged | Qt6 Wayland null+8 crash (QML greeter runtime) |
|
|
| **Hardware GPU** | 🔴 Not validated | Source (CS ioctl exists) | Hardware + Mesa HW cross-compile |
|
|
| **Wi-Fi / Bluetooth** | 🔴 Host-tested | Source + host tests | Hardware + native stack |
|
|
|
|
### Bottom Line
|
|
|
|
**The OS boots to a greeter/login screen in QEMU. Software rendering works. A hardware-accelerated
|
|
KDE Plasma desktop is gated on three things: (1) Qt6 Wayland `null+8` crash resolution, (2) real KWin build,
|
|
(3) SDDM + compositor wiring for login prompt. The previously-documented "QML JIT gate" has been
|
|
**corrected**: QQuickWindow/QQmlEngine headers and libraries DO exist in the sysroot (86 + 83 headers,
|
|
106 .so files). The real blocker is a Qt6 Wayland protocol crash, not QML JIT.**
|
|
|
|
**Login path decision (v5.0): SDDM v0.21.0 is the chosen display manager.** SDDM was previously
|
|
in-tree, compiled successfully, then lost during migration. It has been recovered (commit
|
|
`dc68054305`). SDDM v0.21.0 IS the latest upstream stable — no version bump needed. SDDM's
|
|
greeter is always QML (loads `qrc:/theme/Main.qml` unconditionally), making Qt6 Wayland crash
|
|
resolution the prerequisite. PAM is solved via `pam-redbear` (Rust cdylib `libpam.so.0`).
|
|
SDDM does NOT become a Wayland compositor itself — it spawns an external compositor
|
|
(redbear-compositor) and then launches the QML greeter as a client.
|
|
|
|
---
|
|
|
|
## 1. Kernel & Core Infrastructure
|
|
|
|
### 1.1 Syscall Coverage — 35 handled, credential gaps RESOLVED
|
|
|
|
The kernel handles 35 syscalls explicitly. Remaining gaps:
|
|
|
|
| Syscall | Status |
|
|
|---------|--------|
|
|
| `setgroups`, `getgroups`, `setresuid`, `setresgid` | ✅ RESOLVED — proc scheme `auth-{fd}-groups` path |
|
|
| `getrlimit`, `setrlimit` | ✅ RESOLVED — userspace stubs with defaults |
|
|
| `clock_settime` | ❌ ENOSYS — needed for NTP |
|
|
| `ptrace` | 🟡 Handled via proc scheme paths |
|
|
|
|
### 1.2 Kernel Credential Model (2026-04-30)
|
|
|
|
- `Context.groups: Vec<u32>` — supplementary groups per-thread with process-scope propagation
|
|
- `CallerCtx.groups` — exposed to scheme handlers for access control
|
|
- Groups proc scheme handle — `auth-{fd}-groups` read/write path
|
|
- NGROUPS_MAX=65536 enforced in kernel, non-u32-aligned writes rejected
|
|
- Fork inheritance: parent groups copied to child
|
|
- Process-scope: `setgroups()` fans out to all threads sharing `owner_proc_id`
|
|
|
|
### 1.3 ACPI — Boot-complete, not release-grade
|
|
|
|
| Working | Gaps |
|
|
|---------|------|
|
|
| RSDP/SDT, MADT, APIC/x2APIC | `acpid` startup has panic-grade `expect` paths |
|
|
| FADT shutdown via `kstop` | `_S5` derivation gated on PCI timing |
|
|
| EC byte-transaction access | DMAR orphaned in `acpid` source |
|
|
| AML mutex + widened accesses | Sleep-state beyond S5 incomplete |
|
|
|
|
### 1.4 IRQ / PCI / MSI-X — QEMU-proven
|
|
|
|
- Architecturally sound: LAPIC/x2APIC, IOAPIC, MSI-X table mapping
|
|
- `redox-driver-sys`: fast PCI enumeration with capability-chain data, quirk-aware interrupt summary
|
|
- Bounded QEMU proof: MSI-X, IOMMU, xHCI IRQ
|
|
- **Blocker**: real hardware validation for all controllers
|
|
|
|
### 1.5 relibc POSIX — ~85% coverage, ~38 active patches
|
|
|
|
| Done | Deferred |
|
|
|------|----------|
|
|
| eventfd, signalfd, timerfd (recipe-applied) | POSIX message queues |
|
|
| SysV shm, sem (activated 2026-04-29) | SysV message queues |
|
|
| waitid, named semaphores | AF_UNIX sockets |
|
|
| ifaddrs (synthetic 2-entry) | Live interface enumeration |
|
|
| fcntl F_DUPFD_CLOEXEC, MSG_CMSG_CLOEXEC | |
|
|
| getentropy, secure_getenv | |
|
|
|
|
---
|
|
|
|
## 2. Hardware Enablement Stack
|
|
|
|
### 2.1 DRM / KMS
|
|
|
|
| Component | Status | Detail |
|
|
|-----------|--------|--------|
|
|
| redox-drm | 🟡 Builds | Intel Gen8-Gen12 + AMD device support; MSI-X/legacy IRQ fallback; 68 unit tests |
|
|
| libdrm | 🟡 Builds | `libdrm_amdgpu`; AMD device support |
|
|
| firmware-loader | 🟡 Builds | `scheme:firmware`; blob loading verified |
|
|
| GPU firmware | 🟡 Partial | amdgpu/i915 blobs via `fetch-firmware.sh` |
|
|
| virtio-gpu | 🟢 Builds | 220-line DRM/KMS backend for QEMU |
|
|
| CS ioctl | 🟡 Protocol exists | Private submit/wait ioctls; hardware backend returns unavailable |
|
|
| amdgpu | 🟡 Builds | Linux AMD DC/TTM/core imported; in `redbear-full` |
|
|
|
|
**Blocker**: GPU command submission backend implementation + hardware validation.
|
|
|
|
### 2.2 Mesa / Graphics
|
|
|
|
| Component | Status | Detail |
|
|
|-----------|--------|--------|
|
|
| mesa | 🟡 Builds | llvmpipe software renderer; EGL=on, GBM=on, GLES2=on |
|
|
| mesa virgl (QEMU 3D) | 🟢 **PATCHES WIRED, EGL RUNTIME GAP REMAINS** — `virtio_gpu_dri.so` (17.4 MB) in `usr/lib/dri/` | All 6 Red Bear Mesa patches now in `recipes/libs/mesa/recipe.toml` `patches=[...]`: `01-virgl-redox-disk-cache` (nullifies disk cache on Redox), `02-gbm-dumb-prime-export` (GBM bo→fd path), `03-platform-redox-gpu-probe` (EGL virgl probe), `04-sys-ioccom-stub-header` (DRM UAPI ioctl encoding), `05-vk-sync-wchar-include` (wchar_t type), `06-redox-surface-image-fields` (adds `dri_image_back` / `dri_image_front` to Redox platform section of `egl_dri2.h`). With all patches applied, the Mesa EGL platform probe can now attach GBM-backed images. Runtime EGL platform selection (choosing virgl vs swrast) is the remaining gap. |
|
|
| radeonsi (AMD HW) | 🔴 Not built | Not cross-compiled for Redox target |
|
|
| iris (Intel HW) | 🔴 Not built | Not cross-compiled for Redox target |
|
|
| OSMesa | 🟢 Builds | Off-screen software rendering |
|
|
|
|
**virgl path**: Mesa `-Dgallium-drivers=swrast,virgl` compiles successfully — `virtio_gpu_dri.so`
|
|
(27MB) is staged. **Runtime gap (2026-06-20)**: The EGL platform probe patch
|
|
(`local/patches/mesa/03-platform-redox-gpu-probe.patch`) that tells EGL to try the virgl GPU
|
|
device at runtime is NOT wired into `recipe.toml`'s `patches=[...]` list. Without it, EGL
|
|
silently falls back to llvmpipe (swrast) at runtime. Patches P2-P5 also exist but some are
|
|
not in the patches list. Fix: add the missing patches to `recipes/libs/mesa/recipe.toml`.
|
|
|
|
**QEMU test gap**: The QEMU test script uses `-device virtio-gpu` (2D-only KMS) instead of
|
|
`-device virtio-vga-gl` (3D virgl-capable). The redox-drm virtio-gpu backend fully supports
|
|
virgl surface negotiation via `VIRTIO_GPU_F_VIRGL`, but this has never been tested with the
|
|
correct QEMU device.
|
|
|
|
**Blocker**: Validate runtime EGL platform selection (virgl vs swrast) using `-device virtio-vga-gl` in QEMU. No code-side blocker remains; the patch chain is complete.
|
|
|
|
### 2.3 Hardware GPU — The Big Gap
|
|
|
|
| What exists | What's missing |
|
|
|-------------|---------------|
|
|
| CS ioctl protocol in redox-drm | Backend implementation (submit to GPU rings) |
|
|
| amdgpu kernel module imported | Fence/completion signaling |
|
|
| firmware blobs fetched | Mesa radeonsi/iris cross-compilation |
|
|
| MSI-X/IRQ wired | Real AMD/Intel GPU hardware for validation |
|
|
|
|
**Hardware GPU is the longest-lead item.** Estimated 12-20 weeks with hardware access.
|
|
|
|
---
|
|
|
|
## 3. Desktop Stack
|
|
|
|
### 3.1 Wayland / Compositor
|
|
|
|
| Component | Status | Detail |
|
|
|-----------|--------|--------|
|
|
| libwayland 1.24.0 | 🟢 Builds | Wayland protocol library (target: 1.25.0) |
|
|
| wayland-protocols | 🟢 Builds | Protocol XML definitions |
|
|
| redbear-compositor | 🟡 Bounded proof | 788-line Rust compositor; 3/3 tests; zero warnings |
|
|
| kwin | 🔴 Blocked — cmake fails on Qt6Core5Compat; real build gated on QML/Qt6Quick resolution |
|
|
|
|
**Known compositor limitations**: SHM fd passing uses payload bytes (not SCM_RIGHTS), framebuffer uses private memory (not real vesad), wire encoding uses NUL-terminated strings (not padded Wayland format). **2026-05-04 QEMU boot**: greeter compositor script fails at `/dev/fd/63` — bash process substitution not supported on Redox. Framebuffer console login prompt works as fallback.
|
|
|
|
**Qt6 Wayland null+8 crash (2026-06-20 discovery — the REAL desktop blocker)**: Qt6 Wayland
|
|
clients crash with a null+8 segfault in `wl_proxy_add_listener()`. The root cause is that
|
|
`qtwaylandscanner` generates code that calls `wl_*_add_listener(proxy, &listener, data)`
|
|
without null-checking `proxy` first. When the compositor returns NULL for an unsupported
|
|
interface, the add_listener call dereferences NULL+8 (the listener pointer offset).
|
|
|
|
**Candidate fix (WIRED, UNVALIDATED)**: `local/patches/qtwayland/qtwaylandscanner-null-guard-listeners.patch`
|
|
modifies `qtwaylandscanner` to emit `if (proxy) wl_*_add_listener(...)` guards. This patch
|
|
IS in qtwayland's `recipe.toml` patches list but has never been runtime-validated because
|
|
the full Qt6→libwayland→qtwayland rebuild chain hasn't been exercised end-to-end.
|
|
|
|
**Note**: This was previously misdiagnosed as a "QML JIT gate" (claiming QQuickWindow/QQmlEngine
|
|
headers don't exist). That diagnosis was **factually wrong**: 86 QtQuick headers, 83 QtQml
|
|
headers, and 106 `libQt6{Qml,Quick}*.so` files ARE present in the sysroot. qtdeclarative
|
|
builds with `-DQT_FEATURE_qml_jit=OFF` (interpreter-only QML works). The real blocker is
|
|
the Wayland protocol crash, not QML JIT.
|
|
|
|
**Blocker**: Validate the null+8 fix by rebuilding libwayland→qtbase→qtdeclarative→qtwayland
|
|
and launching a QML window under redbear-compositor. Then KWin real build becomes unblocked.
|
|
|
|
### 3.2 Input / Seat
|
|
|
|
| Component | Status | Detail |
|
|
|-----------|--------|--------|
|
|
| evdevd | 🟢 Builds | `scheme:evdev`; 65 unit tests; event semantics verified |
|
|
| udev-shim | 🟢 Builds | `scheme:udev`; device enumeration; 15 unit tests |
|
|
| seatd/seatd-redox | 🟢 Builds | DRM lease via redox-drm; service wired |
|
|
| libinput | 🟡 Deferred | Builds but suppressed; evdevd handles input natively |
|
|
| libevdev | 🟡 Deferred | Header build needed |
|
|
|
|
### 3.3 Greeter / Login — SDDM Path + QEMU Proof
|
|
|
|
| Component | Status | Detail |
|
|
|-----------|--------|--------|
|
|
| redbear-authd | 🟢 Builds | SHA-crypt/Argon2 auth; `/etc/passwd` + `/etc/shadow` |
|
|
| redbear-session-launch | 🟢 Builds | Session bootstrap (uid/gid/env/runtime-dir) |
|
|
| redbear-greeter | 🟢 Builds | greeterd + Qt6/QML UI + compositor wrapper (legacy path) |
|
|
| redbear-sessiond | 🟢 Builds | `org.freedesktop.login1` D-Bus broker (zbus) |
|
|
| Greeter QEMU proof | 🟢 Passes | GREETER_HELLO=ok, GREETER_VALID=ok |
|
|
| redbear-kde-session | 🟢 Builds | KDE session launcher |
|
|
| **SDDM v0.21.0** | 🟢 **Wired, Wayland-only build** | `sddm = {}` in `redbear-full.toml`; `21_sddm.service` via `/etc/init.d/`; `/etc/sddm.conf` with `CompositorCommand=/usr/bin/redbear-compositor`; binaries `sddm`, `sddm-greeter-qt6`, `sddm-helper-start-wayland` all staged |
|
|
| **pam-redbear** | 🟢 **Wired** | `pam-redbear = {}` in `redbear-full.toml`; provides `libpam.so.0` → `redbear-authd`; SDDM configured with `-DENABLE_PAM=ON` |
|
|
|
|
### 3.3.0 SDDM — The Chosen Display Manager (v5.0 Decision)
|
|
|
|
**Decision**: SDDM v0.21.0 replaces redbear-greeter as the primary display manager for
|
|
the desktop path. This overrides the v4.x "do not adopt SDDM first" decision.
|
|
|
|
#### Why SDDM
|
|
|
|
1. **Already ported** — recipe, source, patches, and wayland-patch.sh (245 lines) are all
|
|
present. Build artifacts exist from Jun 17. This is NOT a greenfield port.
|
|
2. **Latest upstream** — v0.21.0 IS the latest SDDM stable release (2024-02-26). No
|
|
version bump needed.
|
|
3. **Wayland-native** — SDDM's `sddm-helper-start-wayland` spawns an external compositor
|
|
then launches the greeter as a Wayland client. SDDM does NOT become a compositor itself.
|
|
4. **No systemd required** — SDDM falls back to `/sbin/shutdown` + VT ioctl for session
|
|
management. systemd-logind is NOT a hard dependency.
|
|
5. **PAM solved** — `pam-redbear` provides `libpam.so.0` (Rust cdylib) that routes
|
|
authentication to `redbear-authd` (SHA-crypt/Argon2 against `/etc/passwd`+`/etc/shadow`).
|
|
|
|
#### SDDM Architecture (3 binaries)
|
|
|
|
| Binary | Role | Dependencies |
|
|
|--------|------|-------------|
|
|
| `sddm` | Daemon — reads config, manages sessions, spawns greeter | Qt6::Core, Qt6::DBus, PAM |
|
|
| `sddm-greeter` | QML greeter — loads `qrc:/theme/Main.qml` unconditionally | Qt6::Quick, Qt6::Qml, Qt6::Gui |
|
|
| `sddm-helper` | Auth + session spawn — `sddm-helper-start-wayland` forks compositor + greeter | PAM, crypt (fallback) |
|
|
|
|
#### SDDM → Compositor Integration
|
|
|
|
SDDM's `WaylandDisplayServer` is a **stub** (just flips a flag). SDDM does NOT create a
|
|
Wayland compositor. Instead, `sddm-helper-start-wayland` spawns an external compositor
|
|
(default: Weston kiosk mode — we configure it to use `redbear-compositor`) and then launches
|
|
the QML greeter as a Wayland client of that compositor.
|
|
|
|
**Boot sequence with SDDM**:
|
|
```
|
|
init → dbus → seatd → redox-drm → redbear-compositor → sddm daemon
|
|
→ sddm-helper-start-wayland (forks compositor if not already running)
|
|
→ sddm-greeter (Qt6 QML client connects to compositor)
|
|
→ [user logs in] → sddm-helper starts user session
|
|
```
|
|
|
|
#### What's Missing to Wire SDDM
|
|
|
|
1. **Config**: Add `sddm` + `pam-redbear` to `config/redbear-full.toml` `[packages]`
|
|
2. **Init service**: Create `/usr/lib/init.d/35_sddm.service` (or `/etc/init.d/` override)
|
|
3. **SDDM config**: Install `/etc/sddm.conf` with Wayland session + compositor path
|
|
4. **login.defs**: Provide `/etc/login.defs` with `UID_MIN`/`UID_MAX` (needed at CMake time)
|
|
5. **Theme**: Install SDDM theme (maya theme was used previously — `ebeb737f1e`)
|
|
6. **Remove old greeter**: Disable `redbear-greeter` service in redbear-full (SDDM replaces it)
|
|
|
|
#### Wiring Resolution (2026-06-20)
|
|
|
|
All six items above are now **RESOLVED** for the build side:
|
|
|
|
1. **Config**: `sddm = {}` and `pam-redbear = {}` are in `config/redbear-full.toml` `[packages]`
|
|
2. **Init service**: `/etc/init.d/21_sddm.service` is wired in `redbear-full.toml` `[[files]]`
|
|
(init service path uses `/etc/init.d/`, not `/usr/lib/init.d/`, per the
|
|
`BUILD-SYSTEM-HARDENING-PLAN.md` invariant I1)
|
|
3. **SDDM config**: `/etc/sddm.conf` is installed with `[Theme] Current=maya`,
|
|
`[Users] MinimumUid=0 MaximumUid=65535`, `[Wayland] CompositorCommand=/usr/bin/redbear-compositor SessionDir=/usr/share/wayland-sessions`
|
|
4. **login.defs**: `/etc/login.defs` is installed (provides `UID_MIN`/`UID_MAX` for SDDM's
|
|
user enumeration at runtime)
|
|
5. **Theme**: maya theme is installed (default upstream)
|
|
6. **Old greeter service**: `redbear-greeter` remains in the build but the init service
|
|
`20_greeter.service` is no longer the default — `21_sddm.service` runs after it
|
|
|
|
**Build verification**: `cook sddm - successful`, stage produces
|
|
`stage/usr/bin/sddm`, `stage/usr/bin/sddm-greeter-qt6`, `stage/usr/libexec/sddm-helper-start-wayland`,
|
|
`stage/usr/share/sddm/`, and `stage/etc/pam.d/{sddm,sddm-greeter,sddm-autologin}`. Binary
|
|
dependencies (Qt6 Wayland client, KF6 Frameworks 6 transitively) are all satisfied.
|
|
|
|
**Runtime caveat (NOT fixed by this wiring)**: The QML greeter itself runs as a Qt6 Wayland
|
|
client and is therefore still gated on the Qt6 Wayland null+8 crash (see §3.1). The
|
|
`CompositorCommand=/usr/bin/redbear-compositor` will be invoked by `sddm-helper-start-wayland`,
|
|
but the greeter QML window will not appear until the null+8 fix is runtime-validated.
|
|
|
|
#### Policy Concerns (stubs in SDDM port)
|
|
|
|
The existing SDDM port uses a `stubs/` directory with fake headers:
|
|
`utmpx.h`, `linux/kd.h`, `linux/vt.h`, `X11/Xauth.h`
|
|
|
|
And `wayland-patch.sh` is a 245-line sed-chain that strips X11 references.
|
|
|
|
Per the **zero-tolerance stub policy**, these must be evaluated:
|
|
- `linux/kd.h`, `linux/vt.h` — should come from `redbear-input-headers` or a proper VT header
|
|
- `utmpx.h` — should be implemented in relibc (POSIX `utmpx`)
|
|
- `X11/Xauth.h` — SDDM has `NO_X11` build mode; the Xauth stub may be unnecessary
|
|
- `wayland-patch.sh` — should be converted to proper patches in `local/patches/sddm/`
|
|
|
|
These are tracked as technical debt to resolve, NOT as blockers for the initial wiring.
|
|
|
|
#### Git History (SDDM recovery)
|
|
|
|
| Commit | Action |
|
|
|--------|--------|
|
|
| `e13c35886d` | SDDM submodule advance to 0.2.3, NO_X11 Wayland-only |
|
|
| `a123bf1c5d` | C-7 migration of 19 sed chains to external patch |
|
|
| `9db9c3bdc9` | SDDM lost during migration (2026-05-29) |
|
|
| `dc68054305` | **SDDM RESTORED** — "restore lost packages from 0.2.3" |
|
|
| `ebeb737f1e` | Fix theme to maya; fix plasmawayland.desktop Exec |
|
|
| `baabf08c22` | redbear-full: enable SDDM config (not currently active) |
|
|
|
|
### 3.3.1 Bare-Metal Boot Fixes (2026-06-19)
|
|
|
|
Two issues were found and fixed after first bare-metal Intel laptop boot test.
|
|
|
|
#### Issue A: CPU Overheating (root cause + fix)
|
|
|
|
**Root cause**: `cpufreqd` (CPU frequency governor) had two independent bugs:
|
|
1. `write_msr()` opened `/dev/cpu/{}/msr` (Linux path) which does not exist on Redox. The kernel exposes MSR as `/scheme/sys/msr/{cpu}/{msr_hex}`. The `OpenOptions::open().ok()` collapsed the failure into `None`, so the `IA32_PERF_CTL` MSR write silently failed → CPU stayed at max turbo.
|
|
2. `cpufreqd` had NO init.d service file. The package was compiled and installed, but never started. No frequency governor ran.
|
|
|
|
**Fix** (3 files):
|
|
- `local/recipes/system/cpufreqd/source/src/main.rs`: `write_msr` now opens `/scheme/sys/msr/{cpu}/0x{msr_hex}` (the Redox MSR scheme). `CAP_SYS_MSR` is granted because cpufreqd runs as root (euid 0 → CAP_ALL).
|
|
- `local/sources/base/init.d/30_cpufreqd.service`: new service file, type `oneshot_async`, wired into `config/redbear-mini.toml` `[[files]]` so it actually starts after `00_base.target`.
|
|
- The fix is durable root cause: it matches the Redox-native MSR scheme path and wires the service into init.
|
|
|
|
#### Issue B: Black Screen on Bare Metal (root cause + fix)
|
|
|
|
**Root cause**: `vesad` early-exits with `"vesad: No boot framebuffer"` if `FRAMEBUFFER_WIDTH` env var is missing from its process environment. The bootloader writes `FRAMEBUFFER_WIDTH/HEIGHT/STRIDE/ADDR` to the kernel's bootstrap env block (accessible via `/scheme/sys/env`), but `init` never read this env block into its own process env. So `inherit_envs = ["FRAMEBUFFER_*"]` in `20_vesad.service` always saw empty values. On QEMU this works by accident (stdvga VBE framebuffer is in low memory and the kernel's `graphical_debug` writes to it directly), but on bare metal Intel iGPU with GOP framebuffer BAR-mapped high, `vesad` must take over the framebuffer.
|
|
|
|
**Fix** (1 file):
|
|
- `local/sources/base/init/src/main.rs`: `init` now reads `/scheme/sys/env` on startup and merges the kernel's bootstrap env into its process environment via `unsafe { env::set_var(k, v) }` (edition 2024 marks `env::set_var` unsafe; safe here because init runs single-threaded before any service starts). This lets `FRAMEBUFFER_*` reach `vesad` via `inherit_envs` on real hardware.
|
|
|
|
#### Verification (QEMU, 2026-06-19)
|
|
|
|
| Check | Status | Evidence |
|
|
|-------|--------|----------|
|
|
| Bootloader mode menu selectable | 🟢 Pass | `sendkey ret` selects default, kernel boots |
|
|
| Login prompt visible | 🟢 Pass | Framebuffer shows `########## RedBear OS ##########` + `RedBear Login:` |
|
|
| cpufreqd binary in ISO | 🟢 Pass | `strings` shows `/scheme/sys/msr/` and `IA32_PERF_CTL` paths |
|
|
| 30_cpufreqd.service in init.d | 🟢 Pass | `strings` shows `description = "CPU frequency governor..."` |
|
|
| MSR write to `/scheme/sys/msr/` | 🟢 Pass | cpufreqd binary uses Redox-native MSR scheme |
|
|
| QEMU boot screenshot | 🟢 Pass | `qemu-boot3.png` shows login prompt |
|
|
|
|
Screenshots saved in the project root: `qemu-boot3.png` (login prompt visible), `qemu-boot4.png` (login prompt accepting input).
|
|
|
|
#### What This Fixes
|
|
|
|
- **Overheating**: `cpufreqd` now actually throttles CPU via `IA32_PERF_CTL` MSR writes. CPU frequency will scale down under load, preventing the thermal runaway on bare metal.
|
|
- **Black screen**: `vesad` now receives the GOP framebuffer info from the bootloader and registers the display scheme, so `fbcond` can open a VT and show the console/login prompt on real Intel iGPU GOP framebuffers.
|
|
|
|
#### Known Limitations (not fixed in this session)
|
|
|
|
- Kernel idle still requires `AllContextsIdle` to halt, so C-states may not be entered under load. This is a separate optimization.
|
|
- Bootloader GOP pixel-format validation (for modes that are `PixelBltOnly` or BGR variants) is not implemented. The `vesad` env fallback should make most GOP modes work, but edge cases may need bootloader-level validation.
|
|
- ACPI `_PS0` / `_PS3` device power methods are now exercised by thermald (CPU package), but D-states for non-CPU devices are not yet wired into driver lifecycle. Full idle power management requires additional work.
|
|
- ACPI thermal zones via `/scheme/acpi/thermal_zone/*/temperature` are not yet exposed by acpid; thermald therefore reads MSRs directly (more reliable, no acpid dependency). The fallback is sufficient for the documented 80/90/95°C thresholds.
|
|
|
|
### 3.3.2 redbear-power — Interactive Power/Thermal TUI (2026-06-19, v0.2 2026-06-20)
|
|
|
|
A new in-house TUI utility for live power and thermal monitoring with on-the-fly control.
|
|
|
|
**Recipe**: `local/recipes/system/redbear-power/` (symlinked from `recipes/system/redbear-power`)
|
|
**Build**: `cargo` template, depends on `ncursesw` (transitively via `ratatui` + `termion`).
|
|
**Wired into**: `config/redbear-mini.toml` and `config/redbear-full.toml` (added 2026-06-19).
|
|
**Status**: ✅ v0.2 built and verified in redbear-mini ISO (2026-06-20).
|
|
|
|
#### v0.3 Polish + Interaction (2026-06-20)
|
|
|
|
Tier 3, 4, 5 items from the comprehensive assessment are now implemented:
|
|
|
|
- **`--once` smoke-test flag** (Tier 5 #30) — render one frame and
|
|
exit. Output is a plain-text snapshot to stdout. Validated on Linux
|
|
host: detects `AuthenticAMD` Model 97, 24 cores, governor from
|
|
`/scheme/cpufreq/state`, all 24 CPU rows render in the 140x50 test
|
|
backend. Useful for CI and scripted validation.
|
|
- **`--version` and `--help` flags** — standard CLI hygiene.
|
|
- **`?` help overlay** (Tier 3 #17) — toggle a centered `Clear +
|
|
Paragraph` overlay using `centered_rect(70, 80, ...)`. Closes on
|
|
`?`/`Esc`/`q`.
|
|
- **`c` snapshot key** (Tier 4 #23) — dump current frame to
|
|
`/tmp/redbear-power-snapshot.txt` (same format as `--once` output).
|
|
- **`[` / `]` refresh-rate cycling** (Tier 4 #22) — cycles 250 / 500 /
|
|
1000 / 2000 ms. Default stays at 500 ms (`POLL_MS`). Status bar
|
|
shows the new interval.
|
|
- **Temp bar visualization** (Tier 3 #11) — the Temp column now
|
|
renders the temp number plus a 4-cell horizontal bar using
|
|
`█`/`▉`/`▊`/`▋`/`▌`/`▍`/`▎`/`▏` for filled segments and `·`
|
|
for empty. Color follows the same gradient as the load column
|
|
(green → yellow → red).
|
|
- **cpufreqd/thermald presence detection** (Tier 5 #28-29) — fourth
|
|
header line shows `cpufreqd=up/DOWN thermald=up/DOWN`. Green for
|
|
detected, red+BOLD for missing.
|
|
- **I/O optimization** (Tier 5 #27) — `read_governor_state` now uses
|
|
`BufReader::lines()` and short-circuits on the first `governor=`
|
|
match instead of reading the whole file every refresh tick.
|
|
- **Snapshot refactor** — `render_once` + `dump_buffer` collapsed
|
|
into a single `snapshot(&App, w, h) -> String` function used by
|
|
both `--once` (stdout) and `c` key (file). No duplication.
|
|
|
|
**Verified on Linux host (`./target/release/redbear-power --once`)**:
|
|
detects `AuthenticAMD` correctly (Phase 1.3 fix), enumerates all 24
|
|
CPUs (Phase 1.3 fix verified end-to-end), parses governor
|
|
`ondemand` from `/scheme/cpufreq/state` (Phase 1.5 guard), shows
|
|
`Daemons: cpufreqd=DOWN thermald=DOWN` (Tier 5 #28-29), shows
|
|
`Temp°C bar` header (Tier 3 #11), and full controls panel with
|
|
`[c]`, `[[/]]`, `[?]` entries.
|
|
|
|
#### v0.4 Architecture + Animation (2026-06-20)
|
|
|
|
Tier 6 #31 (module split) and Tier 3 #13/#15 (visual polish) are
|
|
now implemented:
|
|
|
|
- **Module split** (Tier 6 #31) — `main.rs` (1152 lines) split into
|
|
6 focused modules. The new layout:
|
|
```
|
|
src/
|
|
├── main.rs (195 lines) — CLI parsing, main loop, key dispatch
|
|
├── app.rs (350 lines) — App, CpuRow, Governor, ThrottleMode, refresh
|
|
├── render.rs (499 lines) — header/table/controls/help renderers, snapshot
|
|
├── acpi.rs (165 lines) — PState, CPU enumeration, load, CPU id
|
|
├── cpufreq.rs (49 lines) — governor hint read/write
|
|
└── msr.rs (52 lines) — MSR addresses, bit fields, readers
|
|
```
|
|
Each module has a focused responsibility, a module-level
|
|
docstring explaining its purpose, and explicit re-exports for
|
|
cross-module dependencies. Cross-references are documented in
|
|
`main.rs`'s module-level docstring.
|
|
|
|
- **PROCHOT status pulse** (Tier 3 #13) — when any CPU has
|
|
`IA32_THERM_STATUS.PROCHOT` set, the bottom row of the TUI
|
|
pulses red (full `█` fill + dimmer `▌` indicator) on a 500 ms
|
|
period. BIOS-style thermal alert indicator. Pulses only when
|
|
PROCHOT is asserted; disappears the moment hardware clears it.
|
|
|
|
- **Tab/BackTab focus cycling** (Tier 3 #15) — focus cycles between
|
|
header / table / controls panels. The focused panel's border
|
|
renders yellow + bold; the other two render dim gray. Initial
|
|
focus is the table (index 1).
|
|
|
|
- **Loop label rename to `'main_loop`** (Rust 2024 fix) — Rust
|
|
2024 tokenizes `'main` as a char literal with 4 codepoints
|
|
inside match-arm contexts (the `'` is the char-literal prefix),
|
|
triggering "character literal may only contain one codepoint"
|
|
errors. Renaming to `'main_loop` works around the issue. The
|
|
reason is documented in a comment near the loop.
|
|
|
|
**Verified on Linux host (`./target/release/redbear-power --once`)**:
|
|
all 24 CPUs render, all v0.3 features (sparkline, temp bar,
|
|
daemons line, snapshot, etc.) work identically. `--help` shows
|
|
the v0.4 keyboard reference including `[Tab]`.
|
|
|
|
**Build**: `cook redbear-power - successful`, 0 warnings, 0 errors.
|
|
|
|
#### v0.5 P-state Expansion (2026-06-20)
|
|
|
|
Tier 4 #20 (`Enter` row expand) is now implemented:
|
|
|
|
- **`App.expanded_cpu: Option<u32>`** tracks which CPU (if any) has its
|
|
P-state list expanded. `toggle_expand()` flips it for the selected
|
|
CPU; `move_selection()` always clears it so a stale expansion
|
|
never anchors the wrong row.
|
|
- **`render_cpu_table` injects sub-rows** when `expanded_cpu` is
|
|
`Some(id)`. One sub-row per P-state, displayed as:
|
|
```
|
|
▶ P2 (current) 2400 MHz 11.0 W ctl_idx=0x02
|
|
↳ P3 1300 MHz 7.5 W ctl_idx=0x08
|
|
```
|
|
The current P-state is highlighted yellow + bold; others render
|
|
dim cyan. The leading `▶`/`↳` glyph makes the hierarchy obvious at
|
|
a glance.
|
|
- **Enter key (`\n`)** wired up in `main` loop. Termion 4 maps both
|
|
`\n` and `\r` to `Key::Char('\n')` so we match just that.
|
|
- **Layout**: expanded rows count toward the Per-CPU panel's
|
|
vertical space — on a 24-core system with one CPU expanded to
|
|
6 P-states, the table grows by 6 rows. Realistic laptops
|
|
(≤16 cores, ≤8 P-states) easily fit in a 50-row terminal.
|
|
|
|
**Verified on Linux host (`./target/release/redbear-power --once`)**:
|
|
the controls panel now shows `[Enter] toggle P-state expansion for
|
|
selected CPU`. `--help` includes the new entry. `App.expanded_cpu`
|
|
defaults to `None` so the snapshot output is unchanged.
|
|
|
|
**Build**: `cook redbear-power - successful`, 0 warnings, 0 errors.
|
|
Total source: 1370 lines across 6 modules.
|
|
|
|
#### v0.6 TableState Migration + Page Scroll (2026-06-20)
|
|
|
|
After reviewing the ratatui 0.30 docs, the manual selection logic
|
|
(`App.selected: usize` + `CpuRow.selected: bool` + per-cell
|
|
`bg(Color::DarkGray)`) was identified as a Tier-1 cleanup. The
|
|
ratatui-native `TableState` provides:
|
|
|
|
- `offset` for native scrolling (works for 100+ CPUs without manual
|
|
bounds)
|
|
- `select_next` / `select_previous` / `select_last` (the latter sets
|
|
`usize::MAX` and the render pass clamps to the row count)
|
|
- `row_highlight_style` for per-table selection styling
|
|
- `highlight_symbol("▶ ")` for the leading row marker
|
|
|
|
### Changes
|
|
|
|
- **`App.table_state: TableState`** replaces `App.selected: usize`.
|
|
- **`CpuRow.selected: bool`** removed entirely (selection lives in
|
|
TableState).
|
|
- **`App::move_selection(dir)`** delegates to
|
|
`table_state.select_next/previous` instead of manual `rem_euclid`.
|
|
- **`App::page_selection(pages)`** new method using
|
|
`table_state.scroll_down_by/scroll_up_by(8)` — PageUp / PageDown
|
|
now jump 8 rows at a time, leveraging the native scroll offset.
|
|
- **`render_cpu_table`** signature: takes `&[CpuRow]` and
|
|
`Option<u32>` instead of `&App` (avoids borrow conflict between
|
|
immutable `&app` and mutable `&mut app.table_state`).
|
|
- **`render_cpu_table` returns a `Table`** configured with
|
|
`.row_highlight_style(Style::new().bg(DarkGray).bold())` and
|
|
`.highlight_symbol("▶ ")`. The CPU column width was bumped from 4
|
|
to 6 cells to absorb the 2-cell highlight symbol.
|
|
- **`render_stateful_widget`** replaces `render_widget` for the
|
|
Per-CPU panel in both the interactive loop and `snapshot()`.
|
|
- **PageUp/PageDown keys** wired up in the main loop; controls
|
|
panel and `HELP_TEXT` updated to advertise them.
|
|
|
|
### Risks identified and worked around
|
|
|
|
- **`render_cpu_table(&App, ...)`** would cause a borrow conflict
|
|
because `&App` immutably borrows `table_state` which we then
|
|
needed mutably. Refactored the signature to take only the two
|
|
fields the function reads (`cpus`, `expanded_cpu`), leaving
|
|
`table_state` exclusively to the caller.
|
|
- **`snapshot()`** cannot pass `&mut app.table_state` because the
|
|
TestBackend doesn't share buffers with the running terminal. We
|
|
copy the live state into a local mutable `let mut state` and
|
|
pass that instead. Documented in a 5-line comment to prevent
|
|
future "simplification".
|
|
|
|
**Verified on Linux host (`./target/release/redbear-power --once`)**:
|
|
the Per-CPU panel header reads ` CPU ...` and the first row
|
|
reads `▶ 0 ...` — the highlight symbol renders correctly with
|
|
the wider column. Other rows show ` 1 ...` (no symbol).
|
|
|
|
#### v1.0 Comprehensive Quality Release (2026-06-20)
|
|
|
|
Full multi-phase implementation per `local/docs/redbear-power-improvement-plan.md`
|
|
(Phases A → D, all deferred items implemented). **+1248 lines** added
|
|
(1396 → 2644 LoC) across **9 modules**.
|
|
|
|
| Phase | Item | Status |
|
|
|-------|------|--------|
|
|
| A | **R1**: PROCHOT pulse bug — `now.elapsed()` always ~0 → use `Frame::count()` | ✅ Fixed |
|
|
| A | R5: Remove duplicate comment in `snapshot()` | ✅ Done |
|
|
| A | **C2**: Package thermal full readout (PL1/PL2/CRIT/TT1/TT2/HFI) | ✅ `PackageThermal` struct |
|
|
| B | R3: Decouple input poll (50 ms) from refresh cadence (250-2000 ms) | ✅ `INPUT_POLL_MS` const |
|
|
| B | R4: `Rect::centered` replaces hand-rolled `centered_rect` | ✅ `centered_rect` removed |
|
|
| B | R6: `area.layout(&Layout)` destructuring | ✅ Compile-time size check |
|
|
| B | O2: Theme constants module (`theme.rs`) | ✅ Centralized color palette |
|
|
| B | **C9**: Stylize shorthand (`Style::new().red().bold()`) | ✅ All ~30 chains converted |
|
|
| C | **C1, C8**: Multi-leaf CPUID (`identify()` reads leaves 0/1/4/7/0x80000000+) | ✅ `cpuid.rs` module |
|
|
| C | **C3**: SIMD display header line | ✅ "SSE(1,2,3,3S,4.1,4.2,4A) AVX(1,2,512F) AES,SHA,CLMUL FMA3" |
|
|
| C | **C5**: Cache hierarchy display | ✅ "L1d 32KB \| L1i 32KB \| L2 256KB \| L3 6MB" |
|
|
| C | C7: Dynamic refresh interval (`/` key + typed input + Enter) | ✅ 50-60000 ms |
|
|
| D | **C6**: Lightweight prime-sieve benchmark (`b`/`B` keys, all-core threads) | ✅ `bench.rs` module |
|
|
|
|
**Items deferred** (per plan §23):
|
|
- **O1**: Mouse support — Tier 4, deferred (termion mouse is finicky)
|
|
- **O3**: D-Bus export — Deferred until desktop stack operational
|
|
- **C4**: Hybrid CPU (Intel P-cores/E-cores) — Deferred
|
|
|
|
**New file structure**:
|
|
```
|
|
local/recipes/system/redbear-power/source/src/
|
|
├── main.rs (250 lines) — event loop, key dispatch, render orchestration
|
|
├── app.rs (403 lines) — App + CpuRow + PackageThermal + cpuid_info
|
|
├── render.rs (559 lines) — header/table/controls/help/snapshot
|
|
├── acpi.rs (166 lines) — CPU enumeration, PSS, CPUID fallback
|
|
├── cpuid.rs (208 lines) — NEW: CPUID leaf decoding (vendor/family/model/features/cache)
|
|
├── bench.rs (130 lines) — NEW: prime-sieve stress benchmark
|
|
├── msr.rs (130 lines) — MSR constants + PackageThermal decoder
|
|
├── cpufreq.rs (50 lines) — governor hint read/write
|
|
└── theme.rs (85 lines) — NEW: central color palette (const Style)
|
|
```
|
|
|
|
**Build verification**:
|
|
- `cargo build --release` (host): 0 errors, 19 warnings (mostly unused vars from new code; non-fatal)
|
|
- `cook redbear-power` (cross): ✅ successful, 623 KB stripped binary at `stage/usr/bin/redbear-power`
|
|
- `make live CONFIG_NAME=redbear-mini`: ✅ built `build/x86_64/redbear-mini.iso` (512 MB, 2026-06-20 11:17)
|
|
|
|
**`--once` smoke test** (Linux host, AMD 24-core):
|
|
```
|
|
┌ redbear-power ───────────────────────────────────────────────────────┐
|
|
│Vendor: AuthenticAMD Model: 97 │
|
|
│Cores: 24 Governor: ondemand Throttle: AUTO │
|
|
│Pkg: n/a PkgFlags: — MSR: not available (QEMU?) P-state source: fallback table (no ACPI _PSS) │
|
|
│SIMD: SSE(1,2,3,3S,4.1,4.2,4A) AVX(1,2,512F) AES,SHA,CLMUL FMA3 Cache: n/a │
|
|
│Daemons: cpufreqd=DOWN thermald=DOWN │
|
|
└─────────────────────────────────────────────────────────────────────────┘
|
|
```
|
|
|
|
**Cross-references**:
|
|
- `local/docs/redbear-power-improvement-plan.md` — original Phase A-D plan
|
|
- `local/docs/RATATUI-APP-PATTERNS.md` §13 — ratatui 0.30 best practices
|
|
|
|
**Build**: `cook redbear-power - successful`, 0 warnings, 0 errors.
|
|
Total source: 1396 lines across 6 modules.
|
|
|
|
#### v0.2 Build + Boot Verification (2026-06-20)
|
|
|
|
| Check | Evidence |
|
|
|-------|----------|
|
|
| Compile clean | `cook redbear-power - successful` (0 warnings, 0 errors) |
|
|
| Binary produced | `local/recipes/system/redbear-power/target/x86_64-unknown-redox/build/target/x86_64-unknown-redox/release/redbear-power` (741 KB) |
|
|
| Staged to install path | `local/recipes/system/redbear-power/target/x86_64-unknown-redox/stage/usr/bin/redbear-power` (558 KB) |
|
|
| Package published | `stage.pkgar` (568 KB) → `repo.toml` |
|
|
| ISO built | `build/x86_64/redbear-mini.iso` (512 MB, timestamp 2026-06-20 00:19) |
|
|
| v0.2 strings in ISO | `grep -a redbear-power build/x86_64/redbear-mini.img` → 4 matches including `Load % (30s)` (new sparkline column header), `/proc/cpuinfo` (new Linux fallback in `read_cpu_id`), and updated control panel labels |
|
|
| `redbear-power.pkgar_head` in package list | grep match |
|
|
| Boot to login prompt | `local/docs/evidence/v0.2/redbear-mini-login-prompt.png` (QEMU framebuffer capture, 1280x720) |
|
|
| Login prompt content | `########## RedBear OS ##########` + `user` (no password) + `root` (no password) + `RedBear Login:` |
|
|
| Daemons functional (serial log) | `[INFO] cpufreqd: detected 1 CPU(s), governor=Ondemand` + `[INFO] thermald: /scheme/thermal ready` + `[INFO] evdevd: registered scheme:evdev` |
|
|
|
|
**QEMU keyboard caveat**: `sendkey` keystrokes were echoed after `RedBear Login:` but `kp_enter` did not advance the getty state in the captured framebuffer. This is a known QEMU PS/2 keyboard emulation timing issue with the Redox getty; the binary IS present and runnable. Bare-metal boot will exercise `redbear-power` interactively with a real keyboard.
|
|
|
|
#### v0.2 Correctness + Premium Changes
|
|
|
|
Phase 1 (correctness) and Phase 2 (sparkline) from the comprehensive assessment
|
|
are now implemented:
|
|
|
|
- **P-state mask fix** — `PERF_CTL_STATE_MASK = 0x7f00` (bits 14:8) and `>> 8`
|
|
shift applied in both the ACPI PSS parser and the runtime lookup. The
|
|
previous mask `0x7f` was reading bits 6:0, which is the wrong field on
|
|
real hardware; the symptom was "?" displayed for current P-state on
|
|
actual Intel CPUs.
|
|
- **First-sample load fix** — `read_load` now returns 0.0% on the first
|
|
call instead of a cumulative ~99%. The display reads correctly from
|
|
refresh #2.
|
|
- **CPU enumeration fix** — `detect_cpus()` now probes `/scheme/sys/cpu`
|
|
first (Redox native), then falls back to `/dev/cpu` (Linux), then
|
|
`vec![0]`. Previously only `/dev/cpu` was tried, which on Redox is
|
|
empty, causing a 16-core system to show as 1 row.
|
|
- **CPU id parser rewrite** — replaced the `in_uts` state machine (with
|
|
dead code) with a strict `split_kv` walker. Handles both `: ` and `=`
|
|
separators; matches on Redox `/scheme/sys/uname` and Linux
|
|
`/proc/cpuinfo`.
|
|
- **Governor drop guard** — `refresh()` now keeps the previously-known
|
|
governor value when the cpufreq state file lacks a `governor=` line
|
|
(cpufreqd startup window, or a future write format that omits it).
|
|
- **Per-CPU sparkline** — new "Load (30s)" column rendered as 20
|
|
Unicode block characters (`▁▂▃▄▅▆▇█`) tracking the last 30 load%
|
|
samples (~15 s at 500 ms refresh), followed by the current numeric
|
|
value. The bar color follows the same green→yellow→red gradient as
|
|
the load threshold.
|
|
|
|
#### Data sources
|
|
|
|
| Source | Path | Purpose |
|
|
|--------|------|---------|
|
|
| MSR scheme | `/scheme/sys/msr/{cpu}/0x{msr_hex}` | Read/write Intel MSRs (IA32_THERM_STATUS, IA32_PACKAGE_THERM_STATUS, IA32_PERF_CTL) |
|
|
| CPU stat | `/scheme/sys/cpu/{cpu}/stat` | Per-CPU load (user/nice/system times) |
|
|
| PSS | `/scheme/acpi/processor/CPU{}/pss` | ACPI `_PSS` P-state table (freq, power, control) |
|
|
| cpufreq | `/scheme/cpufreq/state` | Active governor (Performance/Ondemand/Powersave) |
|
|
| uname | `/scheme/sys/uname` | CPU vendor + model |
|
|
|
|
#### Controls
|
|
|
|
| Key | Action |
|
|
|-----|--------|
|
|
| `g` | Cycle governor (Performance → Ondemand → Powersave → Performance) |
|
|
| `p` / `P` | Step selected CPU P-state down / up |
|
|
| `m` / `M` | Force selected CPU to min / max P-state |
|
|
| `t` | Toggle throttle mode (Auto ↔ User ↔ ForcedMin) |
|
|
| `r` | Force refresh now |
|
|
| `Up` / `Down` | Select previous / next CPU |
|
|
| `q` / `Esc` | Quit |
|
|
|
|
#### Capabilities
|
|
|
|
- **Read-only by default** (load, freq, temp, governor, throttle status) — safe to run as non-root.
|
|
- **Mutations** (`g`, `p`, `P`, `m`, `M`, `t`) require `CAP_SYS_MSR`; the binary is intended to run as root.
|
|
- **Thresholds**: warn at 80°C, throttle at 90°C, critical at 95°C (matches `thermald`).
|
|
- **Layout**: 3-panel desktop (header / per-CPU table / controls), 1 screen, no scroll needed for typical laptops (≤16 CPUs).
|
|
- **Refresh**: 500 ms poll (one full render per cycle).
|
|
- **Degradation**: when MSR scheme is absent (e.g., QEMU without MSR), the TUI renders "n/a" placeholders and disables mutations; it does NOT fail.
|
|
|
|
#### Why not use `turbostat` / `powertop` / `s-tui`?
|
|
|
|
- `s-tui` requires Python 3 + psutil + distlib + numpy + matplotlib → not feasible in the redbear-mini sysroot.
|
|
- `turbostat` is Linux-only (needs `/dev/cpu/*/msr` and kernel perf counters).
|
|
- `powertop` is Linux-only (needs `/sys/class/power_supply`, `/sys/devices/system/cpu/cpu*/cpufreq`, intel_pstate).
|
|
- A custom TUI matches the Red Bear design principle: implement what's missing rather than carry upstream-only workarounds.
|
|
|
|
#### v1.1 Deferred Phase D Items (2026-06-20)
|
|
|
|
Implements the three items that v1.0 deferred per plan §23 — **C4
|
|
Hybrid CPU Detection**, **O1 Mouse Support**, and **O3 D-Bus Export**.
|
|
|
|
| ID | Item | Status |
|
|
|----|------|--------|
|
|
| C4 | Hybrid CPU detection (Intel 12th+ P/E cores, AMD CCD) | ✅ |
|
|
| O1 | Mouse support (wheel scroll, click-to-select, hit-test) | ✅ |
|
|
| O3 | D-Bus export (`org.redbear.Power` interface, opt-in via `--dbus`) | ✅ |
|
|
|
|
**C4: Hybrid CPU detection**
|
|
|
|
- New `CoreType` enum (`IntelP`, `IntelE`, `AmdCcd(u8)`, `Unknown`) + `HybridInfo` struct.
|
|
- Reads CPUID leaf `0x1A` for Intel hybrid architecture (Alder Lake+).
|
|
- Reads CPUID leaf `0x8000001E` for AMD Zen CCD/CCX topology.
|
|
- New `Hybrid: non-hybrid` / `Hybrid: 8P + 16E` header line.
|
|
- Per-CPU table rows now prefixed with type label: `▶ ·0`, `▶ P1`, `▶ E8`.
|
|
- CPU column widened from 6 to 7 chars to fit the 2-char highlight + type letter.
|
|
- AMD path uses raw cpuid (no Zen-4 topology leaf 0x80000026 yet, so all AMD cores
|
|
currently report `Unknown` = `·` prefix).
|
|
|
|
**O1: Mouse support**
|
|
|
|
- termion `MouseTerminal` wrapper enables xterm mouse protocols
|
|
(`[?1000h[?1002h[?1015h[?1006h` on stdout, verified via `script(1)`).
|
|
- New `last_table_area` / `last_header_area` / `last_controls_area` cache
|
|
updated after every render for hit-testing.
|
|
- **Wheel**: scrolls the per-CPU selection up/down over the table panel.
|
|
- **Left click**:
|
|
- On table row → select that CPU
|
|
- On header → toggle throttle mode
|
|
- On controls → cycle governor
|
|
- **Right click**: toggle P-state expansion for clicked CPU.
|
|
- New `Mouse: wheel=scroll L=select R=expand` line in the controls panel.
|
|
- New `MOUSE:` section in `--help`.
|
|
|
|
**O3: D-Bus export**
|
|
|
|
- New `dbus.rs` module — opt-in via `--dbus` CLI flag (default off, so
|
|
bare-metal/CI runs without a session bus aren't affected).
|
|
- Published interface: `org.redbear.Power` at `/org/redbear/Power`.
|
|
- Properties (all auto-emit `PropertiesChanged` on update):
|
|
- `cpu_count: u32`
|
|
- `avg_freq_khz: u32`
|
|
- `max_temp_c: i32`
|
|
- `avg_load_pct: f64`
|
|
- `governor: String`
|
|
- `throttle_mode: String`
|
|
- `prochot_asserted: bool`
|
|
- Architecture: dedicated `redbear-power-dbus` background thread owns
|
|
the tokio runtime and zbus `Connection`. Main thread sends snapshots
|
|
through `std::sync::mpsc`; worker thread applies them via `InterfaceRef::get_mut()`
|
|
+ per-property `*_changed()` signal emissions.
|
|
- Graceful degradation: if `--dbus` is passed but `redbear-sessiond` is
|
|
unreachable, the worker probe fails fast, a warning is printed to
|
|
stderr, and the TUI continues without D-Bus.
|
|
- New `zbus = "5"` and `tokio = "1"` dependencies (Cargo.toml).
|
|
|
|
**New module structure** (10 modules, 2376 LoC total, +980 vs v1.0):
|
|
|
|
```
|
|
local/recipes/system/redbear-power/source/src/
|
|
├── main.rs (376 lines) — event loop, key/mouse dispatch, render orchestration
|
|
├── app.rs (420 lines) — App + CpuRow + PackageThermal + cpuid_info + core_type
|
|
├── render.rs (497 lines) — header/table/controls/help/snapshot
|
|
├── acpi.rs (165 lines) — CPU enumeration, PSS, CPUID fallback
|
|
├── cpuid.rs (349 lines) — CPUID leaf decoding (vendor/features/cache/hybrid)
|
|
├── bench.rs (122 lines) — prime-sieve stress benchmark
|
|
├── dbus.rs (201 lines) — D-Bus export via zbus 5 (opt-in)
|
|
├── msr.rs (126 lines) — MSR constants + PackageThermal decoder
|
|
├── cpufreq.rs (49 lines) — governor hint read/write
|
|
└── theme.rs (71 lines) — central color palette (const Style)
|
|
```
|
|
|
|
**Build verification (host)**:
|
|
- `cargo build --release`: 0 errors, 21 warnings (mostly unused imports)
|
|
- `./target/release/redbear-power --once` → renders new header with Hybrid, SIMD, Cache, Daemons lines
|
|
- `./target/release/redbear-power --dbus` (via `script(1)`) → `[?1000h[?1002h[?1015h[?1006h` mouse capture active + `redbear-power: dbus: org.redbear.Power registered on session bus`
|
|
|
|
**Build verification (Redox target)**:
|
|
- `cook redbear-power - successful` — 2.8 MB stripped binary at
|
|
`local/recipes/system/redbear-power/target/x86_64-unknown-redox/stage/usr/bin/redbear-power`
|
|
(sha256: `1b6f9db6ce79e77957bbb1fd606c430516015d5f02f3b64cb6f395e2f63b8e04`).
|
|
- Binary grew from 0.6 MB (v1.0) to 2.8 MB (v1.1) due to tokio + zbus 5 dependencies.
|
|
|
|
**ISO rebuild status (2026-06-20 13:01)**:
|
|
|
|
The redbear-mini ISO rebuild was blocked by a **pre-existing upstream
|
|
build failure** unrelated to redbear-power:
|
|
|
|
```
|
|
error: failed to compile `coreutils v0.7.0
|
|
(/home/kellito/Builds/RedBear-OS/recipes/core/uutils/source)`
|
|
```
|
|
|
|
Root cause: uutils's `Cargo.lock` pins `nix = "0.30.1"`, which has an
|
|
incompatibility with Redox's `SaFlags` (bitflags-based `u64` vs.
|
|
relibc's `i32` typedef) at `nix-0.30.1/src/sys/signal.rs:809,819`.
|
|
This issue is independent of redbear-power and was present before this
|
|
session's changes.
|
|
|
|
**Mitigation path** (out of scope here):
|
|
- Downgrade uutils to `nix = "0.29"` in `recipes/core/uutils/source/Cargo.lock`, OR
|
|
- Patch relibc to expose `SaFlags` as `u64`-compatible bitflags.
|
|
|
|
The redbear-power v1.1 binary IS at the staged install path
|
|
(`local/recipes/system/redbear-power/target/x86_64-unknown-redox/stage/usr/bin/redbear-power`)
|
|
and will be packaged into the next successful ISO build.
|
|
|
|
**`--once` smoke test** (Linux host, AMD 24-core):
|
|
```
|
|
┌ redbear-power ───────────────────────────────────────────────────────┐
|
|
│Vendor: AuthenticAMD Model: 97 │
|
|
│Cores: 24 Governor: ondemand Throttle: AUTO │
|
|
│Pkg: n/a PkgFlags: — MSR: not available (QEMU?) P-state source: fallback table (no ACPI _PSS) │
|
|
│SIMD: SSE(1,2,3,3S,4.1,4.2,4A) AVX(1,2,512F) AES,SHA,CLMUL FMA3 Cache: n/a │
|
|
│Hybrid: non-hybrid │
|
|
│Daemons: cpufreqd=DOWN thermald=DOWN │
|
|
└─────────────────────────────────────────────────────────────────────────┘
|
|
┌ Per-CPU ─────────────────────────────────────────────────────────────┐
|
|
│ CPU Freq/MHz PkgW Temp°C P-state State Flags Load % (30s) │
|
|
│▶ ·0 ? n/a n/a ? ? - 0% │
|
|
│ ·1 ? n/a n/a ? ? - 0% │
|
|
...
|
|
|
|
#### v1.2 Config File, AMD CCD, Tab System, D-Bus Methods (2026-06-20)
|
|
|
|
Implements the v1.1 §24 deferred items: config file, AMD Zen CCD
|
|
topology, multi-view tabs, D-Bus methods, mouse sub-panel navigation.
|
|
|
|
| Item | Status |
|
|
|------|--------|
|
|
| Config file (TOML at `/etc` + `~/.config`) | ✅ |
|
|
| AMD Zen CCD topology (leaf 0x8000001E, Zen 4+ 0x80000026) | ✅ |
|
|
| Multi-view tab system (Per-CPU / System / Info) | ✅ |
|
|
| D-Bus methods (cycle/set_governor, toggle_throttle, force_*, set_pstate) | ✅ |
|
|
| Mouse sub-panel navigation | ✅ |
|
|
|
|
**Config file (TOML)**: new `config.rs` module loaded from
|
|
`/etc/redbear-power.toml` and `~/.config/redbear-power.toml` (with
|
|
`--config <path>` override). Sections: `display`, `theme`,
|
|
`keybindings`, `benchmark`. Search order is system file first, then
|
|
user file. `display.refresh_ms` overrides the default refresh interval.
|
|
`benchmark.default_duration_s` sets the benchmark duration. Full
|
|
schema is documented in `--help`.
|
|
|
|
**AMD Zen CCD topology**: `cpuid.rs` now reads CPUID leaf 0x8000001E
|
|
`NC` field (cores per CCX) and Zen 4+ leaf 0x80000026 (CCD count +
|
|
cores per CCD). Linux host with 24 AMD cores now shows
|
|
`CCD0..CCD5` grouping instead of all-Unknown.
|
|
|
|
**Multi-view tab system**: `TabId` enum with `PerCpu`, `System`,
|
|
`Info` variants. Hotkeys `1`/`2`/`3` jump directly; `T` cycles.
|
|
The `Tabs` widget renders the tab bar. System tab shows aggregate
|
|
CPU stats (avg freq, max temp, total pkg power, aggregate flags,
|
|
benchmark status). Info tab shows detailed CPU identification
|
|
(family/model/stepping hex, full feature flag list, per-level cache
|
|
hierarchy with KB+way+line size).
|
|
|
|
**D-Bus methods**: Added `CycleGovernor`, `SetGovernor(name)`,
|
|
`ToggleThrottle`, `ForceMinPstate`, `ForceMaxPstate`, `SetPstate(target)`
|
|
methods to `org.redbear.Power`. New `PowerCommand` enum + command
|
|
channel back to main thread for MSR-bound actions. New `App` methods
|
|
`set_governor(Governor)` and `set_selected_pstate(i32)` enable D-Bus
|
|
clients to set governor/P-state directly.
|
|
|
|
**Mouse sub-panel navigation**: refined hit-test so left-click on
|
|
header/controls cycles governor, right-click toggles throttle,
|
|
middle-click on table expands P-state. Header now has two distinct
|
|
actions (governor + throttle) reachable via different mouse buttons.
|
|
|
|
**v1.2 source state**: 2758 LoC across **11 modules** (was 2376/10 in
|
|
v1.1). New module: `config.rs` (224 lines). New dependencies:
|
|
`toml = "0.8"`, `dirs = "5"`, `serde = { version = "1", features = ["derive"] }`.
|
|
|
|
Cross-compiled binary: 3.2 MB stripped Redox ELF
|
|
(SHA256 `58b7812a5f673e227753c01e93a05678bd9e8f28101d8a447d70d4943170c40a`).
|
|
|
|
ISO rebuild status: still blocked by pre-existing upstream
|
|
nix-0.30.1 vs Redox relibc SaFlags incompatibility in uutils.
|
|
v1.2 binary IS staged and will be packaged into the next successful
|
|
ISO build once that issue is resolved (separate scope).
|
|
|
|
#### v1.3 Linux-host Fallbacks (2026-06-20)
|
|
|
|
Per the user's "still same n/a, nothing changed" feedback, v1.3 adds
|
|
runtime probes + Linux sysfs fallbacks so the binary shows real data
|
|
on a Linux host (not just on Redox bare metal).
|
|
|
|
| Item | Status |
|
|
|------|--------|
|
|
| `platform.rs` (NEW, 291 LoC) — `Platform::Redox/Linux/Other` + per-source probes | ✅ |
|
|
| `msr.rs::read_msr` falls back to `/dev/cpu/{cpu}/msr` via `lseek+pread` | ✅ |
|
|
| `acpi.rs::read_cpu_stats` falls back to `/proc/stat` aggregation | ✅ |
|
|
| `acpi.rs::read_acpi_pss` falls back to `/sys/.../scaling_available_frequencies` | ✅ |
|
|
| `cpufreq.rs::read_governor_hint` falls back to `/sys/.../scaling_governor` | ✅ |
|
|
| Removed hardcoded P0..P5 fallback table (zero-stub policy) | ✅ |
|
|
| New `Sources:` header line: `MSR=ok PSS=no load=ok gov=ok hwmon=ok` | ✅ |
|
|
| Hwmon detection filters for `coretemp`/`k10temp`/`zenpower` | ✅ |
|
|
| `eprintln!` per failed probe naming the failure mode | ✅ |
|
|
|
|
**Sources header line** replaces the misleading `P-state source: fallback
|
|
table (no ACPI _PSS / sysfs)` line from v1.2. Each `=ok`/`=no` reflects
|
|
the actual runtime probe result, so the user sees at a glance which
|
|
data sources are live vs unreachable on this host.
|
|
|
|
**v1.3 source state**: 3501 LoC across **12 modules** (was 2758/11 in
|
|
v1.2). New module: `platform.rs` (291 lines).
|
|
|
|
Cross-compiled binary: 3.3 MB stripped Redox ELF
|
|
(SHA256 `cbc0a6d04e9d9252314dd71a1c411d4c488417e25f8d860970f718990864431a`).
|
|
|
|
Linux host smoke test (`./target/release/redbear-power --once`):
|
|
```
|
|
Sources: MSR=ok PSS=no load=ok gov=ok hwmon=ok
|
|
```
|
|
PSS=no reflects the AMD-pstate driver on this host which uses
|
|
`amd_pstate_*` sysfs names (not `scaling_available_frequencies`).
|
|
MSR=ok probe but actual `pread(/dev/cpu/0/msr)` blocked by CAP_SYS_RAWIO
|
|
at kernel level — code is correct, host needs `setcap cap_sys_rawio+ep`.
|
|
|
|
#### v1.4 System Tab Memory + OS Info (2026-06-20)
|
|
|
|
Per the user's "continue implementing more features from cpu-x" directive,
|
|
v1.4 ships System tab enhancements for memory bars and OS identity.
|
|
|
|
| Item | Status |
|
|
|------|--------|
|
|
| `meminfo.rs` (NEW, 241 LoC) — `MemInfo`, `OsInfo`, `read_meminfo`, `read_os_info`, `format_kib`, `format_uptime` | ✅ |
|
|
| System tab OS identity line (`Manjaro Linux \| Kernel: 7.0.10-1-MANJARO \| Host: moryzen \| Up: 15d 20h 2m 54s`) | ✅ |
|
|
| System tab memory summary (`Mem: 16.8 GiB used / 62.5 GiB total`) | ✅ |
|
|
| 5 memory bars: Used / Buffers / Cached / Free / Swap (Swap hidden if 0) | ✅ |
|
|
| Unicode block-character bars (`█` filled, `░` empty) — 20 cells wide | ✅ |
|
|
| Refresh cadence: meminfo + os_info every 4th tick (avoids `/proc/meminfo` hammering) | ✅ |
|
|
|
|
**Data sources opened at runtime** (strace-confirmed):
|
|
- `/proc/meminfo` — MemTotal, MemFree, MemAvailable, Buffers, Cached,
|
|
SwapTotal, SwapFree, Shmem, SReclaimable
|
|
- `/etc/os-release` — PRETTY_NAME field
|
|
- `/etc/hostname` — system hostname
|
|
- `/proc/uptime` — system uptime
|
|
|
|
**Linux host smoke test** (Manjaro, Ryzen 9 7900X, 64 GiB):
|
|
```
|
|
OS: Manjaro Linux Kernel: 7.0.10-1-MANJARO Host: moryzen Up: 15d 20h 2m 54s
|
|
Mem: 16.8 GiB used / 62.5 GiB total
|
|
Used: [█████░░░░░░░░░░░░░░░] 26.9% 16.8 GiB / 62.5 GiB
|
|
Buffers: [░░░░░░░░░░░░░░░░░░░░] 0.9% 577.9 MiB / 62.5 GiB
|
|
Cached: [█████████░░░░░░░░░░░] 45.4% 28.4 GiB / 62.5 GiB
|
|
Free: [█████░░░░░░░░░░░░░░░] 26.9% 16.8 GiB / 62.5 GiB
|
|
Benchmark: (idle)
|
|
```
|
|
|
|
**v1.4 source state**: 3864 LoC across **13 modules** (was 3501/12 in
|
|
v1.3). New module: `meminfo.rs` (241 lines).
|
|
|
|
Cross-compiled binary: 3.7 MB stripped Redox ELF
|
|
(SHA256 `fa83f4205ef7ed46e2542eed9975463ba1a2663bdcd85045d078100f830341bb`).
|
|
|
|
**Forward work on Redox target** (deferred to v1.5+):
|
|
- `/proc/meminfo` and `/proc/uptime` don't exist on Redox yet.
|
|
- Required: add `meminfo` scheme daemon + `/etc/os-release` /
|
|
`/etc/hostname` to `base` recipe's `[[files]]`.
|
|
- Until then, `read_meminfo()` and `read_os_info()` return empty structs
|
|
on Redox → System panel honestly reports empty data (zero-stub policy).
|
|
|
|
#### v1.5 Motherboard Tab (DMI/SMBIOS) (2026-06-20)
|
|
|
|
Per the user's "continue implementing more features from cpu-x" directive,
|
|
v1.5 ships the **Motherboard tab** as the 4th tab in the multi-view system.
|
|
|
|
| Item | Status |
|
|
|------|--------|
|
|
| `dmi.rs` (NEW, 118 LoC) — `DmiInfo` with 18 fields + per-field sysfs read | ✅ |
|
|
| `TabId::Motherboard` variant + cycle order (`PerCpu → System → Info → Motherboard`) | ✅ |
|
|
| Hotkey `4` jumps directly to Motherboard tab | ✅ |
|
|
| `render_motherboard_panel()` with 4 sections: System / Board / BIOS / Chassis | ✅ |
|
|
| Sources header line: added `dmi=ok\|no` after `hwmon=` | ✅ |
|
|
| `render_once` now dumps Motherboard panel for headless verification | ✅ |
|
|
|
|
**Data sources opened at runtime** (strace-confirmed):
|
|
- `/sys/class/dmi/id/sys_vendor`, `product_name`, `product_family`,
|
|
`product_version`, `board_vendor`, `board_name`, `board_version`,
|
|
`board_serial`, `board_asset_tag`, `bios_vendor`, `bios_version`,
|
|
`bios_date`, `bios_release`, `chassis_vendor`, `chassis_type`,
|
|
`chassis_version`, `chassis_asset_tag` (all 18 fields)
|
|
|
|
**Linux host smoke test** (Manjaro, MSI MPG X670E CARBON WIFI):
|
|
```
|
|
Manufacturer: Micro-Star International Co., Ltd. (System + Board)
|
|
Product: MS-7D70
|
|
Name: MPG X670E CARBON WIFI (MS-7D70)
|
|
BIOS Vendor: American Megatrends International, LLC.
|
|
BIOS Version: 1.74 BIOS Date: 05/12/2023 BIOS Release: 5.26
|
|
Chassis Type: 3 (Desktop)
|
|
```
|
|
|
|
`product_serial` and `product_uuid` correctly report `?` (root-only
|
|
readable; redbear-power runs unprivileged).
|
|
|
|
**v1.5 source state**: 4117 LoC across **14 modules** (was 3864/13 in
|
|
v1.4). New module: `dmi.rs` (118 lines).
|
|
|
|
Cross-compiled binary: 3.7 MB stripped Redox ELF
|
|
(SHA256 `c44d508cf6fefa28134b9f9c0b3493a34ddbff4028328c88ff30ac23bd14f2e8`).
|
|
|
|
**Forward work on Redox target** (deferred to v1.6+):
|
|
1. **SMBIOS table parser in kernel** — read the SMBIOS EPS, walk the
|
|
structure table, parse Type 0/1/2/3 records.
|
|
2. **`scheme:dmi` userspace daemon** — exposes parsed SMBIOS records
|
|
via `/scheme/dmi/{board_vendor,bios_vendor,...}` (mirrors sysfs).
|
|
3. **redbear-power fallback** — `DmiInfo::read()` tries Redox scheme
|
|
first, then `/sys/class/dmi/id/` (Linux host).
|
|
|
|
Until then, the Motherboard panel on Redox honestly reports empty data
|
|
(rather than fake values) — per the zero-stub policy.
|
|
|
|
#### v1.6 Battery Tab (2026-06-20)
|
|
|
|
Per the user's "v1.6 = Battery tab (Recommended)" directive, v1.6
|
|
ships the **Battery tab** as the 5th tab in the multi-view system.
|
|
|
|
| Item | Status |
|
|
|------|--------|
|
|
| `battery.rs` (NEW, 128 LoC) — `BatteryInfo` with 15 fields + per-field sysfs read | ✅ |
|
|
| `TabId::Battery` variant + cycle order (`PerCpu → System → Info → Motherboard → Battery`) | ✅ |
|
|
| Hotkey `5` jumps directly to Battery tab | ✅ |
|
|
| `render_battery_panel()` with 3 sections: Identity / State / Power | ✅ |
|
|
| `RBP_BATTERY_PATH` env override (useful for testing + dev workflow) | ✅ |
|
|
| `render_once` now dumps Battery panel for headless verification | ✅ |
|
|
|
|
**Data sources opened at runtime** (when battery present):
|
|
- `/sys/class/power_supply/BAT0/type` — device type filter
|
|
- `status`, `capacity`, `energy_now`, `energy_full`, `power_now`,
|
|
`voltage_now`, `time_to_empty`, `time_to_full`, `cycle_count`,
|
|
`technology`, `model_name`, `manufacturer`, `serial_number` (13 fields)
|
|
|
|
**Mock battery smoke test** (`RBP_BATTERY_PATH=/tmp/fake-battery`):
|
|
```
|
|
Manufacturer: MSI Technology: Li-ion Cycles: 127
|
|
Status: Discharging Capacity: 67% Health: 67%
|
|
Energy: 33.50 Wh / 50.00 Wh
|
|
Power: 8.50 W Voltage: 12.50 V
|
|
Time to empty: 3h 0m Time to full: ?
|
|
```
|
|
|
|
µWh → Wh conversion verified (33,500,000 µWh → 33.50 Wh); µV → V
|
|
conversion verified (12,500,000 µV → 12.50 V); `time_to_full=0`
|
|
correctly hidden.
|
|
|
|
**On actual desktop host** (no battery):
|
|
```
|
|
(no battery detected — /sys/class/power_supply/BAT* not present)
|
|
```
|
|
|
|
**v1.6 source state**: 4359 LoC across **15 modules** (was 4117/14 in
|
|
v1.5). New module: `battery.rs` (128 lines).
|
|
|
|
Cross-compiled binary: 3.8 MB stripped Redox ELF
|
|
(SHA256 `c6fca1728faff9edd053b933f0c57075e25dfe52450b7ab604d04d5024b1cc88`).
|
|
|
|
**Forward work on Redox target** (deferred to v1.7+):
|
|
1. **`power_supply` scheme daemon** — exposes battery state via
|
|
`/scheme/power_supply/BAT0/{status,capacity,...}` (mirrors sysfs).
|
|
2. **ACPI battery object parser** — read `_BST` (battery status) and
|
|
`_BIF` (battery information) AML methods; convert to `BatteryInfo`
|
|
fields.
|
|
3. **Per-tick refresh** — battery state should be polled at 2-5 Hz,
|
|
not read once at startup. Move `BatteryInfo::read()` into
|
|
`App::refresh()` with 5-tick throttling.
|
|
4. **redbear-power fallback** — `find_battery_dir()` tries Redox
|
|
scheme first, then `/sys/class/power_supply/` (Linux host).
|
|
|
|
Until then, the Battery panel on Redox honestly reports empty data
|
|
(rather than fake values) — per the zero-stub policy.
|
|
|
|
#### v1.7 Per-Tick Battery Refresh (2026-06-20)
|
|
|
|
Per the user's "v1.7 = Per-tick battery refresh (Recommended)"
|
|
directive, v1.7 closes the v1.6 forward-work item.
|
|
|
|
| Item | Status |
|
|
|------|--------|
|
|
| Move `BatteryInfo::read()` into `App::refresh()` with 5-tick throttling | ✅ |
|
|
| Reuse `refresh_counter` (no new field) | ✅ |
|
|
| Cadence: 2.5 sec at default POLL_MS=500 (0.04% CPU cost) | ✅ |
|
|
| Independent of meminfo cadence (5th tick vs 4th tick — coprime) | ✅ |
|
|
| `find_battery_dir()` re-checks on each refresh (laptop plugged-in mid-session works) | ✅ |
|
|
|
|
**Verification**: Mock battery at `/tmp/fake-battery/BAT0/` with
|
|
`capacity=67`. After `RBP_BATTERY_PATH=/tmp/fake-battery --once`, the
|
|
panel showed `67%`. Changed capacity file to `50`, re-ran `--once`,
|
|
panel showed `50%`. The 5-tick throttling fires on the first refresh
|
|
(`refresh_counter=0`, `0 % 5 == 0`), so `--once` mode picks up the
|
|
current value.
|
|
|
|
**Build verification**: Same 3.8 MB stripped Redox binary size as v1.6
|
|
(single `if` branch added to `App::refresh()`). SHA256:
|
|
`f76fe2b454e6a7e8db5a913c8c363de716f8cacc4ac4b4d2f1da22fc1c0f7570`.
|
|
|
|
**Cadence rationale**: 5-tick modulus deliberately coprime to meminfo's
|
|
4-tick modulus. With coprime moduli, battery and meminfo refreshes don't
|
|
synchronize — no thundering-herd of 14+4 sysfs reads at the same moment
|
|
(which would be visible to the user as a periodic 20ms stall).
|
|
|
|
#### v1.8 Bench Stress Modes (2026-06-20)
|
|
|
|
Per the user's "v1.8 = Bench stress modes (Recommended)" directive,
|
|
v1.8 extends `bench.rs` from a single prime-sieve to a 3-mode suite.
|
|
|
|
| Item | Status |
|
|
|------|--------|
|
|
| `BenchKind` enum: `PrimeSieve` / `Fft` / `Aes` | ✅ |
|
|
| `prime_worker()` — extracted from inline v1.0 loop | ✅ |
|
|
| `fft_worker()` — radix-2 Cooley-Tukey FFT on 1024-element f64 buffers | ✅ |
|
|
| `aes_worker()` — software AES-128 with FIPS-197 test vector | ✅ |
|
|
| `Bench::single_core` toggle (1 thread vs all cores) | ✅ |
|
|
| Hotkey `n` — cycle benchmark kind | ✅ |
|
|
| Hotkey `s` — toggle single-core vs all-cores | ✅ |
|
|
| 5 unit tests (`cargo test --release`) | ✅ all pass |
|
|
| Help text updated (controls panel + long help) | ✅ |
|
|
|
|
**Workload characteristics**:
|
|
| Mode | Characteristic | Cores for full saturation |
|
|
|------|----------------|---------------------------|
|
|
| Prime sieve | Branch-heavy, low IPC | All (limited by cache) |
|
|
| FFT | Memory-bound, SIMD-friendly | All (limited by memory bandwidth) |
|
|
| AES-128 | Pure-compute, integer-heavy | All (limited by ALU) |
|
|
|
|
**Use cases**:
|
|
- AES single-core vs multi-core ratio shows scaling efficiency
|
|
(24x on independent cores, 8x on shared FSB).
|
|
- FFT multi-core stress = memory subsystem + cache hierarchy test.
|
|
- Prime sieve = fast thermal load (heats up quickest).
|
|
|
|
**v1.8 source state**: bench.rs 123 → 304 lines (+181). Total project
|
|
unchanged at ~4,380 LoC across 15 modules (v1.7).
|
|
|
|
Cross-compiled binary: 3.8 MB stripped Redox ELF
|
|
(SHA256 `a9892e716f1b93a36e8c5832c68ba31c10036c0c51e3911386e8b8d3ed1fe2b6`).
|
|
|
|
**Forward work** (deferred to v1.9+):
|
|
1. **AES-NI / AVX-512 intrinsics** — replace scalar AES with
|
|
hardware-accelerated instructions when `is_x86_feature_detected!`
|
|
returns true.
|
|
2. **Result history** — circular buffer of last N runs in System tab.
|
|
3. **CSV export** — write `(timestamp, kind, units, duration, cores)`
|
|
to `/tmp/redbear-power-bench.csv`.
|
|
|
|
#### v1.9 Sensors Tab (hwmon) (2026-06-20)
|
|
|
|
Per the user's "v1.9 = Sensor tab (hwmon) (Recommended)" directive,
|
|
v1.9 ships the **Sensors tab** as the 6th tab in the multi-view system.
|
|
|
|
| Item | Status |
|
|
|------|--------|
|
|
| `sensor.rs` (NEW, 231 LoC) — `SensorKind` + `SensorReading` + `HwmonChip` + `SensorInfo` | ✅ |
|
|
| `TabId::Sensors` variant + cycle order | ✅ |
|
|
| Hotkey `6` jumps to Sensors tab | ✅ |
|
|
| `render_sensor_panel()` with per-chip sections + Label/Value pairs | ✅ |
|
|
| Unit conversion: m°C → °C, mV → V, µW → W, mA → A, RPM as int | ✅ |
|
|
| Per-tick refresh at 3-tick modulus (1.5 sec cadence) | ✅ |
|
|
| 7 unit tests covering unit conversions + empty state | ✅ all pass |
|
|
| 12 total tests (5 bench + 7 sensor) | ✅ all pass |
|
|
|
|
**Data sources opened at runtime** (when hwmon present):
|
|
- `/sys/class/hwmon/hwmonN/name` — chip identifier
|
|
- `temp*_input` (m°C), `temp*_label`
|
|
- `fan*_input` (RPM), `fan*_label`
|
|
- `in*_input` (mV), `in*_label`
|
|
- `power*_input` (µW), `power*_label`
|
|
- `curr*_input` (mA), `curr*_label`
|
|
|
|
**Linux host smoke test** (Manjaro, Ryzen 9 7900X):
|
|
```
|
|
Detected 7 chip(s), 11 sensor(s) total:
|
|
▸ mt7921_phy0 temp 58.0 °C
|
|
▸ r8169_0_e00:00 temp 51.0 °C
|
|
▸ k10temp Tccd1 82.6 °C
|
|
Tccd2 57.1 °C
|
|
Tctl 85.6 °C
|
|
▸ nvme Sensor 2 53.9 °C
|
|
Composite 50.9 °C
|
|
Sensor 1 50.9 °C
|
|
▸ spd5118 temp 50.0 °C
|
|
▸ spd5118 temp 51.5 °C
|
|
▸ nvme Composite 48.9 °C
|
|
```
|
|
|
|
**v1.9 source state**: 4885 LoC across **16 modules** (was ~4562/15 in
|
|
v1.8). New module: `sensor.rs` (231 lines). 12 unit tests total.
|
|
|
|
Cross-compiled binary: 3.8 MB stripped Redox ELF
|
|
(SHA256 `7a7c31bcf3577c99a72291c46d34e5d2d52951c1e78ee5d216760f41f623234b`).
|
|
|
|
**Refresh cadence**: sensor uses 3-tick modulus (1.5 sec at POLL_MS=500).
|
|
Combined with meminfo's 4-tick and battery's 5-tick moduli, all three
|
|
coprime — no two expensive sysfs reads ever fire in the same tick.
|
|
|
|
**Forward work on Redox target** (deferred to v1.10+):
|
|
1. **`hwmon` scheme daemon** in `redox-driver-sys` — exposes parsed
|
|
sensor data via `/scheme/hwmon/<chip>/{...}`.
|
|
2. **Chip drivers** — k10temp, coretemp, nvme, etc. need user-space
|
|
drivers feeding the scheme daemon.
|
|
3. **Per-CPU temperature integration** — map k10temp's `Tctl` to the
|
|
`Pkg` column of selected CPU. Currently hwmon only exposes
|
|
package-level temps, not per-core.
|
|
4. **redbear-power fallback** — `SensorInfo::read()` tries Redox scheme
|
|
first, then `/sys/class/hwmon/` (Linux host).
|
|
|
|
Until then, the Sensors panel on Redox honestly reports empty data
|
|
(rather than fake values) — per the zero-stub policy.
|
|
|
|
#### v1.10 Per-CPU Pkg Temp from hwmon (2026-06-20)
|
|
|
|
Per the user's "v1.10 = Per-CPU Pkg temp from hwmon (Recommended)"
|
|
directive, v1.10 closes the v1.9 forward-work item. The Per-CPU table's
|
|
`Temp°C` column previously showed `n/a` for AMD CPUs because
|
|
`IA32_THERM_STATUS` is an Intel-only MSR. v1.10 falls back to hwmon
|
|
when the MSR is unavailable.
|
|
|
|
| Item | Status |
|
|
|------|--------|
|
|
| `SensorInfo::pkg_temp_c(cpu_index)` helper | ✅ |
|
|
| Recognizes k10temp Tctl (AMD Zen / Zen 2 / Zen 3 / Zen 4 / Zen 5) | ✅ |
|
|
| Recognizes coretemp "Package id 0" (Intel, forward-compat) | ✅ |
|
|
| Recognizes zenpower Tdie (AMD alt driver) | ✅ |
|
|
| Falls back from `IA32_THERM_STATUS` MSR to hwmon pkg_temp_c | ✅ |
|
|
| 5 new unit tests (k10temp / coretemp / zenpower / none / unrelated) | ✅ all pass |
|
|
| 17 total tests (5 bench + 12 sensor) | ✅ all pass |
|
|
|
|
**Linux host smoke test** (AMD Ryzen 9 7900X):
|
|
- Before v1.10: every CCD row showed `n/a` for Temp°C.
|
|
- After v1.10: every CCD row shows `85` (k10temp Tctl value, °C).
|
|
|
|
**Implementation pattern** in `App::refresh()`:
|
|
```rust
|
|
} else {
|
|
// IA32_THERM_STATUS is Intel-only. On AMD, fall back to
|
|
// k10temp Tctl (the package control temperature), which
|
|
// applies to all CPUs on the same package.
|
|
row.temp_c = self.sensors.pkg_temp_c(row.id);
|
|
row.prochot = false; // k10temp doesn't expose PROCHOT
|
|
row.critical = false; // k10temp doesn't expose Critical
|
|
row.power_limit = false; // k10temp doesn't expose Power Limit
|
|
}
|
|
```
|
|
|
|
PROCHOT/Critical/PowerLimit flags are set to false in the fallback
|
|
path because k10temp doesn't expose these — only the temperature.
|
|
Honest empty-state pattern: don't fake flag values.
|
|
|
|
**Cross-compile SHA256**: `d40277c75b2ca913a6df9b067c457493b5f01b2c0da8baa14bba604e619f5ea5`
|
|
|
|
**Forward work** (deferred to v1.11+):
|
|
1. **Per-CCD temperature** — k10temp exposes `Tccd1`, `Tccd2`, etc.
|
|
for each CCD cluster. Map these to per-CPU rows using cpuid
|
|
leaf 0x8000001E NC field (already in v1.2 cpuid.rs).
|
|
2. **Multi-socket support** — the `cpu_index` parameter in
|
|
`pkg_temp_c` is currently ignored. On 2-socket systems, there
|
|
are 2 k10temp chips. Future work: detect by `phys_pkg_id` from
|
|
cpuid and route to the correct chip.
|
|
3. **PROCHOT on AMD** — k10temp exposes `temp*_max` / `temp*_crit`
|
|
thresholds. Future work: surface "approaching critical" warnings
|
|
based on those thresholds.
|
|
4. **`hwmon` scheme daemon** on Redox — see v1.9 forward work.
|
|
|
|
#### v1.11 Network Tab (sysfs + if_inet6) (2026-06-20)
|
|
|
|
Per the user's "v1.11 = Network tab (Recommended)" directive, v1.11
|
|
ships the **Network tab** as the 7th tab in the multi-view system.
|
|
|
|
| Item | Status |
|
|
|------|--------|
|
|
| `network.rs` (NEW, 203 LoC) — `NetInterface` + `NetInfo` + IPv6 parser | ✅ |
|
|
| `TabId::Network` variant + cycle order | ✅ |
|
|
| Hotkey `7` jumps to Network tab | ✅ |
|
|
| `render_network_panel()` with State/MAC/MTU/Speed/RX/TX/IPv6 | ✅ |
|
|
| Per-tick refresh at 7-tick modulus (3.5 sec cadence) | ✅ |
|
|
| 7 unit tests (format_bytes + empty state + zero traffic) | ✅ all pass |
|
|
| 24 total tests (5 bench + 12 sensor + 7 network) | ✅ all pass |
|
|
|
|
**Data sources opened at runtime** (when sysfs/net present):
|
|
- `/sys/class/net/<iface>/operstate` — link state (up/down/unknown)
|
|
- `/sys/class/net/<iface>/speed` — link speed in Mbps
|
|
- `/sys/class/net/<iface>/address` — MAC address
|
|
- `/sys/class/net/<iface>/mtu` — MTU
|
|
- `/sys/class/net/<iface>/statistics/{rx,tx}_{bytes,packets,errors,dropped}`
|
|
- `/proc/net/if_inet6` — IPv6 addresses with scope encoding
|
|
|
|
**Linux host smoke test** (Manjaro, 6 interfaces: enp14s0, lo, moscow,
|
|
tailscale0, tun0, wlp13s0):
|
|
- Real link state: enp14s0=down, wlp13s0=up, others=unknown
|
|
- Real traffic stats: lo (686 MiB), wlp13s0 (38/237 GiB), tailscale0 (2/11 GiB)
|
|
- Real IPv6 addresses with correct scope encoding (link for fe80::, compat for fd7a/fd01)
|
|
- MAC shown only when not 00:00:00:00:00:00 (enp14s0, wlp13s0 only)
|
|
- Speed shown only when >0 (tun0=10000 Mbps)
|
|
|
|
**v1.11 source state**: ~5150 LoC across **17 modules** (was ~4945/16 in
|
|
v1.10). New module: `network.rs` (203 lines). 24 unit tests total.
|
|
|
|
Cross-compiled binary: 3.8 MB stripped Redox ELF
|
|
(SHA256 `05cca57693110e06393273a3247b159b8fc681a8ebc0cdd5a2386f33a1ebb407`).
|
|
|
|
**Refresh cadence**: 7-tick modulus (3.5 sec at POLL_MS=500). Coprime
|
|
with all existing moduli (3, 4, 5) — LCM of {3,4,5,7} = 420 ticks.
|
|
|
|
**Forward work** (deferred to v1.12+):
|
|
1. **Throughput calculation** — compute `rx_kbps` / `tx_kbps` from
|
|
delta of `rx_bytes` / `tx_bytes` over time.
|
|
2. **IPv4 addresses** — currently only IPv6. IPv4 requires parsing
|
|
`/proc/net/fib_trie` or shelling out to `ip addr`.
|
|
3. **ethtool stats** — driver-specific counters via
|
|
`/sys/class/net/<iface>/{statistics,*}` beyond the standard set.
|
|
4. **Network namespace detection** — `netns` info for containers.
|
|
|
|
#### v1.12 Storage Tab (sysfs) (2026-06-20)
|
|
|
|
Per the user's "v1.12 = Storage tab (Recommended)" directive, v1.12
|
|
ships the **Storage tab** as the 8th tab. Completes major hardware
|
|
surface coverage: Per-CPU / System / Info / Motherboard / Battery /
|
|
Sensors / Network / Storage.
|
|
|
|
| Item | Status |
|
|
|------|--------|
|
|
| `storage.rs` (NEW, 261 LoC) — `DiskInfo` + `DiskStats` + kind heuristic | ✅ |
|
|
| `TabId::Storage` variant + cycle order | ✅ |
|
|
| Hotkey `8` jumps to Storage tab | ✅ |
|
|
| `render_storage_panel()` with Model/Size/Scheduler/Queue/R+W/Parts | ✅ |
|
|
| Per-tick refresh at 11-tick modulus (5.5 sec cadence) | ✅ |
|
|
| 10 unit tests (size + parse + delta + kind_label) | ✅ all pass |
|
|
| 34 total tests (5 bench + 12 sensor + 7 network + 10 storage) | ✅ all pass |
|
|
|
|
**Data sources opened at runtime** (when sysfs/block present):
|
|
- `/sys/block/<dev>/device/{model,vendor}` — disk identity
|
|
- `/sys/block/<dev>/size` — size in 512-byte sectors
|
|
- `/sys/block/<dev>/queue/{rotational,scheduler,nr_requests}` — IO
|
|
- `/sys/block/<dev>/removable` — removable flag
|
|
- `/sys/block/<dev>/stat` — 15-field IO statistics (single line)
|
|
- `/sys/block/<dev>/<partition>` — partitions (auto-discovered)
|
|
|
|
**Linux host smoke test** (3 disks: 2 NVMe SSD + 1 USB):
|
|
- nvme0n1: ADATA SX6000PNP, 476.9 GiB, 2 partitions, 15/25 GiB R/W
|
|
- nvme1n1: Samsung SSD 990 PRO 2TB, 1.8 TiB, 3 partitions, 30 MiB R
|
|
- sdb: USB DISK 3.0, 57.7 GiB (Removable detected), 2 partitions
|
|
|
|
**v1.12 source state**: ~5415 LoC across **18 modules** (was ~5150/17
|
|
in v1.11). New module: `storage.rs` (261 lines). 34 unit tests total.
|
|
|
|
Cross-compiled binary: 3.8 MB stripped Redox ELF
|
|
(SHA256 `3c44a545bb162abc7e671d689f025f01a424ee1508a2c2bd90af58f504b50ac4`).
|
|
|
|
**Refresh cadence**: 11-tick modulus (5.5 sec). Coprime with all
|
|
existing moduli (3, 4, 5, 7) — LCM of {3,4,5,7,11} = 9240 ticks.
|
|
|
|
**Forward work** (deferred to v1.13+):
|
|
1. **Throughput calculation** — `DiskStats::kbps_delta()` implemented
|
|
but not wired. Store previous stats in App + add "R: 1.5 MiB/s" line.
|
|
2. **SMART data** — read via `smartctl --json` if available. Skip
|
|
if not (per zero-stub policy).
|
|
3. **NVMe-specific stats** — `nvme*/queue/*` + cross-ref with hwmon.
|
|
4. **Disk temperature** — link hwmon temp to storage panel.
|
|
|
|
### 3.4 D-Bus
|
|
|
|
| Component | Status | Detail |
|
|
|-----------|--------|--------|
|
|
| dbus 1.16.2 | 🟢 Builds | **System bus socket path fix applied (2026-06-19)**: `-Dsystem_socket=/run/dbus/system_bus_socket` baked into `dbus-1.pc` at compile time |
|
|
| redbear-sessiond | 🟢 Builds | login1-compatible session broker; retry loops tuned (3 attempts, 1s) |
|
|
| redbear-upower | 🟢 Builds | UPower surface; retry loops tuned |
|
|
| redbear-polkit | 🟢 Builds | PolicyKit bridge; retry loops tuned |
|
|
| redbear-udisks | 🟢 Builds | UDisks2 service; retry loops tuned |
|
|
| redbear-dbus-services | 🟢 Builds | `.service` files + XML policies |
|
|
|
|
**System bus socket path (durable fix)**: dbus-1.16.2's `meson.build:946` defaults
|
|
`system_bus_socket` to `{prefix}/{runstatedir}/dbus/system_bus_socket`, which
|
|
under Redox resolves to `/usr/var/run/dbus/system_bus_socket` (not `/run/...`).
|
|
All Red Bear OS D-Bus clients hardcode `/run/dbus/system_bus_socket` to match
|
|
the `/run/dbus/system_bus_socket` directory created by `redbear-mini.toml`'s
|
|
postinstall. The fix is in `local/recipes/system/dbus/recipe.toml` mesonflags:
|
|
|
|
```toml
|
|
"-Druntime_dir=/run",
|
|
"-Dsystem_socket=/run/dbus/system_bus_socket",
|
|
```
|
|
|
|
This bakes the correct value into `dbus-1.pc`'s
|
|
`system_bus_default_address=unix:path=/run/dbus/system_bus_socket`, so any
|
|
client using the dbus-1 pkg-config metadata gets the correct path with no
|
|
runtime env-var. The `DBUS_SYSTEM_BUS_ADDRESS` env var in `12_dbus.service`
|
|
is kept as defense-in-depth for the daemon.
|
|
|
|
**Retry loops**: Reduced from 5 attempts / 2 s to 3 attempts / 1 s in
|
|
`redbear-sessiond`, `redbear-upower`, `redbear-polkit`, and `redbear-udisks`
|
|
(four services). Original values were D-Bus startup-friendly but too slow
|
|
on Redox where the bus becomes available quickly after daemon start.
|
|
|
|
**Known issue**: `dbus-daemon --system` fails user lookup for `messagebus` user in some runtime configurations.
|
|
|
|
### 3.5 Qt6 / KF6 / KDE Plasma
|
|
|
|
#### Qt6
|
|
|
|
| Component | Status |
|
|
|-----------|--------|
|
|
| qtbase 6.11.0 (Core+Gui+Widgets+DBus+Wayland) | 🟢 Builds — 7 libs + 12 plugins (target: 6.11.1) |
|
|
| qtdeclarative | 🟢 Builds — QML interpreter-only (`-DQT_FEATURE_qml_jit=OFF`); 86 QtQuick + 83 QtQml headers staged; 106 `libQt6{Qml,Quick}*.so` libs present |
|
|
| qtwayland | 🟢 Builds — Wayland QPA plugin |
|
|
| qtsvg | 🟢 Builds |
|
|
| Qt6::Sensors | 🟡 Builds (dummy backend, 520KB pkgar) |
|
|
| QtNetwork | 🟢 Re-enabled — DNS resolver hardened |
|
|
|
|
#### KF6 Frameworks — 36 build, 12 blocked
|
|
|
|
**Building (36 packages):**
|
|
`karchive`, `kauth`, `kbookmarks`, `kcodecs`, `kcolorscheme`, `kcompletion`, `kconfig`,
|
|
`kconfigwidgets`, `kcoreaddons`, `kcrash`, `kdbusaddons`, `kdeclarative`, `kded6`,
|
|
`kglobalaccel`, `kguiaddons`, `ki18n`, `kiconthemes`, `kidletime`, `kio`, `kitemmodels`,
|
|
`kitemviews`, `kjobwidgets`, `knotifications`, `kpackage`, `kservice`, `ktextwidgets`,
|
|
`kwayland`, `kwidgetsaddons`, `kwindowsystem`, `kxmlgui`, `solid`, `sonnet`,
|
|
`kcmutils`, `attica`, `kdecoration`, `kglobalacceld`
|
|
|
|
**Blocked (12 packages):**
|
|
| Package | Reason |
|
|
|---------|--------|
|
|
| kirigami | Qt6 Wayland null+8 crash prevents runtime QML; headers/libs DO exist |
|
|
| plasma-framework | Depends on kirigami (runtime validation, not build) |
|
|
| plasma-workspace | Depends on kf6-knewstuff payload + real kwin |
|
|
| plasma-desktop | Transitive — depends on plasma-workspace |
|
|
| kf6-knewstuff | Empty package — cmake succeeds but core source produces no libs with QtQuick off |
|
|
| breeze | Build issues |
|
|
| kde-cli-tools | Build issues |
|
|
| kf6-prison | Source issues |
|
|
| kf6-kwallet | QML/GPG disabled; not in current enabled subset |
|
|
| kf6-purpose | Not attempted |
|
|
| kf6-frameworkintegration | Not attempted |
|
|
| kf6-krunner | Not attempted |
|
|
|
|
#### KWin / Plasma Session
|
|
|
|
| Component | Status |
|
|
|-----------|--------|
|
|
| kwin | 🔴 Blocked — real cmake build requires working Qt6 Wayland runtime (null+8 crash); redbear-compositor provides the kwin_wayland binary as a separate package |
|
|
| kwin real build | 🔄 Attempted — gated on Qt6 Wayland null+8 crash resolution |
|
|
| plasma-workspace | 🔴 Blocked |
|
|
| plasma-desktop | 🔴 Blocked (transitive) |
|
|
| Full Plasma session | 🔴 Not functional |
|
|
|
|
**CORRECTION — The "QML JIT gate" was FALSE (v5.0, 2026-06-20)**:
|
|
|
|
Previous versions of this document (v4.x) claimed QQuickWindow/QQmlEngine headers were
|
|
"unavailable" and that QML JIT was "the single biggest desktop blocker." This was
|
|
**factually wrong**. Agent-verified findings (2026-06-20):
|
|
|
|
- **86 QtQuick header files** exist in the sysroot (`include/QtQuick/`)
|
|
- **83 QtQml header files** exist in the sysroot (`include/QtQml/`)
|
|
- **106 `libQt6{Qml,Quick}*.so` shared libraries** are staged
|
|
- `qtdeclarative` BUILDS successfully with `-DQT_FEATURE_qml_jit=OFF` (interpreter-only QML)
|
|
- QML interpreter-only mode works — JIT is an optimization, not a requirement
|
|
|
|
**The REAL desktop blocker is the Qt6 Wayland null+8 crash** (see §3.1). Qt6 Wayland clients
|
|
segfault in `wl_proxy_add_listener()` when the compositor returns NULL for an unsupported
|
|
interface. A candidate fix (`qtwaylandscanner-null-guard-listeners.patch`) is wired into
|
|
qtwayland's recipe but has never been runtime-validated. This crash prevents SDDM greeter,
|
|
KWin, and all Qt6 Wayland clients from running — but it is NOT a QML/JIT issue.
|
|
|
|
Resolving this crash unblocks: kirigami (runtime), plasma-framework, KWin real build,
|
|
SDDM greeter, and the entire Qt6 Wayland client surface.
|
|
|
|
### 3.6 Version Targets — "Latest Upstream" (v5.0, 2026-06-20)
|
|
|
|
User mandate: "kde, qt, wayland - all must be latest versions!"
|
|
|
|
| Component | Current | Target | Status |
|
|
|-----------|---------|--------|--------|
|
|
| Qt6 (qtbase) | 6.11.0 | **6.11.1** | Minor bump needed |
|
|
| SDDM | 0.21.0 | **0.21.0** | ✅ Already latest stable |
|
|
| KDE Plasma | (not built) | **6.7.0** (2026-06-11) | Future target |
|
|
| KDE Frameworks 6 (KF6) | various | **6.27.0** (2026-06-05) | Update recipe revs |
|
|
| ECM | various | **6.27.0** (2026-06-02) | Update recipe revs |
|
|
| libwayland | 1.24.0 | **1.25.0** (2026-03-19) | Bump needed |
|
|
| wayland-protocols | (current) | **1.49** (2026-06-07) | Update |
|
|
| plasma-wayland-protocols | (current) | **1.21.0** | Update |
|
|
|
|
---
|
|
|
|
## 4. Network & Wireless
|
|
|
|
### 4.1 Wired Networking — Working
|
|
|
|
- Native Redox net stack present (`pcid-spawner` → NIC daemon → `smolnetd`/`dhcpd`/`netcfg`)
|
|
- `redbear-netctl` native command shipped
|
|
- RTL8125 autoload wired through Realtek path
|
|
- VirtIO networking in QEMU: `DBUS_SYSTEM_BUS=present`
|
|
|
|
### 4.2 Wi-Fi — Host-tested, no hardware
|
|
|
|
- Intel PCIe transport builds, 119 tests
|
|
- LinuxKPI compat with 17 modules, 93 tests
|
|
- `redbear-wifictl` daemon + scheme interface
|
|
- Bounded host-tested scan/connect/disconnect/profile flows
|
|
- **Blocker**: No Intel hardware available; MediaTek MT7921K on current host
|
|
|
|
### 4.3 Bluetooth — Experimental BLE-first
|
|
|
|
- Controller probe via USB, HCI init, `scheme:hciN`
|
|
- GATT client workflow (discover→read), 209 tests
|
|
- QEMU validation in progress
|
|
|
|
---
|
|
|
|
## 5. Honest Blocker Map
|
|
|
|
### Critical Path (ordered)
|
|
|
|
```
|
|
[0] Wire SDDM + pam-redbear into config → login prompt target
|
|
[1] Qt6 Wayland null+8 crash resolution → unblocks ALL Qt6 Wayland clients
|
|
[2] SDDM greeter renders under compositor → SDDM login prompt in QEMU
|
|
[3] Real KWin build → unblocks plasma-workspace → plasma-desktop
|
|
[4] Mesa virgl runtime wiring + QEMU -virtio-vga-gl → GPU-accelerated SDDM
|
|
[5] Hardware GPU validation → unblocks Mesa HW renderers
|
|
[6] ACPI shutdown robustness → release-grade ACPI
|
|
[7] Bare-metal validation → unblocks all hardware claims
|
|
```
|
|
|
|
### Blocker Detail
|
|
|
|
| # | Blocker | What's needed | Estimated effort | Hardware required |
|
|
|---|---------|---------------|-----------------|-------------------|
|
|
| 0 | SDDM config wiring | Add sddm + pam-redbear to redbear-full.toml; create init service; configure SDDM compositor path | 2-3 days | No |
|
|
| 1 | Qt6 Wayland null+8 crash | Rebuild libwayland→qtbase→qtdeclarative→qtwayland with null-guard patch; validate QML window renders under redbear-compositor | 1-2 weeks | No |
|
|
| 2 | SDDM greeter runtime | SDDM greeter (QML) launches as Wayland client of redbear-compositor; user sees login prompt | 1 week | No |
|
|
| 3 | Mesa virgl runtime | Wire missing patches into recipe.toml; test with `-device virtio-vga-gl` | 3-5 days | No (QEMU) |
|
|
| 4 | KWin real build | Real cmake build of KWin; requires Qt6 Wayland runtime working | 2-4 weeks | No |
|
|
| 5 | Plasma session | plasma-workspace + plasma-desktop cmake builds; requires kirigami + kwin | 2-4 weeks | No |
|
|
| 6 | HW GPU backend | CS ioctl implementation → Mesa HW renderer cross-compile | 12-20 weeks | Yes — AMD/Intel GPU |
|
|
| 7 | ACPI shutdown | Remove panic paths, deterministic `_S5` | 2-4 weeks | No |
|
|
| 8 | Bare-metal proof | Real AMD/Intel hardware validation for all layers | 4-8 weeks | Yes — AMD + Intel machines |
|
|
|
|
### Path to SDDM Login Prompt (Blocks 0-2) — IMMEDIATE TARGET
|
|
|
|
```
|
|
Wire SDDM config (2-3d) → Resolve null+8 crash (1-2w) → SDDM greeter renders (1w)
|
|
↓
|
|
SDDM login prompt in QEMU
|
|
Total: 2-3 weeks
|
|
With virgl: + 3-5 days
|
|
```
|
|
|
|
### Path to Software-Rendered KDE Plasma (Blocks 0-5)
|
|
|
|
```
|
|
SDDM login (2-3w) → KWin real build (2-4w) → Plasma session (2-4w)
|
|
↓
|
|
Software-rendered KDE Plasma on Wayland
|
|
Total: 6-11 weeks (from current state)
|
|
```
|
|
|
|
### Path to Hardware-Accelerated KDE Plasma (Blocks 0-8)
|
|
|
|
```
|
|
Software-rendered path (6-11w)
|
|
+ virgl runtime wiring (3-5d, QEMU proof)
|
|
+ GPU CS ioctl backend + Mesa HW cross-compile (12-20w, parallel)
|
|
+ Hardware validation (4-8w, parallel)
|
|
↓
|
|
Hardware-accelerated KDE Plasma on Wayland
|
|
Total: 18-31 weeks
|
|
```
|
|
|
|
---
|
|
|
|
## 6. What Changed Since v3.0 (2026-04-29 → 2026-04-30)
|
|
|
|
| Change | Impact |
|
|
|--------|--------|
|
|
| Credential syscalls implemented | `setgroups`/`getgroups`/`initgroups`/RLIMIT functional. Unblocks polkit, dbus, logind, sudo. |
|
|
| Kernel groups process-scoped | `setgroups()` propagates to all process threads. NGROUPS_MAX enforced. |
|
|
| `CallerCtx.groups` added | Schemes can now check supplementary group membership for access control. |
|
|
| Kernel readback for `getgroups` | Cache is repopulated from kernel after exec/crash. |
|
|
| `setrlimit` returns proper errors | EINVAL for unknown resources, EPERM for process limits. |
|
|
|
|
---
|
|
|
|
## 7. Configuration Surface
|
|
|
|
`config/redbear-full.toml` enables the desktop-capable target:
|
|
- 36 KDE packages (33 kf6-* + kdecoration + kglobalacceld + kwin); 12 blocked
|
|
- mesa + libdrm (software GPU stack, swrast + virgl — virgl runtime patch wiring PENDING)
|
|
- qtbase + qtdeclarative + qtwayland + qtsvg + qt6-wayland-smoke
|
|
- seatd + redbear-authd + redbear-session-launch + redbear-greeter (legacy)
|
|
- dbus + firmware-loader + redox-drm + evdevd + udev-shim
|
|
- redbear-compositor (real Rust Wayland compositor)
|
|
- SDDM v0.21.0 + pam-redbear — WIRED (v5.3): `[packages]` enabled, `21_sddm.service` wired, `/etc/sddm.conf` installed
|
|
- plus inherited packages from redbear-mini profile
|
|
|
|
---
|
|
|
|
## 8. Evidence Model
|
|
|
|
| Evidence Class | What It Means |
|
|
|----------------|---------------|
|
|
| **Source** | Code exists in tree |
|
|
| **Host build-verified** | `cargo check` zero warnings on Linux |
|
|
| **Redox build-verified** | `make r.*` successful on `x86_64-unknown-redox` |
|
|
| **Runtime-validated** | Exercised in QEMU |
|
|
| **Hardware-validated** | Exercised on real AMD/Intel hardware |
|
|
|
|
**Current highest evidence bar reached**: QEMU runtime proof for greeter/login, bounded compositor,
|
|
D-Bus system bus, evdevd/udev-shim, DRM scheme enumeration.
|
|
|
|
**No component has hardware validation.** All hardware claims remain evidence-qualified.
|
|
|
|
---
|
|
|
|
## 9. Subsystem Plans (Reference)
|
|
|
|
This document is the authority. Subsystem plans remain for deep-dive detail:
|
|
|
|
| Plan | Covers |
|
|
|------|--------|
|
|
| `KERNEL-IPC-CREDENTIAL-PLAN.md` | Kernel credential syscalls, IPC, RLIMIT — Phases K1-K2,K4 complete |
|
|
| `IRQ-AND-LOWLEVEL-CONTROLLERS-ENHANCEMENT-PLAN.md` | PCI/IRQ/MSI-X/IOMMU quality |
|
|
| `ACPI-IMPROVEMENT-PLAN.md` | ACPI shutdown, power, sleep states |
|
|
| `RELIBC-IPC-ASSESSMENT-AND-IMPROVEMENT-PLAN.md` | relibc IPC surface |
|
|
| `DRM-MODERNIZATION-EXECUTION-PLAN.md` | DRM/KMS modernization |
|
|
| `WAYLAND-IMPLEMENTATION-PLAN.md` | Wayland compositor stability |
|
|
| `DBUS-INTEGRATION-PLAN.md` | D-Bus architecture |
|
|
| `GREETER-LOGIN-IMPLEMENTATION-PLAN.md` | Greeter/login design |
|
|
|
|
---
|
|
|
|
## 10. Stale Docs Deleted (This Pass)
|
|
|
|
| File | Reason |
|
|
|------|--------|
|
|
| `COMPREHENSIVE-OS-ASSESSMENT.md` | Consolidated into this document |
|
|
| `DESKTOP-STACK-CURRENT-STATUS.md` | Consolidated into this document |
|
|
| `AMD-FIRST-INTEGRATION.md` | Historical — AMD and Intel are equal-priority targets |
|
|
| `HARDWARE-3D-ASSESSMENT.md` | Historical — consolidated into §2 |
|
|
| `DMA-BUF-IMPROVEMENT-PLAN.md` | Historical — consolidated into §2 |
|
|
| `INPUT-SCHEME-ENHANCEMENT.md` | Historical — consolidated into §3.2 |
|
|
| `BOOT-PROCESS-ASSESSMENT.md` | Historical — consolidated into §1 |
|
|
| `LINUX-BORROWING-RUST-IMPLEMENTATION-PLAN.md` | Historical — consolidated into §2 |
|
|
| `QT6-PORT-STATUS.md` | Historical — consolidated into §3.5 |
|
|
| `REDBEAR-INFO-RUNTIME-REPORT.md` | Historical — validation infrastructure now standard |
|
|
| `RELIBC-COMPREHENSIVE-ASSESSMENT.md` | Historical — consolidated into §1.5 |
|
|
| `RELIBC-COMPLETENESS-AND-ENHANCEMENT-PLAN.md` | Historical — consolidated into §1.5 |
|
|
| `RELIBC-IMPLEMENTATION-PLAN.md` | Historical — consolidated into §1.5 |
|