URL
https://opencores.org/ocsvn/ion/ion/trunk
Subversion Repositories ion
[/] [ion/] [trunk/] [src/] [memtest/] [memtest.s] - Rev 248
Go to most recent revision | Compare with Previous | Blame | View Log
################################################################################ # memtest.s -- Test external RAM memory (XRAM) #------------------------------------------------------------------------------- # This program tests the external RAM (connected to the core through the cache # module). Currently it only finds the RAM top address if given the bottom # address. Subsequent versions will add some minimal diagnostic capability, # which will be needed when DRAM access is implemented. # # This program does only support a single continuous chunk of RAM. If the cache # module ever supports more than one chunk (e.g. DRAM and SRAM as in the DE-1 # board) this program will be modified accordingly. # # The program assumes there's no useable r/w memory other than the XRAM so it # does not use any memory for variables or stack. #------------------------------------------------------------------------------- # # #------------------------------------------------------------------------------- # To be run from reset vector, standalone. Interrupts must be disabled. # ################################################################################ #---- Test parameters .ifndef XRAM_MAX .set XRAM_MAX, 1024 # max. no. of KB to test for .endif .ifndef XRAM_BASE .set XRAM_BASE, 0x00000000 # 1st XRAM address .endif #---- Set to >0 to enable a few debug messages .set DEBUG, 0 #---- Cache parameters .set ICACHE_NUM_LINES, 256 # no. of lines in the I-Cache .set DCACHE_NUM_LINES, 256 # no. of lines in the D-Cache .set DCACHE_LINE_SIZE, 4 # D-Cache line size in words #---- UART stuff .set UART_BASE, 0x20000000 # UART base address .set UART_TX, 0x0000 # TX reg offset .set UART_STATUS, 0x0004 # status reg offset .set UART_TX_RDY, 0x0001 # tx ready flag mask #---- Debug register block -- 4 read-write, 32-bit registers .set DEBUG_BASE, 0x20010020 # Debug block base #--------------------------------------------------------------------------- .text .align 2 .globl entry .ent entry entry: .set noreorder b start_test nop .ifgt 0 #--- Trap handler address: we don't expect any traps ----------------------- .org 0x0180 interrupt_vector: b interrupt_vector # just freeze there nop .endif #------------------------------------------------------------------------------- start_test: li $a0,0x00000002 mtc0 $a0,$12 # disable interrupts and cache .ifdef TEST_CACHE # if we're going to test the caches then jal init_cache # invalidate all the I- and D-Cache lines now nop jal test_dcache nop .endif la $a0,msg0 jal puts nop li $a0,XRAM_BASE li $a1,8 jal put_hex nop la $a0,crlf jal puts nop la $t0,XRAM_BASE+4 # address of memory word being tested li $t2,XRAM_MAX # max amount of KBs to test for li $t3,1 # (used to decrement $t2) li $t4,0 # no. of KBs found move $t5,$t0 # keep the start addr at hand for comparison sw $zero,0($t0) # clear 1st test word (in case of prev. run) test_loop: lw $t1,0($t0) # read word contents beq $t5,$t1,hit_mirror # if it's the start address, we hit a mirror nop # we rolled off the end of the RAM back here sw $t0,0($t0) # word = word address lw $t1,0($t0) # read word back... bne $t1,$t0,bad_word # ...and if no match, we run off the RAM nop # sub $t2,$t2,$t3 # decrement loop counter... bnez $t2,test_loop # ...and go back if there's more to go addiu $t0,0x400 # in any case, increment test address by 1KB b end_test # end of memory found, result is in $t4 nop hit_mirror: # memory mirror detected .ifgt DEBUG la $a0,msg_mirror jal puts nop .endif b end_test nop bad_word: # readback error detected (maybe r/o area?) .ifgt DEBUG la $a0,msg_bad jal puts nop .endif b end_test nop end_test: # test done, ramtop+4 in $t0, #KB in $t4 la $a0,msg1 # Print ramtop message... jal puts nop addi $a0,$t0,-4 # substract the +4 offset we added before move $sp,$t0 # init SP at the top of RAM space addi $sp,$sp,-16 li $a1,8 jal put_hex nop la $a0,crlf jal puts nop # Clear RAM (only first words, so that simulation is not eternal) .ifgt 1 li $a0,XRAM_BASE addi $a1,$a0,80*4 clear_xram_loop: sw $zero,0($a0) blt $a0,$a1,clear_xram_loop addi $a0,$a0,4 .endif # Test code execution from SRAM. We copy the test_exec_sram routine to # the base of the SRAM and then jal to it li $a1,64 li $a0,XRAM_BASE la $a3,test_exec_sram copy_to_sram: lw $a2,0($a3) nop sw $a2,0($a0) addi $a1,$a1,-4 addi $a3,$a3,4 bnez $a1,copy_to_sram addi $a0,$a0,4 # Optionally dump first few words of XRAM and jump onto XRAM .ifgt 1 la $a0,msg5 jal puts nop li $a0,XRAM_BASE jal dump_hex ori $a1,$zero,20 .endif .ifgt 1 li $a0,XRAM_BASE nop jalr $a0 nop .endif # If all went well in the SRAM we'll have returned here .ifgt 0 # FIXME now we should do some strong test on the RAM to see if it's wired # correctly, using the right timing, etc. # Ok, now we know we have some RAM and stack space we can do some further # testing. # dump the first few words of FLASH la $a0,msg2 jal puts nop # FIXME flash base address is hardcoded li $a0,0xb0000000 jal put_hex ori $a1,$zero,8 la $a0,crlf jal puts nop la $a0,crlf jal puts nop li $a0,0xb0000000 jal dump_hex ori $a1,$zero,6 .endif # Optionally jump to FLASH. This is only useful in simulation. .ifgt 1 .ifdef EXEC_FLASH la $a0,msg8 jal puts nop li $a0,0xb0000000 nop jalr $a0 nop .endif .endif la $a0,msg4 jal puts nop $DONE: j $DONE # ...and freeze here nop test_dcache: # This D-Cache test is extremely simplistic: # We'll make a few I/O writes to a block of test registers and then readback # all the data. The writes and reads will be done as fast as the CPU can, # i.e. back to back. We rely on the I-Cache being active for this. # Note that, while the CPU 'wants' to do the WR cycles back to back, it # can't because of the D-Cache stalls. That's the interaction we want to # test here, for both WR and RD cycles. move $t8,$ra # Save $ra -- remember, there's no stack la $a0,msg9 # Display 'testing cache' message jal puts nop li $a0,DEBUG_BASE # We'll use the debug register block as the li $a1,0x112200aa # target of a series of RD and WR cycles. addi $a2,$a1,0x100 # Load $a1..$a2 with dummy data. addi $a3,$a2,0x100 .align 4 # Align to 16 bytes (start of I-Cache line) sw $a1,0x0($a0) # Perform 3 WR cycles... sw $a2,0x4($a0) sw $a3,0x8($a0) lw $t0,0x0($a0) # ...followed by a RD cycle bne $a1,$t0,test_dcache_ww_error nop lw $t0,0x4($a0) # Verify the other 2 WR cycles bne $a2,$t0,test_dcache_ww_error nop lw $t0,0x8($a0) bne $a3,$t0,test_dcache_ww_error nop la $a0,msg11 # display OK message... jal puts nop test_dcache_done: # ...and quit. move $ra,$t8 # Recover $ra and return jr $ra nop test_dcache_ww_error: # We'll print the same error for all kinds of la $a0,msg10 # failure; we'll diagnose it in simulation. test_dcache_error: jal puts nop b test_dcache_done # This short routine will be copied to RAM and then executed there. This # will test the behavior of the I-Cache. test_exec_sram: #jr $ra #nop sw $ra,0($sp) addi $sp,$sp,-4 la $a0,msg3 la $a1,puts jalr $a1 nop lw $ra,4($sp) jr $ra addi $sp,$sp,4 test_exec_sram_end: nop #---- Functions ---------------------------------------------------------------- # void dump_hex(int *address {a0}, int len {a1}) dump_hex: move $t7,$a0 move $t6,$a1 sw $ra,0($sp) addi $sp,$sp,-4 dump_hex_loop_lines: move $a0,$t7 li $a1,8 jal put_hex nop la $a0,msg6 jal puts nop li $t9,4 dump_hex_loop_words: lw $a0,0($t7) jal put_hex li $a1,8 la $a0,space jal puts addi $t7,4 addi $t9,$t9,-1 bnez $t9,dump_hex_loop_words nop la $a0,crlf jal puts nop addi $t6,$t6,-1 bnez $t6,dump_hex_loop_lines nop la $a0,msg7 jal puts addi $t7,4 # Some debug hex dump... .ifgt 0 lw $ra,4($sp) move $a0,$ra li $a1,8 jal put_hex nop stop_here: j stop_here nop .endif nop lw $ra,4($sp) jr $ra addi $sp,$sp,4 #---- Cache-related functions -------------------------------------------------- # void init_cache(void) -- invalidates all I-Cache lines (uses no RAM) init_cache: li $a0,0x00010000 # Disable cache, enable I-cache line invalidation mfc0 $a1,$12 or $a0,$a0,$a1 mtc0 $a0,$12 # In order to invalidate a I-Cache line we have to write its tag number to # any address while bits CP0[12].17:16=01. The write will be executed as a # regular write too, as a side effect, so we need to choose a harmless # target address. li $a0,XRAM_BASE li $a2,0 li $a1,ICACHE_NUM_LINES-1 inv_i_cache_loop: sw $a2,0($a0) blt $a2,$a1,inv_i_cache_loop addi $a2,1 # Now, the D-Cache is different. To invalidate a D-Cache line you just # read from it (by proper selection of a dummy target address) while bits # CP0[12].17:16=01. The data read is undefined and should be discarded. li $a0,0 # Use any base address that is mapped li $a2,0 li $a1,DCACHE_NUM_LINES-1 inv_d_cache_loop: lw $zero,0($a0) addi $a0,DCACHE_LINE_SIZE*4 blt $a2,$a1,inv_d_cache_loop addi $a2,1 lui $a1,0x0002 # Leave with cache enabled mfc0 $a0,$12 andi $a0,$a0,0xffff or $a1,$a0,$a1 jr $ra mtc0 $a1,$12 #--- Special functions that do not use any RAM --------------------------------- # WARNING: Not for general use! # All parameters in $a0..$a4, stack unused. No attempt to comply with any ABI # has been made. # Since we can't use any RAM, registers have been used with no regard for # intended usage -- have to share reg bank with calling function. # void puts(char *s) -- print zero-terminated string puts: la $a2,UART_BASE # UART base address puts_loop: lb $v0,0($a0) beqz $v0,puts_end addiu $a0,1 puts_wait_tx_rdy: lw $v1,UART_STATUS($a2) andi $v1,$v1,UART_TX_RDY beqz $v1,puts_wait_tx_rdy nop sw $v0,UART_TX($a2) b puts_loop nop puts_end: jr $ra nop # void put_hex(int n, int d) -- print integer as d-digit hex put_hex: la $a2,UART_BASE la $a3,put_hex_table addi $a1,-1 add $a1,$a1,$a1 add $a1,$a1,$a1 put_hex_loop: srlv $v0,$a0,$a1 andi $v0,$v0,0x0f addu $s2,$a3,$v0 lb $v0,0($s2) put_hex_wait_tx_rdy: lw $v1,UART_STATUS($a2) andi $v1,$v1,UART_TX_RDY beqz $v1,put_hex_wait_tx_rdy nop sw $v0,UART_TX($a2) bnez $a1,put_hex_loop addi $a1,-4 jr $ra nop #---- Constant data (note we keep it in the text section) ---------------------- put_hex_table: .ascii "0123456789abcdef" msg0: .ascii "\n\r" .asciz "Scanning external memory at 0x" msg1: .asciz "Found XRAM top at 0x" crlf: .asciz "\n\r" space: .asciz " " msg_mirror: .asciz "hit mirror!\n\r" msg_bad: .asciz "bad readback!\n\r" msg2: .asciz "\n\rDumping the first few words of FLASH at address 0x" msg3: .ascii "\n\rTesting code execution from SRAM... " .asciz "if you see this, it worked\n\r" msg4: .ascii "\n\r\n\r" .asciz "End of test.\n\r\n\r" msg5: .asciz "Dump of first few words of XRAM after initialization:\n\r" msg6: .asciiz ": " msg7: .asciiz "<end of dump>" msg8: .ascii "\n\r" .asciz "Testing execution from 8-bit static memory (FLASH)\n\r" msg9: .asciz "Testing D-Cache with back-to-back pairs of RD & WR cycles...\n\r" msg10: .asciz "Error in WR+WR sequence\n\r" msg11: .asciz "OK\n\r" .set reorder .end entry
Go to most recent revision | Compare with Previous | Blame | View Log