2015-08-05 19:48:30 +02:00
|
|
|
#!/bin/sh
|
|
|
|
#
|
|
|
|
# Unified Segger JLink script for RIOT
|
|
|
|
#
|
|
|
|
# This script is supposed to be called from RIOTs make system,
|
|
|
|
# as it depends on certain environment variables. An
|
|
|
|
#
|
|
|
|
# Global environment variables used:
|
|
|
|
# JLINK: JLink command name, default: "JLinkExe"
|
|
|
|
# JLINK_SERVER: JLink GCB server command name, default: "JLinkGDBDerver"
|
|
|
|
# JLINK_DEVICE: Device identifier used by JLink
|
2016-04-03 13:44:54 +02:00
|
|
|
# JLINK_SERIAL: Device serial used by JLink
|
2015-08-05 19:48:30 +02:00
|
|
|
# JLINK_IF: Interface used by JLink, default: "SWD"
|
|
|
|
# JLINK_SPEED: Interface clock speed to use (in kHz), default "2000"
|
2017-06-30 20:38:17 +02:00
|
|
|
# FLASH_ADDR: Starting address of the target's flash memory, default: "0"
|
2015-08-05 19:48:30 +02:00
|
|
|
# JLINK_PRE_FLASH: Additional JLink commands to execute before flashing
|
|
|
|
# JLINK_POST_FLASH: Additional JLink commands to execute after flashing
|
|
|
|
#
|
|
|
|
# The script supports the following actions:
|
|
|
|
#
|
|
|
|
# flash: flash a given hex file to the target.
|
|
|
|
# hexfile is expected in ihex format and is pointed to
|
|
|
|
# by BINFILE environment variable
|
|
|
|
#
|
|
|
|
# options:
|
|
|
|
# BINFILE: path to the binary file that is flashed
|
|
|
|
#
|
2017-10-29 15:04:48 +01:00
|
|
|
# debug: starts JLink as GDB server in the background and
|
2015-08-05 19:48:30 +02:00
|
|
|
# connects to the server with the GDB client specified by
|
|
|
|
# the board (DBG environment variable)
|
|
|
|
#
|
|
|
|
# options:
|
|
|
|
# GDB_PORT: port opened for GDB connections
|
|
|
|
# TELNET_PORT: port opened for telnet connections
|
2018-03-29 19:54:48 +02:00
|
|
|
# DBG: debugger client command, default: 'gdb -q'
|
2015-08-05 19:48:30 +02:00
|
|
|
# TUI: if TUI!=null, the -tui option will be used
|
|
|
|
# ELFFILE: path to the ELF file to debug
|
|
|
|
#
|
2017-10-29 15:04:48 +01:00
|
|
|
# debug-server: starts JLink as GDB server, but does not connect to
|
2015-08-05 19:48:30 +02:00
|
|
|
# to it with any frontend. This might be useful when using
|
|
|
|
# IDEs.
|
|
|
|
#
|
|
|
|
# options:
|
|
|
|
# GDB_PORT: port opened for GDB connections
|
|
|
|
# TELNET_PORT: port opened for telnet connections
|
|
|
|
#
|
|
|
|
# reset: triggers a hardware reset of the target board
|
|
|
|
#
|
|
|
|
#
|
|
|
|
# @author Hauke Peteresen <hauke.petersen@fu-berlin.de>
|
|
|
|
|
|
|
|
# default GDB port
|
|
|
|
_GDB_PORT=3333
|
|
|
|
# default telnet port
|
|
|
|
_TELNET_PORT=4444
|
|
|
|
# default JLink command, interface and speed
|
|
|
|
_JLINK=JLinkExe
|
|
|
|
_JLINK_SERVER=JLinkGDBServer
|
|
|
|
_JLINK_IF=SWD
|
|
|
|
_JLINK_SPEED=2000
|
2018-01-15 23:00:16 +01:00
|
|
|
# default terminal frontend
|
2018-05-04 11:48:27 +02:00
|
|
|
_JLINK_TERMPROG=${RIOTTOOLS}/pyterm/pyterm
|
2018-01-15 23:00:16 +01:00
|
|
|
_JLINK_TERMFLAGS="-ts 19021"
|
2015-08-05 19:48:30 +02:00
|
|
|
|
|
|
|
#
|
|
|
|
# a couple of tests for certain configuration options
|
|
|
|
#
|
|
|
|
test_config() {
|
|
|
|
if [ -z "${JLINK}" ]; then
|
|
|
|
JLINK=${_JLINK}
|
|
|
|
fi
|
|
|
|
if [ -z "${JLINK_SERVER}" ]; then
|
|
|
|
JLINK_SERVER=${_JLINK_SERVER}
|
|
|
|
fi
|
|
|
|
if [ -z "${JLINK_IF}" ]; then
|
|
|
|
JLINK_IF=${_JLINK_IF}
|
|
|
|
fi
|
|
|
|
if [ -z "${JLINK_SPEED}" ]; then
|
|
|
|
JLINK_SPEED=${_JLINK_SPEED}
|
|
|
|
fi
|
|
|
|
if [ -z "${JLINK_DEVICE}" ]; then
|
|
|
|
echo "Error: No target device defined in JLINK_DEVICE env var"
|
|
|
|
exit 1
|
|
|
|
fi
|
2017-06-30 20:38:17 +02:00
|
|
|
if [ -z "${FLASH_ADDR}" ]; then
|
|
|
|
echo "Error: No flash address defined in FLASH_ADDR env var"
|
|
|
|
exit 1
|
2015-08-05 19:48:30 +02:00
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
test_hexfile() {
|
|
|
|
if [ ! -f "${HEXFILE}" ]; then
|
|
|
|
echo "Error: Unable to locate HEXFILE"
|
|
|
|
echo " (${HEXFILE})"
|
|
|
|
exit 1
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
test_ports() {
|
|
|
|
if [ -z "${GDB_PORT}" ]; then
|
|
|
|
GDB_PORT=${_GDB_PORT}
|
|
|
|
fi
|
|
|
|
if [ -z "${TELNET_PORT}" ]; then
|
|
|
|
TELNET_PORT=${_TELNET_PORT}
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
test_elffile() {
|
|
|
|
if [ ! -f "${ELFFILE}" ]; then
|
|
|
|
echo "Error: Unable to locate ELFFILE"
|
|
|
|
echo " (${ELFFILE})"
|
|
|
|
exit 1
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
test_tui() {
|
|
|
|
if [ -n "${TUI}" ]; then
|
|
|
|
TUI=-tui
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
2016-04-03 13:44:54 +02:00
|
|
|
test_serial() {
|
|
|
|
if [ -n "${JLINK_SERIAL}" ]; then
|
|
|
|
JLINK_SERIAL_SERVER="-select usb='${JLINK_SERIAL}'"
|
|
|
|
JLINK_SERIAL="-SelectEmuBySN '${JLINK_SERIAL}'"
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
2017-10-29 15:04:48 +01:00
|
|
|
test_dbg() {
|
|
|
|
if [ -z "${DBG}" ]; then
|
2018-03-29 19:54:48 +02:00
|
|
|
DBG="${GDB}"
|
2017-10-29 15:04:48 +01:00
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
2018-01-15 23:00:16 +01:00
|
|
|
test_term() {
|
|
|
|
if [ -z "${JLINK_TERMPROG}" ]; then
|
|
|
|
JLINK_TERMPROG="${_JLINK_TERMPROG}"
|
|
|
|
fi
|
|
|
|
if [ -z "${JLINK_TERMFLAGS}" ]; then
|
|
|
|
JLINK_TERMFLAGS="${_JLINK_TERMFLAGS}"
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
2015-08-05 19:48:30 +02:00
|
|
|
#
|
|
|
|
# now comes the actual actions
|
|
|
|
#
|
|
|
|
do_flash() {
|
|
|
|
test_config
|
2016-04-03 13:44:54 +02:00
|
|
|
test_serial
|
2015-08-05 19:48:30 +02:00
|
|
|
test_hexfile
|
|
|
|
# clear any existing contents in burn file
|
|
|
|
/bin/echo -n "" > ${BINDIR}/burn.seg
|
|
|
|
# create temporary burn file
|
|
|
|
if [ ! -z "${JLINK_PRE_FLASH}" ]; then
|
2016-09-30 19:10:01 +02:00
|
|
|
printf "${JLINK_PRE_FLASH}\n" >> ${BINDIR}/burn.seg
|
2015-08-05 19:48:30 +02:00
|
|
|
fi
|
2017-06-30 20:38:17 +02:00
|
|
|
echo "loadbin ${HEXFILE} ${FLASH_ADDR}" >> ${BINDIR}/burn.seg
|
2015-08-05 19:48:30 +02:00
|
|
|
if [ ! -z "${JLINK_POST_FLASH}" ]; then
|
2016-09-30 19:10:01 +02:00
|
|
|
printf "${JLINK_POST_FLASH}\n" >> ${BINDIR}/burn.seg
|
2015-08-05 19:48:30 +02:00
|
|
|
fi
|
2018-05-04 11:48:27 +02:00
|
|
|
cat ${RIOTTOOLS}/jlink/reset.seg >> ${BINDIR}/burn.seg
|
2015-08-05 19:48:30 +02:00
|
|
|
# flash device
|
2016-04-03 13:44:54 +02:00
|
|
|
sh -c "${JLINK} ${JLINK_SERIAL} \
|
|
|
|
-device '${JLINK_DEVICE}' \
|
2015-08-05 19:48:30 +02:00
|
|
|
-speed '${JLINK_SPEED}' \
|
|
|
|
-if '${JLINK_IF}' \
|
2016-07-11 22:34:25 +02:00
|
|
|
-jtagconf -1,-1 \
|
2015-08-05 19:48:30 +02:00
|
|
|
-commandfile '${BINDIR}/burn.seg'"
|
|
|
|
}
|
|
|
|
|
|
|
|
do_debug() {
|
|
|
|
test_config
|
2016-04-03 13:44:54 +02:00
|
|
|
test_serial
|
2015-08-05 19:48:30 +02:00
|
|
|
test_elffile
|
|
|
|
test_ports
|
|
|
|
test_tui
|
2017-10-29 15:04:48 +01:00
|
|
|
test_dbg
|
2015-08-05 19:48:30 +02:00
|
|
|
# start the JLink GDB server
|
2016-04-03 13:44:54 +02:00
|
|
|
sh -c "${JLINK_SERVER} ${JLINK_SERIAL_SERVER} \
|
|
|
|
-device '${JLINK_DEVICE}' \
|
2015-08-05 19:48:30 +02:00
|
|
|
-speed '${JLINK_SPEED}' \
|
|
|
|
-if '${JLINK_IF}' \
|
|
|
|
-port '${GDB_PORT}' \
|
|
|
|
-telnetport '${TELNET_PORT}'" &
|
|
|
|
# save PID for terminating the server afterwards
|
|
|
|
DBG_PID=$?
|
|
|
|
# connect to the GDB server
|
2016-04-22 12:00:10 +02:00
|
|
|
${DBG} -q ${TUI} -ex "tar ext :${GDB_PORT}" ${ELFFILE}
|
2015-08-05 19:48:30 +02:00
|
|
|
# clean up
|
|
|
|
kill ${DBG_PID}
|
|
|
|
}
|
|
|
|
|
|
|
|
do_debugserver() {
|
|
|
|
test_ports
|
|
|
|
test_config
|
2016-04-03 13:44:54 +02:00
|
|
|
test_serial
|
2015-08-05 19:48:30 +02:00
|
|
|
# start the JLink GDB server
|
2016-04-03 13:44:54 +02:00
|
|
|
sh -c "${JLINK_SERVER} ${JLINK_SERIAL_SERVER} \
|
|
|
|
-device '${JLINK_DEVICE}' \
|
2015-08-05 19:48:30 +02:00
|
|
|
-speed '${JLINK_SPEED}' \
|
|
|
|
-if '${JLINK_IF}' \
|
|
|
|
-port '${GDB_PORT}' \
|
|
|
|
-telnetport '${TELNET_PORT}'"
|
|
|
|
}
|
|
|
|
|
|
|
|
do_reset() {
|
|
|
|
test_config
|
2016-04-03 13:44:54 +02:00
|
|
|
test_serial
|
2015-08-05 19:48:30 +02:00
|
|
|
# reset the board
|
2016-04-03 13:44:54 +02:00
|
|
|
sh -c "${JLINK} ${JLINK_SERIAL} \
|
|
|
|
-device '${JLINK_DEVICE}' \
|
2015-08-05 19:48:30 +02:00
|
|
|
-speed '${JLINK_SPEED}' \
|
|
|
|
-if '${JLINK_IF}' \
|
2016-07-11 22:34:25 +02:00
|
|
|
-jtagconf -1,-1 \
|
2018-05-04 11:48:27 +02:00
|
|
|
-commandfile '${RIOTTOOLS}/jlink/reset.seg'"
|
2015-08-05 19:48:30 +02:00
|
|
|
}
|
|
|
|
|
2018-01-15 23:00:16 +01:00
|
|
|
do_term() {
|
|
|
|
test_config
|
|
|
|
test_serial
|
|
|
|
test_term
|
|
|
|
|
|
|
|
# temporary file that save the JLink pid
|
|
|
|
JLINK_PIDFILE=$(mktemp -t "jilnk_pid.XXXXXXXXXX")
|
|
|
|
# will be called by trap
|
|
|
|
cleanup() {
|
|
|
|
JLINK_PID="$(cat ${JLINK_PIDFILE})"
|
|
|
|
kill ${JLINK_PID}
|
|
|
|
rm -r "${JLINK_PIDFILE}"
|
|
|
|
exit 0
|
|
|
|
}
|
|
|
|
# cleanup after script terminates
|
|
|
|
trap "cleanup ${JLINK_PIDFILE}" EXIT
|
|
|
|
# don't trapon Ctrl+C, because JLink keeps running
|
|
|
|
trap '' INT
|
|
|
|
# start Jlink as RTT server
|
2018-09-10 23:28:42 +02:00
|
|
|
sh -c "${JLINK} ${JLINK_SERIAL} \
|
2018-01-15 23:00:16 +01:00
|
|
|
-device '${JLINK_DEVICE}' \
|
|
|
|
-speed '${JLINK_SPEED}' \
|
|
|
|
-if '${JLINK_IF}' \
|
|
|
|
-jtagconf -1,-1 \
|
2018-09-10 20:04:20 +02:00
|
|
|
-commandfile '${RIOTTOOLS}/jlink/term.seg' >/dev/null & \
|
2018-01-15 23:00:16 +01:00
|
|
|
echo \$! > $JLINK_PIDFILE" &
|
|
|
|
|
|
|
|
sh -c "${JLINK_TERMPROG} ${JLINK_TERMFLAGS}"
|
|
|
|
}
|
|
|
|
|
2015-08-05 19:48:30 +02:00
|
|
|
#
|
|
|
|
# parameter dispatching
|
|
|
|
#
|
|
|
|
ACTION="$1"
|
|
|
|
shift # pop $1 from $@
|
|
|
|
|
|
|
|
case "${ACTION}" in
|
|
|
|
flash)
|
|
|
|
echo "### Flashing Target ###"
|
2017-06-30 20:38:17 +02:00
|
|
|
echo "### Flashing at address ${FLASH_ADDR} ###"
|
2015-08-05 19:48:30 +02:00
|
|
|
do_flash "$@"
|
|
|
|
;;
|
|
|
|
debug)
|
|
|
|
echo "### Starting Debugging ###"
|
|
|
|
do_debug "$@"
|
|
|
|
;;
|
|
|
|
debug-server)
|
|
|
|
echo "### Starting GDB Server ###"
|
|
|
|
do_debugserver "$@"
|
|
|
|
;;
|
|
|
|
reset)
|
|
|
|
echo "### Resetting Target ###"
|
|
|
|
do_reset "$@"
|
|
|
|
;;
|
2018-01-15 23:00:16 +01:00
|
|
|
term_rtt)
|
|
|
|
echo "### Starting RTT terminal ###"
|
|
|
|
do_term
|
|
|
|
;;
|
2015-08-05 19:48:30 +02:00
|
|
|
*)
|
|
|
|
echo "Usage: $0 {flash|debug|debug-server|reset}"
|
|
|
|
;;
|
|
|
|
esac
|