mirror of
				https://github.com/openhwgroup/cvw
				synced 2025-02-11 06:05:49 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			136 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			136 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
| #! /usr/bin/python3
 | |
| import fileinput, sys
 | |
| 
 | |
| sys.stderr.write("reminder: parse_qemu.py takes input from stdin\n")
 | |
| parseState = "idle"
 | |
| beginPageFault = 0
 | |
| inPageFault = 0
 | |
| endPageFault = 0
 | |
| CSRs = {}
 | |
| pageFaultCSRs = {}
 | |
| regs = {}
 | |
| pageFaultRegs = {}
 | |
| instrs = {}
 | |
| instrCount = 0
 | |
| returnAdr = 0
 | |
| 
 | |
| def printPC(l):
 | |
|     global parseState, inPageFault, CSRs, pageFaultCSRs, regs, pageFaultCSRs, instrs, instrCount
 | |
|     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]}')
 | |
|         instrCount += 1
 | |
|         if ((instrCount % 100000) == 0):
 | |
|             sys.stderr.write("QEMU parser reached "+str(instrCount)+" instrs\n")
 | |
| 
 | |
| 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
 | |
|     if l.strip() and (not l.startswith("Disassembler")) and (not l.startswith("Please")):
 | |
|         # If we've hit the register file
 | |
|         if l.startswith(' x0/zero'): 
 | |
|             parseState = "regFile"
 | |
|             if not inPageFault:
 | |
|                 instr = instrs[CSRs["pc"]]
 | |
|                 printPC(instr)
 | |
|             parseRegs(l)
 | |
|         # If we've hit a CSR
 | |
|         else:
 | |
|             csr = l.split()[0]
 | |
|             val = int(l.split()[1],16)
 | |
|             # Commented out this conditional because the pageFault instrs don't corrupt CSRs
 | |
|             #if inPageFault:
 | |
|                 # Not sure if these CSRs should be updated or not during page fault.
 | |
|                 #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"):
 | |
|                     # We do update some CSRs
 | |
|                 #    CSRs[csr] = val
 | |
|                 #else:
 | |
|                     # Others we preserve until changed later
 | |
|                 #    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
 | |
|                 pageFaultCSRs[csr] = val
 | |
|             elif pageFaultCSRs and (csr in pageFaultCSRs):
 | |
|                 if (val != pageFaultCSRs[csr]):
 | |
|                     del pageFaultCSRs[csr]
 | |
|                     CSRs[csr] = val
 | |
|             else:
 | |
|                 CSRs[csr] = val
 | |
| 
 | |
| def parseRegs(l):
 | |
|     global parseState, inPageFault, CSRs, pageFaultCSRs, regs, pageFaultCSRs, instrs, pageFaultRegs
 | |
|     if "pc" in l:
 | |
|         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")
 | |
|             beginPageFault = not inPageFault
 | |
|             if beginPageFault:
 | |
|                 returnAdr = int(l.split()[0][2:-1], 16)
 | |
|                 sys.stderr.write('Saving SEPC of '+hex(returnAdr)+'\n')
 | |
|             inPageFault = 1
 | |
|         else: 
 | |
|             endPageFault = inPageFault
 | |
|             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)
 |