| // Copyright 2023 Google LLC |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| #include "VCore.h" // Generated |
| #include "absl/flags/flag.h" |
| #include "absl/flags/parse.h" |
| #include "absl/flags/usage.h" |
| #include "tests/verilator_sim/kelvin/core_if.h" |
| #include "tests/verilator_sim/kelvin/debug_if.h" |
| #include "tests/verilator_sim/kelvin/kelvin_cfg.h" |
| #include "tests/verilator_sim/sysc_tb.h" |
| |
| ABSL_FLAG(int, cycles, 100000000, "Simulation cycles"); |
| ABSL_FLAG(bool, trace, false, "Dump VCD trace"); |
| |
| struct Core_tb : Sysc_tb { |
| sc_in<bool> io_halted; |
| sc_in<bool> io_fault; |
| |
| using Sysc_tb::Sysc_tb; // constructor |
| |
| void posedge() { |
| check(!io_fault, "io_fault"); |
| if (io_halted) sc_stop(); |
| } |
| }; |
| |
| static void Core_run(const char* name, const char* bin, const int cycles, |
| const bool trace) { |
| VCore core(name); |
| Core_tb tb("Core_tb", cycles, /* random= */ false); |
| Core_if mif("Core_if", bin); |
| Debug_if dbg("Debug_if", &mif); |
| |
| sc_signal<bool> io_halted; |
| sc_signal<bool> io_fault; |
| sc_signal<bool> io_debug_req; |
| sc_signal<bool> io_ibus_valid; |
| sc_signal<bool> io_ibus_ready; |
| sc_signal<bool> io_dbus_valid; |
| sc_signal<bool> io_dbus_ready; |
| sc_signal<bool> io_dbus_write; |
| sc_signal<bool> io_iflush_valid; |
| sc_signal<bool> io_iflush_ready; |
| sc_signal<bool> io_dflush_valid; |
| sc_signal<bool> io_dflush_ready; |
| sc_signal<bool> io_dflush_all; |
| sc_signal<bool> io_dflush_clean; |
| sc_signal<bool> io_slog_valid; |
| sc_signal<sc_bv<32> > io_csr_in_value_0; |
| sc_signal<sc_bv<32> > io_csr_in_value_1; |
| sc_signal<sc_bv<32> > io_csr_in_value_2; |
| sc_signal<sc_bv<32> > io_csr_in_value_3; |
| sc_signal<sc_bv<32> > io_csr_in_value_4; |
| sc_signal<sc_bv<32> > io_csr_in_value_5; |
| sc_signal<sc_bv<32> > io_csr_in_value_6; |
| sc_signal<sc_bv<32> > io_csr_in_value_7; |
| sc_signal<sc_bv<32> > io_csr_in_value_8; |
| sc_signal<sc_bv<32> > io_csr_in_value_9; |
| sc_signal<sc_bv<32> > io_csr_in_value_10; |
| sc_signal<sc_bv<32> > io_csr_in_value_11; |
| sc_signal<sc_bv<32> > io_csr_out_value_0; |
| sc_signal<sc_bv<32> > io_csr_out_value_1; |
| sc_signal<sc_bv<32> > io_csr_out_value_2; |
| sc_signal<sc_bv<32> > io_csr_out_value_3; |
| sc_signal<sc_bv<32> > io_csr_out_value_4; |
| sc_signal<sc_bv<32> > io_csr_out_value_5; |
| sc_signal<sc_bv<32> > io_csr_out_value_6; |
| sc_signal<sc_bv<32> > io_csr_out_value_7; |
| sc_signal<sc_bv<32> > io_ibus_addr; |
| sc_signal<sc_bv<256> > io_ibus_rdata; |
| sc_signal<sc_bv<32> > io_dbus_addr; |
| sc_signal<sc_bv<32> > io_dbus_adrx; |
| sc_signal<sc_bv<kDbusBits> > io_dbus_size; |
| sc_signal<sc_bv<kVector> > io_dbus_wdata; |
| sc_signal<sc_bv<kVector / 8> > io_dbus_wmask; |
| sc_signal<sc_bv<kVector> > io_dbus_rdata; |
| sc_signal<sc_bv<5> > io_slog_addr; |
| sc_signal<sc_bv<32> > io_slog_data; |
| sc_signal<sc_bv<4> > io_debug_en; |
| sc_signal<sc_bv<32> > io_debug_addr0; |
| sc_signal<sc_bv<32> > io_debug_addr1; |
| sc_signal<sc_bv<32> > io_debug_addr2; |
| sc_signal<sc_bv<32> > io_debug_addr3; |
| sc_signal<sc_bv<32> > io_debug_inst0; |
| sc_signal<sc_bv<32> > io_debug_inst1; |
| sc_signal<sc_bv<32> > io_debug_inst2; |
| sc_signal<sc_bv<32> > io_debug_inst3; |
| sc_signal<sc_bv<32> > io_debug_cycles; |
| #if 1 |
| sc_signal<bool> io_axi0_write_addr_ready; |
| sc_signal<bool> io_axi0_write_addr_valid; |
| sc_signal<sc_bv<32> > io_axi0_write_addr_bits_addr; |
| sc_signal<sc_bv<kUncId> > io_axi0_write_addr_bits_id; |
| sc_signal<bool> io_axi0_write_data_ready; |
| sc_signal<bool> io_axi0_write_data_valid; |
| sc_signal<sc_bv<kUncBits> > io_axi0_write_data_bits_data; |
| sc_signal<sc_bv<kUncStrb> > io_axi0_write_data_bits_strb; |
| sc_signal<bool> io_axi0_write_resp_ready; |
| sc_signal<bool> io_axi0_write_resp_valid; |
| sc_signal<sc_bv<kUncId> > io_axi0_write_resp_bits_id; |
| sc_signal<sc_bv<2> > io_axi0_write_resp_bits_resp; |
| sc_signal<bool> io_axi0_read_addr_ready; |
| sc_signal<bool> io_axi0_read_addr_valid; |
| sc_signal<sc_bv<32> > io_axi0_read_addr_bits_addr; |
| sc_signal<sc_bv<kUncId> > io_axi0_read_addr_bits_id; |
| sc_signal<bool> io_axi0_read_data_ready; |
| sc_signal<bool> io_axi0_read_data_valid; |
| sc_signal<sc_bv<2> > io_axi0_read_data_bits_resp; |
| sc_signal<sc_bv<kUncId> > io_axi0_read_data_bits_id; |
| sc_signal<sc_bv<kUncBits> > io_axi0_read_data_bits_data; |
| #endif // TODO: Disable if no VCore |
| sc_signal<bool> io_axi1_write_addr_ready; |
| sc_signal<bool> io_axi1_write_addr_valid; |
| sc_signal<sc_bv<32> > io_axi1_write_addr_bits_addr; |
| sc_signal<sc_bv<kUncId> > io_axi1_write_addr_bits_id; |
| sc_signal<bool> io_axi1_write_data_ready; |
| sc_signal<bool> io_axi1_write_data_valid; |
| sc_signal<sc_bv<kUncBits> > io_axi1_write_data_bits_data; |
| sc_signal<sc_bv<kUncStrb> > io_axi1_write_data_bits_strb; |
| sc_signal<bool> io_axi1_write_resp_ready; |
| sc_signal<bool> io_axi1_write_resp_valid; |
| sc_signal<sc_bv<kUncId> > io_axi1_write_resp_bits_id; |
| sc_signal<sc_bv<2> > io_axi1_write_resp_bits_resp; |
| sc_signal<bool> io_axi1_read_addr_ready; |
| sc_signal<bool> io_axi1_read_addr_valid; |
| sc_signal<sc_bv<32> > io_axi1_read_addr_bits_addr; |
| sc_signal<sc_bv<kUncId> > io_axi1_read_addr_bits_id; |
| sc_signal<bool> io_axi1_read_data_ready; |
| sc_signal<bool> io_axi1_read_data_valid; |
| sc_signal<sc_bv<2> > io_axi1_read_data_bits_resp; |
| sc_signal<sc_bv<kUncId> > io_axi1_read_data_bits_id; |
| sc_signal<sc_bv<kUncBits> > io_axi1_read_data_bits_data; |
| |
| io_iflush_ready = 1; |
| io_dflush_ready = 1; |
| |
| tb.io_halted(io_halted); |
| tb.io_fault(io_fault); |
| |
| core.clock(tb.clock); |
| core.reset(tb.reset); |
| core.io_halted(io_halted); |
| core.io_fault(io_fault); |
| core.io_debug_req(io_debug_req); |
| core.io_ibus_valid(io_ibus_valid); |
| core.io_ibus_ready(io_ibus_ready); |
| core.io_dbus_valid(io_dbus_valid); |
| core.io_dbus_ready(io_dbus_ready); |
| core.io_dbus_write(io_dbus_write); |
| core.io_iflush_valid(io_iflush_valid); |
| core.io_iflush_ready(io_iflush_ready); |
| core.io_dflush_valid(io_dflush_valid); |
| core.io_dflush_ready(io_dflush_ready); |
| core.io_dflush_all(io_dflush_all); |
| core.io_dflush_clean(io_dflush_clean); |
| core.io_slog_valid(io_slog_valid); |
| core.io_csr_in_value_0(io_csr_in_value_0); |
| core.io_csr_in_value_1(io_csr_in_value_1); |
| core.io_csr_in_value_2(io_csr_in_value_2); |
| core.io_csr_in_value_3(io_csr_in_value_3); |
| core.io_csr_in_value_4(io_csr_in_value_4); |
| core.io_csr_in_value_5(io_csr_in_value_5); |
| core.io_csr_in_value_6(io_csr_in_value_6); |
| core.io_csr_in_value_7(io_csr_in_value_7); |
| core.io_csr_in_value_8(io_csr_in_value_8); |
| core.io_csr_in_value_9(io_csr_in_value_9); |
| core.io_csr_in_value_10(io_csr_in_value_10); |
| core.io_csr_in_value_11(io_csr_in_value_11); |
| core.io_csr_out_value_0(io_csr_out_value_0); |
| core.io_csr_out_value_1(io_csr_out_value_1); |
| core.io_csr_out_value_2(io_csr_out_value_2); |
| core.io_csr_out_value_3(io_csr_out_value_3); |
| core.io_csr_out_value_4(io_csr_out_value_4); |
| core.io_csr_out_value_5(io_csr_out_value_5); |
| core.io_csr_out_value_6(io_csr_out_value_6); |
| core.io_csr_out_value_7(io_csr_out_value_7); |
| core.io_ibus_addr(io_ibus_addr); |
| core.io_ibus_rdata(io_ibus_rdata); |
| core.io_dbus_addr(io_dbus_addr); |
| core.io_dbus_adrx(io_dbus_adrx); |
| core.io_dbus_size(io_dbus_size); |
| core.io_dbus_wdata(io_dbus_wdata); |
| core.io_dbus_wmask(io_dbus_wmask); |
| core.io_dbus_rdata(io_dbus_rdata); |
| core.io_slog_addr(io_slog_addr); |
| core.io_slog_data(io_slog_data); |
| core.io_debug_en(io_debug_en); |
| core.io_debug_addr_0(io_debug_addr0); |
| core.io_debug_addr_1(io_debug_addr1); |
| core.io_debug_addr_2(io_debug_addr2); |
| core.io_debug_addr_3(io_debug_addr3); |
| core.io_debug_inst_0(io_debug_inst0); |
| core.io_debug_inst_1(io_debug_inst1); |
| core.io_debug_inst_2(io_debug_inst2); |
| core.io_debug_inst_3(io_debug_inst3); |
| core.io_debug_cycles(io_debug_cycles); |
| |
| mif.clock(tb.clock); |
| mif.reset(tb.reset); |
| mif.io_ibus_valid(io_ibus_valid); |
| mif.io_ibus_ready(io_ibus_ready); |
| mif.io_ibus_addr(io_ibus_addr); |
| mif.io_ibus_rdata(io_ibus_rdata); |
| mif.io_dbus_valid(io_dbus_valid); |
| mif.io_dbus_ready(io_dbus_ready); |
| mif.io_dbus_write(io_dbus_write); |
| mif.io_dbus_addr(io_dbus_addr); |
| mif.io_dbus_adrx(io_dbus_adrx); |
| mif.io_dbus_size(io_dbus_size); |
| mif.io_dbus_wdata(io_dbus_wdata); |
| mif.io_dbus_wmask(io_dbus_wmask); |
| mif.io_dbus_rdata(io_dbus_rdata); |
| |
| dbg.clock(tb.clock); |
| dbg.reset(tb.reset); |
| dbg.io_slog_valid(io_slog_valid); |
| dbg.io_slog_addr(io_slog_addr); |
| dbg.io_slog_data(io_slog_data); |
| |
| #define BINDAXI(a) \ |
| core.a(a); \ |
| mif.a(a) |
| #if 1 |
| BINDAXI(io_axi0_write_addr_ready); |
| BINDAXI(io_axi0_write_addr_valid); |
| BINDAXI(io_axi0_write_addr_bits_addr); |
| BINDAXI(io_axi0_write_addr_bits_id); |
| BINDAXI(io_axi0_write_data_ready); |
| BINDAXI(io_axi0_write_data_valid); |
| BINDAXI(io_axi0_write_data_bits_data); |
| BINDAXI(io_axi0_write_data_bits_strb); |
| BINDAXI(io_axi0_write_resp_ready); |
| BINDAXI(io_axi0_write_resp_valid); |
| BINDAXI(io_axi0_write_resp_bits_id); |
| BINDAXI(io_axi0_write_resp_bits_resp); |
| BINDAXI(io_axi0_read_addr_ready); |
| BINDAXI(io_axi0_read_addr_valid); |
| BINDAXI(io_axi0_read_addr_bits_addr); |
| BINDAXI(io_axi0_read_addr_bits_id); |
| BINDAXI(io_axi0_read_data_ready); |
| BINDAXI(io_axi0_read_data_valid); |
| BINDAXI(io_axi0_read_data_bits_resp); |
| BINDAXI(io_axi0_read_data_bits_id); |
| BINDAXI(io_axi0_read_data_bits_data); |
| #endif // TODO: Disable if no VCore |
| BINDAXI(io_axi1_write_addr_ready); |
| BINDAXI(io_axi1_write_addr_valid); |
| BINDAXI(io_axi1_write_addr_bits_addr); |
| BINDAXI(io_axi1_write_addr_bits_id); |
| BINDAXI(io_axi1_write_data_ready); |
| BINDAXI(io_axi1_write_data_valid); |
| BINDAXI(io_axi1_write_data_bits_data); |
| BINDAXI(io_axi1_write_data_bits_strb); |
| BINDAXI(io_axi1_write_resp_ready); |
| BINDAXI(io_axi1_write_resp_valid); |
| BINDAXI(io_axi1_write_resp_bits_id); |
| BINDAXI(io_axi1_write_resp_bits_resp); |
| BINDAXI(io_axi1_read_addr_ready); |
| BINDAXI(io_axi1_read_addr_valid); |
| BINDAXI(io_axi1_read_addr_bits_addr); |
| BINDAXI(io_axi1_read_addr_bits_id); |
| BINDAXI(io_axi1_read_data_ready); |
| BINDAXI(io_axi1_read_data_valid); |
| BINDAXI(io_axi1_read_data_bits_resp); |
| BINDAXI(io_axi1_read_data_bits_id); |
| BINDAXI(io_axi1_read_data_bits_data); |
| |
| if (trace) { |
| tb.trace(core); |
| } |
| |
| tb.start(); |
| } |
| |
| int sc_main(int argc, char *argv[]) { |
| absl::SetProgramUsageMessage("Kelvin SystemC simulation tool"); |
| auto out_args = absl::ParseCommandLine(argc, argv); |
| argc = out_args.size(); |
| argv = &out_args[0]; |
| if (argc != 2) { |
| fprintf(stderr, "Need one binary input file\n"); |
| return 1; |
| } |
| const char* path = argv[1]; |
| |
| Core_run(Sysc_tb::get_name(argv[0]), path, absl::GetFlag(FLAGS_cycles), |
| absl::GetFlag(FLAGS_trace)); |
| return 0; |
| } |