MIPS:解析和修改字符串

时间:2019-03-23 13:34:44

标签: assembly mips mars-simulator

我必须在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应该包含写换行符的地址。 但是我可能会误解了指令保存字节和加载字节的实际工作方式。

任何帮助或建议都将不胜感激:)

1 个答案:

答案 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。