UHCI Host Controller Process Error at random

Question about which tools to use, bugs, the best way to implement a function, etc should go here. Don't forget to see if your question is answered in the wiki first! When in doubt post here.
Post Reply
venos
Posts: 13
Joined: Wed Oct 30, 2024 6:13 am

UHCI Host Controller Process Error at random

Post by venos »

Hi all,

I'm writing a USB stack at present. I have what I thought (manifestly not...) a working UHCI driver, albeit a very simple one. It worked, and I managed to get configuration descriptors and assign an address to the first (and, at present, only) device on the bus.

However, as I perturb the code in the rest of the kernel, I randomly get Host Controller Process Errors (status = 0x30), which according to the datasheet can be caused by either invalid TD length, or invalid PID. I have neither, I've checked (code below, along with status output) - and, worse than that, sometimes it's the initial "how big is the config descriptor" transfer that does it, sometimes the "set address" transfer, and sometimes the "get the actual config" transfer, seemingly at random.

To clear it, or change which transfer errors, I can simply change some code, almost at random.

To me, this suggests timing, however AFAIK there aren't any timing requirements on configuring the frame list, so long as Terminate is set to false *last*. Alternatively, I could see it maybe being memory ordering? Beyond that, not 100% sure.

I've taken some liberties with the startup timing, but running in Qemu I wouldn't have thought that crucial at this stage, though I'm acutely aware I should really hook that up to the HPET, even if I'm polling.

Any advice would be appreciated, as I'm at a genuine loss. I've attached code below, and a screenshot of the relevant debugging output.

Code (this is the whole driver, so excuse the length):

Code: Select all

use alloc::boxed::Box;
use alloc::vec::Vec;
use anyhow::{anyhow, Result};
use bitfield::bitfield;
use core::arch::asm;
use core::ops::BitOr;
use core::slice;
use pci_types::{ConfigRegionAccess, PciAddress, PciHeader, CommandRegister};
use x86_64::instructions::port::Port;
use x86_64::PhysAddr;

use crate::drivers::pcie;
use crate::driver;
use crate::memory;
use crate::dma::arena;
use crate::drivers::usb::usb;

const USBCMD: u16 = 0x00;
const USBSTS: u16 = 0x02;
const USBINTR: u16 = 0x04;
const FRBASEADD: u16 = 0x08;
const PORTSC1: u16 = 0x10;
const PORTSC2: u16 = 0x12;

const HOST_CONTROLLER_RUN: u16 = 1 << 0;
const HOST_CONTROLLER_RESET: u16 = 1 << 1;
const GLOBAL_RESET: u16 = 1 << 2;
const MAX_PACKET_SIZE_64: u16 = 1 << 7;

const STATUS_HALTED: u16 = 1 << 5;
const STATUS_INT: u16 = 1 << 0;

const SHORT_PACKET_INTERRUPT: u16 = 1 << 3;
const INTERRUPT_ON_COMPLETE: u16 = 1 << 2;
const RESUME_INTERRUPT: u16 = 1 << 1;
const TIMEOUT_CRC_INTERRUPT: u16 = 1 << 0;

const PORT_RESET: u16 = 1 << 9;
const PORT_ALWAYS_1: u16 = 1 << 7;
const PORT_ENABLE_CHANGE: u16 = 1 << 3;
const PORT_ENABLE: u16 = 1 << 2;
const PORT_STATUS_CHANGE: u16 = 1 << 1;
const PORT_CONNECTION_STATUS: u16 = 1 << 0;

bitfield! {
    #[derive(Default)]
    pub struct QueueHead(u32);

    element_link_pointer, set_element_link_pointer: 31, 4;
    qh_td_select, set_qh_td_select: 1;
    terminate, set_terminate: 0;
}

impl QueueHead {
    fn set_element_link_pointer_phys(&mut self, link_pointer: PhysAddr) {
	self.set_element_link_pointer(link_pointer.as_u64() as u32 >> 4);
    }
}

bitfield! {
    pub struct FrameListPointer(u32);

    frame_list_pointer, set_frame_list_pointer: 31, 4;
    qh_td_select, set_qh_td_select: 1;
    terminate, set_terminate: 0;
}

