mod channel;
pub use channel::*;
use crate::Session;
use crate::{Core, MemoryInterface, config::MemoryRegion};
use std::ops::Range;
use std::thread;
use std::time::Duration;
use std::time::Instant;
use zerocopy::{FromBytes, IntoBytes};
#[derive(Debug)]
pub struct Rtt {
ptr: u64,
pub up_channels: Vec<UpChannel>,
pub down_channels: Vec<DownChannel>,
}
#[repr(C)]
#[derive(FromBytes)]
struct RttControlBlockHeaderInner {
id: [u8; 16],
max_up_channels: u32,
max_down_channels: u32,
}
enum RttControlBlockHeader {
Header32(RttControlBlockHeaderInner),
Header64(RttControlBlockHeaderInner),
}
impl RttControlBlockHeader {
pub fn try_from_header(is_64_bit: bool, mem: &[u8]) -> Option<Self> {
if is_64_bit {
RttControlBlockHeaderInner::read_from_prefix(mem)
.map(|(header, _)| Self::Header64(header))
.ok()
} else {
RttControlBlockHeaderInner::read_from_prefix(mem)
.map(|(header, _)| Self::Header32(header))
.ok()
}
}
pub const fn minimal_header_size() -> usize {
std::mem::size_of::<RttControlBlockHeaderInner>()
}
pub const fn header_size(&self) -> usize {
std::mem::size_of::<RttControlBlockHeaderInner>()
}
pub fn id(&self) -> [u8; 16] {
match self {
RttControlBlockHeader::Header32(x) => x.id,
RttControlBlockHeader::Header64(x) => x.id,
}
}
pub fn max_up_channels(&self) -> usize {
match self {
RttControlBlockHeader::Header32(x) => x.max_up_channels as usize,
RttControlBlockHeader::Header64(x) => x.max_up_channels as usize,
}
}
pub fn max_down_channels(&self) -> usize {
match self {
RttControlBlockHeader::Header32(x) => x.max_down_channels as usize,
RttControlBlockHeader::Header64(x) => x.max_down_channels as usize,
}
}
pub fn channel_buffer_size(&self) -> usize {
match self {
RttControlBlockHeader::Header32(_x) => RttChannelBufferInner::<u32>::size(),
RttControlBlockHeader::Header64(_x) => RttChannelBufferInner::<u64>::size(),
}
}
pub fn total_rtt_buffer_size(&self) -> usize {
let total_number_of_channels = self.max_up_channels() + self.max_down_channels();
let channel_size = self.channel_buffer_size();
self.header_size() + channel_size * total_number_of_channels
}
pub fn parse_channel_buffers(&self, mem: &[u8]) -> Result<Vec<RttChannelBuffer>, Error> {
let buffers = match self {
RttControlBlockHeader::Header32(_) => {
<[RttChannelBufferInner<u32>]>::ref_from_bytes(mem)
.map_err(|_| Error::ControlBlockNotFound)?
.iter()
.cloned()
.map(RttChannelBuffer::from)
.collect::<Vec<RttChannelBuffer>>()
}
RttControlBlockHeader::Header64(_) => {
<[RttChannelBufferInner<u64>]>::ref_from_bytes(mem)
.map_err(|_| Error::ControlBlockNotFound)?
.iter()
.cloned()
.map(RttChannelBuffer::from)
.collect::<Vec<RttChannelBuffer>>()
}
};
Ok(buffers)
}
}
impl Rtt {
pub const RTT_ID: [u8; 16] = *b"SEGGER RTT\0\0\0\0\0\0";
pub fn attach_at(
core: &mut Core,
ptr: u64,
) -> Result<Rtt, Error> {
let is_64_bit = core.is_64_bit();
let mut mem = [0u32; RttControlBlockHeader::minimal_header_size() / 4];
core.read(ptr, &mut mem.as_mut_bytes()[0..Self::RTT_ID.len()])?;
core.read_32(
ptr + Self::RTT_ID.len() as u64,
&mut mem[Self::RTT_ID.len() / 4..],
)?;
let rtt_header = RttControlBlockHeader::try_from_header(is_64_bit, mem.as_bytes())
.ok_or(Error::ControlBlockNotFound)?;
let rtt_id = rtt_header.id();
if rtt_id != Self::RTT_ID {
tracing::trace!(
"Expected control block to start with RTT ID: {:?}\n. Got instead: {:?}",
String::from_utf8_lossy(&Self::RTT_ID),
String::from_utf8_lossy(&rtt_id)
);
return Err(Error::ControlBlockNotFound);
}
let max_up_channels = rtt_header.max_up_channels();
let max_down_channels = rtt_header.max_down_channels();
if max_up_channels > 255 || max_down_channels > 255 {
return Err(Error::ControlBlockCorrupted(format!(
"Unexpected array sizes at {ptr:#010x}: max_up_channels={max_up_channels} max_down_channels={max_down_channels}"
)));
}
let channel_buffer_len = rtt_header.total_rtt_buffer_size() - rtt_header.header_size();
let mut mem = vec![0; channel_buffer_len / 4];
core.read_32(ptr + rtt_header.header_size() as u64, &mut mem)?;
let mut up_channels = Vec::new();
let mut down_channels = Vec::new();
let channel_buffer_size = rtt_header.channel_buffer_size();
let up_channels_start = 0;
let up_channels_len = max_up_channels * channel_buffer_size;
let up_channels_raw_buffer = &mem.as_bytes()[up_channels_start..][..up_channels_len];
let up_channels_buffer = rtt_header.parse_channel_buffers(up_channels_raw_buffer)?;
let down_channels_start = up_channels_start + up_channels_len;
let down_channels_len = max_down_channels * channel_buffer_size;
let down_channels_raw_buffer = &mem.as_bytes()[down_channels_start..][..down_channels_len];
let down_channels_buffer = rtt_header.parse_channel_buffers(down_channels_raw_buffer)?;
let mut offset = ptr + rtt_header.header_size() as u64 + up_channels_start as u64;
for (channel_index, buffer) in up_channels_buffer.into_iter().enumerate() {
let buffer_size = buffer.size() as u64;
if let Some(chan) = Channel::from(core, channel_index, offset, buffer)? {
up_channels.push(UpChannel(chan));
} else {
tracing::warn!("Buffer for up channel {channel_index} not initialized");
}
offset += buffer_size;
}
let mut offset = ptr + rtt_header.header_size() as u64 + down_channels_start as u64;
for (channel_index, buffer) in down_channels_buffer.into_iter().enumerate() {
let buffer_size = buffer.size() as u64;
if let Some(chan) = Channel::from(core, channel_index, offset, buffer)? {
down_channels.push(DownChannel(chan));
} else {
tracing::warn!("Buffer for down channel {channel_index} not initialized");
}
offset += buffer_size;
}
Ok(Rtt {
ptr,
up_channels,
down_channels,
})
}
pub fn attach_region(core: &mut Core, region: &ScanRegion) -> Result<Rtt, Error> {
let ptr = Self::find_contol_block(core, region)?;
Self::attach_at(core, ptr)
}
pub fn attach(core: &mut Core) -> Result<Rtt, Error> {
Self::attach_region(core, &ScanRegion::default())
}
pub fn find_contol_block(core: &mut Core, region: &ScanRegion) -> Result<u64, Error> {
let ranges = match region.clone() {
ScanRegion::Exact(addr) => {
tracing::debug!("Scanning at exact address: {:#010x}", addr);
return Ok(addr);
}
ScanRegion::Ram => {
tracing::debug!("Scanning whole RAM");
core.memory_regions()
.filter_map(MemoryRegion::as_ram_region)
.map(|r| r.range.clone())
.collect()
}
ScanRegion::Ranges(regions) if regions.is_empty() => {
tracing::debug!(
"ELF file has no RTT block symbol, and this target does not support automatic scanning"
);
return Err(Error::NoControlBlockLocation);
}
ScanRegion::Ranges(regions) => {
tracing::debug!("Scanning regions: {:#010x?}", region);
regions
}
};
let mut instances = ranges
.into_iter()
.filter_map(|range| {
let range_len = range.end.checked_sub(range.start)?;
let Ok(range_len) = usize::try_from(range_len) else {
tracing::warn!("Region too long ({} bytes), ignoring", range_len);
return None;
};
let mut mem = vec![0; range_len];
core.read(range.start, &mut mem).ok()?;
let offset = mem
.windows(Self::RTT_ID.len())
.position(|w| w == Self::RTT_ID)?;
let target_ptr = range.start + offset as u64;
Some(target_ptr)
})
.collect::<Vec<_>>();
match instances.len() {
0 => Err(Error::ControlBlockNotFound),
1 => Ok(instances.remove(0)),
_ => Err(Error::MultipleControlBlocksFound(instances)),
}
}
pub fn ptr(&self) -> u64 {
self.ptr
}
pub fn up_channels(&mut self) -> &mut [UpChannel] {
&mut self.up_channels
}
pub fn down_channels(&mut self) -> &mut [DownChannel] {
&mut self.down_channels
}
pub fn up_channel(&mut self, channel: usize) -> Option<&mut UpChannel> {
self.up_channels.get_mut(channel)
}
pub fn down_channel(&mut self, channel: usize) -> Option<&mut DownChannel> {
self.down_channels.get_mut(channel)
}
pub fn control_block_size() -> usize {
RttControlBlockHeader::minimal_header_size()
}
}
#[derive(Clone, Debug, Default)]
pub enum ScanRegion {
#[default]
Ram,
Ranges(Vec<Range<u64>>),
Exact(u64),
}
impl ScanRegion {
pub fn range(range: Range<u64>) -> Self {
Self::Ranges(vec![range])
}
}
#[derive(thiserror::Error, Debug, docsplay::Display)]
pub enum Error {
NoControlBlockLocation,
ControlBlockNotFound,
MultipleControlBlocksFound(Vec<u64>),
ControlBlockCorrupted(String),
IncorrectCoreSpecified(usize, usize),
Probe(#[from] crate::Error),
MemoryRead(String),
Other(#[from] anyhow::Error),
ReadPointerChanged,
MissingChannel(usize),
}
fn display_list(list: &[u64]) -> String {
list.iter()
.map(|ptr| format!("{ptr:#010x}"))
.collect::<Vec<_>>()
.join(", ")
}
fn try_attach_to_rtt_inner(
mut try_attach_once: impl FnMut() -> Result<Rtt, Error>,
timeout: Duration,
) -> Result<Rtt, Error> {
let t = Instant::now();
let mut attempt = 1;
loop {
tracing::debug!("Initializing RTT (attempt {attempt})...");
match try_attach_once() {
err @ Err(Error::NoControlBlockLocation) => return err,
Err(_) if t.elapsed() < timeout => {
attempt += 1;
tracing::debug!("Failed to initialize RTT. Retrying until timeout.");
thread::sleep(Duration::from_millis(50));
}
other => return other,
}
}
}
pub fn try_attach_to_rtt(
core: &mut Core<'_>,
timeout: Duration,
rtt_region: &ScanRegion,
) -> Result<Rtt, Error> {
try_attach_to_rtt_inner(|| Rtt::attach_region(core, rtt_region), timeout)
}
pub fn try_attach_to_rtt_shared(
session: &parking_lot::FairMutex<Session>,
core_id: usize,
timeout: Duration,
rtt_region: &ScanRegion,
) -> Result<Rtt, Error> {
try_attach_to_rtt_inner(
|| {
let mut session_handle = session.lock();
let mut core = session_handle.core(core_id)?;
Rtt::attach_region(&mut core, rtt_region)
},
timeout,
)
}
#[cfg(test)]
mod test {
use super::*;
#[test]
fn test_how_control_block_list_looks() {
let error = Error::MultipleControlBlocksFound(vec![0x2000, 0x3000]);
assert_eq!(
error.to_string(),
"Multiple control blocks found in target memory: 0x00002000, 0x00003000."
);
}
}