artiq-zynq/src/runtime/src/main.rs

198 lines
5.5 KiB
Rust
Raw Normal View History

2020-04-11 20:19:39 +08:00
#![no_std]
#![no_main]
2020-04-25 20:45:25 +08:00
#![recursion_limit="1024"] // for futures_util::select!
#![feature(alloc_error_handler)]
#![feature(panic_info_message)]
2020-04-11 20:19:39 +08:00
extern crate alloc;
2020-04-11 21:32:44 +08:00
use core::{cmp, str};
2020-07-13 16:06:05 +08:00
use log::{info, warn, error};
2020-04-11 22:24:23 +08:00
2020-07-13 14:59:56 +08:00
use libboard_zynq::{timer::GlobalTimer, time::Milliseconds, devc, slcr};
2020-07-13 16:06:05 +08:00
use libasync::{task, block_async};
2020-05-01 08:18:36 +08:00
use libsupport_zynq::ram;
2020-07-07 19:40:32 +08:00
use libregister::RegisterW;
2020-07-13 16:06:05 +08:00
use nb;
use void::Void;
use embedded_hal::timer::CountDown;
2020-04-11 21:32:44 +08:00
2020-06-13 15:33:17 +08:00
mod sd_reader;
2020-06-13 16:44:54 +08:00
mod config;
2020-07-06 12:04:00 +08:00
mod net_settings;
mod proto_core_io;
mod proto_async;
2020-07-13 15:00:53 +08:00
mod proto_mgmt;
2020-04-13 13:48:08 +08:00
mod comms;
2020-06-01 18:02:21 +08:00
mod rpc;
2020-05-07 13:52:40 +08:00
#[path = "../../../build/pl.rs"]
2020-04-11 21:32:44 +08:00
mod pl;
2020-04-12 20:15:01 +08:00
mod rtio;
2020-04-13 13:48:08 +08:00
mod kernel;
2020-04-24 14:37:16 +08:00
mod moninj;
mod load_pl;
mod eh_artiq;
mod panic;
2020-07-13 14:59:56 +08:00
mod logger;
2020-07-13 15:00:53 +08:00
mod mgmt;
2020-07-16 11:47:55 +08:00
mod analyzer;
2020-04-11 21:32:44 +08:00
fn init_gateware() {
2020-07-07 19:40:32 +08:00
// Set up PS->PL clocks
slcr::RegisterBlock::unlocked(|slcr| {
// As we are touching the mux, the clock may glitch, so reset the PL.
slcr.fpga_rst_ctrl.write(
slcr::FpgaRstCtrl::zeroed()
.fpga0_out_rst(true)
.fpga1_out_rst(true)
.fpga2_out_rst(true)
.fpga3_out_rst(true)
);
slcr.fpga0_clk_ctrl.write(
slcr::Fpga0ClkCtrl::zeroed()
.src_sel(slcr::PllSource::IoPll)
.divisor0(8)
.divisor1(1)
);
slcr.fpga_rst_ctrl.write(
slcr::FpgaRstCtrl::zeroed()
);
});
if devc::DevC::new().is_done() {
2020-05-04 22:27:15 +08:00
info!("gateware already loaded");
2020-05-07 12:43:53 +08:00
// Do not load again: assume that the gateware already present is
// what we want (e.g. gateware configured via JTAG before PS
// startup, or by FSBL).
2020-07-06 00:54:44 +08:00
// Make sure that the PL/PS interface is enabled (e.g. OpenOCD does not enable it).
slcr::RegisterBlock::unlocked(|slcr| {
slcr.init_postload_fpga();
});
2020-05-04 22:27:15 +08:00
} else {
// Load from SD card
match load_pl::load_bitstream_from_sd() {
Ok(_) => info!("Bitstream loaded successfully!"),
Err(e) => info!("Failure loading bitstream: {}", e),
}
2020-05-04 22:27:15 +08:00
}
}
fn identifier_read(buf: &mut [u8]) -> &str {
unsafe {
pl::csr::identifier::address_write(0);
let len = pl::csr::identifier::data_read();
let len = cmp::min(len, buf.len() as u8);
for i in 0..len {
pl::csr::identifier::address_write(1 + i);
buf[i as usize] = pl::csr::identifier::data_read();
}
str::from_utf8_unchecked(&buf[..len as usize])
}
}
fn init_rtio(timer: GlobalTimer, cfg: &config::Config) {
let clock_sel =
if let Ok(rtioclk) = cfg.read_str("rtioclk") {
match rtioclk.as_ref() {
"internal" => {
info!("using internal RTIO clock");
0
},
"external" => {
info!("using external RTIO clock");
1
},
other => {
warn!("RTIO clock specification '{}' not recognized", other);
info!("using internal RTIO clock");
0
},
}
} else {
info!("using internal RTIO clock (default)");
0
};
unsafe {
pl::csr::rtio_crg::pll_reset_write(1);
pl::csr::rtio_crg::clock_sel_write(clock_sel);
pl::csr::rtio_crg::pll_reset_write(0);
}
let mut countdown = timer.countdown();
countdown.start(Milliseconds(1));
2020-07-13 16:06:05 +08:00
nb::block!(countdown.wait()).unwrap();
let locked = unsafe { pl::csr::rtio_crg::pll_locked_read() != 0 };
if !locked {
panic!("RTIO PLL failed to lock");
}
2020-04-11 22:24:23 +08:00
2020-04-26 11:51:21 +08:00
unsafe {
pl::csr::rtio_core::reset_phy_write(1);
}
}
2020-07-13 16:06:05 +08:00
fn wait_for_async_rtio_error() -> nb::Result<(), Void> {
unsafe {
if pl::csr::rtio_core::async_error_read() != 0 {
Ok(())
} else {
Err(nb::Error::WouldBlock)
}
}
}
async fn report_async_rtio_errors() {
loop {
let _ = block_async!(wait_for_async_rtio_error()).await;
unsafe {
let errors = pl::csr::rtio_core::async_error_read();
if errors & 1 != 0 {
error!("RTIO collision involving channel {}",
pl::csr::rtio_core::collision_channel_read());
}
if errors & 2 != 0 {
error!("RTIO busy error involving channel {}",
pl::csr::rtio_core::busy_channel_read());
}
if errors & 4 != 0 {
error!("RTIO sequence error involving channel {}",
pl::csr::rtio_core::sequence_error_channel_read());
}
pl::csr::rtio_core::async_error_write(errors);
}
}
}
2020-07-13 14:59:56 +08:00
static mut LOG_BUFFER: [u8; 1<<17] = [0; 1<<17];
#[no_mangle]
pub fn main_core0() {
let timer = GlobalTimer::start();
2020-07-13 14:59:56 +08:00
let buffer_logger = unsafe {
logger::BufferLogger::new(&mut LOG_BUFFER[..])
};
buffer_logger.set_uart_log_level(log::LevelFilter::Debug);
buffer_logger.register();
log::set_max_level(log::LevelFilter::Debug);
2020-07-13 14:59:56 +08:00
info!("NAR3/Zynq7000 starting...");
ram::init_alloc_linker();
init_gateware();
info!("detected gateware: {}", identifier_read(&mut [0; 64]));
2020-04-26 11:51:21 +08:00
2020-07-08 19:24:26 +08:00
let cfg = match config::Config::new() {
Ok(cfg) => cfg,
Err(err) => {
warn!("config initialization failed: {}", err);
config::Config::new_dummy()
}
};
init_rtio(timer, &cfg);
2020-07-13 16:06:05 +08:00
task::spawn(report_async_rtio_errors());
2020-07-08 19:24:26 +08:00
comms::main(timer, &cfg);
2020-04-11 20:19:39 +08:00
}