impl FrameListPointer {
    fn set_frame_list_pointer_phys(&mut self, link_pointer: PhysAddr) {
	self.set_frame_list_pointer(link_pointer.as_u64() as u32 >> 4);
    }
}

impl Default for FrameListPointer {
    fn default() -> Self { Self(1) }
}

bitfield! {
    #[derive(Default)]
    pub struct TransferDescriptor(u128);

    link_pointer, set_link_pointer: 31, 4;
    depth_breadth_select, set_depth_breadth_select: 2;
    qh_td_select, set_qh_td_select: 1;
    terminate, set_terminate: 0;

    short_packet_detect, set_short_packet_detect: 61;
    error_count, set_error_count: 60, 59;
    low_speed, set_low_speed: 58;
    isochronous, set_isochronous: 57;
    interrupt_on_complete, set_interrupt_on_complete: 56;
    status_active, set_status_active: 55;
    status_stalled, _: 54;
    status_buffer_error, _: 53;
    status_babble, _: 52;
    status_nak, _: 51;
    status_crc_timeout, _: 50;
    status_bitstuff_error, _: 49;
    actual_length, _: 42, 32;

    max_length, set_max_length: 95, 85;
    toggle, set_toggle: 83;
    endpoint, set_endpoint: 82, 79;
    address, set_address: 78, 72;
    packet_identification, set_packet_identification: 71, 64;

    buffer_pointer, set_buffer_pointer: 127, 96;
}

impl TransferDescriptor {
    fn set_link_pointer_phys_addr(&mut self, link_pointer: PhysAddr) {
	self.set_link_pointer((link_pointer.as_u64() >> 4).into());
    }

    fn set_buffer_pointer_phys_addr(&mut self, buffer_pointer: PhysAddr) {
	self.set_buffer_pointer(buffer_pointer.as_u64().into());
    }
}

pub struct UhciBus<'a> {
    pci_address: PciAddress,
    base_io: u16,

    frame_list: &'a mut [FrameListPointer],

    port1: bool,
    port2: bool,
}

unsafe impl Send for UhciBus<'_> { }
unsafe impl Sync for UhciBus<'_> { }

impl<'a> UhciBus<'a> {
    pub fn new(pci_address: PciAddress, uhci_base: u16) -> UhciBus<'a> {
	let pci_config_access = pcie::PciConfigAccess::new();
	let mut device_header = PciHeader::new(pci_address);

	// Disable BIOS emulation, enable interrutps
	unsafe {
	    pci_config_access.write(pci_address, 0xC0, 0x8F00);
	    pci_config_access.write(pci_address, 0xC0, 0x2000);
	}

	// Enable busmaster
	device_header.update_command(pci_config_access, |command_reg| command_reg.bitor(
	    CommandRegister::BUS_MASTER_ENABLE | CommandRegister::MEMORY_ENABLE | CommandRegister::IO_ENABLE));

	// Reset the controller
	unsafe {
	    let mut cmd_reg = Port::<u16>::new(uhci_base + USBCMD);
	    let mut port_sts = Port::<u16>::new(uhci_base + USBSTS);

	    cmd_reg.write(GLOBAL_RESET);

	    for _ in 0 .. 1000000 { unsafe { asm!("nop"); } }

	    cmd_reg.write(0);

	    if cmd_reg.read() != 0 {
		panic!("Command reg is not 0");
	    }

	    if port_sts.read() != STATUS_HALTED {
		panic!("Status is not halted");
	    }

	    // Clear status by writing 1s
	    port_sts.write(0xFF);

	    cmd_reg.write(HOST_CONTROLLER_RESET);
	    for _ in 0 .. 1000000 { unsafe { asm!("nop"); } }
	    if (cmd_reg.read() & HOST_CONTROLLER_RESET) != 0 {
		panic!("UHCI controller did not properly reset");
	    }
	}

	// Create a transfer frame page in DMA memory, and configure the controller with the physical address
	let (frame_list_virt, frame_list_phys) = memory::kernel_allocate(
	    4096, memory::MemoryAllocationType::DMA,
	    memory::MemoryAllocationOptions::Arbitrary,
	    memory::MemoryAccessRestriction::Kernel)
	    .expect("Unable to allocate a frame list memory region");
	let frame_list_phys_addr = frame_list_phys[0].as_u64();
	if frame_list_phys_addr >= 1 << 32 {
	    panic!("Frame list region is out of bounds, in higher half of physical memory");
	}

