如何在EDUMIPS64模拟器中打印数组或格式化字符串?

时间:2017-05-02 02:43:26

标签: assembly mips simulator mips64

此计划的文档非常有限。为了做到这一点,我几乎没有什么可以借鉴的。除了程序的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

您可以查看here。 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---

请注意,我们必须向$14提供start_address标签,该标签不得包含任何内容(只是一些可用的空格)。在此之前,格式化的字符串必须在此之后,syscall 5的参数必须存在。其他标签可以在之后或之前发布。

1 个答案:

答案 0 :(得分:1)

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

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

因此,您需要在数据部分保留一些空间,然后先存储格式字符串的地址,然后再存储任何参数。然后将该段内存的起始地址放在r3中。例如,如果您的格式字符串为r14,并且您要打印的两个值位于"%d, %d"r2,则r3 sw和{{1}直接在你存储格式字符串的地址之后。

例如:

r2