我需要将C代码从下面更改为MIPS代码,但是我是MIPS的新手,并坚持使用它。
这是C代码:
int main()
{
int a, b, result;
if(a == b)
result = a*b;
else
result = assess(a, b);
return result;
}
int assess(int a, int b)
{
if(b<a)
return upgrade(a, b);
else
return demote(a, b);
}
int upgrade(int a, int b)
{
return 4*(a+b);
}
int demote(int a, int b)
{
return 4*(b-a);
}
这里是我编写的MIPS代码,该代码无法正常工作(我知道这里有主要的错误和错误)。因为我不熟悉该语言,所以我的主要问题是使用堆栈,返回和调用函数。
.data
a:.word 8
b:.word 8
result:.word 0
main:
li $s0 a
li $s1 b
li $s3 result
beq $s0,$s1,Resultmul ELSE
add $s3,$s3,assess
assess:
blt $s1,$s0,upgrade
bge $s1,$0,demote
Resultmul :
mul $s3,$s1,$0
upgrade:
addi $sp,$sp,-4
sw $0,0($sp)
add $t1,$a0,$a1
mul $t1,$t1,4
add $v0,$s0,$zero
lw $s0,0($sp)
addi $sp,$sp,4
jr $ra
demote:
addi $sp,$sp,-4
sw $0,0($sp)
sub $t1,$a0,$a1
mul $t1,$t1,4
add $v0,$s0,$zero
lw $s0,0($sp)
addi $sp,$sp,4
jr $ra
[如果有人可以帮助,那将是救生员。
我不会为您提供完整的解决方案,因此您可以从练习中学习,但是我建议您使用模板来工作,例如此处的模板。
我使用Visual Studio Code(具有MIPS支持和更好的MIPS支持以突出显示),其中每个空格或制表符都使我可以折叠它(这就是为什么到处都有这些空格的原因)和QtSpim运行此命令并获取输出64
。
而且,我习惯于使用制表符进行编码;对我来说更清楚了,但可能对您而言不是,所以很抱歉,如果您必须删除所有标签和注释。
######################## pseudo ####################################
#
# int main()
# {
# int a, b, result;
# if(a == b)
# result = a*b;
# else
# result = assess(a, b);
# return result;
# }
#
# int assess(int a, int b)
# {
# if(b<a)
# return upgrade(a, b);
# else
# return demote(a, b);
# }
#
# int upgrade(int a, int b)
# {
# return 4*(a+b);
# }
#
# int demote(int a, int b)
# {
# return 4*(b-a);
# }
#
###################### DATA Segment ################################
.data
A:
.word 8
B:
.word 8
result:
.word 0
###################### CODE Segment ################################
.text
.globl main
main:
这里您犯了一个小错误:您已经存储个单词,所以您也应该load个单词。否则,您必须输入li $t0, 8
。
# int A = A, B = B, result
lw $s0, A # $s0 = A
lw $s1, B # $s1 = B
lw $s2, result # $s2 = result
# if (a == b)
bne $s0, $s1, noteq # if $s0 != $s1 then noteq
# result = multiply(a,b);
move $a0, $s0 # $a0 = $s0
move $a1, $s1 # $a1 = $s1
jal multiply # jump to multiply and save position to $ra
sw $v0, result #
b end # branch to end
# else
noteq:
# result = assess(a,b);
move $a0, $s0 # $a0 = $s0
move $a1, $s1 # $a1 = $s1
# jal assess # jump to assess and save position to $ra
sw $v0, result #
b end # branch to end (this rule can be left out)
end:
# printf("%i", result)
li $v0, 1 # $v0 = 1
lw $a0, result #
syscall
# exit()
li $v0, 10 # $v0 = 10
syscall
由于它们是伪代码中的函数,因此在您的程序集中也应将它们视为函数。表示它们是通过j
(用于退出之类的非返回函数)或jal
和(以jr
返回)调用的。
我做了一个完全不必要的功能multiply
,向您展示了模板,这对于较大的功能非常方便。
###################### FUNC Segment ################################
###################### FUNCTION ####################################
# multiply(A, B)
#
# Purpose: <General description>
######################## i/0 #######################################
# Input:
# $a0 = A
# $a1 = B
# Output:
# $v0 = value
# Registers being used:
# $s0 = A
# $s1 = B
# $s2 = value
######################## pseudo ####################################
#
# int multiply(int A, int B)
# {
# return A * B;
# }
#
######################## <code> ####################################
multiply:#(A, B)
总是存储要覆盖的寄存器的内容,因此您可以调用其他函数而不会丢失任何内容。还应立即初始化存储在新寄存器中$a0-$a3
中的参数,因为使用syscall
打印内容时可能会覆盖这些参数。
# store(&return, parameters that are about overwritten)
sub $sp, $sp, 16 # $sp = $sp - 16
sw $ra, 0($sp) #
sw $s0, 4($sp) #
sw $s1, 8($sp) #
sw $s2, 12($sp) #
# int A = A, B = B, value
move $s0, $a0 # $s0 = $a0
move $s1, $a1 # $s1 = $a1
这是该函数的非常短的正文。如您所知,存储所有这些参数是愚蠢的,因此请不要使用这些开销函数。
# value = A * B;
mul $s2, $s0, $s1
这是处理函数的返回。在较大的函数中,大多数时候您将需要标签来跳转到最终处理程序。我总是在foo
之类的函数foo_label1
中调用标签,但这只是我的建议。
move $v0, $s2 # $v0 = $s2
# restore()
lw $ra, 0($sp) #
lw $s0, 4($sp) #
lw $s1, 8($sp) #
lw $s2, 12($sp) #
addi $sp, $sp, 12 # $sp = $sp + 12
# return index
jr $ra # jump to $ra
######################## </code> ###################################
这里是其他功能的空模板。
###################### FUNCTION ####################################
# <name of function>
#
# Purpose: <General description>
######################## i/0 #######################################
# Input:
# $a0 =
# $a1 =
# $a2 =
# $a3 =
# Output:
# $v0 =
# Registers being used:
# $t0 =
# $t1 =
# $t2 =
# $s0 =
# $s1 =
# $s2 =
# $s3 =
# $s4 =
######################## pseudo ####################################
#
# int assess(int a, int b)
# {
# if(b<a)
# return upgrade(a, b);
# else
# return demote(a, b);
# }
#
######################## <code> ####################################
#
######################## </code> ###################################
P.S。我已经重命名了您的变量名,因为b
可能会导致错误。