	let frame_list = unsafe {
	    slice::from_raw_parts_mut (frame_list_virt.as_mut_ptr::<FrameListPointer>(), 1024)
	};
	frame_list.fill_with(Default::default);

	// Set transfer frame base address
	unsafe {
	    let mut frbase_reg = Port::<u32>::new(uhci_base + FRBASEADD);
	    frbase_reg.write(frame_list_phys_addr as u32);

	    // Receive all interrupts
	    let mut usbintr_reg = Port::<u16>::new(uhci_base + USBINTR);
	    usbintr_reg.write(SHORT_PACKET_INTERRUPT | INTERRUPT_ON_COMPLETE | RESUME_INTERRUPT | TIMEOUT_CRC_INTERRUPT);
	}

	// Check which ports are present
	let (port1_present, port2_present) = unsafe {
	    let mut port_sc1 = Port::<u16>::new(uhci_base + PORTSC1);
	    let mut port_sc2 = Port::<u16>::new(uhci_base + PORTSC2);

	    let sc1_val = port_sc1.read();
	    let sc2_val = port_sc2.read();

	    (sc1_val != 0xFFFF && sc1_val & PORT_ALWAYS_1 != 0, sc2_val != 0xFFFF && sc2_val & PORT_ALWAYS_1 != 0)
	};

	// Start the controller
	unsafe {
	    let mut cmd_reg = Port::<u16>::new(uhci_base + USBCMD);
	    let mut port_sts = Port::<u16>::new(uhci_base + USBSTS);

	    cmd_reg.write(HOST_CONTROLLER_RUN | MAX_PACKET_SIZE_64);
	    for _ in 0 .. 1000000 { unsafe { asm!("nop"); } }

	    if (port_sts.read() & STATUS_HALTED) != 0 {
		panic!("Status is halted");
	    }
	}

	// Check which ports have devices
	let (port1, port2) = unsafe {
	    let mut port_sc1 = Port::<u16>::new(uhci_base + PORTSC1);
	    let mut port_sc2 = Port::<u16>::new(uhci_base + PORTSC2);

	    let sc1_val = port_sc1.read();
	    let sc2_val = port_sc2.read();

	    (sc1_val & PORT_CONNECTION_STATUS != 0 && port1_present, sc2_val & PORT_CONNECTION_STATUS != 0 && port2_present)
	};

	UhciBus {
	    pci_address: pci_address,
	    base_io: uhci_base,

	    frame_list: frame_list,
	    port1: port1,
	    port2: port2,
	}
    }

    fn reset_port(&self, port: u8) -> Result<()> {
	if port > 2 {
	    return Err(anyhow!("Invalid port number: {}", port));
	}

	if (port == 1 && !self.port1) || (port == 2 && !self.port2) {
	    return Err(anyhow!("Port {} is not attached", port));
	}

	unsafe {
	    let port_offset = if port == 1 { PORTSC1 } else { PORTSC2 };
	    let mut port_sc = Port::<u16>::new(self.base_io + port_offset);

	    // Reset the port
	    let mut current = port_sc.read();
	    port_sc.write(current | PORT_RESET);
	    for _ in 0 .. 1000000 { asm!("nop"); }
	    current = port_sc.read();
	    port_sc.write(current & !PORT_RESET);
	    for _ in 0 .. 1000000 { asm!("nop"); }

	    // Clear any status change and enable
	    current = port_sc.read();
	    port_sc.write(current | PORT_STATUS_CHANGE);
	    current = port_sc.read();
	    port_sc.write(current | PORT_ENABLE);
	    for _ in 0 .. 1000000 { asm!("nop"); }

	    // Clear status change, and make sure we're still enabled
	    current = port_sc.read();
	    port_sc.write(current | PORT_STATUS_CHANGE | PORT_ENABLE_CHANGE | PORT_ENABLE);
	    for _ in 0 .. 1000000 { asm!("nop"); }

	    if port_sc.read() & PORT_ENABLE == 0 {
		return Err(anyhow!("Port not enabled after reset"));
	    }
	    if port_sc.read() & PORT_CONNECTION_STATUS == 0 {
		return Err(anyhow!("Port not connected after reset"));
	    }
	}

	Ok(())
    }
}
    
