kernel/mem: Add read-first semantic emulation code.

This commit is contained in:
Marcelina Kościelnicka 2022-01-27 23:26:56 +01:00
parent 9a2294f285
commit bac750fb99
2 changed files with 116 additions and 0 deletions

View File

@ -1560,3 +1560,107 @@ void Mem::emulate_rd_srst_over_ce(int idx) {
port.ce_over_srst = true;
port.en = module->Or(NEW_ID, port.en, port.srst);
}
bool Mem::emulate_read_first_ok() {
if (wr_ports.empty())
return false;
SigSpec clk = wr_ports[0].clk;
bool clk_polarity = wr_ports[0].clk_polarity;
for (auto &port: wr_ports) {
if (!port.clk_enable)
return false;
if (port.clk != clk)
return false;
if (port.clk_polarity != clk_polarity)
return false;
}
bool found_read_first = false;
for (auto &port: rd_ports) {
if (!port.clk_enable)
return false;
if (port.clk != clk)
return false;
if (port.clk_polarity != clk_polarity)
return false;
// No point doing this operation if there is no read-first relationship
// in the first place.
for (int j = 0; j < GetSize(wr_ports); j++)
if (!port.transparency_mask[j] && !port.collision_x_mask[j])
found_read_first = true;
}
return found_read_first;
}
void Mem::emulate_read_first(FfInitVals *initvals) {
log_assert(emulate_read_first_ok());
for (int i = 0; i < GetSize(rd_ports); i++)
for (int j = 0; j < GetSize(wr_ports); j++)
if (rd_ports[i].transparency_mask[j])
emulate_transparency(j, i, initvals);
for (int i = 0; i < GetSize(rd_ports); i++)
for (int j = 0; j < GetSize(wr_ports); j++) {
log_assert(!rd_ports[i].transparency_mask[j]);
rd_ports[i].collision_x_mask[j] = false;
rd_ports[i].transparency_mask[j] = true;
}
for (auto &port: wr_ports) {
Wire *new_data = module->addWire(NEW_ID, GetSize(port.data));
Wire *new_addr = module->addWire(NEW_ID, GetSize(port.addr));
auto compressed = port.compress_en();
Wire *new_en = module->addWire(NEW_ID, GetSize(compressed.first));
FfData ff_data(module, initvals, NEW_ID);
FfData ff_addr(module, initvals, NEW_ID);
FfData ff_en(module, initvals, NEW_ID);
ff_data.width = GetSize(port.data);
ff_data.has_clk = true;
ff_data.sig_clk = port.clk;
ff_data.pol_clk = port.clk_polarity;
ff_data.sig_d = port.data;
ff_data.sig_q = new_data;;
ff_data.val_init = Const(State::Sx, ff_data.width);
ff_data.emit();
ff_addr.width = GetSize(port.addr);
ff_addr.has_clk = true;
ff_addr.sig_clk = port.clk;
ff_addr.pol_clk = port.clk_polarity;
ff_addr.sig_d = port.addr;
ff_addr.sig_q = new_addr;;
ff_addr.val_init = Const(State::Sx, ff_addr.width);
ff_addr.emit();
ff_en.width = GetSize(compressed.first);
ff_en.has_clk = true;
ff_en.sig_clk = port.clk;
ff_en.pol_clk = port.clk_polarity;
ff_en.sig_d = compressed.first;
ff_en.sig_q = new_en;;
ff_en.val_init = Const(State::S0, ff_en.width);
ff_en.emit();
port.data = new_data;
port.addr = new_addr;
port.en = port.decompress_en(compressed.second, new_en);
}
}
std::pair<SigSpec, std::vector<int>> MemWr::compress_en() {
SigSpec sig = en[0];
std::vector<int> swizzle;
SigBit prev_bit = en[0];
int idx = 0;
for (auto &bit: en) {
if (bit != prev_bit) {
sig.append(bit);
prev_bit = bit;
idx++;
}
swizzle.push_back(idx);
}
log_assert(idx + 1 == GetSize(sig));
return {sig, swizzle};
}
SigSpec MemWr::decompress_en(const std::vector<int> &swizzle, SigSpec sig) {
SigSpec res;
for (int i: swizzle)
res.append(sig[i]);
return res;
}

View File

@ -74,6 +74,9 @@ struct MemWr : RTLIL::AttrObject {
res[i] = State(sub >> i & 1);
return res;
}
std::pair<SigSpec, std::vector<int>> compress_en();
SigSpec decompress_en(const std::vector<int> &swizzle, SigSpec sig);
};
struct MemInit : RTLIL::AttrObject {
@ -209,6 +212,15 @@ struct Mem : RTLIL::AttrObject {
// emulation logic.
void emulate_rd_srst_over_ce(int idx);
// Returns true iff emulate_read_first makes sense to call.
bool emulate_read_first_ok();
// Emulates all read-first read-write port relationships in terms of
// all-transparent ports, by delaying all write ports by one cycle.
// This can only be used when all read ports and all write ports are
// in the same clock domain.
void emulate_read_first(FfInitVals *initvals);
Mem(Module *module, IdString memid, int width, int start_offset, int size) : module(module), memid(memid), packed(false), mem(nullptr), cell(nullptr), width(width), start_offset(start_offset), size(size) {}
};