mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
a354b10964
On error Jlink exits with a no error code by default. From the JLink User Guide: '-ExitOnError' has the same meaning as the 'exitonerror' command 'exitonerror' command This command toggles whether J-Link Commander exits on error or not. 1: J-Link Commander will now exit on Error. 0: J-Link Commander will no longer exit on Error. Executing 'flash/reset' without a board connected now correctly returns an error. For 'term' it does not show an error due to the way it is handled internally. It also returns an error when the board fails to do an operation when it is in a state where it cannot be flashed for example.
295 lines
8.1 KiB
Bash
Executable File
295 lines
8.1 KiB
Bash
Executable File
#!/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
|
|
# JLINK_SERIAL: Device serial used by JLink
|
|
# JLINK_IF: Interface used by JLink, default: "SWD"
|
|
# JLINK_SPEED: Interface clock speed to use (in kHz), default "2000"
|
|
# FLASH_ADDR: Starting address of the target's flash memory, default: "0"
|
|
# IMAGE_OFFSET: Offset from the targets flash memory, for flashing the image
|
|
# 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 <binfile>
|
|
#
|
|
# flash given binary format file to the target.
|
|
#
|
|
# options:
|
|
# <binfile>: path to the binary file that is flashed
|
|
#
|
|
# debug: debug <elffile>
|
|
#
|
|
# starts JLink as GDB server in the background and
|
|
# connects to the server with the GDB client specified by
|
|
# the board (DBG environment variable)
|
|
#
|
|
# options:
|
|
# <elffile>: path to the ELF file to debug
|
|
# GDB_PORT: port opened for GDB connections
|
|
# TELNET_PORT: port opened for telnet connections
|
|
# DBG: debugger client command, default: 'gdb -q'
|
|
# TUI: if TUI!=null, the -tui option will be used
|
|
#
|
|
# debug-server: starts JLink as GDB server, but does not connect to
|
|
# 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>
|
|
|
|
# Set IMAGE_OFFSET to zero by default.
|
|
: ${IMAGE_OFFSET:=0}
|
|
|
|
# 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
|
|
# default terminal frontend
|
|
_JLINK_TERMPROG=${RIOTTOOLS}/pyterm/pyterm
|
|
_JLINK_TERMFLAGS="-ts 19021"
|
|
|
|
#
|
|
# 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
|
|
if [ -z "${FLASH_ADDR}" ]; then
|
|
echo "Error: No flash address defined in FLASH_ADDR env var"
|
|
exit 1
|
|
fi
|
|
}
|
|
|
|
test_binfile() {
|
|
if [ ! -f "${BINFILE}" ]; then
|
|
echo "Error: Unable to locate BINFILE"
|
|
echo " (${BINFILE})"
|
|
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
|
|
}
|
|
|
|
test_serial() {
|
|
if [ -n "${JLINK_SERIAL}" ]; then
|
|
JLINK_SERIAL_SERVER="-select usb='${JLINK_SERIAL}'"
|
|
JLINK_SERIAL="-SelectEmuBySN '${JLINK_SERIAL}'"
|
|
fi
|
|
}
|
|
|
|
test_dbg() {
|
|
if [ -z "${DBG}" ]; then
|
|
DBG="${GDB}"
|
|
fi
|
|
}
|
|
|
|
test_term() {
|
|
if [ -z "${JLINK_TERMPROG}" ]; then
|
|
JLINK_TERMPROG="${_JLINK_TERMPROG}"
|
|
fi
|
|
if [ -z "${JLINK_TERMFLAGS}" ]; then
|
|
JLINK_TERMFLAGS="${_JLINK_TERMFLAGS}"
|
|
fi
|
|
}
|
|
|
|
#
|
|
# now comes the actual actions
|
|
#
|
|
do_flash() {
|
|
BINFILE=$1
|
|
test_config
|
|
test_serial
|
|
test_binfile
|
|
# clear any existing contents in burn file
|
|
/bin/echo -n "" > ${BINDIR}/burn.seg
|
|
# create temporary burn file
|
|
if [ ! -z "${JLINK_PRE_FLASH}" ]; then
|
|
printf "${JLINK_PRE_FLASH}\n" >> ${BINDIR}/burn.seg
|
|
fi
|
|
# address to flash is hex formatted, as required by JLink
|
|
ADDR_TO_FLASH=$(printf "0x%08x\n" "$((${FLASH_ADDR} + ${IMAGE_OFFSET}))")
|
|
echo "loadbin ${BINFILE} ${ADDR_TO_FLASH}" >> ${BINDIR}/burn.seg
|
|
if [ ! -z "${JLINK_POST_FLASH}" ]; then
|
|
printf "${JLINK_POST_FLASH}\n" >> ${BINDIR}/burn.seg
|
|
fi
|
|
cat ${RIOTTOOLS}/jlink/reset.seg >> ${BINDIR}/burn.seg
|
|
# flash device
|
|
sh -c "${JLINK} ${JLINK_SERIAL} \
|
|
-ExitOnError 1 \
|
|
-device '${JLINK_DEVICE}' \
|
|
-speed '${JLINK_SPEED}' \
|
|
-if '${JLINK_IF}' \
|
|
-jtagconf -1,-1 \
|
|
-commandfile '${BINDIR}/burn.seg'"
|
|
}
|
|
|
|
do_debug() {
|
|
ELFFILE=$1
|
|
test_config
|
|
test_serial
|
|
test_elffile
|
|
test_ports
|
|
test_tui
|
|
test_dbg
|
|
# start the JLink GDB server
|
|
sh -c "${JLINK_SERVER} ${JLINK_SERIAL_SERVER} \
|
|
-device '${JLINK_DEVICE}' \
|
|
-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
|
|
${DBG} -q ${TUI} -ex "tar ext :${GDB_PORT}" ${ELFFILE}
|
|
# clean up
|
|
kill ${DBG_PID}
|
|
}
|
|
|
|
do_debugserver() {
|
|
test_ports
|
|
test_config
|
|
test_serial
|
|
# start the JLink GDB server
|
|
sh -c "${JLINK_SERVER} ${JLINK_SERIAL_SERVER} \
|
|
-device '${JLINK_DEVICE}' \
|
|
-speed '${JLINK_SPEED}' \
|
|
-if '${JLINK_IF}' \
|
|
-port '${GDB_PORT}' \
|
|
-telnetport '${TELNET_PORT}'"
|
|
}
|
|
|
|
do_reset() {
|
|
test_config
|
|
test_serial
|
|
# reset the board
|
|
sh -c "${JLINK} ${JLINK_SERIAL} \
|
|
-ExitOnError 1 \
|
|
-device '${JLINK_DEVICE}' \
|
|
-speed '${JLINK_SPEED}' \
|
|
-if '${JLINK_IF}' \
|
|
-jtagconf -1,-1 \
|
|
-commandfile '${RIOTTOOLS}/jlink/reset.seg'"
|
|
}
|
|
|
|
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
|
|
sh -c "${JLINK} ${JLINK_SERIAL} \
|
|
-ExitOnError 1 \
|
|
-device '${JLINK_DEVICE}' \
|
|
-speed '${JLINK_SPEED}' \
|
|
-if '${JLINK_IF}' \
|
|
-jtagconf -1,-1 \
|
|
-commandfile '${RIOTTOOLS}/jlink/term.seg' >/dev/null & \
|
|
echo \$! > $JLINK_PIDFILE" &
|
|
|
|
sh -c "${JLINK_TERMPROG} ${JLINK_TERMFLAGS}"
|
|
}
|
|
|
|
#
|
|
# parameter dispatching
|
|
#
|
|
ACTION="$1"
|
|
shift # pop $1 from $@
|
|
|
|
case "${ACTION}" in
|
|
flash)
|
|
echo "### Flashing Target ###"
|
|
echo "### Flashing at base address ${FLASH_ADDR} with offset ${IMAGE_OFFSET} ###"
|
|
do_flash "$@"
|
|
;;
|
|
debug)
|
|
echo "### Starting Debugging ###"
|
|
do_debug "$@"
|
|
;;
|
|
debug-server)
|
|
echo "### Starting GDB Server ###"
|
|
do_debugserver "$@"
|
|
;;
|
|
reset)
|
|
echo "### Resetting Target ###"
|
|
do_reset "$@"
|
|
;;
|
|
term_rtt)
|
|
echo "### Starting RTT terminal ###"
|
|
do_term
|
|
;;
|
|
*)
|
|
echo "Usage: $0 {flash|debug|debug-server|reset}"
|
|
echo " flash <binfile>"
|
|
echo " debug <elffile>"
|
|
;;
|
|
esac
|