impl<'a> usb::UsbHCI for UhciBus<'a> {
    fn get_ports(&self) -> Vec<usb::Port> {
	let mut ports: Vec<usb::Port> = Vec::new();
	if self.port1 {
	    if let Err(e) = self.reset_port(1) {
		log::info!("{}", e);
	    } else {
		ports.push(usb::Port {
		    num: 1,
		    status: usb::PortStatus::CONNECTED,
		    speed: usb::PortSpeed::LOW_SPEED,  // TODO (here and below): actually check the speed of the port
		});
	    }
	}
	if self.port2 {
	    if let Err(e) = self.reset_port(2) {
		log::info!("{}", e);
	    } else {
		ports.push(usb::Port {
		    num: 2,
		    status: usb::PortStatus::DISCONNECTED,
		    speed: usb::PortSpeed::LOW_SPEED,
		});
	    }
	}

	ports
    }

    fn transfer(&mut self, address: u8, transfer: usb::UsbTransfer, arena: &arena::Arena) {
	let (queue_head, queue_head_phys) = arena.acquire_default::<QueueHead>(0x10).unwrap();
	let mut transfer_descriptors_to_check: Vec<&TransferDescriptor> = Vec::new();

	match transfer.transfer_type {
	    usb::TransferType::ControlRead(setup_packet) => {
		let (packet, packet_phys) = arena.acquire::<usb::SetupPacket>(0, &setup_packet).unwrap();

		let (data_out_td, data_out_td_phys) = arena.acquire_default::<TransferDescriptor>(0x10).unwrap();
		data_out_td.set_link_pointer(0);
		data_out_td.set_depth_breadth_select(true);
		data_out_td.set_qh_td_select(false);
		data_out_td.set_terminate(true);
		data_out_td.set_error_count(3);
		data_out_td.set_low_speed(true);
		data_out_td.set_interrupt_on_complete(true);
		data_out_td.set_status_active(true);
		data_out_td.set_max_length(0x7FF);
		data_out_td.set_toggle(false);
		data_out_td.set_endpoint(0);
		data_out_td.set_address(address.into());
		data_out_td.set_packet_identification(0xE1);  // OUT
		data_out_td.set_buffer_pointer(0);

		let (data_in_td, data_in_td_phys) = arena.acquire_default::<TransferDescriptor>(0x10).unwrap();
		data_in_td.set_link_pointer_phys_addr(data_out_td_phys);
		data_in_td.set_depth_breadth_select(true);
		data_in_td.set_qh_td_select(false);
		data_in_td.set_terminate(false);
		data_in_td.set_error_count(3);
		data_in_td.set_low_speed(true);
		data_in_td.set_status_active(true);
		data_in_td.set_max_length((setup_packet.length - 1) as u128);
		data_in_td.set_toggle(true);
		data_in_td.set_endpoint(0);
		data_in_td.set_address(address.into());
		data_in_td.set_packet_identification(0x69);  // IN
		data_in_td.set_buffer_pointer_phys_addr(transfer.buffer_phys_ptr);
		
		let (setup_td, setup_td_phys) = arena.acquire_default::<TransferDescriptor>(0x10).unwrap();
		setup_td.set_link_pointer_phys_addr(data_in_td_phys);
		setup_td.set_depth_breadth_select(true);  // Depth first
		setup_td.set_qh_td_select(false);  // Next one is a transfer descriptor
		setup_td.set_terminate(false);
		setup_td.set_error_count(3);
		setup_td.set_low_speed(true);  // Low speed, as we don't know if it can do high speed yet
		setup_td.set_status_active(true);
		setup_td.set_max_length(7);  // 8 bytes
		setup_td.set_toggle(false);
		setup_td.set_endpoint(0);
		setup_td.set_address(address.into());
		setup_td.set_packet_identification(0x2d);  // SETUP
		setup_td.set_buffer_pointer_phys_addr(packet_phys);

		queue_head.set_element_link_pointer_phys(setup_td_phys);
		queue_head.set_qh_td_select(false);  // This is a queue of TDs
		queue_head.set_terminate(false);

		transfer_descriptors_to_check.push(setup_td);
		transfer_descriptors_to_check.push(data_in_td);
		transfer_descriptors_to_check.push(data_out_td);

		log::info!("Setup 0x{:x}, in 0x{:x}, out 0x{:x}", setup_td_phys.as_u64(), data_in_td_phys.as_u64(), data_out_td_phys.as_u64());
	    },
	    usb::TransferType::ControlNoData(setup_packet) => {
		let (packet, packet_phys) = arena.acquire::<usb::SetupPacket>(0, &setup_packet).unwrap();

		let (data_in_td, data_in_td_phys) = arena.acquire_default::<TransferDescriptor>(0x10).unwrap();
		data_in_td.set_link_pointer(0);
		data_in_td.set_depth_breadth_select(true);
		data_in_td.set_qh_td_select(false);
		data_in_td.set_terminate(true);
		data_in_td.set_error_count(3);
		data_in_td.set_low_speed(true);
		data_in_td.set_interrupt_on_complete(true);
		data_in_td.set_status_active(true);
		data_in_td.set_max_length(0x7FF);
		data_in_td.set_toggle(true);
		data_in_td.set_endpoint(0);
		data_in_td.set_address(address.into());
		data_in_td.set_packet_identification(0x69);  // IN
		data_in_td.set_buffer_pointer(0);
		
		let (setup_td, setup_td_phys) = arena.acquire_default::<TransferDescriptor>(0x10).unwrap();
		setup_td.set_link_pointer_phys_addr(data_in_td_phys);
		setup_td.set_depth_breadth_select(true);  // Depth first
		setup_td.set_qh_td_select(false);  // Next one is a transfer descriptor
		setup_td.set_terminate(false);
		setup_td.set_error_count(3);
		setup_td.set_low_speed(true);  // Low speed, as we don't know if it can do high speed yet
		setup_td.set_status_active(true);
		setup_td.set_max_length(7);  // 8 bytes
		setup_td.set_toggle(false);
		setup_td.set_endpoint(0);
		setup_td.set_address(address.into());
		setup_td.set_packet_identification(0x2d);  // SETUP
		setup_td.set_buffer_pointer_phys_addr(packet_phys);

		queue_head.set_element_link_pointer_phys(setup_td_phys);
		queue_head.set_qh_td_select(false);  // This is a queue of TDs
		queue_head.set_terminate(false);

		transfer_descriptors_to_check.push(setup_td);
		transfer_descriptors_to_check.push(data_in_td);

		log::info!("Setup 0x{:x}, in 0x{:x}", setup_td_phys.as_u64(), data_in_td_phys.as_u64());
	    },
	    _ => unimplemented!(),
	}
	
	// Clear any lingering interrupts
	unsafe {
	    let mut port_sts = Port::<u16>::new(self.base_io + USBSTS);
	    port_sts.write(STATUS_INT);
	}

	for i in 0 .. 1024 {
	    // Set QH/TD first, and element pointer second, to avoid a race with the UHCI controller
	    self.frame_list[i].set_qh_td_select(true);  // Entry is a QH
	    self.frame_list[i].set_frame_list_pointer_phys(queue_head_phys);
	    self.frame_list[i].set_terminate(false);
	}

	if transfer.poll {
	    // Loop until done
	    // TODO: implement PCI interrupt routing
	    unsafe {
		let mut port_sts = Port::<u16>::new(self.base_io + USBSTS);
		while (port_sts.read() & STATUS_INT) == 0 {
		    for (i, td) in transfer_descriptors_to_check.iter().enumerate() {
			log::info!("{} - length {:x}, pid {:x}, active = {}", i, td.max_length(), td.packet_identification(), td.status_active());
			if td.status_stalled() {
			    log::info!("  Stalled");
			}
			if td.status_buffer_error() {
			    log::info!("  Buffer error");
			}
			if td.status_babble() {
			    log::info!("  Babble");
			}
			if td.status_nak() {
			    log::info!("  Nak");
			}
			if td.status_crc_timeout() {
			    log::info!("  CRC/Timeout");
			}
			if td.status_bitstuff_error() {
			    log::info!("  Bitstuff error");
			}
		    }
		    if (port_sts.read() & STATUS_HALTED) != 0 {
			log::info!("A fatal error occurred: {:x}", port_sts.read());

			for (i, td) in transfer_descriptors_to_check.iter().enumerate() {
			    log::info!("{} - length {:x}, pid {:x}, active = {}", i, td.max_length(), td.packet_identification(), td.status_active());
			    if td.status_stalled() {
				log::info!("  Stalled");
			    }
			    if td.status_buffer_error() {
				log::info!("  Buffer error");
			    }
			    if td.status_babble() {
				log::info!("  Babble");
			    }
			    if td.status_nak() {
				log::info!("  Nak");
			    }
			    if td.status_crc_timeout() {
				log::info!("  CRC/Timeout");
			    }
			    if td.status_bitstuff_error() {
				log::info!("  Bitstuff error");
			    }
			}
			panic!("Status is halted");
		    }
		    asm!("pause");
		}
	    }

	    log::info!("A");
	} else {
	    unimplemented!();
	}
    }
}

