Skip to content
This repository has been archived by the owner on Mar 26, 2024. It is now read-only.

Commit

Permalink
Add a register method and use it in SWD example
Browse files Browse the repository at this point in the history
At least the J-Link clone I have (serial 123456, HW 7.00, FW
"J-Link ARM-OB STM32 compiled Aug 22 2012 19:52:04") absolutely requires
a registered connection, at least for SWD usage. Nothing ever gets ACKed otherwise.

OpenOCD always calls jaylink_register (with all zeros for info)
when the REGISTER capability is present.
  • Loading branch information
valpackett committed Jan 21, 2021
1 parent 20e3f18 commit 4f04da1
Show file tree
Hide file tree
Showing 2 changed files with 47 additions and 2 deletions.
6 changes: 5 additions & 1 deletion examples/swdump.rs
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@
//!
//! This example assumes that the above "bug" (or misspecification?) is present.
use jaylink::{CommunicationSpeed, Interface, JayLink};
use jaylink::{Capabilities, CommunicationSpeed, Connection, Interface, JayLink};
use log::trace;
use std::{cmp, fmt};
use structopt::StructOpt;
Expand Down Expand Up @@ -296,6 +296,10 @@ fn run(opts: Opts) -> Result<(), SwdError> {
let khz = cmp::min(opts.speed.unwrap_or(200), 0xfffe);
probe.set_speed(CommunicationSpeed::khz(khz).unwrap())?;

if probe.capabilities().contains(Capabilities::REGISTER) {
probe.register(Connection::default())?;
}

probe.swj_seq()?;

let dpidr = probe.raw_read(Port::Debug, 0b0000)?;
Expand Down
43 changes: 42 additions & 1 deletion src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -90,7 +90,7 @@ pub use self::interface::{Interface, InterfaceIter, Interfaces};
use self::bits::IteratorExt as _;
use self::error::ResultExt as _;
use bitflags::bitflags;
use byteorder::{LittleEndian, ReadBytesExt};
use byteorder::{BigEndian, LittleEndian, ReadBytesExt, WriteBytesExt};
use io::Cursor;
use log::{debug, trace, warn};
use std::cell::{Cell, RefCell, RefMut};
Expand Down Expand Up @@ -161,6 +161,14 @@ enum Command {

ReadConfig = 0xF2,
WriteConfig = 0xF3,

Register = 0x09,
}

#[repr(u8)]
enum RegisterCommand {
Register = 0x64,
Unregister = 0x65,
}

#[repr(u8)]
Expand Down Expand Up @@ -207,6 +215,15 @@ impl SwoStatus {
}
}

#[derive(Default, Debug)]
pub struct Connection {
handle: u16,
pid: u32,
hid: Option<std::net::Ipv4Addr>,
iid: u8,
cid: u8,
}

/// A handle to a J-Link USB device.
///
/// This is the main interface type of this library. There are multiple ways of obtaining an
Expand Down Expand Up @@ -899,6 +916,30 @@ impl JayLink {
Ok(())
}

/// Registers a connection on the device.
///
/// This requires the [`REGISTER`] capability.
///
/// **Note**: This may be **required** on some devices for SWD to work at all.
///
/// [`REGISTER`]: struct.Capabilities.html#associatedconstant.REGISTER
pub fn register(&self, conn: Connection) -> Result<()> {
self.require_capabilities(Capabilities::REGISTER)?;
let mut buf = Vec::with_capacity(14);
buf.push(Command::Register as u8);
buf.push(RegisterCommand::Register as u8);
buf.write_u32::<LittleEndian>(conn.pid).unwrap();
buf.write_u32::<BigEndian>(conn.hid.unwrap_or(std::net::Ipv4Addr::LOCALHOST).into())
.unwrap();
buf.push(conn.iid);
buf.push(conn.cid);
buf.write_u16::<LittleEndian>(conn.handle).unwrap();
self.write_cmd(&buf)?;
let mut buf = [0; 76];
self.read(&mut buf)?;
Ok(())
}

/// Performs a JTAG I/O operation.
///
/// This will shift out data on `TMS` (pin 7) and `TDI` (pin 5), while reading data shifted
Expand Down

0 comments on commit 4f04da1

Please sign in to comment.