blob: cf45955d32e45ff68de9d40a99e2927e71564d6a [file] [log] [blame]
// 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 "VVRegfile.h"
#include "tests/verilator_sim/kelvin/kelvin_cfg.h"
#include "tests/verilator_sim/sysc_tb.h"
struct VRegfile_tb : Sysc_tb {
sc_out<bool> io_read_0_valid;
sc_out<sc_bv<6> > io_read_0_addr;
sc_out<bool> io_read_0_tag;
sc_in<sc_bv<kVector> > io_read_0_data;
sc_out<bool> io_read_1_valid;
sc_out<sc_bv<6> > io_read_1_addr;
sc_out<bool> io_read_1_tag;
sc_in<sc_bv<kVector> > io_read_1_data;
sc_out<bool> io_read_2_valid;
sc_out<sc_bv<6> > io_read_2_addr;
sc_out<bool> io_read_2_tag;
sc_in<sc_bv<kVector> > io_read_2_data;
sc_out<bool> io_read_3_valid;
sc_out<sc_bv<6> > io_read_3_addr;
sc_out<bool> io_read_3_tag;
sc_in<sc_bv<kVector> > io_read_3_data;
sc_out<bool> io_read_4_valid;
sc_out<sc_bv<6> > io_read_4_addr;
sc_out<bool> io_read_4_tag;
sc_in<sc_bv<kVector> > io_read_4_data;
sc_out<bool> io_read_5_valid;
sc_out<sc_bv<6> > io_read_5_addr;
sc_out<bool> io_read_5_tag;
sc_in<sc_bv<kVector> > io_read_5_data;
sc_out<bool> io_read_6_valid;
sc_out<sc_bv<6> > io_read_6_addr;
sc_out<bool> io_read_6_tag;
sc_in<sc_bv<kVector> > io_read_6_data;
sc_out<bool> io_scalar_0_valid;
sc_out<sc_bv<32> > io_scalar_0_data;
sc_out<bool> io_scalar_1_valid;
sc_out<sc_bv<32> > io_scalar_1_data;
sc_out<bool> io_write_0_valid;
sc_out<sc_bv<6> > io_write_0_addr;
sc_out<sc_bv<kVector> > io_write_0_data;
sc_out<bool> io_write_1_valid;
sc_out<sc_bv<6> > io_write_1_addr;
sc_out<sc_bv<kVector> > io_write_1_data;
sc_out<bool> io_write_2_valid;
sc_out<sc_bv<6> > io_write_2_addr;
sc_out<sc_bv<kVector> > io_write_2_data;
sc_out<bool> io_write_3_valid;
sc_out<sc_bv<6> > io_write_3_addr;
sc_out<sc_bv<kVector> > io_write_3_data;
sc_out<bool> io_write_4_valid;
sc_out<sc_bv<6> > io_write_4_addr;
sc_out<sc_bv<kVector> > io_write_4_data;
sc_out<bool> io_write_5_valid;
sc_out<sc_bv<6> > io_write_5_addr;
sc_out<sc_bv<kVector> > io_write_5_data;
sc_out<bool> io_whint_0_valid;
sc_out<sc_bv<6> > io_whint_0_addr;
sc_out<bool> io_whint_1_valid;
sc_out<sc_bv<6> > io_whint_1_addr;
sc_out<bool> io_whint_2_valid;
sc_out<sc_bv<6> > io_whint_2_addr;
sc_out<bool> io_whint_3_valid;
sc_out<sc_bv<6> > io_whint_3_addr;
sc_out<bool> io_transpose_valid;
sc_out<sc_bv<6> > io_transpose_addr;
sc_out<sc_bv<ctz(kVector / 32)> > io_transpose_index;
sc_in<sc_bv<kVector> > io_transpose_data;
sc_out<bool> io_conv_valid;
sc_out<bool> io_conv_ready;
sc_out<bool> io_conv_op_conv;
sc_out<bool> io_conv_op_init;
sc_out<bool> io_conv_op_tran;
sc_out<bool> io_conv_op_wclr;
sc_out<sc_bv<6> > io_conv_addr1;
sc_out<sc_bv<6> > io_conv_addr2;
sc_out<sc_bv<2> > io_conv_mode;
sc_out<sc_bv<ctz(kVector / 32)> > io_conv_index;
sc_out<sc_bv<9> > io_conv_abias;
sc_out<sc_bv<9> > io_conv_bbias;
sc_out<bool> io_conv_asign;
sc_out<bool> io_conv_bsign;
sc_out<bool> io_vrfsb_set_valid;
sc_out<sc_bv<128> > io_vrfsb_set_bits;
sc_in<sc_bv<128> > io_vrfsb_data;
using Sysc_tb::Sysc_tb;
void posedge() {
if (init_ < 64) {
io_write_0_valid = true;
io_write_0_addr = init_;
sc_bv<kVector> wdata;
for (int i = 0; i < kVector / 32; ++i) {
uint32_t data = rand_uint32();
vreg_[init_][i] = data;
wdata.set_word(i, data);
}
io_write_0_data = wdata;
init_++;
return;
}
if (init_ < 72) {
io_write_0_valid = false;
io_write_0_addr = 0;
io_write_0_data = 0;
init_++;
return;
}
conv_prev_ = conv_curr_;
conv_curr_ = io_conv_valid && io_conv_ready;
// Inputs.
bool wvalid[6];
uint8_t waddr[6];
if ((cycle() & 1023) < 64) {
// Periodically mask writes so convolution can burst.
memset(wvalid, 0, sizeof(wvalid));
} else {
// Randomly write.
for (int i = 0; i < 6; ++i) {
wvalid[i] = rand_bool();
}
}
RandWriteAddresses(waddr, conv_curr_);
write_mask1_ = write_mask0_;
write_mask0_ = 0;
for (int i = 0; i < 6; ++i) {
if (wvalid[i]) {
write_mask0_ |= 1ull << waddr[i];
}
}
const uint64_t write_mask = write_mask0_ | write_mask1_;
io_write_0_valid = wvalid[0];
io_write_1_valid = wvalid[1];
io_write_2_valid = wvalid[2];
io_write_3_valid = wvalid[3];
io_write_4_valid = wvalid[4];
io_write_5_valid = wvalid[5];
io_write_0_addr = waddr[0];
io_write_1_addr = waddr[1];
io_write_2_addr = waddr[2];
io_write_3_addr = waddr[3];
io_write_4_addr = waddr[4];
io_write_5_addr = waddr[5];
bool pvalid[4];
uint8_t paddr[4];
for (int i = 0; i < 4; ++i) {
pvalid[i] = rand_bool();
paddr[i] = rand_uint32(0, 63);
}
io_whint_0_valid = pvalid[0];
io_whint_1_valid = pvalid[1];
io_whint_2_valid = pvalid[2];
io_whint_3_valid = pvalid[3];
io_whint_0_addr = paddr[0];
io_whint_1_addr = paddr[1];
io_whint_2_addr = paddr[2];
io_whint_3_addr = paddr[3];
#define WRITE_DATA(idx) \
{ \
sc_bv<kVector> wdata; \
for (int i = 0; i < kVector / 32; ++i) { \
wdata.set_word(i, rand_uint32()); \
} \
io_write_##idx##_data = wdata; \
}
WRITE_DATA(0);
WRITE_DATA(1);
WRITE_DATA(2);
WRITE_DATA(3);
WRITE_DATA(4);
WRITE_DATA(5);
uint8_t conv_addr1 = 0;
uint8_t conv_addr2 = 0;
uint8_t conv_index = rand_uint32(0, (kVector / 32) - 1);
bool conv_valid = rand_uint32(0, 7) != 0 &&
RandConvAddr(conv_addr1, conv_addr2, write_mask);
bool conv_ready = rand_bool();
bool conv_op_wclr = rand_uint32(0, 7) == 0 && (write_mask >> 48) == 0;
bool conv_op_conv = !conv_op_wclr;
bool conv_op_init = false;
bool conv_op_tran = false;
bool scalar0valid = rand_int(0, 7) == 0;
bool scalar1valid = rand_int(0, 7) == 0;
io_conv_valid = conv_valid;
io_conv_ready = conv_ready;
io_conv_op_conv = conv_op_conv;
io_conv_op_init = conv_op_init;
io_conv_op_tran = conv_op_tran;
io_conv_op_wclr = conv_op_wclr;
io_conv_addr1 = conv_addr1;
io_conv_addr2 = conv_addr2;
io_conv_mode = 0;
io_conv_index = conv_index;
io_conv_abias = rand_uint32();
io_conv_bbias = rand_uint32();
io_conv_asign = rand_bool();
io_conv_bsign = rand_bool();
io_read_0_valid = rand_bool();
io_read_1_valid = rand_bool() && !scalar0valid;
io_read_2_valid = rand_bool();
io_read_3_valid = rand_bool();
io_read_4_valid = rand_bool() && !scalar1valid;
io_read_5_valid = rand_bool();
io_read_6_valid = rand_bool();
io_read_0_addr = rand_uint32(0, 63);
io_read_1_addr = rand_uint32(0, 63);
io_read_2_addr = rand_uint32(0, 63);
io_read_3_addr = rand_uint32(0, 63);
io_read_4_addr = rand_uint32(0, 63);
io_read_5_addr = rand_uint32(0, 63);
io_read_6_addr = rand_uint32(0, 63);
io_read_0_tag = rand_bool();
io_read_1_tag = rand_bool();
io_read_2_tag = rand_bool();
io_read_3_tag = rand_bool();
io_read_4_tag = rand_bool();
io_read_5_tag = rand_bool();
io_read_6_tag = rand_bool();
io_scalar_0_valid = scalar0valid;
io_scalar_1_valid = scalar1valid;
io_scalar_0_data = rand_uint32();
io_scalar_1_data = rand_uint32();
uint8_t transpose_addr = 0;
bool transpose_valid =
rand_bool() && RandTransposeAddr(transpose_addr, write_mask);
io_transpose_valid =
transpose_valid && !conv_valid && !conv_curr_ && !conv_prev_;
io_transpose_addr = transpose_addr;
io_transpose_index = rand_uint32(0, (kVector / 32) - 1);
// Read.
#define READ_CHECK(idx) \
if (read_valid_[idx] || read_scalar_valid_[idx]) { \
for (int i = 0; i < kVector / 32; ++i) { \
uint32_t value = read_scalar_valid_[idx] ? read_scalar_data_[idx] \
: vreg_[read_addr_[idx]][i]; \
check(io_read_##idx##_data.read().get_word(i) == value, "read data"); \
} \
}
READ_CHECK(0);
READ_CHECK(1);
READ_CHECK(2);
READ_CHECK(3);
READ_CHECK(4);
READ_CHECK(5);
// Reads from convolution accumulator writes will be ignored by schedule.
#define READ_LATCH(idx) \
read_valid_[idx] = io_read_##idx##_valid.read() && \
!((conv_curr_ || conv_prev_) && \
io_read_##idx##_addr.read().get_word(0) >= 48); \
read_addr_[idx] = io_read_##idx##_addr.read().get_word(0); \
if (idx == 1) { \
read_scalar_valid_[idx] = io_scalar_0_valid; \
read_scalar_data_[idx] = io_scalar_0_data.read().get_word(0); \
} \
if (idx == 4) { \
read_scalar_valid_[idx] = io_scalar_1_valid; \
read_scalar_data_[idx] = io_scalar_1_data.read().get_word(0); \
}
READ_LATCH(0);
READ_LATCH(1);
READ_LATCH(2);
READ_LATCH(3);
READ_LATCH(4);
READ_LATCH(5);
// Transpose.
for (int i = 0; i < kVector / 32; ++i) {
if (transpose_valid_) {
check(io_transpose_data.read().get_word(i) ==
vreg_[transpose_addr_ + i][transpose_index_],
"transpose data");
}
}
transpose_valid_ = io_transpose_valid;
transpose_addr_ = io_transpose_addr.read().get_word(0);
transpose_index_ = io_transpose_index.read().get_word(0);
// Scoreboard.
io_vrfsb_set_valid = rand_bool();
io_vrfsb_set_bits = ScoreboardSet();
for (int i = 0; i < 2; ++i) {
uint64_t sb_ref = scoreboard_[i];
uint64_t sb_dut =
io_vrfsb_data.read().get_word(2 * i + 0) |
(uint64_t(io_vrfsb_data.read().get_word(2 * i + 1)) << 32);
check(sb_ref == sb_dut, "Scoreboard");
}
// Update model.
#define WRITE_UPDATE(idx) \
if (io_write_##idx##_valid) { \
int addr = io_write_##idx##_addr.read().get_word(0); \
uint64_t clear = ~(1ull << addr); \
scoreboard_[0] &= clear; \
scoreboard_[1] &= clear; \
for (int i = 0; i < kVector / 32; ++i) { \
vreg_[addr][i] = io_write_##idx##_data.read().get_word(i); \
} \
}
#define WHINT_UPDATE(idx) \
if (io_whint_##idx##_valid) { \
int addr = io_whint_##idx##_addr.read().get_word(0); \
uint64_t clear = ~(1ull << addr); \
scoreboard_[0] &= clear; \
scoreboard_[1] &= clear; \
}
WRITE_UPDATE(0);
WRITE_UPDATE(1);
WRITE_UPDATE(2);
WRITE_UPDATE(3);
WRITE_UPDATE(4);
WRITE_UPDATE(5);
WHINT_UPDATE(0);
WHINT_UPDATE(1);
WHINT_UPDATE(2);
WHINT_UPDATE(3);
if (wclr_) {
const uint64_t clear = ~(kVector == 128 ? 0x000f000000000000llu
: kVector == 256 ? 0x00ff000000000000llu
: 0xffff000000000000llu);
scoreboard_[0] &= clear;
scoreboard_[1] &= clear;
}
// delayed one cycle beyond io.conv.wclr, no forwarding to read ports.
wclr_ = io_conv_valid && io_conv_ready && io_conv_op_wclr;
if (io_vrfsb_set_valid) {
scoreboard_[0] |= io_vrfsb_set_bits.read().get_word(0);
scoreboard_[0] |= uint64_t(io_vrfsb_set_bits.read().get_word(1)) << 32;
scoreboard_[1] |= io_vrfsb_set_bits.read().get_word(2);
scoreboard_[1] |= uint64_t(io_vrfsb_set_bits.read().get_word(3)) << 32;
}
if (io_conv_valid && io_conv_ready) {
if (io_conv_op_conv) {
uint8_t addr1 = io_conv_addr1.read().get_word(0);
uint8_t addr2 = io_conv_addr2.read().get_word(0);
uint32_t abias = io_conv_abias.read().get_word(0);
uint32_t bbias = io_conv_bbias.read().get_word(0);
bool asign = io_conv_asign;
bool bsign = io_conv_bsign;
int index = io_conv_index.read().get_word(0);
for (int i = 0; i < kVector / 32; ++i) {
for (int j = 0; j < kVector / 32; ++j) {
uint32_t adata = vreg_[addr1 + i][index];
uint32_t bdata = vreg_[addr2][j];
vacc_[i][j] +=
DotProduct(adata, bdata, abias, bbias, asign, bsign);
}
}
}
if (io_conv_op_wclr) {
for (int i = 0; i < kVector / 32; ++i) {
for (int j = 0; j < kVector / 32; ++j) {
constexpr int interleave[4] = {0, 2, 1, 3};
const int rbase = i & ~3;
const int rquad = i & 3;
const int word = j;
const int si = rbase + interleave[word & 3];
const int sj = rquad * (kVector / 32 / 4) + (word / 4);
vreg_[si + 48][sj] = vacc_[i][j];
vacc_[i][j] = 0;
}
}
}
}
}
private:
int init_ = 0;
bool wclr_ = false;
uint32_t vreg_[64][kVector / 32];
uint32_t vacc_[kVector / 32][kVector / 32] = {0};
bool read_valid_[6] = {false};
uint8_t read_addr_[6];
bool read_scalar_valid_[6] = {false};
uint32_t read_scalar_data_[6];
bool transpose_valid_ = false;
uint8_t transpose_addr_;
uint8_t transpose_index_;
uint64_t write_mask0_ = 0; // current cycle
uint64_t write_mask1_ = 0; // previous cycle
uint64_t scoreboard_[2] = {0};
bool conv_curr_ = false;
bool conv_prev_ = false;
void RandWriteAddresses(uint8_t addr[6], bool exclude_accumulators) {
uint8_t active[64];
int mark = 0;
memset(active, 0, 64);
while (mark < 6) {
int r = rand_uint32(0, exclude_accumulators ? 47 : 63);
if (active[r]) continue;
active[r] = 1;
addr[mark] = r;
++mark;
}
}
sc_bv<128> ScoreboardSet() {
sc_bv<128> data = 0;
int n = rand_int(4, 8);
for (int i = 0; i < n; ++i) {
data.set_bit(rand_int(0, 127), sc_dt::Log_1);
}
return data;
}
bool RandConvAddr(uint8_t& addr1, uint8_t& addr2, uint64_t writevalid) {
const int range1 = 16;
uint8_t bank1 = rand_uint32(0, 2);
uint8_t bank2 = rand_uint32(0, 2);
while (bank1 == bank2) {
bank2 = rand_uint32(0, 2);
}
addr1 = bank1 * 16;
addr2 = bank2 * 16 + rand_uint32(0, 15);
// Do not overlap with an active write (which takes 2 cycles).
for (int i = addr1; i < addr1 + range1; ++i) {
if (writevalid & (1ull << i)) return false;
}
if (writevalid & (1ull << addr2)) return false;
for (int i = 48; i < 64; ++i) {
if (writevalid & (1ull << i)) return false;
}
return true;
}
bool RandTransposeAddr(uint8_t& addr, uint64_t writevalid) {
int bank = rand_uint32(0, 3);
addr = 0;
// Do not transpose from an active write (which takes 2 cycles).
for (int attempt = 0; attempt < 4; ++attempt) {
uint8_t test = (bank + attempt) & 3;
if ((writevalid & (0xffffull << (test * 16))) == 0) {
addr = test * 16;
return true;
}
}
return false;
}
uint32_t DotProduct(uint32_t adata, uint32_t bdata, uint32_t abias,
uint32_t bbias, bool asign, bool bsign) {
int32_t dotp = 0;
int32_t s_abias = int32_t(abias << 23) >> 23;
int32_t s_bbias = int32_t(bbias << 23) >> 23;
for (int i = 0; i < 4; ++i) {
int32_t s_adata = int32_t(uint8_t(adata >> (8 * i)));
int32_t s_bdata = int32_t(uint8_t(bdata >> (8 * i)));
if (asign) {
s_adata = int8_t(s_adata);
}
if (bsign) {
s_bdata = int8_t(s_bdata);
}
dotp += (s_adata + s_abias) * (s_bdata + s_bbias);
}
return dotp;
}
};
static void VRegfile_test(char* name, int loops, bool trace) {
sc_signal<bool> clock;
sc_signal<bool> reset;
sc_signal<bool> io_read_0_valid;
sc_signal<sc_bv<6> > io_read_0_addr;
sc_signal<bool> io_read_0_tag;
sc_signal<sc_bv<kVector> > io_read_0_data;
sc_signal<bool> io_read_1_valid;
sc_signal<sc_bv<6> > io_read_1_addr;
sc_signal<bool> io_read_1_tag;
sc_signal<sc_bv<kVector> > io_read_1_data;
sc_signal<bool> io_read_2_valid;
sc_signal<sc_bv<6> > io_read_2_addr;
sc_signal<bool> io_read_2_tag;
sc_signal<sc_bv<kVector> > io_read_2_data;
sc_signal<bool> io_read_3_valid;
sc_signal<sc_bv<6> > io_read_3_addr;
sc_signal<bool> io_read_3_tag;
sc_signal<sc_bv<kVector> > io_read_3_data;
sc_signal<bool> io_read_4_valid;
sc_signal<sc_bv<6> > io_read_4_addr;
sc_signal<bool> io_read_4_tag;
sc_signal<sc_bv<kVector> > io_read_4_data;
sc_signal<bool> io_read_5_valid;
sc_signal<sc_bv<6> > io_read_5_addr;
sc_signal<bool> io_read_5_tag;
sc_signal<sc_bv<kVector> > io_read_5_data;
sc_signal<bool> io_read_6_valid;
sc_signal<sc_bv<6> > io_read_6_addr;
sc_signal<bool> io_read_6_tag;
sc_signal<sc_bv<kVector> > io_read_6_data;
sc_signal<bool> io_scalar_0_valid;
sc_signal<sc_bv<32> > io_scalar_0_data;
sc_signal<bool> io_scalar_1_valid;
sc_signal<sc_bv<32> > io_scalar_1_data;
sc_signal<bool> io_write_0_valid;
sc_signal<sc_bv<6> > io_write_0_addr;
sc_signal<sc_bv<kVector> > io_write_0_data;
sc_signal<bool> io_write_1_valid;
sc_signal<sc_bv<6> > io_write_1_addr;
sc_signal<sc_bv<kVector> > io_write_1_data;
sc_signal<bool> io_write_2_valid;
sc_signal<sc_bv<6> > io_write_2_addr;
sc_signal<sc_bv<kVector> > io_write_2_data;
sc_signal<bool> io_write_3_valid;
sc_signal<sc_bv<6> > io_write_3_addr;
sc_signal<sc_bv<kVector> > io_write_3_data;
sc_signal<bool> io_write_4_valid;
sc_signal<sc_bv<6> > io_write_4_addr;
sc_signal<sc_bv<kVector> > io_write_4_data;
sc_signal<bool> io_write_5_valid;
sc_signal<sc_bv<6> > io_write_5_addr;
sc_signal<sc_bv<kVector> > io_write_5_data;
sc_signal<bool> io_whint_0_valid;
sc_signal<sc_bv<6> > io_whint_0_addr;
sc_signal<bool> io_whint_1_valid;
sc_signal<sc_bv<6> > io_whint_1_addr;
sc_signal<bool> io_whint_2_valid;
sc_signal<sc_bv<6> > io_whint_2_addr;
sc_signal<bool> io_whint_3_valid;
sc_signal<sc_bv<6> > io_whint_3_addr;
sc_signal<bool> io_transpose_valid;
sc_signal<sc_bv<6> > io_transpose_addr;
sc_signal<sc_bv<ctz(kVector / 32)> > io_transpose_index;
sc_signal<sc_bv<kVector> > io_transpose_data;
sc_signal<bool> io_conv_valid;
sc_signal<bool> io_conv_ready;
sc_signal<bool> io_conv_op_conv;
sc_signal<bool> io_conv_op_init;
sc_signal<bool> io_conv_op_tran;
sc_signal<bool> io_conv_op_wclr;
sc_signal<sc_bv<6> > io_conv_addr1;
sc_signal<sc_bv<6> > io_conv_addr2;
sc_signal<sc_bv<2> > io_conv_mode;
sc_signal<sc_bv<ctz(kVector / 32)> > io_conv_index;
sc_signal<sc_bv<9> > io_conv_abias;
sc_signal<sc_bv<9> > io_conv_bbias;
sc_signal<bool> io_conv_asign;
sc_signal<bool> io_conv_bsign;
sc_signal<sc_bv<128> > io_vrfsb_set_bits;
sc_signal<sc_bv<128> > io_vrfsb_data;
sc_signal<bool> io_vrfsb_set_valid;
VRegfile_tb tb("VRegfile_tb", loops, true /*random*/);
VVRegfile vrf(name);
if (trace) {
tb.trace(vrf);
}
vrf.clock(tb.clock);
vrf.reset(tb.reset);
BIND2(tb, vrf, io_read_0_valid);
BIND2(tb, vrf, io_read_0_addr);
BIND2(tb, vrf, io_read_0_tag);
BIND2(tb, vrf, io_read_0_data);
BIND2(tb, vrf, io_read_1_valid);
BIND2(tb, vrf, io_read_1_addr);
BIND2(tb, vrf, io_read_1_tag);
BIND2(tb, vrf, io_read_1_data);
BIND2(tb, vrf, io_read_2_valid);
BIND2(tb, vrf, io_read_2_addr);
BIND2(tb, vrf, io_read_2_tag);
BIND2(tb, vrf, io_read_2_data);
BIND2(tb, vrf, io_read_3_valid);
BIND2(tb, vrf, io_read_3_addr);
BIND2(tb, vrf, io_read_3_tag);
BIND2(tb, vrf, io_read_3_data);
BIND2(tb, vrf, io_read_4_valid);
BIND2(tb, vrf, io_read_4_addr);
BIND2(tb, vrf, io_read_4_tag);
BIND2(tb, vrf, io_read_4_data);
BIND2(tb, vrf, io_read_5_valid);
BIND2(tb, vrf, io_read_5_addr);
BIND2(tb, vrf, io_read_5_tag);
BIND2(tb, vrf, io_read_5_data);
BIND2(tb, vrf, io_read_6_valid);
BIND2(tb, vrf, io_read_6_addr);
BIND2(tb, vrf, io_read_6_tag);
BIND2(tb, vrf, io_read_6_data);
BIND2(tb, vrf, io_scalar_0_valid);
BIND2(tb, vrf, io_scalar_0_data);
BIND2(tb, vrf, io_scalar_1_valid);
BIND2(tb, vrf, io_scalar_1_data);
BIND2(tb, vrf, io_write_0_valid);
BIND2(tb, vrf, io_write_0_addr);
BIND2(tb, vrf, io_write_0_data);
BIND2(tb, vrf, io_write_1_valid);
BIND2(tb, vrf, io_write_1_addr);
BIND2(tb, vrf, io_write_1_data);
BIND2(tb, vrf, io_write_2_valid);
BIND2(tb, vrf, io_write_2_addr);
BIND2(tb, vrf, io_write_2_data);
BIND2(tb, vrf, io_write_3_valid);
BIND2(tb, vrf, io_write_3_addr);
BIND2(tb, vrf, io_write_3_data);
BIND2(tb, vrf, io_write_4_valid);
BIND2(tb, vrf, io_write_4_addr);
BIND2(tb, vrf, io_write_4_data);
BIND2(tb, vrf, io_write_5_valid);
BIND2(tb, vrf, io_write_5_addr);
BIND2(tb, vrf, io_write_5_data);
BIND2(tb, vrf, io_whint_0_valid);
BIND2(tb, vrf, io_whint_0_addr);
BIND2(tb, vrf, io_whint_1_valid);
BIND2(tb, vrf, io_whint_1_addr);
BIND2(tb, vrf, io_whint_2_valid);
BIND2(tb, vrf, io_whint_2_addr);
BIND2(tb, vrf, io_whint_3_valid);
BIND2(tb, vrf, io_whint_3_addr);
BIND2(tb, vrf, io_transpose_valid);
BIND2(tb, vrf, io_transpose_addr);
BIND2(tb, vrf, io_transpose_index);
BIND2(tb, vrf, io_transpose_data);
BIND2(tb, vrf, io_conv_valid);
BIND2(tb, vrf, io_conv_ready);
BIND2(tb, vrf, io_conv_op_conv);
BIND2(tb, vrf, io_conv_op_init);
BIND2(tb, vrf, io_conv_op_tran);
BIND2(tb, vrf, io_conv_op_wclr);
BIND2(tb, vrf, io_conv_addr1);
BIND2(tb, vrf, io_conv_addr2);
BIND2(tb, vrf, io_conv_mode);
BIND2(tb, vrf, io_conv_index);
BIND2(tb, vrf, io_conv_abias);
BIND2(tb, vrf, io_conv_bbias);
BIND2(tb, vrf, io_conv_asign);
BIND2(tb, vrf, io_conv_bsign);
BIND2(tb, vrf, io_vrfsb_set_valid);
BIND2(tb, vrf, io_vrfsb_set_bits);
BIND2(tb, vrf, io_vrfsb_data);
tb.start();
}
int sc_main(int argc, char* argv[]) {
VRegfile_test(Sysc_tb::get_name(argv[0]), 1000000, false);
return 0;
}