pub fn init() {
    let uhci_driver = UhciDriver {};
    driver::register_driver(Box::new(uhci_driver));
}

pub struct UhciDriver {}
impl driver::Driver for UhciDriver {
    fn init(&self, info: &Box<dyn driver::DeviceTypeIdentifier>) {
	log::info!("Initialising UHCI controller");

	let pci_info = if let Some(pci_info) = info.as_any().downcast_ref::<pcie::PciDeviceType>() {
	    pci_info
	} else {
	    return;
	};

	let bar = pcie::get_bar(pci_info.clone(), 4).expect("Unable to find UHCI BAR");

	// UHCI is guaranteed to be I/O space
	let uhci_base = bar.unwrap_io();
	usb::register_hci(Box::new(UhciBus::new(pci_info.address, uhci_base as u16)));
    }

    fn check_device(&self, info: &Box<dyn driver::DeviceTypeIdentifier>) -> bool {
	if let Some(pci_info) = info.as_any().downcast_ref::<pcie::PciDeviceType>() {
	    pci_info.base_class == 0x0C &&
		pci_info.sub_class == 0x03 &&
		pci_info.interface == 0x00
	} else {
	    false
	}
    }

    fn check_new_device(&self, info: &Box<dyn driver::DeviceTypeIdentifier>) -> bool {
	true // Not yet implemented
    }
}
Debugging output:
Screenshot From 2025-03-17 13-36-58.png
Screenshot From 2025-03-17 13-36-58.png (24.48 KiB) Viewed 4381 times
User avatar
BenLunt
Member
Member
Posts: 959
Joined: Sat Nov 22, 2014 6:33 pm
Location: USA
Contact:

