Compare commits
No commits in common. "03c30636e97aec2ff7de686fb3730a93c7783566" and "05119c1fc8ff9ec00967301e7528269ddf5248cd" have entirely different histories.
03c30636e9
...
05119c1fc8
@ -6,14 +6,17 @@ module cache_controller #(
|
||||
parameter int NUM_WAYS = 4,
|
||||
parameter int BLOCK_SIZE = 32
|
||||
) (
|
||||
input logic clk,
|
||||
input logic [ADDR_SIZE - 1:0] addr,
|
||||
input logic write_enable,
|
||||
input logic [$clog2(NUM_WAYS) - 1:0] replace_way,
|
||||
input logic [$clog2(NUM_WAYS) - 1:0] populate_way,
|
||||
output logic cru_enable,
|
||||
output logic [$clog2(NUM_WAYS) - 1:0] write_way,
|
||||
output logic [$clog2(NUM_SETS) - 1:0] set,
|
||||
output logic [ADDR_SIZE - $clog2(NUM_SETS) - $clog2(BLOCK_SIZE / 4) - 1:0] tag
|
||||
input logic replace_way,
|
||||
input logic [NUM_WAYS - 1:0] hits,
|
||||
input logic [NUM_WAYS - 1:0] valid_flags,
|
||||
output logic [SetSize - 1:0] set,
|
||||
output logic [TagSize - 1:0] tag,
|
||||
output logic [WaySize - 1:0] way,
|
||||
output logic hit,
|
||||
output logic cru_enable
|
||||
);
|
||||
localparam int NumBlockBytes = BLOCK_SIZE / 4;
|
||||
localparam int ByteOffsetSize = $clog2(NumBlockBytes);
|
||||
@ -29,40 +32,61 @@ module cache_controller #(
|
||||
|
||||
typedef enum logic [1:0] {
|
||||
READ = 'b00,
|
||||
WRITE_POPULATE = 'b10,
|
||||
WRITE_REPLACE = 'b11
|
||||
WRITE_UNVALID = 'b10,
|
||||
REPLACE = 'b11
|
||||
} cache_state_t;
|
||||
|
||||
cache_addr_t packed_addr;
|
||||
cache_addr packed_addr;
|
||||
cache_state_t state;
|
||||
|
||||
logic [WaySize - 1:0] next_populate_way;
|
||||
|
||||
logic [WaySize - 1:0] valid_encode, next_unvalid_way;
|
||||
PriorityEncoder #(
|
||||
.N(WaySize)
|
||||
) valid_flags_encoder (
|
||||
.data_in(valid_flags),
|
||||
.data_out(valid_encode),
|
||||
.valid(valid_flags_encoder_valid)
|
||||
);
|
||||
PriorityEncoder #(
|
||||
.N(WaySize)
|
||||
) read_way_encoder (
|
||||
.data_in(hits),
|
||||
.data_out(read_way),
|
||||
.valid(read_way_encoder_valid)
|
||||
);
|
||||
|
||||
logic valid;
|
||||
always_comb begin
|
||||
packed_addr = cache_addr_t'(addr);
|
||||
packed_addr = cache_addr'(addr);
|
||||
set = packed_addr.set;
|
||||
tag = packed_addr.tag;
|
||||
|
||||
hit = |hits;
|
||||
valid = &valid_flags;
|
||||
|
||||
state = cache_state_t'{write_enable, valid};
|
||||
|
||||
case (state)
|
||||
READ: begin
|
||||
cru_enable = 0;
|
||||
write_way = 'dx;
|
||||
if (read_way_encoder_valid) way = read_way;
|
||||
else way = 'd0;
|
||||
end
|
||||
WRITE_POPULATE: begin
|
||||
WRITE_UNVALID: begin
|
||||
cru_enable = 0;
|
||||
write_way = next_populate_way;
|
||||
way = next_unvalid_way;
|
||||
end
|
||||
WRITE_REPLACE: begin
|
||||
REPLACE: begin
|
||||
cru_enable = 1;
|
||||
write_way = replace_way;
|
||||
way = replace_way;
|
||||
end
|
||||
default: begin
|
||||
cru_enable = 0;
|
||||
write_way = 'dx;
|
||||
way = 'dx;
|
||||
end
|
||||
endcase
|
||||
next_populate_way = populate_way + 'd1;
|
||||
|
||||
if (valid_flags_encoder_valid) next_unvalid_way = valid_encode + 'd1;
|
||||
else next_unvalid_way = 'd0;
|
||||
end
|
||||
endmodule
|
||||
|
@ -8,36 +8,20 @@ module cache_memory #(
|
||||
) (
|
||||
input logic clk,
|
||||
input logic rst,
|
||||
input logic [$clog2(NUM_WAYS) - 1:0] write_way,
|
||||
input logic [$clog2(NUM_SETS) - 1:0] set,
|
||||
input logic [ADDR_SIZE - $clog2(NUM_SETS) - $clog2(BLOCK_SIZE / 4) - 1:0] tag,
|
||||
input logic [WaySize - 1:0] way,
|
||||
input logic [SetSize - 1:0] set,
|
||||
input logic [TagSize - 1:0] tag,
|
||||
input logic write_enable,
|
||||
input logic [BLOCK_SIZE - 1:0] write_data,
|
||||
output logic [BLOCK_SIZE - 1:0] read_data,
|
||||
output logic [$clog2(NUM_WAYS) - 1:0] populate_way,
|
||||
output logic hit
|
||||
output logic [NUM_WAYS - 1:0] hits,
|
||||
output logic [NUM_WAYS - 1:0] valid_flags
|
||||
);
|
||||
localparam int NumBlockBytes = BLOCK_SIZE / 4;
|
||||
localparam int ByteOffsetSize = $clog2(NumBlockBytes);
|
||||
localparam int WaySize = $clog2(NUM_WAYS);
|
||||
localparam int SetSize = $clog2(NUM_SETS);
|
||||
localparam int TagSize = ADDR_SIZE - SetSize - ByteOffsetSize;
|
||||
|
||||
logic [NUM_WAYS - 1:0] hits;
|
||||
logic [WaySize-1:0] way;
|
||||
logic [WaySize-1:0] read_way;
|
||||
priority_encoder #(.N(WaySize)) read_way_encoder (
|
||||
.data_in(hits),
|
||||
.data_out(read_way),
|
||||
.valid(hit)
|
||||
);
|
||||
|
||||
always_comb begin
|
||||
if(write_enable)
|
||||
way = write_way;
|
||||
else
|
||||
way = read_way;
|
||||
end
|
||||
|
||||
typedef struct packed {
|
||||
logic [BLOCK_SIZE - 1:0] data;
|
||||
@ -66,18 +50,10 @@ module cache_memory #(
|
||||
ways[way][set].valid <= 1;
|
||||
end
|
||||
end
|
||||
|
||||
logic [NUM_WAYS - 1:0] valid_flags;
|
||||
always_comb begin
|
||||
for (int i = 0; i < NUM_WAYS; i++) begin
|
||||
valid_flags[i] = ways[i][set].valid;
|
||||
hits[i] = ways[i][set].valid && (tag == ways[i][set].tag);
|
||||
end
|
||||
end
|
||||
|
||||
priority_encoder #(.N(WaySize)) populate_way_encoder (
|
||||
.data_in(valid_flags),
|
||||
.data_out(populate_way),
|
||||
.valid('dz)
|
||||
);
|
||||
endmodule
|
||||
|
@ -15,17 +15,17 @@ module two_way_lru_cache #(
|
||||
);
|
||||
localparam int NumWays = 2;
|
||||
localparam int NumBlockBytes = BLOCK_SIZE / 4;
|
||||
localparam int ByteOffsetSize = $clog2(NumBlockBytes);
|
||||
localparam int WaySize = $clog2(NumWays);
|
||||
localparam int ByteOffsetSize = $clog2(NUM_BLOCK_BYTES);
|
||||
localparam int WaySize = $clog2(NUM_WAYS);
|
||||
localparam int SetSize = $clog2(NUM_SETS);
|
||||
localparam int TagSize = ADDR_SIZE - SetSize - ByteOffsetSize;
|
||||
localparam int TagSize = ADDR_SIZE - SET_SIZE - BYTE_OFFSET_SIZE;
|
||||
|
||||
logic [$clog2(NumWays) - 1:0] populate_way;
|
||||
logic read_valid;
|
||||
logic [NUM_WAYS - 1:0] valid_flags;
|
||||
logic [NUM_WAYS - 1:0] hits;
|
||||
|
||||
logic [WaySize - 1:0] way;
|
||||
logic [SetSize - 1:0] set;
|
||||
logic [TagSize - 1:0] tag;
|
||||
logic [WAY_SIZE - 1:0] way;
|
||||
logic [SET_SIZE - 1:0] set;
|
||||
logic [TAG_SIZE - 1:0] tag;
|
||||
|
||||
cache_memory #(
|
||||
.ADDR_SIZE (ADDR_SIZE),
|
||||
@ -35,20 +35,20 @@ module two_way_lru_cache #(
|
||||
) cache_memory (
|
||||
.clk(clk),
|
||||
.rst(rst),
|
||||
.write_way(write_way),
|
||||
.way(way),
|
||||
.set(set),
|
||||
.tag(tag),
|
||||
.write_enable(write_enable),
|
||||
.write_data(write_data),
|
||||
.read_data(read_data),
|
||||
.populate_way(populate_way),
|
||||
.hit(hit)
|
||||
.hits(hits),
|
||||
.valid_flags(valid_flags)
|
||||
);
|
||||
|
||||
two_way_lru_cru #(
|
||||
.ADDR_SIZE (ADDR_SIZE),
|
||||
.NUM_SETS (NUM_SETS),
|
||||
.BLOCK_SIZE(BLOCK_SIZE)
|
||||
.BLOCK_SIZE
|
||||
) cache_replace_unit (
|
||||
.clk(clk),
|
||||
.rst(rst),
|
||||
@ -60,16 +60,19 @@ module two_way_lru_cache #(
|
||||
cache_controller #(
|
||||
.ADDR_SIZE (ADDR_SIZE),
|
||||
.NUM_SETS (NUM_SETS),
|
||||
.NUM_WAYS (NumWays),
|
||||
.NUM_WAYS (NUM_WAYS),
|
||||
.BLOCK_SIZE(BLOCK_SIZE)
|
||||
) cache_controller (
|
||||
.clk(clk),
|
||||
.addr(addr),
|
||||
.write_enable(write_enable),
|
||||
.replace_way(replace_preferred_way),
|
||||
.populate_way(populate_way),
|
||||
.cru_enable(cru_enable),
|
||||
.write_way(write_way),
|
||||
.hits(hits),
|
||||
.valid_flags(valid_flags),
|
||||
.set(set),
|
||||
.tag(tag)
|
||||
.tag(tag),
|
||||
.way(way),
|
||||
.hit(hit),
|
||||
.cru_enable(cru_enable)
|
||||
);
|
||||
endmodule
|
||||
|
@ -13,8 +13,8 @@ module two_way_lru_cru #(
|
||||
);
|
||||
localparam int NumBlocksBytes = BLOCK_SIZE / 4;
|
||||
localparam int ByteOffsetSize = $clog2(NumBlocksBytes);
|
||||
localparam int SetSize = $clog2(NUM_SETS);
|
||||
localparam int TagSize = ADDR_SIZE - SetSize - ByteOffsetSize;
|
||||
localparam int SetSize = $clog2(NumSets);
|
||||
localparam int TagSize = ADDR_SIZE - SetSize - ByteOffSetSize;
|
||||
|
||||
typedef struct packed {
|
||||
logic [ByteOffsetSize - 1:0] byte_offset;
|
||||
|
@ -1,7 +0,0 @@
|
||||
# Script to initialize a Xilinx Vivado project
|
||||
# Make sure to run this script from the project root directory
|
||||
create_project RVSCC project
|
||||
create_fileset -blockset rtl
|
||||
create_fileset -blockset fw
|
||||
add_files -fileset rtl rtl
|
||||
add_files -fileset sim_1 test
|
@ -4,14 +4,14 @@ module test_cache_memory ();
|
||||
logic clk;
|
||||
logic rst;
|
||||
|
||||
logic [dut.WaySize-1:0] write_way;
|
||||
logic [dut.WaySize-1:0] way;
|
||||
logic [dut.SetSize-1:0] set;
|
||||
logic [dut.TagSize-1:0] tag;
|
||||
logic write_enable;
|
||||
logic [31:0] write_data;
|
||||
logic [31:0] read_data;
|
||||
logic read_valid;
|
||||
logic [dut.WaySize-1:0] populate_way;
|
||||
logic [1:0] hits;
|
||||
logic [1:0] valid_flags;
|
||||
cache_memory #(
|
||||
.ADDR_SIZE (32),
|
||||
.NUM_SETS (4),
|
||||
@ -20,28 +20,30 @@ module test_cache_memory ();
|
||||
) dut (
|
||||
.clk(clk),
|
||||
.rst(rst),
|
||||
.write_way(write_way),
|
||||
.way(way),
|
||||
.set(set),
|
||||
.tag(tag),
|
||||
.write_enable(write_enable),
|
||||
.write_data(write_data),
|
||||
.read_data(read_data),
|
||||
.read_valid(read_valid),
|
||||
.populate_way(populate_way)
|
||||
.hits(hits),
|
||||
.valid_flags(valid_flags)
|
||||
);
|
||||
|
||||
localparam int ClockCycle = 2;
|
||||
always #(ClockCycle/2) clk = !clk;
|
||||
always #ClockCycle clk = !clk;
|
||||
|
||||
logic [31:0] write_value;
|
||||
|
||||
initial begin
|
||||
$dumpfile("cache.vcd");
|
||||
$dumpvars;
|
||||
clk = 0;
|
||||
rst = 1;
|
||||
#ClockCycle;
|
||||
rst = 0;
|
||||
|
||||
write_way = 0;
|
||||
way = 0;
|
||||
set = 0;
|
||||
tag = 27'($urandom);
|
||||
write_enable = 1;
|
||||
@ -50,13 +52,11 @@ module test_cache_memory ();
|
||||
#ClockCycle;
|
||||
write_enable = 0;
|
||||
tag += 1;
|
||||
#1;
|
||||
assert (read_valid == 0)
|
||||
assert (valid_flags == 'b00)
|
||||
else $error("Valid flags does not match");
|
||||
#ClockCycle;
|
||||
tag -= 1;
|
||||
#1;
|
||||
assert (read_valid == 1)
|
||||
tag -= 1;
|
||||
assert (valid_flags == 'b01)
|
||||
else $error("Valid flags does not match");
|
||||
$finish;
|
||||
end
|
||||
|
Loading…
Reference in New Issue
Block a user