b9874d0941
Add redbear-usb-storage-check in-guest binary that validates USB mass storage read and write I/O: discovers /scheme/disk/ devices, writes a test pattern to sector 2048, reads it back, verifies match, restores original content. Updates test-usb-storage-qemu.sh with write-proof verification step. Includes all accumulated Red Bear OS work: kernel patches, relibc patches, driver infrastructure, DRM/GPU, KDE recipes, firmware, validation tooling, build system hardening, and documentation.
106 lines
2.9 KiB
Rust
106 lines
2.9 KiB
Rust
use driver_graphics::GraphicsScheme;
|
|
use event::{user_data, EventQueue};
|
|
use pcid_interface::{irq_helpers::pci_allocate_interrupt_vector, PciFunctionHandle};
|
|
use std::{
|
|
io::{Read, Write},
|
|
os::fd::AsRawFd,
|
|
};
|
|
|
|
mod device;
|
|
use self::device::Device;
|
|
|
|
fn main() {
|
|
pcid_interface::pci_daemon(daemon);
|
|
}
|
|
|
|
fn daemon(daemon: daemon::Daemon, mut pcid_handle: PciFunctionHandle) -> ! {
|
|
let pci_config = pcid_handle.config();
|
|
|
|
let mut name = pci_config.func.name();
|
|
name.push_str("_ihdg");
|
|
|
|
common::setup_logging(
|
|
"graphics",
|
|
"pci",
|
|
&name,
|
|
common::output_level(),
|
|
common::file_level(),
|
|
);
|
|
|
|
log::info!("IHDG {}", pci_config.func.display());
|
|
|
|
let device = Device::new(&mut pcid_handle, &pci_config.func)
|
|
.expect("ihdgd: failed to initialize device");
|
|
|
|
let irq_file = pci_allocate_interrupt_vector(&mut pcid_handle, "ihdgd");
|
|
|
|
// Needs to be before GraphicsScheme::new to avoid a deadlock due to initnsmgr blocking on
|
|
// /scheme/event as it is already blocked on opening /scheme/display.ihdg.*.
|
|
// FIXME change the initnsmgr to not block on openat for the target scheme.
|
|
let event_queue: EventQueue<Source> =
|
|
EventQueue::new().expect("ihdgd: failed to create event queue");
|
|
|
|
let mut scheme = GraphicsScheme::new(device, format!("display.ihdg.{}", name), false);
|
|
|
|
user_data! {
|
|
enum Source {
|
|
Input,
|
|
Irq,
|
|
Scheme,
|
|
}
|
|
}
|
|
|
|
event_queue
|
|
.subscribe(
|
|
scheme.inputd_event_handle().as_raw_fd() as usize,
|
|
Source::Input,
|
|
event::EventFlags::READ,
|
|
)
|
|
.unwrap();
|
|
event_queue
|
|
.subscribe(
|
|
irq_file.irq_handle().as_raw_fd() as usize,
|
|
Source::Irq,
|
|
event::EventFlags::READ,
|
|
)
|
|
.unwrap();
|
|
event_queue
|
|
.subscribe(
|
|
scheme.event_handle().raw(),
|
|
Source::Scheme,
|
|
event::EventFlags::READ,
|
|
)
|
|
.unwrap();
|
|
|
|
libredox::call::setrens(0, 0).expect("ihdgd: failed to enter null namespace");
|
|
|
|
daemon.ready();
|
|
|
|
let all = [Source::Input, Source::Irq, Source::Scheme];
|
|
for event in all
|
|
.into_iter()
|
|
.chain(event_queue.map(|e| e.expect("ihdgd: failed to get next event").user_data))
|
|
{
|
|
match event {
|
|
Source::Input => scheme.handle_vt_events(),
|
|
Source::Irq => {
|
|
let mut irq = [0; 8];
|
|
irq_file.irq_handle().read(&mut irq).unwrap();
|
|
if scheme.adapter_mut().handle_irq() {
|
|
irq_file.irq_handle().write(&mut irq).unwrap();
|
|
|
|
scheme.adapter_mut().handle_events();
|
|
scheme.tick().unwrap();
|
|
}
|
|
}
|
|
Source::Scheme => {
|
|
scheme
|
|
.tick()
|
|
.expect("ihdgd: failed to handle scheme events");
|
|
}
|
|
}
|
|
}
|
|
|
|
std::process::exit(0);
|
|
}
|