0

该程序的文档非常有限。为了做到这一点,我几乎没有什么可以借鉴的。除了程序的PDF,只有这个:

.data
format_str:     .asciiz "%dth of %s:\n%s version %i.%i is being tested!"
s1:             .asciiz "June"
s2:             .asciiz "EduMIPS64"
fs_addr:        .space  4
                .word   5
s1_addr:        .space  4
s2_addr:        .space  4
                .word   0
                .word   5
test:
.code
        daddi r5, r0, format_str
        sw r5, fs_addr(r0)
        daddi r2, r0, s1
        daddi r3, r0, s2
        sd r2, s1_addr(r0)
        sd r3, s2_addr(r0)
        daddi r14, r0, fs_addr
        syscall 5
        syscall 0

你可以在这里查看。EDU/WINMIPS64 与常规 MIPS 程序集有很大不同

有几段解释这一点,没有太大帮助。无论如何,这里会打印一个格式化的字符串,以及一些字符串参数(存储在 s1 和 s2 中)和整数参数(它们存储在哪里?)。

我有两个数组存储在内存中。我已经亲自对它们执行了指令,现在我想打印它们。我如何在这样的格式化字符串上提供这两个整数(它们是双字,即需要存储 8 个字节)?帮助材料没有解释。

这是到目前为止我设法创建的代码(大量评论):

....
dadd $s4, $zero, $zero                  ; i = 0
printOutput:                                    ; print results from the 2 arrays
        beq $s4, 960, end                       ; if (i = size = 960 / 8) goto end
        dadd $s1, $s4, $s2                      ; full address of 1st array
        dadd $s0, $s4, $s3                      ; full address of 2nd array

        daddi $a3, $zero, printString           ; address ofstring to be printed stored in $a3
        ld $v0, 0($s1)                          ; $v0 will be used to print 1st array[i]. Is this right?
        ld $v1, 0($s2)                          ; $v1 will be used to print 2nd array[i]. Is this right? Which register to use for supplying a formatted string to print integers? It is not explained anywhere! 
        dadd $14, $zero, $a3                    ; print string. $14 is the register to syscall instructions. But i'm just guessing with this. I really don't know how it should handle. I just supplied $a3 because it seems more intuitive.

        syscall 5                               ; prints ouput (on the MEM stage)
        daddi $s4, $s4, 8                       ; i++
        j printOutput
end:

如果有人知道如何做到这一点,如果他/她可以分享,我将非常感激。任何地方都没有这方面的例子。谢谢。

更新

在迈克尔的帮助下,通过反复试验,我找到了问题的主要原因。我们为输出字符串和其他内存地址提供标签的顺序非常敏感。通过反复试验,我发现我们必须遵守以下顺序:

.data
format_string .asciiz "%d and %i etc.."
start_address .space  4
syscallArg1   .space  4                   ; 1st parameter to syscall 5
syscallArg2   .space  4                   ; 2nd parameter to syscall 5
---other labels go here---
.text
---code---

请注意,我们必须提供$14start_address 标签,它不能包含任何内容(只是一些可用的空白空间)。在此之前,格式化的字符串必须经过,在此之后,syscall 5必须存在的参数。其他标签可以在所有这些之后或之前。

4

1 回答 1

2

手册说的是,您应该将格式字符串的地址,后跟零个或多个参数(例如整数或字符串地址)存储在连续内存中,然后r14在执行之前加载该内存的起始地址sycall 5

他们的示例代码对我来说看起来有点不正确,因为他们应该使用sw而不是sd存储r2和存储r3在内存中(假设字符串地址实际上适合 32 位)。

所以你需要在数据部分保留一些空间,然后首先存储你的格式字符串的地址,然后是任何参数。然后将那块内存的起始地址放在r14. 例如,如果您的格式字符串是"%d, %d"并且您要打印的两个值在其中,r2并且r3您将直接在存储格式字符串地址的位置之后。sw r2r3


例如:

.data
format_str: .asciiz "%d"
printf_args: .space 8
...
.code
daddi r5, r0, format_str
sw r5, printf_args(r0)
daddi r5, r0, 123
sw r5, printf_args+4(r0)
daddi r14, r0, printf_args
syscall 5
于 2017-05-02T09:26:35.223 回答