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.
135 lines
3.8 KiB
Rust
135 lines
3.8 KiB
Rust
use std::io::{Read, Write};
|
|
use std::os::unix::io::AsRawFd;
|
|
use std::usize;
|
|
|
|
use event::{user_data, EventQueue};
|
|
use pcid_interface::PciFunctionHandle;
|
|
use redox_scheme::scheme::register_sync_scheme;
|
|
use redox_scheme::Socket;
|
|
use scheme_utils::ReadinessBased;
|
|
|
|
#[cfg(any(target_arch = "x86", target_arch = "x86_64"))]
|
|
pub mod device;
|
|
|
|
fn main() {
|
|
pcid_interface::pci_daemon(daemon);
|
|
}
|
|
|
|
#[cfg(any(target_arch = "x86", target_arch = "x86_64"))]
|
|
fn daemon(daemon: daemon::Daemon, pcid_handle: PciFunctionHandle) -> ! {
|
|
let pci_config = pcid_handle.config();
|
|
|
|
let mut name = pci_config.func.name();
|
|
name.push_str("_ac97");
|
|
|
|
let bar0 = pci_config.func.bars[0].expect_port();
|
|
let bar1 = pci_config.func.bars[1].expect_port();
|
|
|
|
let irq = pci_config
|
|
.func
|
|
.legacy_interrupt_line
|
|
.expect("ac97d: no legacy interrupts supported");
|
|
|
|
println!(" + ac97 {}", pci_config.func.display());
|
|
|
|
common::setup_logging(
|
|
"audio",
|
|
"pci",
|
|
&name,
|
|
common::output_level(),
|
|
common::file_level(),
|
|
);
|
|
|
|
common::acquire_port_io_rights().expect("ac97d: failed to set I/O privilege level to Ring 3");
|
|
|
|
let mut irq_file = irq.irq_handle("ac97d");
|
|
|
|
let socket = Socket::nonblock().expect("ac97d: failed to create socket");
|
|
let mut device =
|
|
unsafe { device::Ac97::new(bar0, bar1).expect("ac97d: failed to allocate device") };
|
|
let mut readiness_based = ReadinessBased::new(&socket, 16);
|
|
|
|
user_data! {
|
|
enum Source {
|
|
Irq,
|
|
Scheme,
|
|
}
|
|
}
|
|
|
|
let event_queue = EventQueue::<Source>::new().expect("ac97d: Could not create event queue.");
|
|
event_queue
|
|
.subscribe(
|
|
irq_file.as_raw_fd() as usize,
|
|
Source::Irq,
|
|
event::EventFlags::READ,
|
|
)
|
|
.unwrap();
|
|
event_queue
|
|
.subscribe(
|
|
socket.inner().raw(),
|
|
Source::Scheme,
|
|
event::EventFlags::READ,
|
|
)
|
|
.unwrap();
|
|
|
|
register_sync_scheme(&socket, "audiohw", &mut device)
|
|
.expect("ac97d: failed to register audiohw scheme to namespace");
|
|
daemon.ready();
|
|
|
|
libredox::call::setrens(0, 0).expect("ac97d: failed to enter null namespace");
|
|
|
|
let all = [Source::Irq, Source::Scheme];
|
|
for event in all
|
|
.into_iter()
|
|
.chain(event_queue.map(|e| e.expect("ac97d: failed to get next event").user_data))
|
|
{
|
|
match event {
|
|
Source::Irq => {
|
|
let mut irq = [0; 8];
|
|
irq_file.read(&mut irq).unwrap();
|
|
|
|
if !device.irq() {
|
|
continue;
|
|
}
|
|
irq_file.write(&mut irq).unwrap();
|
|
|
|
readiness_based
|
|
.poll_all_requests(&mut device)
|
|
.expect("ac97d: failed to poll requests");
|
|
readiness_based
|
|
.write_responses()
|
|
.expect("ac97d: failed to write to socket");
|
|
|
|
/*
|
|
let next_read = device_irq.next_read();
|
|
if next_read > 0 {
|
|
return Ok(Some(next_read));
|
|
}
|
|
*/
|
|
}
|
|
Source::Scheme => {
|
|
readiness_based
|
|
.read_and_process_requests(&mut device)
|
|
.expect("ac97d: failed to read from socket");
|
|
readiness_based
|
|
.write_responses()
|
|
.expect("ac97d: failed to write to socket");
|
|
|
|
/*
|
|
let next_read = device.borrow().next_read();
|
|
if next_read > 0 {
|
|
return Ok(Some(next_read));
|
|
}
|
|
*/
|
|
}
|
|
}
|
|
}
|
|
|
|
std::process::exit(0);
|
|
}
|
|
|
|
#[cfg(not(any(target_arch = "x86", target_arch = "x86_64")))]
|
|
fn daemon(daemon: daemon::Daemon, pcid_handle: PciFunctionHandle) -> ! {
|
|
unimplemented!()
|
|
}
|