3

グループプロジェクトについていくつか質問があります。最初の質問は、MIDI ファイルの再生に関するものです。現在、私たちのプログラムは約 75% の精度でタイプ 0 ファイルを再生できます (それを測定できる場合)。基本的に、曲の「ほとんど」が正しい (または非常に近い) ものになります。明らかにすべてのサウンドを再生しているわけではなく、再生するサウンドの一部は歪んでいます。では、ほとんどの音を正しく再生している場合、他の音を聞き逃したり歪ませたりする原因は何でしょうか?

2 番目の質問は、割り込みハンドラに関するものです。プログラムがXバイト前後にスキップするようにポインターを調整しようとしていますが、プログラムの実行中に割り込みを試みると、プログラムが停止してロックされます。割り込みを段階的に実行すると、停止しません。私は途中か、完全に的外れのようです (これは以前のバージョンで動作していましたが、おそらく再生機能が正しく記述されていませんでした)。推測する必要がある場合、問題は WITH_STATUS および WITHOUT_STATUS 関数に関係していると思います。そうですか?

よろしくお願いします。

PLAY_MIDI:          
        addi    $s5, $0, 0          # temporaries to control speed and volume
        addi    $t6, $0, 100   
        addi    $t9, $0, 0          
        addi    $sp, $sp, -4            # add a byte to the stack pointer and store the return address there
        sw  $ra, 0($sp)
        addi    $t4, $0, 3
LOOP:
        lh  $t0, DAT_Format
        bnez    $t0, NO_SUPPORTED
        addi    $s3, $s3, 1
        addi    $s1, $s1, -1    
        lbu     $t9, 0($s3)             # look for status byte
        blez    $s1, OFF_LOOP
        bltu    $t9, 0x80, LOOP
        bgeu    $t9, 0xa0, LOOP
WITH_STATUS:
        lbu     $t9, 0($s3)
        lb  $t6, -1($s3)            #sleep time 
        lb  $a1, 1($s3)         # load note
        lb  $a2, 2($s3)             # load volume
        addi    $s3, $s3, 4
        addi    $s1, $s1, -4
        move    $s2, $s1        # *** copy $s1 to $s2  for pause/play
        j   SLEEP
WITHOUT_STATUS:
        lb  $t6, -1($s3)            #sleep time
        lb  $a1, 0($s3)         # load note
        lb  $a2, 1($s3)             # load volume
        addi    $s3, $s3, 3         #increase stack
        addi    $s1, $s1, -3
        move    $s2, $s1        # ** copy $s1 to $s2    for pause/play
SLEEP:

        mul     $t6, $t6, $t4
        li  $v0, 32             # sleep until the next note should be played
        addu    $a0, $0, $t6            # sleep for $t6 ms
        syscall

        and     $a0, $t9,0x0F           # load channel
        and     $t9,$t9,0x000000F0      # load Command

                                               ######################################################
                                               ## copy $s1 to $s2 for use differentiating between  ##
        beq     $t9, 0x90, PLAY_NOTE_ON     ## stop and pause                   ##
        beq     $t9, 0x80, PLAY_NOTE_OFF        ## $s2 will hold last position of $s1               ##
        bgeu    $t9, 0xa0, LOOP             ######################################################

AFTER_PLAY:         
        blez    $s1, OFF_LOOP
        lbu     $t0, 0($s3)
        bgeu    $t0, 0x80, WITH_STATUS
        j   WITHOUT_STATUS

OFF_LOOP:       
        bne $s1, $s2, PAUSE_LOOP        # get the return address from the stack and decrease the pointer
        lw  $ra, 0($sp)
        addi    $sp, $sp, 4
        jr  $ra             # jump to the address we just loaded
PLAY_NOTE_ON:
        li  $v0, 60             # add a byte to the stack pointer and store the return address there
        syscall 
        j   AFTER_PLAY          

PLAY_NOTE_OFF:
        li  $v0, 61
        syscall
        j   AFTER_PLAY

NO_SUPPORTED:
j EXIT

#### Below is part of the interrupt handler ####
SKIP_FORWARD:       addi    $s3, $s3, 1         # by adding to $s3 program will move pointer ahead 
        la  $a0, skip_forward       # changed from $s3 to $s1 ***
        li  $v0, 4
        syscall
        jr  $ra

SKIP_BACKWARD:      addi    $s3, $s3, -1            # by subtracting from $s3 program will move pointer backwards
        la  $a0, skip_backward      # changed $s3 to $s1
        li  $v0, 4
        syscall 
        jr  $ra
4

0 に答える 0