225 lines
6.6 KiB
Rust
225 lines
6.6 KiB
Rust
use core::convert::TryInto;
|
|
|
|
use armv4t_emu::{reg, Memory};
|
|
use gdbstub::arch;
|
|
use gdbstub::arch::arm::reg::id::ArmCoreRegId;
|
|
use gdbstub::target;
|
|
use gdbstub::target::ext::base::singlethread::{ResumeAction, SingleThreadOps, StopReason};
|
|
use gdbstub::target::ext::breakpoints::WatchKind;
|
|
use gdbstub::target::{Target, TargetError, TargetResult};
|
|
|
|
use crate::emu::{Emu, Event};
|
|
|
|
// Additional GDB extensions
|
|
|
|
mod extended_mode;
|
|
mod monitor_cmd;
|
|
mod section_offsets;
|
|
mod target_description_xml_override;
|
|
|
|
/// Turn a `ArmCoreRegId` into an internal register number of `armv4t_emu`.
|
|
fn cpu_reg_id(id: ArmCoreRegId) -> Option<u8> {
|
|
match id {
|
|
ArmCoreRegId::Gpr(i) => Some(i),
|
|
ArmCoreRegId::Sp => Some(reg::SP),
|
|
ArmCoreRegId::Lr => Some(reg::LR),
|
|
ArmCoreRegId::Pc => Some(reg::PC),
|
|
ArmCoreRegId::Cpsr => Some(reg::CPSR),
|
|
_ => None,
|
|
}
|
|
}
|
|
|
|
impl Target for Emu {
|
|
type Arch = arch::arm::Armv4t;
|
|
type Error = &'static str;
|
|
|
|
fn base_ops(&mut self) -> target::ext::base::BaseOps<Self::Arch, Self::Error> {
|
|
target::ext::base::BaseOps::SingleThread(self)
|
|
}
|
|
|
|
fn sw_breakpoint(&mut self) -> Option<target::ext::breakpoints::SwBreakpointOps<Self>> {
|
|
Some(self)
|
|
}
|
|
|
|
fn hw_watchpoint(&mut self) -> Option<target::ext::breakpoints::HwWatchpointOps<Self>> {
|
|
Some(self)
|
|
}
|
|
|
|
fn extended_mode(&mut self) -> Option<target::ext::extended_mode::ExtendedModeOps<Self>> {
|
|
Some(self)
|
|
}
|
|
|
|
fn monitor_cmd(&mut self) -> Option<target::ext::monitor_cmd::MonitorCmdOps<Self>> {
|
|
Some(self)
|
|
}
|
|
|
|
fn section_offsets(&mut self) -> Option<target::ext::section_offsets::SectionOffsetsOps<Self>> {
|
|
Some(self)
|
|
}
|
|
|
|
fn target_description_xml_override(
|
|
&mut self,
|
|
) -> Option<target::ext::target_description_xml_override::TargetDescriptionXmlOverrideOps<Self>>
|
|
{
|
|
Some(self)
|
|
}
|
|
}
|
|
|
|
impl SingleThreadOps for Emu {
|
|
fn resume(
|
|
&mut self,
|
|
action: ResumeAction,
|
|
check_gdb_interrupt: &mut dyn FnMut() -> bool,
|
|
) -> Result<StopReason<u32>, Self::Error> {
|
|
let event = match action {
|
|
ResumeAction::Step => match self.step() {
|
|
Some(e) => e,
|
|
None => return Ok(StopReason::DoneStep),
|
|
},
|
|
ResumeAction::Continue => {
|
|
let mut cycles = 0;
|
|
loop {
|
|
if let Some(event) = self.step() {
|
|
break event;
|
|
};
|
|
|
|
// check for GDB interrupt every 1024 instructions
|
|
cycles += 1;
|
|
if cycles % 1024 == 0 && check_gdb_interrupt() {
|
|
return Ok(StopReason::GdbInterrupt);
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
Ok(match event {
|
|
Event::Halted => StopReason::Halted,
|
|
Event::Break => StopReason::HwBreak,
|
|
Event::WatchWrite(addr) => StopReason::Watch {
|
|
kind: WatchKind::Write,
|
|
addr,
|
|
},
|
|
Event::WatchRead(addr) => StopReason::Watch {
|
|
kind: WatchKind::Read,
|
|
addr,
|
|
},
|
|
})
|
|
}
|
|
|
|
fn read_registers(&mut self, regs: &mut arch::arm::reg::ArmCoreRegs) -> TargetResult<(), Self> {
|
|
let mode = self.cpu.mode();
|
|
|
|
for i in 0..13 {
|
|
regs.r[i] = self.cpu.reg_get(mode, i as u8);
|
|
}
|
|
regs.sp = self.cpu.reg_get(mode, reg::SP);
|
|
regs.lr = self.cpu.reg_get(mode, reg::LR);
|
|
regs.pc = self.cpu.reg_get(mode, reg::PC);
|
|
regs.cpsr = self.cpu.reg_get(mode, reg::CPSR);
|
|
|
|
Ok(())
|
|
}
|
|
|
|
fn write_registers(&mut self, regs: &arch::arm::reg::ArmCoreRegs) -> TargetResult<(), Self> {
|
|
let mode = self.cpu.mode();
|
|
|
|
for i in 0..13 {
|
|
self.cpu.reg_set(mode, i, regs.r[i as usize]);
|
|
}
|
|
self.cpu.reg_set(mode, reg::SP, regs.sp);
|
|
self.cpu.reg_set(mode, reg::LR, regs.lr);
|
|
self.cpu.reg_set(mode, reg::PC, regs.pc);
|
|
self.cpu.reg_set(mode, reg::CPSR, regs.cpsr);
|
|
|
|
Ok(())
|
|
}
|
|
|
|
fn read_register(
|
|
&mut self,
|
|
reg_id: arch::arm::reg::id::ArmCoreRegId,
|
|
dst: &mut [u8],
|
|
) -> TargetResult<(), Self> {
|
|
if let Some(i) = cpu_reg_id(reg_id) {
|
|
let w = self.cpu.reg_get(self.cpu.mode(), i);
|
|
dst.copy_from_slice(&w.to_le_bytes());
|
|
Ok(())
|
|
} else {
|
|
Err(().into())
|
|
}
|
|
}
|
|
|
|
fn write_register(
|
|
&mut self,
|
|
reg_id: arch::arm::reg::id::ArmCoreRegId,
|
|
val: &[u8],
|
|
) -> TargetResult<(), Self> {
|
|
let w = u32::from_le_bytes(
|
|
val.try_into()
|
|
.map_err(|_| TargetError::Fatal("invalid data"))?,
|
|
);
|
|
if let Some(i) = cpu_reg_id(reg_id) {
|
|
self.cpu.reg_set(self.cpu.mode(), i, w);
|
|
Ok(())
|
|
} else {
|
|
Err(().into())
|
|
}
|
|
}
|
|
|
|
fn read_addrs(&mut self, start_addr: u32, data: &mut [u8]) -> TargetResult<(), Self> {
|
|
for (addr, val) in (start_addr..).zip(data.iter_mut()) {
|
|
*val = self.mem.r8(addr)
|
|
}
|
|
Ok(())
|
|
}
|
|
|
|
fn write_addrs(&mut self, start_addr: u32, data: &[u8]) -> TargetResult<(), Self> {
|
|
for (addr, val) in (start_addr..).zip(data.iter().copied()) {
|
|
self.mem.w8(addr, val)
|
|
}
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
impl target::ext::breakpoints::SwBreakpoint for Emu {
|
|
fn add_sw_breakpoint(&mut self, addr: u32) -> TargetResult<bool, Self> {
|
|
self.breakpoints.push(addr);
|
|
Ok(true)
|
|
}
|
|
|
|
fn remove_sw_breakpoint(&mut self, addr: u32) -> TargetResult<bool, Self> {
|
|
match self.breakpoints.iter().position(|x| *x == addr) {
|
|
None => return Ok(false),
|
|
Some(pos) => self.breakpoints.remove(pos),
|
|
};
|
|
|
|
Ok(true)
|
|
}
|
|
}
|
|
|
|
impl target::ext::breakpoints::HwWatchpoint for Emu {
|
|
fn add_hw_watchpoint(&mut self, addr: u32, kind: WatchKind) -> TargetResult<bool, Self> {
|
|
match kind {
|
|
WatchKind::Write => self.watchpoints.push(addr),
|
|
WatchKind::Read => self.watchpoints.push(addr),
|
|
WatchKind::ReadWrite => self.watchpoints.push(addr),
|
|
};
|
|
|
|
Ok(true)
|
|
}
|
|
|
|
fn remove_hw_watchpoint(&mut self, addr: u32, kind: WatchKind) -> TargetResult<bool, Self> {
|
|
let pos = match self.watchpoints.iter().position(|x| *x == addr) {
|
|
None => return Ok(false),
|
|
Some(pos) => pos,
|
|
};
|
|
|
|
match kind {
|
|
WatchKind::Write => self.watchpoints.remove(pos),
|
|
WatchKind::Read => self.watchpoints.remove(pos),
|
|
WatchKind::ReadWrite => self.watchpoints.remove(pos),
|
|
};
|
|
|
|
Ok(true)
|
|
}
|
|
}
|