URL
https://opencores.org/ocsvn/zipcpu/zipcpu/trunk
Subversion Repositories zipcpu
[/] [zipcpu/] [trunk/] [bench/] [cpp/] [zippy_tb.cpp] - Rev 27
Go to most recent revision | Compare with Previous | Blame | View Log
/////////////////////////////////////////////////////////////////////////////// // // Filename: zippy_tb.cpp // // Project: Zip CPU -- a small, lightweight, RISC CPU soft core // // Purpose: A bench simulator for the CPU. Eventually, you should be // able to give this program the name of a piece of compiled // code to load into memory. For now, we hand assemble with the // computers help. // // // Creator: Dan Gisselquist, Ph.D. // Gisselquist Tecnology, LLC // /////////////////////////////////////////////////////////////////////////////// // // Copyright (C) 2015, Gisselquist Technology, LLC // // This program is free software (firmware): you can redistribute it and/or // modify it under the terms of the GNU General Public License as published // by the Free Software Foundation, either version 3 of the License, or (at // your option) any later version. // // This program is distributed in the hope that it will be useful, but WITHOUT // ANY WARRANTY; without even the implied warranty of MERCHANTIBILITY or // FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License // for more details. // // License: GPL, v3, as defined and found on www.gnu.org, // http://www.gnu.org/licenses/gpl.html // // /////////////////////////////////////////////////////////////////////////////// // // #include <signal.h> #include <time.h> #include <unistd.h> #include <ctype.h> #include <ncurses.h> #include "verilated.h" #include "Vzipsystem.h" #include "testb.h" // #include "twoc.h" // #include "qspiflashsim.h" #include "memsim.h" #include "zopcodes.h" #include "zparser.h" #define CMD_REG 0 #define CMD_DATA 1 #define CMD_HALT (1<<10) #define CMD_STALL (1<<9) #define CMD_STEP (1<<8) #define CMD_INT (1<<7) #define CMD_RESET (1<<6) // No particular "parameters" need definition or redefinition here. class ZIPPY_TB : public TESTB<Vzipsystem> { public: unsigned long m_mem_size; MEMSIM m_mem; // QSPIFLASHSIM m_flash; FILE *dbg_fp; bool dbg_flag, bomb; ZIPPY_TB(void) : m_mem_size(1<<20), m_mem(m_mem_size) { //dbg_fp = fopen("dbg.txt", "w"); dbg_fp = NULL; dbg_flag = false; bomb = false; } void reset(void) { // m_flash.debug(false); TESTB<Vzipsystem>::reset(); } bool on_tick(void) { tick(); return true; } void showval(int y, int x, const char *lbl, unsigned int v) { mvprintw(y,x, "%s: 0x%08x", lbl, v); } void dispreg(int y, int x, const char *n, unsigned int v) { // 4,4,8,1 = 17 of 20, +3 = 19 mvprintw(y, x, "%s: 0x%08x", n, v); } void showreg(int y, int x, const char *n, int r) { // 4,4,8,1 = 17 of 20, +3 = 19 mvprintw(y, x, "%s: 0x%08x", n, m_core->v__DOT__thecpu__DOT__regset[r]); addch( ((r == m_core->v__DOT__thecpu__DOT__dcdA) &&(m_core->v__DOT__thecpu__DOT__dcdvalid) &&(m_core->v__DOT__thecpu__DOT__dcdA_rd)) ?'a':' '); addch( ((r == m_core->v__DOT__thecpu__DOT__dcdB) &&(m_core->v__DOT__thecpu__DOT__dcdvalid) &&(m_core->v__DOT__thecpu__DOT__dcdB_rd)) ?'b':' '); addch( ((r == m_core->v__DOT__thecpu__DOT__wr_reg_id) &&(m_core->v__DOT__thecpu__DOT__wr_reg_ce)) ?'W':' '); } void showins(int y, const char *lbl, const int ce, const int valid, const int gie, const int stall, const unsigned int pc) { char line[80]; if (ce) mvprintw(y, 0, "Ck "); else mvprintw(y, 0, " "); if (stall) printw("Stl "); else printw(" "); printw("%s: 0x%08x", lbl, pc); if (valid) { if (gie) attroff(A_BOLD); else attron(A_BOLD); zipi_to_string(m_mem[pc], line); printw(" %-24s", &line[1]); } else { attroff(A_BOLD); printw(" (0x%08x)%28s", m_mem[pc],""); } attroff(A_BOLD); } void dbgins(const char *lbl, const int ce, const int valid, const int gie, const int stall, const unsigned int pc) { char line[80]; if (!dbg_fp) return; if (ce) fprintf(dbg_fp, "%s Ck ", lbl); else fprintf(dbg_fp, "%s ", lbl); if (stall) fprintf(dbg_fp, "Stl "); else fprintf(dbg_fp, " "); fprintf(dbg_fp, "0x%08x: ", pc); if (valid) { zipi_to_string(m_mem[pc], line); fprintf(dbg_fp, " %-20s\n", &line[1]); } else { fprintf(dbg_fp, " (0x%08x)\n", m_mem[pc]); } } void show_state(void) { int ln= 0; mvprintw(ln,0, "Peripherals-SS"); ln++; /* showval(ln, 1, "TRAP", m_core->v__DOT__trap_data); mvprintw(ln, 17, "%s%s", ((m_core->v__DOT__sys_cyc) &&(m_core->v__DOT__sys_we) &&(m_core->v__DOT__sys_addr == 0))?"W":" ", (m_core->v__DOT__trap_int)?"I":" "); */ showval(ln, 1, "PIC ", m_core->v__DOT__pic_data); showval(ln,21, "WDT ", m_core->v__DOT__watchdog__DOT__r_value); showval(ln,41, "CACH", m_core->v__DOT__manualcache__DOT__cache_base); showval(ln,61, "PIC2", m_core->v__DOT__ctri__DOT__r_int_state); ln++; showval(ln, 1, "TMRA", m_core->v__DOT__timer_a__DOT__r_value); showval(ln,21, "TMRB", m_core->v__DOT__timer_b__DOT__r_value); showval(ln,41, "TMRB", m_core->v__DOT__timer_c__DOT__r_value); showval(ln,61, "JIF ", m_core->v__DOT__jiffies__DOT__r_counter); ln++; showval(ln, 1, "UTSK", m_core->v__DOT__utc_data); showval(ln,21, "UOST", m_core->v__DOT__uoc_data); showval(ln,41, "UPST", m_core->v__DOT__upc_data); showval(ln,61, "UICT", m_core->v__DOT__uic_data); ln++; mvprintw(ln, 40, "%s %s", (m_core->v__DOT__cpu_halt)? "CPU-HALT": " ", (m_core->v__DOT__cpu_reset)?"CPU-RESET":" "); ln++; mvprintw(ln, 40, "%s %s %s 0x%02x", (m_core->v__DOT__cmd_halt)? "HALT": " ", (m_core->v__DOT__cmd_reset)?"RESET":" ", (m_core->v__DOT__cmd_step)? "STEP" :" ", (m_core->v__DOT__cmd_addr)&0x3f); if (m_core->v__DOT__thecpu__DOT__gie) attroff(A_BOLD); else attron(A_BOLD); mvprintw(ln, 0, "Supervisor Registers"); ln++; showreg(ln, 1, "sR0 ", 0); showreg(ln,21, "sR1 ", 1); showreg(ln,41, "sR2 ", 2); showreg(ln,61, "sR3 ", 3); ln++; showreg(ln, 1, "sR4 ", 4); showreg(ln,21, "sR5 ", 5); showreg(ln,41, "sR6 ", 6); showreg(ln,61, "sR7 ", 7); ln++; showreg(ln, 1, "sR8 ", 8); showreg(ln,21, "sR9 ", 9); showreg(ln,41, "sR10", 10); showreg(ln,61, "sR11", 11); ln++; showreg(ln, 1, "sR12", 12); showreg(ln,21, "sSP ", 13); mvprintw(ln,41, "sCC :%s%s%s%s%s%s%s%s", (m_core->v__DOT__thecpu__DOT__trap)?"TRP":" ", (m_core->v__DOT__thecpu__DOT__step)?"STP":" ", (m_core->v__DOT__thecpu__DOT__sleep)?"SLP":" ", (m_core->v__DOT__thecpu__DOT__gie)?"GIE":" ", (m_core->v__DOT__thecpu__DOT__iflags&8)?"V":" ", (m_core->v__DOT__thecpu__DOT__iflags&4)?"N":" ", (m_core->v__DOT__thecpu__DOT__iflags&2)?"C":" ", (m_core->v__DOT__thecpu__DOT__iflags&1)?"Z":" "); mvprintw(ln,61, "sPC : 0x%08x", m_core->v__DOT__thecpu__DOT__ipc); ln++; if (m_core->v__DOT__thecpu__DOT__gie) attron(A_BOLD); else attroff(A_BOLD); mvprintw(ln, 0, "User Registers"); ln++; showreg(ln, 1, "uR0 ", 16); showreg(ln,21, "uR1 ", 17); showreg(ln,41, "uR2 ", 18); showreg(ln,61, "uR3 ", 19); ln++; showreg(ln, 1, "uR4 ", 20); showreg(ln,21, "uR5 ", 21); showreg(ln,41, "uR6 ", 22); showreg(ln,61, "uR7 ", 23); ln++; showreg(ln, 1, "uR8 ", 24); showreg(ln,21, "uR9 ", 25); showreg(ln,41, "uR10", 26); showreg(ln,61, "uR11", 27); ln++; showreg(ln, 1, "uR12", 28); showreg(ln,21, "uSP ", 29); mvprintw(ln,41, "uCC :%s%s%s%s%s%s%s%s", (m_core->v__DOT__thecpu__DOT__trap)?"TRP":" ", (m_core->v__DOT__thecpu__DOT__step)?"STP":" ", (m_core->v__DOT__thecpu__DOT__sleep)?"SLP":" ", (m_core->v__DOT__thecpu__DOT__gie)?"GIE":" ", (m_core->v__DOT__thecpu__DOT__flags&8)?"V":" ", (m_core->v__DOT__thecpu__DOT__flags&4)?"N":" ", (m_core->v__DOT__thecpu__DOT__flags&2)?"C":" ", (m_core->v__DOT__thecpu__DOT__flags&1)?"Z":" "); mvprintw(ln,61, "uPC : 0x%08x", m_core->v__DOT__thecpu__DOT__upc); attroff(A_BOLD); ln+=1; mvprintw(ln, 0, "PFPIPE: rda=%08x/%d, bas=%08x, off=%08x, nv=%03x, ackw=%d", m_core->v__DOT__thecpu__DOT__pf__DOT__r_addr, m_core->v__DOT__thecpu__DOT__pf__DOT__r_cv, m_core->v__DOT__thecpu__DOT__pf__DOT__r_cache_base, m_core->v__DOT__thecpu__DOT__pf__DOT__r_cache_offset, m_core->v__DOT__thecpu__DOT__pf__DOT__r_nvalid, m_core->v__DOT__thecpu__DOT__pf__DOT__r_acks_waiting); ln++; mvprintw(ln, 0, "PF BUS: %3s %3s %s @0x%08x[0x%08x] -> %s %s %08x", (m_core->v__DOT__thecpu__DOT__pf_cyc)?"CYC":" ", (m_core->v__DOT__thecpu__DOT__pf_stb)?"STB":" ", " ", // (m_core->v__DOT__thecpu__DOT__pf_we )?"WE":" ", (m_core->v__DOT__thecpu__DOT__pf_addr), 0, // (m_core->v__DOT__thecpu__DOT__pf_data), (m_core->v__DOT__thecpu__DOT__pf_ack)?"ACK":" ", (m_core->v__DOT__cpu_stall)?"STL":" ", (m_core->v__DOT__wb_data)); ln++; mvprintw(ln, 0, "MEMBUS: %3s %3s %s @0x%08x[0x%08x] -> %s %s %08x", (m_core->v__DOT__thecpu__DOT__mem_cyc)?"CYC":" ", (m_core->v__DOT__thecpu__DOT__mem_stb)?"STB":" ", (m_core->v__DOT__thecpu__DOT__mem_we )?"WE":" ", (m_core->v__DOT__thecpu__DOT__mem_addr), (m_core->v__DOT__thecpu__DOT__mem_data), (m_core->v__DOT__thecpu__DOT__mem_ack)?"ACK":" ", (m_core->v__DOT__cpu_stall)?"STL":" ", (m_core->v__DOT__thecpu__DOT__mem_result)); ln++; mvprintw(ln, 0, "SYSBUS: %3s %3s %s @0x%08x[0x%08x] -> %s %s %08x", (m_core->o_wb_cyc)?"CYC":" ", (m_core->o_wb_stb)?"STB":" ", (m_core->o_wb_we )?"WE":" ", (m_core->o_wb_addr), (m_core->o_wb_data), (m_core->i_wb_ack)?"ACK":" ", (m_core->i_wb_stall)?"STL":" ", (m_core->i_wb_data)); ln+=2; showins(ln, "I ", !m_core->v__DOT__thecpu__DOT__dcd_stalled, m_core->v__DOT__thecpu__DOT__pf_valid, //m_core->v__DOT__thecpu__DOT__instruction_gie, m_core->v__DOT__thecpu__DOT__gie, 0, // m_core->v__DOT__thecpu__DOT__instruction_pc); ln++; m_core->v__DOT__thecpu__DOT__pf_pc); ln++; showins(ln, "Dc", m_core->v__DOT__thecpu__DOT__dcd_ce, m_core->v__DOT__thecpu__DOT__dcdvalid, m_core->v__DOT__thecpu__DOT__dcd_gie, m_core->v__DOT__thecpu__DOT__dcd_stalled, m_core->v__DOT__thecpu__DOT__dcd_pc-1); ln++; showins(ln, "Op", m_core->v__DOT__thecpu__DOT__op_ce, m_core->v__DOT__thecpu__DOT__opvalid, m_core->v__DOT__thecpu__DOT__op_gie, m_core->v__DOT__thecpu__DOT__op_stall, m_core->v__DOT__thecpu__DOT__op_pc-1); ln++; showins(ln, "Al", m_core->v__DOT__thecpu__DOT__alu_ce, m_core->v__DOT__thecpu__DOT__alu_pc_valid, m_core->v__DOT__thecpu__DOT__alu_gie, m_core->v__DOT__thecpu__DOT__alu_stall, m_core->v__DOT__thecpu__DOT__alu_pc-1); ln++; mvprintw(ln-5, 48,"%s %s", (m_core->v__DOT__thecpu__DOT__op_break)?"OB":" ", (m_core->v__DOT__thecpu__DOT__clear_pipeline)?"CLRP":" "); mvprintw(ln-4, 48, (m_core->v__DOT__thecpu__DOT__new_pc)?"new-pc":" "); printw("(%s:%02x,%x)", (m_core->v__DOT__thecpu__DOT__set_cond)?"SET":" ", (m_core->v__DOT__thecpu__DOT__opF&0x0ff), (m_core->v__DOT__thecpu__DOT__op_gie) ? (m_core->v__DOT__thecpu__DOT__w_uflags) : (m_core->v__DOT__thecpu__DOT__w_iflags)); printw("(%s%s%s:%02x)", (m_core->v__DOT__thecpu__DOT__opF_wr)?"OF":" ", (m_core->v__DOT__thecpu__DOT__alF_wr)?"FL":" ", (m_core->v__DOT__thecpu__DOT__wr_flags_ce)?"W":" ", (m_core->v__DOT__thecpu__DOT__alu_flags)); /* mvprintw(ln-3, 48, "dcdI : 0x%08x", m_core->v__DOT__thecpu__DOT__dcdI); mvprintw(ln-2, 48, "r_opB: 0x%08x", m_core->v__DOT__thecpu__DOT__opB); */ mvprintw(ln-3, 48, "Op(%x)%8x,%8x->", m_core->v__DOT__thecpu__DOT__opn, m_core->v__DOT__thecpu__DOT__r_opA, m_core->v__DOT__thecpu__DOT__r_opB); if (m_core->v__DOT__thecpu__DOT__alu_valid) printw("%08x", m_core->v__DOT__thecpu__DOT__alu_result); else printw("%8s",""); mvprintw(ln-1, 48, "MEM: %s%s %s%s %s %-5s", (m_core->v__DOT__thecpu__DOT__opvalid_mem)?"M":" ", (m_core->v__DOT__thecpu__DOT__mem_ce)?"CE":" ", (m_core->v__DOT__thecpu__DOT__mem_we)?"Wr ":"Rd ", (m_core->v__DOT__thecpu__DOT__mem_stalled)?"PIPE":" ", (m_core->v__DOT__thecpu__DOT__mem_valid)?"MEMV":" ", zop_regstr[(m_core->v__DOT__thecpu__DOT__mem_wreg&0x1f)^0x10]); } unsigned int cmd_read(unsigned int a) { if (dbg_fp) { dbg_flag= true; fprintf(dbg_fp, "CMD-READ(%d)\n", a); } wb_write(CMD_REG, CMD_HALT|(a&0x3f)); while((wb_read(CMD_REG) & CMD_STALL) == 0) ; unsigned int v = wb_read(CMD_DATA); if (dbg_flag) fprintf(dbg_fp, "CMD-READ(%d) = 0x%08x\n", a, v); dbg_flag = false; return v; } bool halted(void) { return (m_core->v__DOT__cmd_halt != 0); } void read_state(void) { int ln= 0; mvprintw(ln,0, "Peripherals-RS"); ln++; showval(ln, 1, "PIC ", cmd_read(32+ 0)); showval(ln,21, "WDT ", cmd_read(32+ 1)); showval(ln,41, "CACH", cmd_read(32+ 2)); showval(ln,61, "PIC2", cmd_read(32+ 3)); ln++; showval(ln, 1, "TMRA", cmd_read(32+ 4)); showval(ln,21, "TMRB", cmd_read(32+ 5)); showval(ln,41, "TMRC", cmd_read(32+ 6)); showval(ln,61, "JIF ", cmd_read(32+ 7)); ln++; showval(ln, 1, "UTSK", cmd_read(32+12)); showval(ln,21, "UMST", cmd_read(32+13)); showval(ln,41, "UPST", cmd_read(32+14)); showval(ln,61, "UAST", cmd_read(32+15)); ln++; ln++; unsigned int cc = cmd_read(14); if (dbg_fp) fprintf(dbg_fp, "CC = %08x, gie = %d\n", cc, m_core->v__DOT__thecpu__DOT__gie); if (cc & 0x020) attroff(A_BOLD); else attron(A_BOLD); mvprintw(ln, 0, "Supervisor Registers"); ln++; dispreg(ln, 1, "sR0 ", cmd_read(0)); dispreg(ln,21, "sR1 ", cmd_read(1)); dispreg(ln,41, "sR2 ", cmd_read(2)); dispreg(ln,61, "sR3 ", cmd_read(3)); ln++; dispreg(ln, 1, "sR4 ", cmd_read(4)); dispreg(ln,21, "sR5 ", cmd_read(5)); dispreg(ln,41, "sR6 ", cmd_read(6)); dispreg(ln,61, "sR7 ", cmd_read(7)); ln++; dispreg(ln, 1, "sR8 ", cmd_read( 8)); dispreg(ln,21, "sR9 ", cmd_read( 9)); dispreg(ln,41, "sR10", cmd_read(10)); dispreg(ln,61, "sR11", cmd_read(11)); ln++; dispreg(ln, 1, "sR12", cmd_read(12)); dispreg(ln,21, "sSP ", cmd_read(13)); mvprintw(ln,41, "sCC :%s%s%s%s%s%s%s", (cc & 0x040)?"STP":" ", (cc & 0x020)?"GIE":" ", (cc & 0x010)?"SLP":" ", (cc&8)?"V":" ", (cc&4)?"N":" ", (cc&2)?"C":" ", (cc&1)?"Z":" "); mvprintw(ln,61, "sPC : 0x%08x", cmd_read(15)); ln++; if (cc & 0x020) attron(A_BOLD); else attroff(A_BOLD); mvprintw(ln, 0, "User Registers"); ln++; dispreg(ln, 1, "uR0 ", cmd_read(16)); dispreg(ln,21, "uR1 ", cmd_read(17)); dispreg(ln,41, "uR2 ", cmd_read(18)); dispreg(ln,61, "uR3 ", cmd_read(19)); ln++; dispreg(ln, 1, "uR4 ", cmd_read(20)); dispreg(ln,21, "uR5 ", cmd_read(21)); dispreg(ln,41, "uR6 ", cmd_read(22)); dispreg(ln,61, "uR7 ", cmd_read(23)); ln++; dispreg(ln, 1, "uR8 ", cmd_read(24)); dispreg(ln,21, "uR9 ", cmd_read(25)); dispreg(ln,41, "uR10", cmd_read(26)); dispreg(ln,61, "uR11", cmd_read(27)); ln++; dispreg(ln, 1, "uR12", cmd_read(28)); dispreg(ln,21, "uSP ", cmd_read(29)); cc = cmd_read(30); mvprintw(ln,41, "uCC :%s%s%s%s%s%s%s", (cc&0x040)?"STP":" ", (cc&0x020)?"GIE":" ", (cc&0x010)?"SLP":" ", (cc&8)?"V":" ", (cc&4)?"N":" ", (cc&2)?"C":" ", (cc&1)?"Z":" "); mvprintw(ln,61, "uPC : 0x%08x", cmd_read(31)); attroff(A_BOLD); ln+=2; ln+=3; showins(ln, "I ", !m_core->v__DOT__thecpu__DOT__dcd_stalled, m_core->v__DOT__thecpu__DOT__pf_valid, m_core->v__DOT__thecpu__DOT__gie, 0, // m_core->v__DOT__thecpu__DOT__instruction_pc); ln++; m_core->v__DOT__thecpu__DOT__pf_pc); ln++; showins(ln, "Dc", m_core->v__DOT__thecpu__DOT__dcd_ce, m_core->v__DOT__thecpu__DOT__dcdvalid, m_core->v__DOT__thecpu__DOT__dcd_gie, m_core->v__DOT__thecpu__DOT__dcd_stalled, m_core->v__DOT__thecpu__DOT__dcd_pc-1); ln++; showins(ln, "Op", m_core->v__DOT__thecpu__DOT__op_ce, m_core->v__DOT__thecpu__DOT__opvalid, m_core->v__DOT__thecpu__DOT__op_gie, m_core->v__DOT__thecpu__DOT__op_stall, m_core->v__DOT__thecpu__DOT__op_pc-1); ln++; showins(ln, "Al", m_core->v__DOT__thecpu__DOT__alu_ce, m_core->v__DOT__thecpu__DOT__alu_pc_valid, m_core->v__DOT__thecpu__DOT__alu_gie, m_core->v__DOT__thecpu__DOT__alu_stall, m_core->v__DOT__thecpu__DOT__alu_pc-1); ln++; } void tick(void) { int gie = m_core->v__DOT__thecpu__DOT__gie; /* m_core->i_qspi_dat = m_flash(m_core->o_qspi_cs_n, m_core->o_qspi_sck, m_core->o_qspi_dat); */ int stb = m_core->o_wb_stb; if ((m_core->o_wb_addr & (-1<<20))!=1) stb = 0; m_mem(m_core->o_wb_cyc, m_core->o_wb_stb, m_core->o_wb_we, m_core->o_wb_addr & ((1<<20)-1), m_core->o_wb_data, m_core->i_wb_ack, m_core->i_wb_stall,m_core->i_wb_data); if ((m_core->o_wb_cyc)&&(m_core->o_wb_stb)&&(!stb)) m_core->i_wb_ack = 1; if ((dbg_flag)&&(dbg_fp)) { fprintf(dbg_fp, "DBG %s %s %s @0x%08x/%d[0x%08x] %s %s [0x%08x] %s %s %s%s%s%s%s%s%s%s\n", (m_core->i_dbg_cyc)?"CYC":" ", (m_core->i_dbg_stb)?"STB": ((m_core->v__DOT__dbg_stb)?"DBG":" "), ((m_core->i_dbg_we)?"WE":" "), (m_core->i_dbg_addr),0, m_core->i_dbg_data, (m_core->o_dbg_ack)?"ACK":" ", (m_core->o_dbg_stall)?"STALL":" ", (m_core->o_dbg_data), (m_core->v__DOT__cpu_halt)?"CPU-HALT ":"", (m_core->v__DOT__cpu_dbg_stall)?"CPU-DBG_STALL":"", (m_core->v__DOT__thecpu__DOT__dcdvalid)?"DCDV ":"", (m_core->v__DOT__thecpu__DOT__opvalid)?"OPV ":"", (m_core->v__DOT__thecpu__DOT__pf_cyc)?"PCYC ":"", (m_core->v__DOT__thecpu__DOT__mem_cyc)?"MCYC ":"", (m_core->v__DOT__thecpu__DOT__alu_wr)?"ALUW ":"", (m_core->v__DOT__thecpu__DOT__alu_ce)?"ALCE ":"", (m_core->v__DOT__thecpu__DOT__alu_valid)?"ALUV ":"", (m_core->v__DOT__thecpu__DOT__mem_valid)?"MEMV ":""); fprintf(dbg_fp, " SYS %s %s %s @0x%08x/%d[0x%08x] %s [0x%08x]\n", (m_core->v__DOT__sys_cyc)?"CYC":" ", (m_core->v__DOT__sys_stb)?"STB":" ", (m_core->v__DOT__sys_we)?"WE":" ", (m_core->v__DOT__sys_addr), (m_core->v__DOT__dbg_addr), (m_core->v__DOT__sys_data), (m_core->v__DOT__dbg_ack)?"ACK":" ", (m_core->v__DOT__wb_data)); } if (dbg_fp) fprintf(dbg_fp, "CEs %d/0x%08x,%d/0x%08x DCD: ->%02x, OP: ->%02x, ALU: halt=%d,%d ce=%d, valid=%d, wr=%d Reg=%02x, IPC=%08x, UPC=%08x\n", m_core->v__DOT__thecpu__DOT__dcd_ce, m_core->v__DOT__thecpu__DOT__dcd_pc, m_core->v__DOT__thecpu__DOT__op_ce, m_core->v__DOT__thecpu__DOT__op_pc, m_core->v__DOT__thecpu__DOT__dcdA, m_core->v__DOT__thecpu__DOT__opR, m_core->v__DOT__cmd_halt, m_core->v__DOT__cpu_halt, m_core->v__DOT__thecpu__DOT__alu_ce, m_core->v__DOT__thecpu__DOT__alu_valid, m_core->v__DOT__thecpu__DOT__alu_wr, m_core->v__DOT__thecpu__DOT__alu_reg, m_core->v__DOT__thecpu__DOT__ipc, m_core->v__DOT__thecpu__DOT__upc); if ((dbg_fp)&&(!gie)&&(m_core->v__DOT__thecpu__DOT__w_release_from_interrupt)) { fprintf(dbg_fp, "RELEASE: int=%d, %d/%02x[%08x] ?/%02x[0x%08x], ce=%d %d,%d,%d\n", m_core->v__DOT__pic_interrupt, m_core->v__DOT__thecpu__DOT__wr_reg_ce, m_core->v__DOT__thecpu__DOT__wr_reg_id, m_core->v__DOT__thecpu__DOT__wr_reg_vl, m_core->v__DOT__cmd_addr, m_core->v__DOT__dbg_idata, m_core->v__DOT__thecpu__DOT__master_ce, m_core->v__DOT__thecpu__DOT__alu_wr, m_core->v__DOT__thecpu__DOT__alu_valid, m_core->v__DOT__thecpu__DOT__mem_valid); } else if ((dbg_fp)&&(gie)&&(m_core->v__DOT__thecpu__DOT__w_switch_to_interrupt)) { fprintf(dbg_fp, "SWITCH: %d/%02x[%08x] ?/%02x[0x%08x], ce=%d %d,%d,%d, F%02x,%02x\n", m_core->v__DOT__thecpu__DOT__wr_reg_ce, m_core->v__DOT__thecpu__DOT__wr_reg_id, m_core->v__DOT__thecpu__DOT__wr_reg_vl, m_core->v__DOT__cmd_addr, m_core->v__DOT__dbg_idata, m_core->v__DOT__thecpu__DOT__master_ce, m_core->v__DOT__thecpu__DOT__alu_wr, m_core->v__DOT__thecpu__DOT__alu_valid, m_core->v__DOT__thecpu__DOT__mem_valid, m_core->v__DOT__thecpu__DOT__w_iflags, m_core->v__DOT__thecpu__DOT__w_uflags); fprintf(dbg_fp, "\tbrk=%d,%d\n", m_core->v__DOT__thecpu__DOT__break_en, m_core->v__DOT__thecpu__DOT__op_break); } TESTB<Vzipsystem>::tick(); if ((dbg_fp)&&(gie != m_core->v__DOT__thecpu__DOT__gie)) { fprintf(dbg_fp, "SWITCH FROM %s to %s: sPC = 0x%08x uPC = 0x%08x pf_pc = 0x%08x\n", (gie)?"User":"Supervisor", (gie)?"Supervisor":"User", m_core->v__DOT__thecpu__DOT__ipc, m_core->v__DOT__thecpu__DOT__upc, m_core->v__DOT__thecpu__DOT__pf_pc); } if (dbg_fp) { dbgins("Op - ", m_core->v__DOT__thecpu__DOT__op_ce, m_core->v__DOT__thecpu__DOT__opvalid, m_core->v__DOT__thecpu__DOT__op_gie, m_core->v__DOT__thecpu__DOT__op_stall, m_core->v__DOT__thecpu__DOT__op_pc-1); dbgins("Al - ", m_core->v__DOT__thecpu__DOT__alu_ce, m_core->v__DOT__thecpu__DOT__alu_pc_valid, m_core->v__DOT__thecpu__DOT__alu_gie, m_core->v__DOT__thecpu__DOT__alu_stall, m_core->v__DOT__thecpu__DOT__alu_pc-1); } } bool test_success(void) { return ((!m_core->v__DOT__thecpu__DOT__gie) &&(m_core->v__DOT__thecpu__DOT__sleep)); } bool test_failure(void) { return ((m_core->v__DOT__thecpu__DOT__alu_pc_valid) &&(m_mem[m_core->v__DOT__thecpu__DOT__alu_pc-1] == 0x2f0f7fff)); } void wb_write(unsigned a, unsigned int v) { mvprintw(0,35, "%40s", ""); mvprintw(0,40, "wb_write(%d,%x)", a, v); m_core->i_dbg_cyc = 1; m_core->i_dbg_stb = 1; m_core->i_dbg_we = 1; m_core->i_dbg_addr = a & 1; m_core->i_dbg_data = v; tick(); while(m_core->o_dbg_stall) tick(); m_core->i_dbg_stb = 0; while(!m_core->o_dbg_ack) tick(); // Release the bus m_core->i_dbg_cyc = 0; m_core->i_dbg_stb = 0; tick(); mvprintw(0,35, "%40s", ""); mvprintw(0,40, "wb_write -- complete"); } unsigned long wb_read(unsigned a) { unsigned int v; mvprintw(0,35, "%40s", ""); mvprintw(0,40, "wb_read(0x%08x)", a); m_core->i_dbg_cyc = 1; m_core->i_dbg_stb = 1; m_core->i_dbg_we = 0; m_core->i_dbg_addr = a & 1; tick(); while(m_core->o_dbg_stall) tick(); m_core->i_dbg_stb = 0; while(!m_core->o_dbg_ack) tick(); v = m_core->o_dbg_data; // Release the bus m_core->i_dbg_cyc = 0; m_core->i_dbg_stb = 0; tick(); mvprintw(0,35, "%40s", ""); mvprintw(0,40, "wb_read = 0x%08x", v); return v; } }; void usage(void) { printf("USAGE: zippy_tb [-a] <testfile.out>\n"); printf("\n"); printf("\tWhere testfile.out is an output file from the assembler.\n"); printf("\t-a\tSets the testbench to run automatically without any\n"); printf("\t\tuser interaction.\n"); printf("\n"); printf("\tUser Commands:\n"); printf("\t\tWhen the test bench is run interactively, the following\n"); printf("\t\tkey strokes are recognized:\n"); printf("\t\t\'h\'\tHalt the processor using the external interface.\n"); printf("\t\t\'g\'\tLet the processor run at full throttle with no.\n"); printf("\t\t\tuser intervention.\n"); printf("\t\t\'q\'\tQuit the simulation.\n"); printf("\t\t\'r\'\tReset the processor.\n"); printf("\t\t\'s\'\tStep the CPU using the external stepping command\n"); printf("\t\t\tThis may consume more than one tick.\n"); printf("\t\t\'t\'\tClock a single tick through the system.\n"); } int main(int argc, char **argv) { Verilated::commandArgs(argc, argv); ZIPPY_TB *tb = new ZIPPY_TB(); bool autorun = false, exit_on_done = false; // mem[0x00000] = 0xbe000010; // Halt instruction unsigned int mptr = 0; if (argc <= 1) { usage(); exit(-1); } else { for(int argn=1; argn<argc; argn++) { if (argv[argn][0] == '-') { switch(argv[argn][1]) { case 'a': autorun = true; break; case 'e': exit_on_done = true; break; case 'h': usage(); exit(0); break; default: usage(); exit(-1); break; } } else if (access(argv[argn], R_OK)==0) { FILE *fp = fopen(argv[argn], "r"); if (fp == NULL) { printf("Cannot open %s\n", argv[argn]); perror("O/S Err: "); exit(-1); } mptr += fread(&tb->m_mem[mptr], sizeof(ZIPI), tb->m_mem_size - mptr, fp); fclose(fp); } } } if (autorun) { bool done = false; printf("Running in non-interactive mode\n"); tb->reset(); for(int i=0; i<2; i++) tb->tick(); tb->m_core->v__DOT__cmd_halt = 0; while(!done) { tb->tick(); // tb->m_core->v__DOT__thecpu__DOT__step = 0; // tb->m_core->v__DOT__cmd_halt = 0; // tb->m_core->v__DOT__cmd_step = 0; printf("PC = %08x:%08x (%08x)\n", tb->m_core->v__DOT__thecpu__DOT__ipc, tb->m_core->v__DOT__thecpu__DOT__upc, tb->m_core->v__DOT__thecpu__DOT__alu_pc); done = (tb->test_success())||(tb->test_failure()); } } else { // Interactive initscr(); raw(); noecho(); keypad(stdscr, true); tb->reset(); for(int i=0; i<2; i++) tb->tick(); tb->m_core->v__DOT__cmd_halt = 0; int chv = 'q'; bool done = false, halted = true, manual = true; halfdelay(1); // tb->wb_write(CMD_REG, CMD_HALT | CMD_RESET); // while((tb->wb_read(CMD_REG) & (CMD_HALT|CMD_STALL))==(CMD_HALT|CMD_STALL)) // tb->show_state(); while(!done) { chv = getch(); switch(chv) { case 'h': case 'H': tb->wb_write(CMD_REG, CMD_HALT); if (!halted) erase(); halted = true; break; case 'g': case 'G': tb->wb_write(CMD_REG, 0); if (halted) erase(); halted = false; manual = false; break; case 'q': case 'Q': done = true; break; case 'r': case 'R': tb->wb_write(CMD_REG, CMD_RESET|CMD_HALT); halted = true; erase(); break; case 's': case 'S': if (manual) erase(); tb->wb_write(CMD_REG, CMD_STEP); manual = false; break; case 't': case 'T': if (!manual) erase(); manual = true; // tb->m_core->v__DOT__thecpu__DOT__step = 0; // tb->m_core->v__DOT__cmd_halt = 0; // tb->m_core->v__DOT__cmd_step = 0; tb->tick(); break; case ERR: default: if (!manual) tb->tick(); } if (manual) { tb->show_state(); } else if (halted) { if (tb->dbg_fp) fprintf(tb->dbg_fp, "\n\nREAD-STATE ******\n"); tb->read_state(); } else tb->show_state(); if (tb->m_core->i_rst) done =true; if (tb->bomb) done = true; if (exit_on_done) { if (tb->test_success()) done = true; if (tb->test_failure()) done = true; } } endwin(); } #ifdef MANUAL_STEPPING_MODE else { // Manual stepping mode tb->show_state(); while('q' != tolower(chv = getch())) { tb->tick(); tb->show_state(); if (tb->test_success()) break; else if (tb->test_failure()) break; } } #endif printf("Clocks used : %08x\n", tb->m_core->v__DOT__mtc_data); printf("Instructions Issued : %08x\n", tb->m_core->v__DOT__mic_data); if (tb->m_core->v__DOT__mtc_data != 0) printf("Instructions / Clock: %.2f\n", (double)tb->m_core->v__DOT__mic_data / (double)tb->m_core->v__DOT__mtc_data); if (tb->test_success()) printf("SUCCESS!\n"); else if (tb->test_failure()) printf("TEST FAILED!\n"); else printf("User quit\n"); exit(0); }
Go to most recent revision | Compare with Previous | Blame | View Log