我必须在MIPS中编写一个程序,在其中我必须打开一个文件,该文件的名称由用户输入,并使用MARS模拟器对其进行解析。 但是,由于syscall 8(它从用户输入中读取一个字符串)遵循C函数fgets的语义,因此必须尝试从字符串中删除换行\ n字符,然后才能尝试打开文件。
我能够解析字符串(在这里,我打印每个字符的ASCII代码以用于调试)。 但是,当我尝试使用
行将换行字符更改为NULL字符时sb $zero 0($t1)
火星在执行过程中遇到异常:
“运行时异常为0x00400050:地址超出范围0x00000000”
如果我将这一行注释掉,程序将正常运行并打印字符串的每个ascii代码。
.data
prompt : .asciiz "enter filename:\n"
lf : .asciiz "\n"
space : .asciiz " "
.text
la $a0 prompt # asking user for filename
jal print_string
jal read_string
la $t0 ($a0) #copying the string address
lb $t1 0($t0) #loading the first byte
parsing_string : #loop
beq $t1 0 remove_line_feed # reaching the NULL character '\0'
move $a0 $t1 #printing the ascii code
li $v0 1
syscall
la $a0 space #printing a space
li $v0 4
syscall
add $t0 $t0 1 # loading
lbu $t1 ($t0) # the next byte
j parsing_string
remove_line_feed :
sb $zero 0($t1) # runtime exception if i uncomment this line
j end
end :
li $v0 10 # syscall 10 : end program
syscall
read_string : #read a string in $a0
li $a1 100 # $a1 = maximum number of characters to read
li $v0 8 # syscall 8 : fgets()
syscall
jr $ra
print_string : #print string in $a0
li $v0 4 # syscall 4 : print string
syscall
jr $ra
我不明白我在做什么错,在程序的这一点上,寄存器$ t1应该包含写换行符的地址。 但是我可能会误解了指令保存字节和加载字节的实际工作方式。
任何帮助或建议都将不胜感激:)
答案 0 :(得分:0)
在asm代码中
sb $zero 0($t1) # runtime exception if i uncomment this line
您将$t1
用作要清除的字节的地址。
但是不是。 你用过
la $t0 ($a0) #copying the string address
,您所有的访问都使用$t0
。
在parsing_string
函数的末尾,您读取地址$t0
上的字符并将其放入$t1
:
lbu $t1 ($t0) # the next byte
然后返回parsing_string
,测试t1是否为0
beq $t1 0 remove_line_feed # reaching the NULL character '\0'
转到remove_line_feed
remove_line_feed :
sb $zero 0($t1) # runtime exception if i uncomment this line
j end
您尝试在地址0+$t1
上写零的位置(刚刚测试过等于零)。
并且模拟器正确地告诉您您尝试在地址0处写入并生成错误。
如果您将代码更改为
remove_line_feed :
sb $zero 0($t0) # runtime exception if i uncomment this line
j end
这将消除运行时错误。
但是,这不会使您的代码正确。您要做的是清除{\ {1}}之前的\ em,您的代码应该是
\0
或者,您可以更改比较,而不是搜索字符串的最后remove_line_feed :
sb $zero -1($t0)
j end
,而是搜索10(ascii换行符)并将相同的char替换为0。