Re: UHCI Host Controller Process Error at random

Post by BenLunt »

Hi,

In the screen shot, you have (assuming a low-speed device is attached):

Code: Select all

1 - length 8, pid 69, active = true
Is that one TD? If so, you are trying to get 9 bytes from a low-speed device that can only transfer 8 bytes per TD. For 9 bytes, you must have:

Code: Select all

SETUP (0x2D)
IN 8 (0x69)
IN 1 (0x69)
STATUS(0xE1)
Again, assuming the attached device is a low-speed device. This would explain the "Invalid TD Length" error.

Ben
- https://www.fysnet.net/the_universal_serial_bus.htm
venos
Posts: 13
Joined: Wed Oct 30, 2024 6:13 am

Re: UHCI Host Controller Process Error at random

Post by venos »

No dice, same issue. However, that was a bug I'd have ran into sooner or later.

See attached for evidence.

With that said, this implies there are likely other parts of the spec(s) I've missed or misread, I'm going to try going over this with a finer toothed comb and see what I can find.

The specs are complex and there are several of them that affect this, so I'm not overly surprised I missed one or two things.
Screenshot From 2025-03-18 16-01-04.png
Screenshot From 2025-03-18 16-01-04.png (21.77 KiB) Viewed 4273 times
venos
Posts: 13
Joined: Wed Oct 30, 2024 6:13 am

Re: UHCI Host Controller Process Error at random

Post by venos »

And thus do we move from "why doesn't this work" to "how did this ever work" to "hooray!".

Queue heads should be two pointers, not one. And apparently somehow by sheer luck I got away with that... sometimes.

Oops :)
sounds
Member
Member
Posts: 124
Joined: Sat Feb 04, 2012 5:03 pm

Re: UHCI Host Controller Process Error at random

Post by sounds »

Great to hear it's working better now!
Post Reply