aboutsummaryrefslogtreecommitdiff
path: root/Debug
diff options
context:
space:
mode:
authorLorenz Meier <lm@inf.ethz.ch>2014-03-03 10:32:01 +0100
committerLorenz Meier <lm@inf.ethz.ch>2014-03-03 10:32:01 +0100
commit5cefd4811f7913402baabf93939ed4fbf4727654 (patch)
tree5e97cf7a45c4c5b7fd6e01e05e2ec87bd2aaf2aa /Debug
parent7bb583d6e21a0fa7e51f147d23997aaeb7e218c9 (diff)
parent3dd3ba4637bfe6d665f20c1e5712ac22131b5b22 (diff)
downloadpx4-firmware-mavlink_variable_length.tar.gz
px4-firmware-mavlink_variable_length.tar.bz2
px4-firmware-mavlink_variable_length.zip
Merged with master, cleanup of varlength prototypemavlink_variable_length
Diffstat (limited to 'Debug')
-rw-r--r--Debug/ARMv7M2
-rw-r--r--Debug/Nuttx.py371
-rw-r--r--Debug/PX42
-rw-r--r--Debug/dot.gdbinit13
-rw-r--r--Debug/olimex-px4fmu-debug.cfg22
-rw-r--r--Debug/openocd.gdbinit21
-rw-r--r--Debug/px4fmu-v1-board.cfg38
-rwxr-xr-xDebug/runopenocd.sh5
-rw-r--r--Debug/stm32f4x.cfg64
9 files changed, 472 insertions, 66 deletions
diff --git a/Debug/ARMv7M b/Debug/ARMv7M
index 803d96453..4c539b345 100644
--- a/Debug/ARMv7M
+++ b/Debug/ARMv7M
@@ -56,7 +56,7 @@ define vecstate
if $mmfsr & (1<<3)
printf " during exception return"
end
- if $mmfsr & (1<<0)
+ if $mmfsr & (1<<1)
printf " during data access"
end
if $mmfsr & (1<<0)
diff --git a/Debug/Nuttx.py b/Debug/Nuttx.py
new file mode 100644
index 000000000..093edd0e0
--- /dev/null
+++ b/Debug/Nuttx.py
@@ -0,0 +1,371 @@
+# GDB/Python functions for dealing with NuttX
+
+import gdb, gdb.types
+
+class NX_register_set(object):
+ """Copy of the registers for a given context"""
+
+ v7_regmap = {
+ 'R13': 0,
+ 'SP': 0,
+ 'PRIORITY': 1,
+ 'R4': 2,
+ 'R5': 3,
+ 'R6': 4,
+ 'R7': 5,
+ 'R8': 6,
+ 'R9': 7,
+ 'R10': 8,
+ 'R11': 9,
+ 'EXC_RETURN': 10,
+ 'R0': 11,
+ 'R1': 12,
+ 'R2': 13,
+ 'R3': 14,
+ 'R12': 15,
+ 'R14': 16,
+ 'LR': 16,
+ 'R15': 17,
+ 'PC': 17,
+ 'XPSR': 18,
+ }
+
+ v7em_regmap = {
+ 'R13': 0,
+ 'SP': 0,
+ 'PRIORITY': 1,
+ 'R4': 2,
+ 'R5': 3,
+ 'R6': 4,
+ 'R7': 5,
+ 'R8': 6,
+ 'R9': 7,
+ 'R10': 8,
+ 'R11': 9,
+ 'EXC_RETURN': 10,
+ 'R0': 27,
+ 'R1': 28,
+ 'R2': 29,
+ 'R3': 30,
+ 'R12': 31,
+ 'R14': 32,
+ 'LR': 32,
+ 'R15': 33,
+ 'PC': 33,
+ 'XPSR': 34,
+ }
+
+ regs = dict()
+
+ def __init__(self, xcpt_regs):
+ if xcpt_regs is None:
+ self.regs['R0'] = long(gdb.parse_and_eval('$r0'))
+ self.regs['R1'] = long(gdb.parse_and_eval('$r1'))
+ self.regs['R2'] = long(gdb.parse_and_eval('$r2'))
+ self.regs['R3'] = long(gdb.parse_and_eval('$r3'))
+ self.regs['R4'] = long(gdb.parse_and_eval('$r4'))
+ self.regs['R5'] = long(gdb.parse_and_eval('$r5'))
+ self.regs['R6'] = long(gdb.parse_and_eval('$r6'))
+ self.regs['R7'] = long(gdb.parse_and_eval('$r7'))
+ self.regs['R8'] = long(gdb.parse_and_eval('$r8'))
+ self.regs['R9'] = long(gdb.parse_and_eval('$r9'))
+ self.regs['R10'] = long(gdb.parse_and_eval('$r10'))
+ self.regs['R11'] = long(gdb.parse_and_eval('$r11'))
+ self.regs['R12'] = long(gdb.parse_and_eval('$r12'))
+ self.regs['R13'] = long(gdb.parse_and_eval('$r13'))
+ self.regs['SP'] = long(gdb.parse_and_eval('$sp'))
+ self.regs['R14'] = long(gdb.parse_and_eval('$r14'))
+ self.regs['LR'] = long(gdb.parse_and_eval('$lr'))
+ self.regs['R15'] = long(gdb.parse_and_eval('$r15'))
+ self.regs['PC'] = long(gdb.parse_and_eval('$pc'))
+ self.regs['XPSR'] = long(gdb.parse_and_eval('$xpsr'))
+ else:
+ for key in self.v7em_regmap.keys():
+ self.regs[key] = int(xcpt_regs[self.v7em_regmap[key]])
+
+
+ @classmethod
+ def with_xcpt_regs(cls, xcpt_regs):
+ return cls(xcpt_regs)
+
+ @classmethod
+ def for_current(cls):
+ return cls(None)
+
+ def __format__(self, format_spec):
+ return format_spec.format(
+ registers = self.registers
+ )
+
+ @property
+ def registers(self):
+ return self.regs
+
+
+class NX_task(object):
+ """Reference to a NuttX task and methods for introspecting it"""
+
+ def __init__(self, tcb_ptr):
+ self._tcb = tcb_ptr.dereference()
+ self._group = self._tcb['group'].dereference()
+ self.pid = tcb_ptr['pid']
+
+ @classmethod
+ def for_tcb(cls, tcb):
+ """return a task with the given TCB pointer"""
+ pidhash_sym = gdb.lookup_global_symbol('g_pidhash')
+ pidhash_value = pidhash_sym.value()
+ pidhash_type = pidhash_sym.type
+ for i in range(pidhash_type.range()[0],pidhash_type.range()[1]):
+ pidhash_entry = pidhash_value[i]
+ if pidhash_entry['tcb'] == tcb:
+ return cls(pidhash_entry['tcb'])
+ return None
+
+ @classmethod
+ def for_pid(cls, pid):
+ """return a task for the given PID"""
+ pidhash_sym = gdb.lookup_global_symbol('g_pidhash')
+ pidhash_value = pidhash_sym.value()
+ pidhash_type = pidhash_sym.type
+ for i in range(pidhash_type.range()[0],pidhash_type.range()[1]):
+ pidhash_entry = pidhash_value[i]
+ if pidhash_entry['pid'] == pid:
+ return cls(pidhash_entry['tcb'])
+ return None
+
+ @staticmethod
+ def pids():
+ """return a list of all PIDs"""
+ pidhash_sym = gdb.lookup_global_symbol('g_pidhash')
+ pidhash_value = pidhash_sym.value()
+ pidhash_type = pidhash_sym.type
+ result = []
+ for i in range(pidhash_type.range()[0],pidhash_type.range()[1]):
+ entry = pidhash_value[i]
+ pid = int(entry['pid'])
+ if pid is not -1:
+ result.append(pid)
+ return result
+
+ @staticmethod
+ def tasks():
+ """return a list of all tasks"""
+ tasks = []
+ for pid in NX_task.pids():
+ tasks.append(NX_task.for_pid(pid))
+ return tasks
+
+ def _state_is(self, state):
+ """tests the current state of the task against the passed-in state name"""
+ statenames = gdb.types.make_enum_dict(gdb.lookup_type('enum tstate_e'))
+ if self._tcb['task_state'] == statenames[state]:
+ return True
+ return False
+
+ @property
+ def stack_used(self):
+ """calculate the stack used by the thread"""
+ if 'stack_used' not in self.__dict__:
+ stack_base = self._tcb['stack_alloc_ptr'].cast(gdb.lookup_type('unsigned char').pointer())
+ if stack_base == 0:
+ self.__dict__['stack_used'] = 0
+ else:
+ stack_limit = self._tcb['adj_stack_size']
+ for offset in range(0, stack_limit):
+ if stack_base[offset] != 0xff:
+ break
+ self.__dict__['stack_used'] = stack_limit - offset
+ return self.__dict__['stack_used']
+
+ @property
+ def name(self):
+ """return the task's name"""
+ return self._tcb['name'].string()
+
+ @property
+ def state(self):
+ """return the name of the task's current state"""
+ statenames = gdb.types.make_enum_dict(gdb.lookup_type('enum tstate_e'))
+ for name,value in statenames.iteritems():
+ if value == self._tcb['task_state']:
+ return name
+ return 'UNKNOWN'
+
+ @property
+ def waiting_for(self):
+ """return a description of what the task is waiting for, if it is waiting"""
+ if self._state_is('TSTATE_WAIT_SEM'):
+ waitsem = self._tcb['waitsem'].dereference()
+ waitsem_holder = waitsem['holder']
+ holder = NX_task.for_tcb(waitsem_holder['htcb'])
+ if holder is not None:
+ return '{}({})'.format(waitsem.address, holder.name)
+ else:
+ return '{}(<bad holder>)'.format(waitsem.address)
+ if self._state_is('TSTATE_WAIT_SIG'):
+ return 'signal'
+ return None
+
+ @property
+ def is_waiting(self):
+ """tests whether the task is waiting for something"""
+ if self._state_is('TSTATE_WAIT_SEM') or self._state_is('TSTATE_WAIT_SIG'):
+ return True
+
+ @property
+ def is_runnable(self):
+ """tests whether the task is runnable"""
+ if (self._state_is('TSTATE_TASK_PENDING') or
+ self._state_is('TSTATE_TASK_READYTORUN') or
+ self._state_is('TSTATE_TASK_RUNNING')):
+ return True
+ return False
+
+ @property
+ def file_descriptors(self):
+ """return a dictionary of file descriptors and inode pointers"""
+ filelist = self._group['tg_filelist']
+ filearray = filelist['fl_files']
+ result = dict()
+ for i in range(filearray.type.range()[0],filearray.type.range()[1]):
+ inode = long(filearray[i]['f_inode'])
+ if inode != 0:
+ result[i] = inode
+ return result
+
+ @property
+ def registers(self):
+ if 'registers' not in self.__dict__:
+ registers = dict()
+ if self._state_is('TSTATE_TASK_RUNNING'):
+ registers = NX_register_set.for_current().registers
+ else:
+ context = self._tcb['xcp']
+ regs = context['regs']
+ registers = NX_register_set.with_xcpt_regs(regs).registers
+
+ self.__dict__['registers'] = registers
+ return self.__dict__['registers']
+
+ def __repr__(self):
+ return "<NX_task {}>".format(self.pid)
+
+ def __str__(self):
+ return "{}:{}".format(self.pid, self.name)
+
+ def __format__(self, format_spec):
+ return format_spec.format(
+ pid = self.pid,
+ name = self.name,
+ state = self.state,
+ waiting_for = self.waiting_for,
+ stack_used = self.stack_used,
+ stack_limit = self._tcb['adj_stack_size'],
+ file_descriptors = self.file_descriptors,
+ registers = self.registers
+ )
+
+class NX_show_task (gdb.Command):
+ """(NuttX) prints information about a task"""
+
+ def __init__(self):
+ super(NX_show_task, self).__init__("show task", gdb.COMMAND_USER)
+
+ def invoke(self, arg, from_tty):
+ t = NX_task.for_pid(int(arg))
+ if t is not None:
+ my_fmt = 'PID:{pid} name:{name} state:{state}\n'
+ my_fmt += ' stack used {stack_used} of {stack_limit}\n'
+ if t.is_waiting:
+ my_fmt += ' waiting for {waiting_for}\n'
+ my_fmt += ' open files: {file_descriptors}\n'
+ my_fmt += ' R0 {registers[R0]:#010x} {registers[R1]:#010x} {registers[R2]:#010x} {registers[R3]:#010x}\n'
+ my_fmt += ' R4 {registers[R4]:#010x} {registers[R5]:#010x} {registers[R6]:#010x} {registers[R7]:#010x}\n'
+ my_fmt += ' R8 {registers[R8]:#010x} {registers[R9]:#010x} {registers[R10]:#010x} {registers[R11]:#010x}\n'
+ my_fmt += ' R12 {registers[PC]:#010x}\n'
+ my_fmt += ' SP {registers[SP]:#010x} LR {registers[LR]:#010x} PC {registers[PC]:#010x} XPSR {registers[XPSR]:#010x}\n'
+ print format(t, my_fmt)
+
+class NX_show_tasks (gdb.Command):
+ """(NuttX) prints a list of tasks"""
+
+ def __init__(self):
+ super(NX_show_tasks, self).__init__('show tasks', gdb.COMMAND_USER)
+
+ def invoke(self, args, from_tty):
+ tasks = NX_task.tasks()
+ for t in tasks:
+ print format(t, '{pid:<2} {name:<16} {state:<20} {stack_used:>4}/{stack_limit:<4}')
+
+NX_show_task()
+NX_show_tasks()
+
+class NX_show_heap (gdb.Command):
+ """(NuttX) prints the heap"""
+
+ def __init__(self):
+ super(NX_show_heap, self).__init__('show heap', gdb.COMMAND_USER)
+ if gdb.lookup_type('struct mm_allocnode_s').sizeof == 8:
+ self._allocflag = 0x80000000
+ self._allocnodesize = 8
+ else:
+ self._allocflag = 0x8000
+ self._allocnodesize = 4
+
+ def _node_allocated(self, allocnode):
+ if allocnode['preceding'] & self._allocflag:
+ return True
+ return False
+
+ def _node_size(self, allocnode):
+ return allocnode['size'] & ~self._allocflag
+
+ def _print_allocations(self, region_start, region_end):
+ if region_start >= region_end:
+ raise gdb.GdbError('heap region {} corrupt'.format(hex(region_start)))
+ nodecount = region_end - region_start
+ print 'heap {} - {}'.format(region_start, region_end)
+ cursor = 1
+ while cursor < nodecount:
+ allocnode = region_start[cursor]
+ if self._node_allocated(allocnode):
+ state = ''
+ else:
+ state = '(free)'
+ print ' {} {} {}'.format(allocnode.address + 8, self._node_size(allocnode), state)
+ cursor += self._node_size(allocnode) / self._allocnodesize
+
+ def invoke(self, args, from_tty):
+ heap = gdb.lookup_global_symbol('g_mmheap').value()
+ nregions = heap['mm_nregions']
+ region_starts = heap['mm_heapstart']
+ region_ends = heap['mm_heapend']
+ print '{} heap(s)'.format(nregions)
+ # walk the heaps
+ for i in range(0, nregions):
+ self._print_allocations(region_starts[i], region_ends[i])
+
+NX_show_heap()
+
+class NX_show_interrupted_thread (gdb.Command):
+ """(NuttX) prints the register state of an interrupted thread when in interrupt/exception context"""
+
+ def __init__(self):
+ super(NX_show_interrupted_thread, self).__init__('show interrupted-thread', gdb.COMMAND_USER)
+
+ def invoke(self, args, from_tty):
+ regs = gdb.lookup_global_symbol('current_regs').value()
+ if regs is 0:
+ raise gdb.GdbError('not in interrupt context')
+ else:
+ registers = NX_register_set.with_xcpt_regs(regs)
+ my_fmt = ''
+ my_fmt += ' R0 {registers[R0]:#010x} {registers[R1]:#010x} {registers[R2]:#010x} {registers[R3]:#010x}\n'
+ my_fmt += ' R4 {registers[R4]:#010x} {registers[R5]:#010x} {registers[R6]:#010x} {registers[R7]:#010x}\n'
+ my_fmt += ' R8 {registers[R8]:#010x} {registers[R9]:#010x} {registers[R10]:#010x} {registers[R11]:#010x}\n'
+ my_fmt += ' R12 {registers[PC]:#010x}\n'
+ my_fmt += ' SP {registers[SP]:#010x} LR {registers[LR]:#010x} PC {registers[PC]:#010x} XPSR {registers[XPSR]:#010x}\n'
+ print format(registers, my_fmt)
+
+NX_show_interrupted_thread()
diff --git a/Debug/PX4 b/Debug/PX4
index 085cffe43..e99228ee0 100644
--- a/Debug/PX4
+++ b/Debug/PX4
@@ -27,7 +27,7 @@ define _perf_print
# PC_COUNT
if $hdr->type == 0
set $count = (struct perf_ctr_count *)$hdr
- printf "%llu events,\n", $count->event_count;
+ printf "%llu events\n", $count->event_count
end
# PC_ELPASED
if $hdr->type == 1
diff --git a/Debug/dot.gdbinit b/Debug/dot.gdbinit
new file mode 100644
index 000000000..d70410bc7
--- /dev/null
+++ b/Debug/dot.gdbinit
@@ -0,0 +1,13 @@
+# copy the file to .gdbinit in your Firmware tree, and adjust the path
+# below to match your system
+# For example:
+# target extended /dev/serial/by-id/usb-Black_Sphere_Technologies_Black_Magic_Probe_DDE5A1C4-if00
+# target extended /dev/ttyACM4
+
+
+monitor swdp_scan
+attach 1
+monitor vector_catch disable hard
+set mem inaccessible-by-default off
+set print pretty
+source Debug/PX4
diff --git a/Debug/olimex-px4fmu-debug.cfg b/Debug/olimex-px4fmu-debug.cfg
new file mode 100644
index 000000000..61d70070d
--- /dev/null
+++ b/Debug/olimex-px4fmu-debug.cfg
@@ -0,0 +1,22 @@
+# program a bootable device load on a mavstation
+# To run type openocd -f mavprogram.cfg
+
+source [find interface/olimex-arm-usb-ocd-h.cfg]
+source [find px4fmu-v1-board.cfg]
+
+init
+halt
+
+# Find the flash inside this CPU
+flash probe 0
+
+# erase it (128 pages) then program and exit
+
+#flash erase_sector 0 0 127
+# stm32f1x mass_erase 0
+
+# It seems that Pat's image has a start address offset of 0x1000 but the vectors need to be at zero, so fixbin.sh moves things around
+#flash write_bank 0 fixed.bin 0
+#flash write_image firmware.elf
+#shutdown
+
diff --git a/Debug/openocd.gdbinit b/Debug/openocd.gdbinit
new file mode 100644
index 000000000..92d78b58d
--- /dev/null
+++ b/Debug/openocd.gdbinit
@@ -0,0 +1,21 @@
+target remote :3333
+
+# Don't let GDB get confused while stepping
+define hook-step
+ mon cortex_m maskisr on
+end
+define hookpost-step
+ mon cortex_m maskisr off
+end
+
+mon init
+mon stm32_init
+# mon reset halt
+mon poll
+mon cortex_m maskisr auto
+set mem inaccessible-by-default off
+set print pretty
+source Debug/PX4
+
+echo PX4 resumed, press ctrl-c to interrupt\n
+continue
diff --git a/Debug/px4fmu-v1-board.cfg b/Debug/px4fmu-v1-board.cfg
new file mode 100644
index 000000000..19b862a2d
--- /dev/null
+++ b/Debug/px4fmu-v1-board.cfg
@@ -0,0 +1,38 @@
+# The latest defaults in OpenOCD 0.7.0 are actually prettymuch correct for the px4fmu
+
+# increase working area to 32KB for faster flash programming
+set WORKAREASIZE 0x8000
+
+source [find target/stm32f4x.cfg]
+
+# needed for px4
+reset_config trst_only
+
+proc stm32_reset {} {
+ reset halt
+# FIXME - needed to init periphs on reset
+# 0x40023800 RCC base
+# 0x24 RCC_APB2 0x75933
+# RCC_APB2 0
+}
+
+# perform init that is required on each connection to the target
+proc stm32_init {} {
+
+ # force jtag to not shutdown during sleep
+ #uint32_t cr = getreg32(STM32_DBGMCU_CR);
+ #cr |= DBGMCU_CR_STANDBY | DBGMCU_CR_STOP | DBGMCU_CR_SLEEP;
+ #putreg32(cr, STM32_DBGMCU_CR);
+ mww 0xe0042004 00000007
+}
+
+# if srst is not fitted use SYSRESETREQ to
+# perform a soft reset
+cortex_m reset_config sysresetreq
+
+# Let GDB directly program elf binaries
+gdb_memory_map enable
+
+# doesn't work yet
+gdb_flash_program disable
+
diff --git a/Debug/runopenocd.sh b/Debug/runopenocd.sh
new file mode 100755
index 000000000..6258fccfb
--- /dev/null
+++ b/Debug/runopenocd.sh
@@ -0,0 +1,5 @@
+#!/bin/bash
+
+DIR="$( cd "$( dirname "${BASH_SOURCE[0]}" )" && pwd )"
+
+openocd -f interface/olimex-arm-usb-ocd-h.cfg -f $DIR/px4fmu-v1-board.cfg
diff --git a/Debug/stm32f4x.cfg b/Debug/stm32f4x.cfg
deleted file mode 100644
index 28bfcfbbb..000000000
--- a/Debug/stm32f4x.cfg
+++ /dev/null
@@ -1,64 +0,0 @@
-# script for stm32f2xxx
-
-if { [info exists CHIPNAME] } {
- set _CHIPNAME $CHIPNAME
-} else {
- set _CHIPNAME stm32f4xxx
-}
-
-if { [info exists ENDIAN] } {
- set _ENDIAN $ENDIAN
-} else {
- set _ENDIAN little
-}
-
-# Work-area is a space in RAM used for flash programming
-# By default use 64kB
-if { [info exists WORKAREASIZE] } {
- set _WORKAREASIZE $WORKAREASIZE
-} else {
- set _WORKAREASIZE 0x10000
-}
-
-# JTAG speed should be <= F_CPU/6. F_CPU after reset is 8MHz, so use F_JTAG = 1MHz
-#
-# Since we may be running of an RC oscilator, we crank down the speed a
-# bit more to be on the safe side. Perhaps superstition, but if are
-# running off a crystal, we can run closer to the limit. Note
-# that there can be a pretty wide band where things are more or less stable.
-jtag_khz 1000
-
-jtag_nsrst_delay 100
-jtag_ntrst_delay 100
-
-#jtag scan chain
-if { [info exists CPUTAPID ] } {
- set _CPUTAPID $CPUTAPID
-} else {
- # See STM Document RM0033
- # Section 32.6.3 - corresponds to Cortex-M3 r2p0
- set _CPUTAPID 0x4ba00477
-}
-jtag newtap $_CHIPNAME cpu -irlen 4 -ircapture 0x1 -irmask 0xf -expected-id $_CPUTAPID
-
-if { [info exists BSTAPID ] } {
- set _BSTAPID $BSTAPID
-} else {
- # See STM Document RM0033
- # Section 32.6.2
- #
- set _BSTAPID 0x06413041
-}
-jtag newtap $_CHIPNAME bs -irlen 5 -expected-id $_BSTAPID
-
-set _TARGETNAME $_CHIPNAME.cpu
-target create $_TARGETNAME cortex_m3 -endian $_ENDIAN -chain-position $_TARGETNAME -rtos auto
-
-$_TARGETNAME configure -work-area-phys 0x20000000 -work-area-size $_WORKAREASIZE -work-area-backup 0
-
-set _FLASHNAME $_CHIPNAME.flash
-flash bank $_FLASHNAME stm32f2x 0 0 0 0 $_TARGETNAME
-
-# if srst is not fitted use SYSRESETREQ to
-# perform a soft reset
-cortex_m3 reset_config sysresetreq