根据规范我的输出应该是这样的:
00:00
00:01
00:12
01:23
12:34
23:45
34:56
45:67
56:78
67:89
78:9A
89:AB
9A:BC
AB:CD
BC:DE
CD:EF
DE:F0
EF:01
F0:12
01:23
12:34
等等。我编写了2个汇编子程序,一个用于执行十六进制转换,另一个用于从测试程序中打印数字,我现在得到的输出开始类似于它的外观:
0:
1:
2:
3:
4:
5:
6:
7:
8:
9:
A:
B:
C:
D:
E:
F:
0:
1:
2:
3:
4:
5:
所以看起来我正确得到最右边的数字,但我该如何得到其余的数字?它在assignment中说
示例:内存位置0x0047114包含二进制值0101 0011(十六进制0x53)内存位置0x0047115包含二进制文件 值0001 0110(十六进制0x16)存储单元0x0047116包含 二进制值0000 0000(十六进制0x00)内存位置 0x0047117包含二进制值0000 0000(十六进制0x00) 调用者将寄存器r4设置为0x0047114并调用puttime; puttime打印 16:53在一条新线上。
我最近发布的其他问题有更多背景知识,这些问题具有相同的作业,但侧重于其他部分:
https://stackoverflow.com/questions/12105322/how-to-proceed-coding-for-this-assignment
Difference between load word and move?
https://stackoverflow.com/questions/12152323/why-is-load-word-not-working
我的转换程序是:
.global hexasc
.text
.align 2
hexasc: movi r8, 0x09
movi r9, 0x0f
andi r4, r4, 0x0f #keep only 4 bits
bgt r4, r8, L1 #is x>9?
movi r2, 0x30
add r2, r2, r4
andi r2, r2, 0xff
ret
L1: movi r2, 0x37
add r2, r2, r4
andi r2, r2, 0xff
ret
我认为以上是正确且有效的,但我不确定如何编程获取变量的中间部分并将其传递给我的子程序:
.macro PUSH reg
subi sp, sp, 4
stw \reg, 0(sp)
.endm
.macro POP reg
ldw \reg, 0(sp)
addi sp, sp, 4
.endm
.global puttime
.text
.align 2
puttime:
PUSH r31
ldw r8,0(r4) # put the variable in register 8
PUSH r8
nop
movi r4, 0x0A # put newline char in register 4
movia r9,putchar # copy subroutine address to a register
callr r9 # call subroutine via register
POP r8
mov r4, r8
call hexasc
mov r4, r2
movia r9,putchar # copy subroutine address to a register
callr r9
movi r4, 0x3A
movia r9, putchar
callr r9
POP r31
ret
非常感谢任何帮助。
答案 0 :(得分:1)
我认为您需要两次调用hexasc,输入数据中的每个十六进制数字一次。在每次调用hexasc之间,您需要将输入数据右移4位,因为hexasc仅转换输入数据的底部4位。
顺便说一下,使用PUSH和POP不是惯用的MIPS代码。在MIPS中,通常在进入例程时从堆栈指针中减去一个常量值,并在从路由退出时添加常量值。像这样:
puttime:
subi sp, 8 # reserve space for 2 words on the stack
sw r31, 4(sp)
...
lw r31, 4(sp)
addi sp, 8
ret