2021-06-10 13:58:20 +00:00
#! /usr/bin/python3
import fileinput , sys
sys . stderr . write ( " reminder: this script takes input from stdin \n " )
parseState = " idle "
2021-07-09 21:03:47 +00:00
beginPageFault = 0
2021-06-10 13:58:20 +00:00
inPageFault = 0
2021-07-09 21:03:47 +00:00
endPageFault = 0
2021-06-10 13:58:20 +00:00
CSRs = { }
pageFaultCSRs = { }
regs = { }
pageFaultRegs = { }
instrs = { }
2021-07-03 21:25:19 +00:00
instrCount = 0
2021-07-09 21:03:47 +00:00
returnAdr = 0
2021-06-10 13:58:20 +00:00
def printPC ( l ) :
2021-07-03 21:25:19 +00:00
global parseState , inPageFault , CSRs , pageFaultCSRs , regs , pageFaultCSRs , instrs , instrCount
2021-06-10 13:58:20 +00:00
if not inPageFault :
inst = l . split ( )
if len ( inst ) > 3 :
print ( f ' => { inst [ 1 ] } : \t { inst [ 2 ] } { inst [ 3 ] } ' )
else :
print ( f ' => { inst [ 1 ] } : \t { inst [ 2 ] } ' )
print ( f ' { inst [ 0 ] } 0x { inst [ 1 ] } ' )
2021-07-03 21:25:19 +00:00
instrCount + = 1
if ( ( instrCount % 100000 ) == 0 ) :
sys . stderr . write ( " QEMU parser reached " + str ( instrCount ) + " instrs \n " )
2021-06-10 13:58:20 +00:00
def printCSRs ( ) :
global parseState , inPageFault , CSRs , pageFaultCSRs , regs , pageFaultCSRs , instrs
if not inPageFault :
for ( csr , val ) in CSRs . items ( ) :
print ( ' {} {} {:#x} {} ' . format ( csr , ' ' * ( 15 - len ( csr ) ) , val , val ) )
print ( ' ----- ' )
def parseCSRs ( l ) :
global parseState , inPageFault , CSRs , pageFaultCSRs , regs , pageFaultCSRs , instrs
2021-07-09 21:03:47 +00:00
if l . strip ( ) and ( not l . startswith ( " Disassembler " ) ) and ( not l . startswith ( " Please " ) ) and not inPageFault :
if l . startswith ( ' x0/zero ' ) :
2021-06-10 13:58:20 +00:00
parseState = " regFile "
instr = instrs [ CSRs [ " pc " ] ]
printPC ( instr )
parseRegs ( l )
else :
csr = l . split ( ) [ 0 ]
val = int ( l . split ( ) [ 1 ] , 16 )
2021-07-09 21:03:47 +00:00
# Commented out this conditional because the pageFault instrs don't corrupt CSRs
#if inPageFault:
2021-06-17 04:50:02 +00:00
# Not sure if these CSRs should be updated or not during page fault.
2021-07-09 21:03:47 +00:00
#if l.startswith("mstatus") or l.startswith("mepc") or l.startswith("mcause") or l.startswith("mtval") or l.startswith("sepc") or l.startswith("scause") or l.startswith("stval"):
2021-06-26 12:30:58 +00:00
# We do update some CSRs
2021-07-09 21:03:47 +00:00
# CSRs[csr] = val
#else:
2021-06-26 12:30:58 +00:00
# Others we preserve until changed later
2021-07-09 21:03:47 +00:00
# pageFaultCSRs[csr] = val
#elif pageFaultCSRs and (csr in pageFaultCSRs):
# if (val != pageFaultCSRs[csr]):
# del pageFaultCSRs[csr]
# CSRs[csr] = val
#else:
# CSRs[csr] = val
#
# However SEPC and STVAL do get corrupted upon exiting
if endPageFault and ( ( csr == ' sepc ' ) or ( csr == ' stval ' ) ) :
CSRs [ csr ] = returnAdr
2021-06-10 13:58:20 +00:00
else :
CSRs [ csr ] = val
def parseRegs ( l ) :
global parseState , inPageFault , CSRs , pageFaultCSRs , regs , pageFaultCSRs , instrs
2021-07-09 21:03:47 +00:00
if " pc " in l :
2021-06-10 13:58:20 +00:00
printCSRs ( )
# New non-disassembled instruction
parseState = " CSRs "
parseCSRs ( l )
elif l . startswith ( ' -------- ' ) :
# End of disassembled instruction
printCSRs ( )
parseState = " idle "
else :
s = l . split ( )
for i in range ( 0 , len ( s ) , 2 ) :
if ' / ' in s [ i ] :
reg = s [ i ] . split ( ' / ' ) [ 1 ]
val = int ( s [ i + 1 ] , 16 )
if inPageFault :
pageFaultRegs [ reg ] = val
else :
if pageFaultRegs and ( reg in pageFaultRegs ) :
if ( val != pageFaultRegs [ reg ] ) :
del pageFaultRegs [ reg ]
regs [ reg ] = val
else :
regs [ reg ] = val
val = regs [ reg ]
print ( ' {} {} {:#x} {} ' . format ( reg , ' ' * ( 15 - len ( reg ) ) , val , val ) )
else :
sys . stderr . write ( " Whoops. Expected a list of reg file regs; got: \n " + l )
#############
# Main Code #
#############
for l in fileinput . input ( ) :
if l . startswith ( ' qemu-system-riscv64: QEMU: Terminated via GDBstub ' ) :
break
elif l . startswith ( ' IN: ' ) :
# New disassembled instr
parseState = " instr "
elif ( parseState == " instr " ) and l . startswith ( ' 0x ' ) :
if " out of bounds " in l :
sys . stderr . write ( " Detected QEMU page fault error \n " )
2021-07-09 21:03:47 +00:00
beginPageFault = ~ ( inPageFault )
if beginPageFault :
returnAdr = int ( l . split ( ) [ 0 ] [ 2 : - 1 ] , 16 )
2021-06-10 13:58:20 +00:00
inPageFault = 1
else :
2021-07-09 21:03:47 +00:00
endPageFault = inPageFault
2021-06-10 13:58:20 +00:00
inPageFault = 0
adr = int ( l . split ( ) [ 0 ] [ 2 : - 1 ] , 16 )
instrs [ adr ] = l
parseState = " CSRs "
elif parseState == " CSRs " :
parseCSRs ( l )
elif parseState == " regFile " :
parseRegs ( l )