diff --git a/src/libboard_artiq/src/cxp_ctrl.rs b/src/libboard_artiq/src/cxp_ctrl.rs new file mode 100644 index 0000000..f634b7f --- /dev/null +++ b/src/libboard_artiq/src/cxp_ctrl.rs @@ -0,0 +1,261 @@ +use core::slice; + +use byteorder::{ByteOrder, NetworkEndian}; +use io::Cursor; +use libboard_zynq::{println, time::Milliseconds, timer::GlobalTimer}; + +use crate::{cxp_proto::{print_packet, Error, RXPacket, TXPacket, DATA_MAXSIZE}, + mem::mem::CXP_MEM, + pl::csr::CXP}; + +const BUF_LEN: usize = 0x800; +const TRANSMISSION_TIMEOUT: u64 = 200; + +fn packet_pending(channel: u8) -> bool { + unsafe { (CXP[channel as usize].rx_pending_packet_read)() == 1 } +} + +fn receive(channel: u8) -> Result, Error> { + if packet_pending(channel) { + let channel = channel as usize; + unsafe { + let read_buffer_ptr = (CXP[channel].rx_read_ptr_read)() as usize; + println!("buffer ptr = {}", read_buffer_ptr); + let ptr = (CXP_MEM[channel].base + CXP_MEM[channel].size / 2 + read_buffer_ptr * BUF_LEN) as *mut u32; + + let mut reader = Cursor::new(slice::from_raw_parts_mut(ptr as *mut u8, BUF_LEN)); + let packet_type = (CXP[channel].rx_packet_type_read)(); + println!("packet_type = {}", packet_type); + // DEBUG: + // println!("RX MEM before reading"); + // print_packet(&reader.get_ref()[0..40]); + + let packet = RXPacket::read_from(&mut reader, packet_type); + println!("{:X?}", packet); + + (CXP[channel].rx_pending_packet_write)(1); + Ok(Some(packet?)) + } + } else { + Ok(None) + } +} + +fn receive_timeout(channel: u8, timeout_ms: u64) -> Result { + // assume timer was initialized successfully + let timer = unsafe { GlobalTimer::get() }; + let limit = timer.get_time() + Milliseconds(timeout_ms); + while timer.get_time() < limit { + match receive(channel)? { + None => (), + Some(packet) => return Ok(packet), + } + } + Err(Error::TimedOut) +} + +fn send_data_packet(channel: u8, packet: &TXPacket) -> Result<(), Error> { + // assume tx is enabled + let channel = channel as usize; + unsafe { + while (CXP[channel].tx_writer_busy_read)() == 1 {} + let ptr = CXP_MEM[channel].base as *mut u32; + let mut writer = Cursor::new(slice::from_raw_parts_mut(ptr as *mut u8, BUF_LEN)); + + packet.write_to(&mut writer)?; + // DEBUG: + // println!("TX MEM after writing"); + // print_packet(&writer.get_ref()[0..40]); + + (CXP[channel].tx_writer_word_len_write)(writer.position() as u16 / 4); + (CXP[channel].tx_writer_stb_write)(1); + } + + Ok(()) +} + +pub fn send_test_packet(channel: u8) -> Result<(), Error> { + // assume tx is enabled + let channel = channel as usize; + unsafe { + while (CXP[channel].tx_writer_busy_read)() == 1 {} + (CXP[channel].tx_writer_stb_testseq_write)(1); + } + Ok(()) +} + +// +// +// DEBUG: +// +// +pub fn rx_debug_mem_print(channel: u8) { + unsafe { + let ptr = CXP_MEM[channel as usize].base as *mut u32; + let arr = slice::from_raw_parts_mut(ptr as *mut u8, BUF_LEN * 4); + print_packet(arr); + } +} + +pub fn print_decode_error(channel: u8) { + unsafe { + println!( + "CH#{} Decode error = {}", + channel, + (CXP[channel as usize].rx_reader_decode_err_read)() + ); + } +} + +// +// +// CTRL packet +// +// + +// Section 9.6.1.2 (CXP-001-2021) +// CTRL packet need to be tagged for CXP 2.0 or greater +static mut TAG: u8 = 0; + +pub fn reset_tag() { + unsafe { TAG = 0 } +} + +fn increment_tag() { + unsafe { TAG = TAG.wrapping_add(1) }; +} + +fn check_tag(tag: Option) -> Result<(), Error> { + unsafe { + if tag.is_some() && tag != Some(TAG) { + Err(Error::TagMismatch) + } else { + Ok(()) + } + } +} + +fn check_length(length: u32) -> Result<(), Error> { + if length > DATA_MAXSIZE as u32 || length == 0 { + Err(Error::LengthOutOfRange) + } else { + Ok(()) + } +} + +fn process_ack_packet(channel: u8, timeout: u64) -> Result<(), Error> { + match receive_timeout(channel, timeout) { + Ok(RXPacket::CtrlAck { tag }) => { + check_tag(tag)?; + Ok(()) + } + Ok(RXPacket::CtrlDelay { tag, time }) => { + check_tag(tag)?; + + // info!("delaying by {} ms ....", time); + process_ack_packet(channel, time as u64) + } + Ok(_) => Err(Error::UnexpectedReply), + Err(e) => Err(e), + } +} + +fn process_reply_packet(channel: u8, timeout: u64, expected_length: u32) -> Result<[u8; DATA_MAXSIZE], Error> { + match receive_timeout(channel, timeout) { + Ok(RXPacket::CtrlReply { tag, length, data }) => { + check_tag(tag)?; + + if length != expected_length { + return Err(Error::UnexpectedReply); + }; + + Ok(data) + } + Ok(RXPacket::CtrlDelay { tag, time }) => { + check_tag(tag)?; + + // info!("delaying by {} ms ....", time); + process_reply_packet(channel, time as u64, expected_length) + } + Ok(_) => Err(Error::UnexpectedReply), + Err(e) => Err(e), + } +} + +pub fn write_bytes_no_ack(channel: u8, addr: u32, val: &[u8], with_tag: bool) -> Result<(), Error> { + let length = val.len() as u32; + check_length(length)?; + + let mut data: [u8; DATA_MAXSIZE] = [0; DATA_MAXSIZE]; + data[..length as usize].clone_from_slice(val); + + let tag: Option = if with_tag { Some(unsafe { TAG }) } else { None }; + send_data_packet( + channel, + &TXPacket::CtrlWrite { + tag, + addr, + length, + data, + }, + ) +} + +pub fn write_bytes(channel: u8, addr: u32, val: &[u8], with_tag: bool) -> Result<(), Error> { + write_bytes_no_ack(channel, addr, val, with_tag)?; + process_ack_packet(channel, TRANSMISSION_TIMEOUT)?; + + // DEBUG + print_decode_error(channel); + + if with_tag { + increment_tag(); + }; + Ok(()) +} + +pub fn write_u32(channel: u8, addr: u32, val: u32, with_tag: bool) -> Result<(), Error> { + write_bytes(channel, addr, &val.to_be_bytes(), with_tag) +} + +pub fn write_u64(channel: u8, addr: u32, val: u64, with_tag: bool) -> Result<(), Error> { + write_bytes(channel, addr, &val.to_be_bytes(), with_tag) +} + +fn read(channel: u8, addr: u32, length: u32, with_tag: bool) -> Result<(), Error> { + check_length(length)?; + let tag: Option = if with_tag { Some(unsafe { TAG }) } else { None }; + send_data_packet(channel, &TXPacket::CtrlRead { tag, addr, length }) +} + +pub fn read_bytes(channel: u8, addr: u32, bytes: &mut [u8], with_tag: bool) -> Result<(), Error> { + let length = bytes.len() as u32; + read(channel, addr, length, with_tag)?; + + let data = process_reply_packet(channel, TRANSMISSION_TIMEOUT, length)?; + bytes.clone_from_slice(&data[..length as usize]); + + // DEBUG + print_decode_error(channel); + + if with_tag { + increment_tag(); + }; + Ok(()) +} + +pub fn read_u32(channel: u8, addr: u32, with_tag: bool) -> Result { + let mut bytes: [u8; 4] = [0; 4]; + read_bytes(channel, addr, &mut bytes, with_tag)?; + let val = NetworkEndian::read_u32(&bytes); + + Ok(val) +} + +pub fn read_u64(channel: u8, addr: u32, with_tag: bool) -> Result { + let mut bytes: [u8; 8] = [0; 8]; + read_bytes(channel, addr, &mut bytes, with_tag)?; + let val = NetworkEndian::read_u64(&bytes); + + Ok(val) +}