我在调试用ARM7微控制器(AT91SAM7S64)的汇编语言编写的简单程序时遇到问题。我使用gcc,gdb和OpenOCD。 我的程序被正确加载到目标并且工作得很好(它闪烁了一个led)。但是当我调用'next'命令时,gdb会跳过某些源代码行。
以下是源代码片段:
Reset_Handler:
LDR R0, =0x0100
LDR R1, =PIOA_PER
STR R0, [R1]
LDR R1, =PIOA_OER
STR R0, [R1]
uuu:
bl wait;
LDR R1, =PIOA_SODR
STR R0, [R1]
uuu1:
bl wait;
LDR R2, =PIOA_CODR
STR R0, [R2]
b uuu;
@ one second delay
wait:
.............
.............
.end
要获得gdb输出(见下文),我使用了“目标模拟”而不是真正的目标,但结果是相同的。
(gdb) target sim
Connected to the simulator.
(gdb) load
Loading section .text, size 0xc8 vma 0x100000
Start address 0x100000
Transfer rate: 1600 bits in <1 sec.
(gdb) b Reset_Handler
Breakpoint 1 at 0x100064: file main.s, line 59.
(gdb) run
Starting program: C:\Arm\Projects\Asm/./main.elf
Breakpoint 1, Reset_Handler () at main.s:60
60 LDR R1, =PIOA_PER
(gdb) n
61 STR R0, [R1]
(gdb) n
63 LDR R1, =PIOA_OER
(gdb) n
64 STR R0, [R1]
(gdb) n
uuu () at main.s:66
66 bl wait;
(gdb) n
67 LDR R1, =PIOA_SODR
(gdb) n
68 STR R0, [R1]
(gdb) n <<<<<--------- Here the problem begins
67 LDR R1, =PIOA_SODR
(gdb) n
68 STR R0, [R1]
(gdb) n
67 LDR R1, =PIOA_SODR
(gdb) n
68 STR R0, [R1]
(gdb) stepi <<<<<------ Doing a 'stepi' command allows to pass below 'uuu1' label
uuu1 () at main.s:70
70 bl wait;
(gdb) n
71 LDR R2, =PIOA_CODR
(gdb) n
72 STR R0, [R2]
(gdb) n
73 b uuu;
(gdb) n <<<<<--------- Here the problem begins again
71 LDR R2, =PIOA_CODR
(gdb) n
72 STR R0, [R2]
(gdb) n
73 b uuu;
(gdb) n
71 LDR R2, =PIOA_CODR
(gdb) where
#0 uuu1 () at main.s:71
#1 0x00100084 in uuu1 () at main.s:70
Backtrace stopped: previous frame identical to this frame (corrupt stack?)
(gdb)
似乎gdb假设'uuu1'是一个单独的函数,并且由于某种原因跳过它。如果我删除'uuu1'标签,问题就会消失。此标签不在任何地方使用,但gdb行为看起来很奇怪。我一直试图找到任何解决方案很长一段时间,但取得了重大成果。使用gcc选项'-fomit-frame-pointer'没有帮助。 我该怎么办?
gdb和gcc的版本:
arm-none-eabi-gdb --version
GNU gdb (GDB) 7.1
..........
This GDB was configured as "--host=i686-pc-mingw32 --target=arm-none-eabi".
arm-none-eabi-gcc --version
arm-none-eabi-gcc (GCC) 4.5.1
我的MakeFile:
TRGT = arm-none-eabi-
CC = $(TRGT)gcc
CP = $(TRGT)objcopy
AS = $(TRGT)gcc -x assembler-with-cpp
#AS = $(TRGT)as
LD = $(TRGT)ld
OBJDUMP = $(TRGT)objdump
LD_SCRIPT = main.ld
MCU = arm7tdmi
#DEBUG = stabs
DEBUG = dwarf-2
ASFLAGS = -mcpu=$(MCU) -g$(DEBUG)
LDFLAGS = -T $(LD_SCRIPT)
all: main.elf main.lss
@echo Done!
main.elf : main.o
@echo Linking $<
$(CC) -nostartfiles $(LDFLAGS) $< -o $@
main.o : main.s
@echo Compiling $<
$(AS) -c $(ASFLAGS) $< -o $@
提前感谢您的帮助!
答案 0 :(得分:1)
在代码问题中尝试使用“si”代替n。
“n”或多或少是下一个语句,“si”是下一个asm指令。
如果asm的调试代码不正确,“si”可能仍然允许你执行它,利用了在这种情况下“statement”和“assembler”级别相同的事实。
答案 1 :(得分:0)
通过汇编程序说明,使用ni
代替next
和si
代替step
。
答案 2 :(得分:0)
我知道这有点旧,但您应该尝试将-O0
标志添加到编译器中。我阻止gcc进行任何优化,这可能导致像你得到的问题。