Files
PeakRDL-regblock/peakrdl/regblock/cpuif/axi4lite/axi4lite_tmpl.sv
2022-01-31 23:11:31 -08:00

103 lines
4.3 KiB
Systemverilog

enum logic [1:0] {
CPUIF_IDLE,
CPUIF_BRESP,
CPUIF_RRESP
} cpuif_state;
logic cpuif_prev_was_rd;
always_ff {{get_always_ff_event(cpuif.reset)}} begin
if({{get_resetsignal(cpuif.reset)}}) begin
cpuif_state <= CPUIF_IDLE;
cpuif_prev_was_rd <= '0;
cpuif_req <= '0;
cpuif_req_is_wr <= '0;
cpuif_addr <= '0;
cpuif_wr_data <= '0;
{{cpuif.signal("arready")}} <= '0;
{{cpuif.signal("awready")}} <= '0;
{{cpuif.signal("wready")}} <= '0;
{{cpuif.signal("bvalid")}} <= '0;
{{cpuif.signal("bresp")}} <= '0;
{{cpuif.signal("rvalid")}} <= '0;
{{cpuif.signal("rdata")}} <= '0;
{{cpuif.signal("rresp")}} <= '0;
end else begin
// Load response transfers as they arrive
if(cpuif_rd_ack) begin
{{cpuif.signal("rvalid")}} <= '1;
{{cpuif.signal("rdata")}} <= cpuif_rd_data;
if(cpuif_rd_err) {{cpuif.signal("rresp")}} <= 2'b10; // SLVERR
else {{cpuif.signal("rresp")}} <= 2'b00; // OKAY
end
if(cpuif_wr_ack) begin
{{cpuif.signal("bvalid")}} <= '1;
if(cpuif_wr_err) {{cpuif.signal("bresp")}} <= 2'b10; // SLVERR
else {{cpuif.signal("bresp")}} <= 2'b00; // OKAY
end
// Transaction state machine
case(cpuif_state)
CPUIF_IDLE: begin
// round-robin arbitrate between read/write requests
// Allow read if previous transfer was not a read, or no write is active
if({{cpuif.signal("arvalid")}} && (!cpuif_prev_was_rd || !{{cpuif.signal("awvalid")}} || !{{cpuif.signal("wvalid")}})) begin
cpuif_req <= '1;
cpuif_req_is_wr <= '0;
{%- if cpuif.data_width == 8 %}
cpuif_addr <= {{cpuif.signal("araddr")}}[{{cpuif.addr_width-1}}:0];
{%- else %}
cpuif_addr <= { {{-cpuif.signal("araddr")}}[{{cpuif.addr_width-1}}:{{clog2(cpuif.data_width_bytes)}}], {{clog2(cpuif.data_width_bytes)}}'b0};
{%- endif %}
{{cpuif.signal("arready")}} <= '1;
cpuif_state <= CPUIF_RRESP;
end else if({{cpuif.signal("awvalid")}} && {{cpuif.signal("wvalid")}}) begin
{{cpuif.signal("awready")}} <= '1;
{{cpuif.signal("wready")}} <= '1;
if({{cpuif.signal("wstrb")}} != {{"%d'b" % cpuif.data_width_bytes}}{{"1" * cpuif.data_width_bytes}}) begin
// Unaligned writes or use of byte strobes is not supported yet
{{cpuif.signal("bvalid")}} <= '1;
{{cpuif.signal("bresp")}} <= 2'b10; // SLVERR
end else begin
cpuif_req <= '1;
cpuif_req_is_wr <= '1;
{%- if cpuif.data_width == 8 %}
cpuif_addr <= {{cpuif.signal("awaddr")}}[{{cpuif.addr_width-1}}:0];
{%- else %}
cpuif_addr <= { {{-cpuif.signal("awaddr")}}[{{cpuif.addr_width-1}}:{{clog2(cpuif.data_width_bytes)}}], {{clog2(cpuif.data_width_bytes)}}'b0};
{%- endif %}
cpuif_wr_data <= {{cpuif.signal("wdata")}};
end
cpuif_state <= CPUIF_BRESP;
end
end
CPUIF_BRESP: begin
cpuif_req <= '0;
{{cpuif.signal("awready")}} <= '0;
{{cpuif.signal("wready")}} <= '0;
cpuif_prev_was_rd <= '0;
if({{cpuif.signal("bvalid")}} && {{cpuif.signal("bready")}}) begin
{{cpuif.signal("bvalid")}} <= '0;
cpuif_state <= CPUIF_IDLE;
end
end
CPUIF_RRESP: begin
cpuif_req <= '0;
{{cpuif.signal("arready")}} <= '0;
cpuif_prev_was_rd <= '1;
if({{cpuif.signal("rvalid")}} && {{cpuif.signal("rready")}}) begin
{{cpuif.signal("rvalid")}} <= '0;
cpuif_state <= CPUIF_IDLE;
end
end
default: begin
cpuif_state <= CPUIF_IDLE;
end
endcase
end
end