init commit
This commit is contained in:
263
examples/zedboard/src/bin/uart-blocking.rs
Normal file
263
examples/zedboard/src/bin/uart-blocking.rs
Normal file
@ -0,0 +1,263 @@
|
||||
#![no_std]
|
||||
#![no_main]
|
||||
|
||||
use axi_uart16550::AxiUart16550;
|
||||
use axi_uartlite::AxiUartlite;
|
||||
use core::panic::PanicInfo;
|
||||
use cortex_ar::asm::nop;
|
||||
use embassy_executor::Spawner;
|
||||
use embassy_time::{Duration, Ticker};
|
||||
use embedded_hal::digital::StatefulOutputPin;
|
||||
use embedded_io::Write;
|
||||
use fugit::RateExtU32;
|
||||
use log::{error, info};
|
||||
use zedboard::PS_CLOCK_FREQUENCY;
|
||||
use zynq7000_hal::{
|
||||
BootMode,
|
||||
clocks::Clocks,
|
||||
configure_level_shifter,
|
||||
gic::{GicConfigurator, GicInterruptHelper, Interrupt},
|
||||
gpio::{EmioPin, GpioPins, PinState},
|
||||
gtc::Gtc,
|
||||
uart::{ClkConfigRaw, Uart, UartConfig},
|
||||
};
|
||||
|
||||
use zynq7000::{PsPeripherals, slcr::LevelShifterConfig};
|
||||
use zynq7000_rt as _;
|
||||
|
||||
const INIT_STRING: &str = "-- Zynq 7000 Zedboard blocking UART example --\n\r";
|
||||
|
||||
const AXI_UARTLITE_BASE_ADDR: u32 = 0x42C0_0000;
|
||||
const AXI_UAR16550_BASE_ADDR: u32 = 0x43C0_0000;
|
||||
|
||||
/// Entry point (not called like a normal main function)
|
||||
#[unsafe(no_mangle)]
|
||||
pub extern "C" fn boot_core(cpu_id: u32) -> ! {
|
||||
if cpu_id != 0 {
|
||||
panic!("unexpected CPU ID {}", cpu_id);
|
||||
}
|
||||
main();
|
||||
}
|
||||
|
||||
#[derive(Debug, Copy, Clone, PartialEq)]
|
||||
pub enum UartSel {
|
||||
Uart0 = 0b000,
|
||||
Uartlite = 0b001,
|
||||
Uart16550 = 0b010,
|
||||
Uart0ToUartlite = 0b011,
|
||||
Uart0ToUart16550 = 0b100,
|
||||
UartliteToUart16550 = 0b101,
|
||||
}
|
||||
|
||||
pub struct UartMultiplexer {
|
||||
sel_pins: [EmioPin; 3],
|
||||
}
|
||||
|
||||
impl UartMultiplexer {
|
||||
pub fn new(mut sel_pins: [EmioPin; 3]) -> Self {
|
||||
for pin in sel_pins.iter_mut() {
|
||||
pin.into_output(PinState::Low);
|
||||
}
|
||||
Self { sel_pins }
|
||||
}
|
||||
|
||||
pub fn select(&mut self, sel: UartSel) {
|
||||
// TODO: A pin group switcher would be nice to do this in one go.
|
||||
match sel {
|
||||
UartSel::Uart0 => {
|
||||
self.sel_pins[2].set_low().unwrap();
|
||||
self.sel_pins[1].set_low().unwrap();
|
||||
self.sel_pins[0].set_low().unwrap();
|
||||
}
|
||||
UartSel::Uartlite => {
|
||||
self.sel_pins[2].set_low().unwrap();
|
||||
self.sel_pins[1].set_low().unwrap();
|
||||
self.sel_pins[0].set_high().unwrap();
|
||||
}
|
||||
UartSel::Uart16550 => {
|
||||
self.sel_pins[2].set_low().unwrap();
|
||||
self.sel_pins[1].set_high().unwrap();
|
||||
self.sel_pins[0].set_low().unwrap();
|
||||
}
|
||||
UartSel::Uart0ToUartlite => {
|
||||
self.sel_pins[2].set_low().unwrap();
|
||||
self.sel_pins[1].set_high().unwrap();
|
||||
self.sel_pins[0].set_high().unwrap();
|
||||
}
|
||||
UartSel::Uart0ToUart16550 => {
|
||||
self.sel_pins[2].set_high().unwrap();
|
||||
self.sel_pins[1].set_low().unwrap();
|
||||
self.sel_pins[0].set_low().unwrap();
|
||||
}
|
||||
UartSel::UartliteToUart16550 => {
|
||||
self.sel_pins[2].set_high().unwrap();
|
||||
self.sel_pins[1].set_low().unwrap();
|
||||
self.sel_pins[0].set_high().unwrap();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
#[embassy_executor::main]
|
||||
#[unsafe(export_name = "main")]
|
||||
async fn main(_spawner: Spawner) -> ! {
|
||||
// Enable PS-PL level shifters.
|
||||
configure_level_shifter(LevelShifterConfig::EnableAll);
|
||||
let dp = PsPeripherals::take().unwrap();
|
||||
// Clock was already initialized by PS7 Init TCL script or FSBL, we just read it.
|
||||
let clocks = Clocks::new_from_regs(PS_CLOCK_FREQUENCY).unwrap();
|
||||
// Set up the global interrupt controller.
|
||||
let mut gic = GicConfigurator::new_with_init(dp.gicc, dp.gicd);
|
||||
gic.enable_all_interrupts();
|
||||
gic.set_all_spi_interrupt_targets_cpu0();
|
||||
gic.enable();
|
||||
unsafe {
|
||||
gic.enable_interrupts();
|
||||
}
|
||||
let mut gpio_pins = GpioPins::new(dp.gpio);
|
||||
|
||||
// Set up global timer counter and embassy time driver.
|
||||
let gtc = Gtc::new(dp.gtc, clocks.arm_clocks());
|
||||
zynq7000_embassy::init(clocks.arm_clocks(), gtc);
|
||||
|
||||
// Set up the UART, we are logging with it.
|
||||
let uart_clk_config = ClkConfigRaw::new_autocalc_with_error(clocks.io_clocks(), 115200)
|
||||
.unwrap()
|
||||
.0;
|
||||
let uart_tx = gpio_pins.mio.mio48.into_uart();
|
||||
let uart_rx = gpio_pins.mio.mio49.into_uart();
|
||||
let mut log_uart = Uart::new_with_mio(
|
||||
dp.uart_1,
|
||||
UartConfig::new_with_clk_config(uart_clk_config),
|
||||
(uart_tx, uart_rx),
|
||||
)
|
||||
.unwrap();
|
||||
log_uart.write_all(INIT_STRING.as_bytes()).unwrap();
|
||||
|
||||
// Safety: Co-operative multi-tasking is used.
|
||||
unsafe { zynq7000_hal::log::init_unsafe_single_core(log_uart, log::LevelFilter::Trace, false) };
|
||||
|
||||
// UART0 routed through EMIO to PL pins.
|
||||
let mut uart_0 =
|
||||
Uart::new_with_emio(dp.uart_0, UartConfig::new_with_clk_config(uart_clk_config)).unwrap();
|
||||
// Safety: Valid address of AXI UARTLITE.
|
||||
let mut uartlite = unsafe { AxiUartlite::new(AXI_UARTLITE_BASE_ADDR) };
|
||||
|
||||
// TODO: Can we determine/read the clock frequency to the FPGAs as well?
|
||||
let (clk_config, error) =
|
||||
axi_uart16550::ClkConfig::new_autocalc_with_error(100.MHz(), 115200).unwrap();
|
||||
assert!(error < 0.02);
|
||||
let mut uart_16550 = unsafe {
|
||||
AxiUart16550::new(
|
||||
AXI_UAR16550_BASE_ADDR,
|
||||
axi_uart16550::UartConfig::new_with_clk_config(clk_config),
|
||||
)
|
||||
};
|
||||
|
||||
let boot_mode = BootMode::new();
|
||||
info!("Boot mode: {:?}", boot_mode);
|
||||
|
||||
let mut ticker = Ticker::every(Duration::from_millis(1000));
|
||||
let mut mio_led = gpio_pins.mio.mio7.into_output(PinState::Low);
|
||||
|
||||
let mut emio_leds: [EmioPin; 8] = [
|
||||
gpio_pins.emio.take(0).unwrap(),
|
||||
gpio_pins.emio.take(1).unwrap(),
|
||||
gpio_pins.emio.take(2).unwrap(),
|
||||
gpio_pins.emio.take(3).unwrap(),
|
||||
gpio_pins.emio.take(4).unwrap(),
|
||||
gpio_pins.emio.take(5).unwrap(),
|
||||
gpio_pins.emio.take(6).unwrap(),
|
||||
gpio_pins.emio.take(7).unwrap(),
|
||||
];
|
||||
for led in emio_leds.iter_mut() {
|
||||
led.into_output(PinState::Low);
|
||||
}
|
||||
|
||||
let mut uart_mux = UartMultiplexer::new([
|
||||
gpio_pins.emio.take(8).unwrap(),
|
||||
gpio_pins.emio.take(9).unwrap(),
|
||||
gpio_pins.emio.take(10).unwrap(),
|
||||
]);
|
||||
let mut current_sel = UartSel::Uart0;
|
||||
uart_mux.select(current_sel);
|
||||
let mut led_idx = 0;
|
||||
loop {
|
||||
mio_led.toggle().unwrap();
|
||||
|
||||
emio_leds[led_idx].toggle().unwrap();
|
||||
led_idx += 1;
|
||||
if led_idx >= emio_leds.len() {
|
||||
led_idx = 0;
|
||||
}
|
||||
uart_0
|
||||
.write_all("Hello, World from UART0!\n\r".as_bytes())
|
||||
.unwrap();
|
||||
uartlite
|
||||
.write_all("Hello, World from AXI UARTLITE!\n\r".as_bytes())
|
||||
.unwrap();
|
||||
uart_16550
|
||||
.write_all("Hello, World from AXI UART16550!\n\r".as_bytes())
|
||||
.unwrap();
|
||||
|
||||
uart_0.flush().unwrap();
|
||||
uartlite.flush().unwrap();
|
||||
uart_16550.flush().unwrap();
|
||||
match current_sel {
|
||||
UartSel::Uart0 => current_sel = UartSel::Uartlite,
|
||||
UartSel::Uartlite => current_sel = UartSel::Uart16550,
|
||||
UartSel::Uart16550 => current_sel = UartSel::Uart0,
|
||||
UartSel::Uart0ToUartlite | UartSel::Uart0ToUart16550 | UartSel::UartliteToUart16550 => {
|
||||
}
|
||||
}
|
||||
uart_mux.select(current_sel);
|
||||
ticker.next().await; // Wait for the next cycle of the ticker
|
||||
}
|
||||
}
|
||||
|
||||
#[unsafe(no_mangle)]
|
||||
pub extern "C" fn _irq_handler() {
|
||||
let mut gic_helper = GicInterruptHelper::new();
|
||||
let irq_info = gic_helper.acknowledge_interrupt();
|
||||
match irq_info.interrupt() {
|
||||
Interrupt::Sgi(_) => (),
|
||||
Interrupt::Ppi(ppi_interrupt) => {
|
||||
if ppi_interrupt == zynq7000_hal::gic::PpiInterrupt::GlobalTimer {
|
||||
unsafe {
|
||||
zynq7000_embassy::on_interrupt();
|
||||
}
|
||||
}
|
||||
}
|
||||
Interrupt::Spi(_spi_interrupt) => (),
|
||||
Interrupt::Invalid(_) => (),
|
||||
Interrupt::Spurious => (),
|
||||
}
|
||||
gic_helper.end_of_interrupt(irq_info);
|
||||
}
|
||||
|
||||
#[unsafe(no_mangle)]
|
||||
pub extern "C" fn _abort_handler() {
|
||||
loop {
|
||||
nop();
|
||||
}
|
||||
}
|
||||
|
||||
#[unsafe(no_mangle)]
|
||||
pub extern "C" fn _undefined_handler() {
|
||||
loop {
|
||||
nop();
|
||||
}
|
||||
}
|
||||
|
||||
#[unsafe(no_mangle)]
|
||||
pub extern "C" fn _prefetch_handler() {
|
||||
loop {
|
||||
nop();
|
||||
}
|
||||
}
|
||||
|
||||
/// Panic handler
|
||||
#[panic_handler]
|
||||
fn panic(info: &PanicInfo) -> ! {
|
||||
error!("Panic: {:?}", info);
|
||||
loop {}
|
||||
}
|
Reference in New Issue
Block a user