1
0
mirror of https://github.com/corundum/corundum.git synced 2025-01-16 08:12:53 +08:00

merged changes in pcie

This commit is contained in:
Alex Forencich 2019-08-19 14:32:55 -07:00
commit 0fd9bc7376

View File

@ -58,7 +58,9 @@ module pcie_us_axi_dma_rd #
// Length field width // Length field width
parameter LEN_WIDTH = 20, parameter LEN_WIDTH = 20,
// Tag field width // Tag field width
parameter TAG_WIDTH = 8 parameter TAG_WIDTH = 8,
// Operation tag width
parameter OP_TAG_WIDTH = AXI_ID_WIDTH < PCIE_TAG_WIDTH ? AXI_ID_WIDTH : PCIE_TAG_WIDTH
) )
( (
input wire clk, input wire clk,
@ -156,7 +158,8 @@ parameter AXIS_PCIE_WORD_SIZE = AXIS_PCIE_DATA_WIDTH/AXIS_PCIE_WORD_WIDTH;
parameter OFFSET_WIDTH = $clog2(AXIS_PCIE_DATA_WIDTH/8); parameter OFFSET_WIDTH = $clog2(AXIS_PCIE_DATA_WIDTH/8);
parameter CYCLE_COUNT_WIDTH = 13-AXI_BURST_SIZE; parameter CYCLE_COUNT_WIDTH = 13-AXI_BURST_SIZE;
parameter STATUS_FIFO_ADDR_WIDTH = 5; parameter OP_TABLE_READ_COUNT_WIDTH = PCIE_TAG_WIDTH+1;
parameter OP_TABLE_WRITE_COUNT_WIDTH = LEN_WIDTH;
// bus width assertions // bus width assertions
initial begin initial begin
@ -184,6 +187,11 @@ initial begin
$error("Error: AXI_MAX_BURST_LEN must be between 1 and 256 (instance %m)"); $error("Error: AXI_MAX_BURST_LEN must be between 1 and 256 (instance %m)");
$finish; $finish;
end end
if (AXI_ID_WIDTH < OP_TAG_WIDTH) begin
$error("Error: AXI_ID_WIDTH must be at least OP_TAG_WIDTH (instance %m)");
$finish;
end
end end
localparam [3:0] localparam [3:0]
@ -232,7 +240,8 @@ localparam [2:0]
TLP_STATE_HEADER = 3'd1, TLP_STATE_HEADER = 3'd1,
TLP_STATE_START = 3'd2, TLP_STATE_START = 3'd2,
TLP_STATE_TRANSFER = 3'd3, TLP_STATE_TRANSFER = 3'd3,
TLP_STATE_WAIT_END = 3'd4; TLP_STATE_DROP_TAG = 3'd4,
TLP_STATE_WAIT_END = 3'd5;
reg [2:0] tlp_state_reg = TLP_STATE_IDLE, tlp_state_next; reg [2:0] tlp_state_reg = TLP_STATE_IDLE, tlp_state_next;
@ -241,8 +250,6 @@ reg transfer_in_save;
reg tag_table_we_req; reg tag_table_we_req;
reg status_fifo_we;
reg tlp_cmd_ready; reg tlp_cmd_ready;
reg finish_tag; reg finish_tag;
@ -268,7 +275,7 @@ reg bubble_cycle_reg = 1'b0, bubble_cycle_next;
reg first_cycle_reg = 1'b0, first_cycle_next; reg first_cycle_reg = 1'b0, first_cycle_next;
reg last_cycle_reg = 1'b0, last_cycle_next; reg last_cycle_reg = 1'b0, last_cycle_next;
reg [PCIE_TAG_WIDTH-1:0] pcie_tag_reg = {PCIE_TAG_WIDTH{1'b0}}, pcie_tag_next; reg [PCIE_TAG_WIDTH-1:0] pcie_tag_reg = {PCIE_TAG_WIDTH{1'b0}}, pcie_tag_next;
reg [TAG_WIDTH-1:0] tag_reg = {TAG_WIDTH{1'b0}}, tag_next; reg [OP_TAG_WIDTH-1:0] op_tag_reg = {OP_TAG_WIDTH{1'b0}}, op_tag_next;
reg final_cpl_reg = 1'b0, final_cpl_next; reg final_cpl_reg = 1'b0, final_cpl_next;
reg [OFFSET_WIDTH-1:0] offset_reg = {OFFSET_WIDTH{1'b0}}, offset_next; reg [OFFSET_WIDTH-1:0] offset_reg = {OFFSET_WIDTH{1'b0}}, offset_next;
@ -276,25 +283,16 @@ reg [OFFSET_WIDTH-1:0] first_cycle_offset_reg = {OFFSET_WIDTH{1'b0}}, first_cycl
reg [OFFSET_WIDTH-1:0] last_cycle_offset_reg = {OFFSET_WIDTH{1'b0}}, last_cycle_offset_next; reg [OFFSET_WIDTH-1:0] last_cycle_offset_reg = {OFFSET_WIDTH{1'b0}}, last_cycle_offset_next;
reg [AXI_ADDR_WIDTH-1:0] tlp_cmd_addr_reg = {AXI_ADDR_WIDTH{1'b0}}, tlp_cmd_addr_next; reg [AXI_ADDR_WIDTH-1:0] tlp_cmd_addr_reg = {AXI_ADDR_WIDTH{1'b0}}, tlp_cmd_addr_next;
reg [OP_TAG_WIDTH-1:0] tlp_cmd_op_tag_reg = {OP_TAG_WIDTH{1'b0}}, tlp_cmd_op_tag_next;
reg [TAG_WIDTH-1:0] tlp_cmd_tag_reg = {TAG_WIDTH{1'b0}}, tlp_cmd_tag_next; reg [TAG_WIDTH-1:0] tlp_cmd_tag_reg = {TAG_WIDTH{1'b0}}, tlp_cmd_tag_next;
reg [PCIE_TAG_WIDTH-1:0] tlp_cmd_pcie_tag_reg = {PCIE_TAG_WIDTH{1'b0}}, tlp_cmd_pcie_tag_next; reg [PCIE_TAG_WIDTH-1:0] tlp_cmd_pcie_tag_reg = {PCIE_TAG_WIDTH{1'b0}}, tlp_cmd_pcie_tag_next;
reg tlp_cmd_last_reg = 1'b0, tlp_cmd_last_next; reg tlp_cmd_last_reg = 1'b0, tlp_cmd_last_next;
reg tlp_cmd_valid_reg = 1'b0, tlp_cmd_valid_next; reg tlp_cmd_valid_reg = 1'b0, tlp_cmd_valid_next;
reg [AXI_ADDR_WIDTH-1:0] tag_table_axi_addr[(2**PCIE_TAG_WIDTH)-1:0]; reg [AXI_ADDR_WIDTH-1:0] tag_table_axi_addr[(2**PCIE_TAG_WIDTH)-1:0];
reg [TAG_WIDTH-1:0] tag_table_tag[(2**PCIE_TAG_WIDTH)-1:0]; reg [OP_TAG_WIDTH-1:0] tag_table_op_tag[(2**PCIE_TAG_WIDTH)-1:0];
reg tag_table_last[(2**PCIE_TAG_WIDTH)-1:0];
reg tag_table_we_tlp_reg = 1'b0, tag_table_we_tlp_next; reg tag_table_we_tlp_reg = 1'b0, tag_table_we_tlp_next;
reg [STATUS_FIFO_ADDR_WIDTH+1-1:0] status_fifo_wr_ptr_reg = 0, status_fifo_wr_ptr_next;
reg [STATUS_FIFO_ADDR_WIDTH+1-1:0] status_fifo_rd_ptr_reg = 0, status_fifo_rd_ptr_next;
reg [TAG_WIDTH-1:0] status_fifo_tag[(2**STATUS_FIFO_ADDR_WIDTH)-1:0];
reg status_fifo_last[(2**STATUS_FIFO_ADDR_WIDTH)-1:0];
reg status_fifo_completion[(2**STATUS_FIFO_ADDR_WIDTH)-1:0];
reg [TAG_WIDTH-1:0] status_fifo_wr_tag;
reg status_fifo_wr_last;
reg status_fifo_wr_completion;
reg [10:0] max_read_request_size_dw_reg = 11'd0; reg [10:0] max_read_request_size_dw_reg = 11'd0;
reg s_axis_rc_tready_reg = 1'b0, s_axis_rc_tready_next; reg s_axis_rc_tready_reg = 1'b0, s_axis_rc_tready_next;
@ -303,6 +301,7 @@ reg s_axis_read_desc_ready_reg = 1'b0, s_axis_read_desc_ready_next;
reg [TAG_WIDTH-1:0] m_axis_read_desc_status_tag_reg = {TAG_WIDTH{1'b0}}, m_axis_read_desc_status_tag_next; reg [TAG_WIDTH-1:0] m_axis_read_desc_status_tag_reg = {TAG_WIDTH{1'b0}}, m_axis_read_desc_status_tag_next;
reg m_axis_read_desc_status_valid_reg = 1'b0, m_axis_read_desc_status_valid_next; reg m_axis_read_desc_status_valid_reg = 1'b0, m_axis_read_desc_status_valid_next;
reg [AXI_ID_WIDTH-1:0] m_axi_awid_reg = {AXI_ID_WIDTH{1'b0}}, m_axi_awid_next;
reg [AXI_ADDR_WIDTH-1:0] m_axi_awaddr_reg = {AXI_ADDR_WIDTH{1'b0}}, m_axi_awaddr_next; reg [AXI_ADDR_WIDTH-1:0] m_axi_awaddr_reg = {AXI_ADDR_WIDTH{1'b0}}, m_axi_awaddr_next;
reg [7:0] m_axi_awlen_reg = 8'd0, m_axi_awlen_next; reg [7:0] m_axi_awlen_reg = 8'd0, m_axi_awlen_next;
reg m_axi_awvalid_reg = 1'b0, m_axi_awvalid_next; reg m_axi_awvalid_reg = 1'b0, m_axi_awvalid_next;
@ -337,7 +336,7 @@ assign s_axis_read_desc_ready = s_axis_read_desc_ready_reg;
assign m_axis_read_desc_status_tag = m_axis_read_desc_status_tag_reg; assign m_axis_read_desc_status_tag = m_axis_read_desc_status_tag_reg;
assign m_axis_read_desc_status_valid = m_axis_read_desc_status_valid_reg; assign m_axis_read_desc_status_valid = m_axis_read_desc_status_valid_reg;
assign m_axi_awid = {AXI_ID_WIDTH{1'b0}}; assign m_axi_awid = m_axi_awid_reg;
assign m_axi_awaddr = m_axi_awaddr_reg; assign m_axi_awaddr = m_axi_awaddr_reg;
assign m_axi_awlen = m_axi_awlen_reg; assign m_axi_awlen = m_axi_awlen_reg;
assign m_axi_awsize = $clog2(AXI_STRB_WIDTH); assign m_axi_awsize = $clog2(AXI_STRB_WIDTH);
@ -362,6 +361,7 @@ wire [3:0] first_be = 4'b1111 << req_pcie_addr_reg[1:0];
wire [3:0] last_be = 4'b1111 >> (3 - ((req_pcie_addr_reg[1:0] + req_tlp_count_next[1:0] - 1) & 3)); wire [3:0] last_be = 4'b1111 >> (3 - ((req_pcie_addr_reg[1:0] + req_tlp_count_next[1:0] - 1) & 3));
wire [10:0] dword_count = (req_tlp_count_next + req_pcie_addr_reg[1:0] + 3) >> 2; wire [10:0] dword_count = (req_tlp_count_next + req_pcie_addr_reg[1:0] + 3) >> 2;
// PCIe tag management
wire [PCIE_TAG_WIDTH-1:0] new_tag; wire [PCIE_TAG_WIDTH-1:0] new_tag;
wire new_tag_valid; wire new_tag_valid;
reg new_tag_ready; reg new_tag_ready;
@ -389,6 +389,69 @@ pcie_tag_manager_inst (
.active_tags(active_tags) .active_tags(active_tags)
); );
// operation tag management
wire [OP_TAG_WIDTH-1:0] op_table_start_ptr;
wire op_table_start_ptr_valid;
reg [TAG_WIDTH-1:0] op_table_start_tag;
reg op_table_start_en;
reg [OP_TAG_WIDTH-1:0] op_table_finish_ptr;
reg op_table_finish_en;
reg [OP_TAG_WIDTH-1:0] op_table_read_start_ptr;
reg op_table_read_start_commit;
reg op_table_read_start_en;
reg [OP_TAG_WIDTH-1:0] op_table_read_finish_ptr;
reg op_table_read_finish_en;
reg [OP_TAG_WIDTH-1:0] op_table_write_start_ptr;
reg op_table_write_start_commit;
reg op_table_write_start_en;
reg [OP_TAG_WIDTH-1:0] op_table_write_finish_ptr;
reg op_table_write_finish_en;
reg [2**OP_TAG_WIDTH-1:0] op_table_active = 0;
reg [TAG_WIDTH-1:0] op_table_tag [2**OP_TAG_WIDTH-1:0];
reg op_table_init [2**OP_TAG_WIDTH-1:0];
reg op_table_read_init [2**OP_TAG_WIDTH-1:0];
reg op_table_read_commit [2**OP_TAG_WIDTH-1:0];
reg [OP_TABLE_READ_COUNT_WIDTH-1:0] op_table_read_count_start [2**OP_TAG_WIDTH-1:0];
reg [OP_TABLE_READ_COUNT_WIDTH-1:0] op_table_read_count_finish [2**OP_TAG_WIDTH-1:0];
reg op_table_write_init [2**OP_TAG_WIDTH-1:0];
reg op_table_write_commit [2**OP_TAG_WIDTH-1:0];
reg [OP_TABLE_WRITE_COUNT_WIDTH-1:0] op_table_write_count_start [2**OP_TAG_WIDTH-1:0];
reg [OP_TABLE_WRITE_COUNT_WIDTH-1:0] op_table_write_count_finish [2**OP_TAG_WIDTH-1:0];
priority_encoder #(
.WIDTH(2**OP_TAG_WIDTH),
.LSB_PRIORITY("HIGH")
)
op_table_start_ptr_enc_inst (
.input_unencoded(~op_table_active),
.output_valid(op_table_start_ptr_valid),
.output_encoded(op_table_start_ptr),
.output_unencoded()
);
integer i;
initial begin
for (i = 0; i < 2**OP_TAG_WIDTH; i = i + 1) begin
op_table_tag[i] = 0;
op_table_init[i] = 0;
op_table_read_init[i] = 0;
op_table_read_commit[i] = 0;
op_table_read_count_start[i] = 0;
op_table_read_count_finish[i] = 0;
op_table_write_init[i] = 0;
op_table_write_commit[i] = 0;
op_table_write_count_start[i] = 0;
op_table_write_count_finish[i] = 0;
end
for (i = 0; i < 2**PCIE_TAG_WIDTH; i = i + 1) begin
tag_table_axi_addr[i] = 0;
tag_table_op_tag[i] = 0;
end
end
always @* begin always @* begin
req_state_next = REQ_STATE_IDLE; req_state_next = REQ_STATE_IDLE;
@ -400,6 +463,7 @@ always @* begin
req_tlp_count_next = req_tlp_count_reg; req_tlp_count_next = req_tlp_count_reg;
tlp_cmd_addr_next = tlp_cmd_addr_reg; tlp_cmd_addr_next = tlp_cmd_addr_reg;
tlp_cmd_op_tag_next = tlp_cmd_op_tag_reg;
tlp_cmd_tag_next = tlp_cmd_tag_reg; tlp_cmd_tag_next = tlp_cmd_tag_reg;
tlp_cmd_pcie_tag_next = tlp_cmd_pcie_tag_reg; tlp_cmd_pcie_tag_next = tlp_cmd_pcie_tag_reg;
tlp_cmd_last_next = tlp_cmd_last_reg; tlp_cmd_last_next = tlp_cmd_last_reg;
@ -450,11 +514,17 @@ always @* begin
m_axis_rq_tuser_int[59:28] = 32'd0; // parity m_axis_rq_tuser_int[59:28] = 32'd0; // parity
new_tag_ready = 1'b0; new_tag_ready = 1'b0;
op_table_start_tag = s_axis_read_desc_tag;
op_table_start_en = 1'b0;
op_table_read_start_ptr = tlp_cmd_op_tag_reg;
op_table_read_start_commit = 1'b0;
op_table_read_start_en = 1'b0;
// TLP segmentation and request generation // TLP segmentation and request generation
case (req_state_reg) case (req_state_reg)
REQ_STATE_IDLE: begin REQ_STATE_IDLE: begin
s_axis_read_desc_ready_next = enable && !tlp_cmd_valid_reg; s_axis_read_desc_ready_next = enable && !tlp_cmd_valid_reg && op_table_start_ptr_valid;
if (s_axis_read_desc_ready && s_axis_read_desc_valid) begin if (s_axis_read_desc_ready && s_axis_read_desc_valid) begin
s_axis_read_desc_ready_next = 1'b0; s_axis_read_desc_ready_next = 1'b0;
@ -462,6 +532,9 @@ always @* begin
req_axi_addr_next = s_axis_read_desc_axi_addr; req_axi_addr_next = s_axis_read_desc_axi_addr;
req_op_count_next = s_axis_read_desc_len; req_op_count_next = s_axis_read_desc_len;
tlp_cmd_tag_next = s_axis_read_desc_tag; tlp_cmd_tag_next = s_axis_read_desc_tag;
tlp_cmd_op_tag_next = op_table_start_ptr;
op_table_start_tag = s_axis_read_desc_tag;
op_table_start_en = 1'b1;
req_state_next = REQ_STATE_START; req_state_next = REQ_STATE_START;
end else begin end else begin
req_state_next = REQ_STATE_IDLE; req_state_next = REQ_STATE_IDLE;
@ -503,6 +576,10 @@ always @* begin
tlp_cmd_last_next = req_op_count_next == 0; tlp_cmd_last_next = req_op_count_next == 0;
tlp_cmd_valid_next = 1'b1; tlp_cmd_valid_next = 1'b1;
op_table_read_start_ptr = tlp_cmd_op_tag_reg;
op_table_read_start_commit = req_op_count_next == 0;
op_table_read_start_en = 1'b1;
if (req_op_count_next != 0) begin if (req_op_count_next != 0) begin
req_state_next = REQ_STATE_START; req_state_next = REQ_STATE_START;
end else begin end else begin
@ -542,6 +619,10 @@ always @* begin
tlp_cmd_last_next = req_op_count_next == 0; tlp_cmd_last_next = req_op_count_next == 0;
tlp_cmd_valid_next = 1'b1; tlp_cmd_valid_next = 1'b1;
op_table_read_start_ptr = tlp_cmd_op_tag_reg;
op_table_read_start_commit = req_op_count_next == 0;
op_table_read_start_en = 1'b1;
if (req_op_count_next != 0) begin if (req_op_count_next != 0) begin
req_state_next = REQ_STATE_START; req_state_next = REQ_STATE_START;
end else begin end else begin
@ -564,8 +645,6 @@ always @* begin
tag_table_we_tlp_next = 1'b0; tag_table_we_tlp_next = 1'b0;
status_fifo_we = 1'b0;
s_axis_rc_tready_next = 1'b0; s_axis_rc_tready_next = 1'b0;
m_axis_read_desc_status_tag_next = m_axis_read_desc_status_tag_reg; m_axis_read_desc_status_tag_next = m_axis_read_desc_status_tag_reg;
@ -587,18 +666,13 @@ always @* begin
first_cycle_next = first_cycle_reg; first_cycle_next = first_cycle_reg;
last_cycle_next = last_cycle_reg; last_cycle_next = last_cycle_reg;
pcie_tag_next = pcie_tag_reg; pcie_tag_next = pcie_tag_reg;
tag_next = tag_reg; op_tag_next = op_tag_reg;
final_cpl_next = final_cpl_reg; final_cpl_next = final_cpl_reg;
offset_next = offset_reg; offset_next = offset_reg;
first_cycle_offset_next = first_cycle_offset_reg; first_cycle_offset_next = first_cycle_offset_reg;
last_cycle_offset_next = last_cycle_offset_reg; last_cycle_offset_next = last_cycle_offset_reg;
status_fifo_rd_ptr_next = status_fifo_rd_ptr_reg; m_axi_awid_next = m_axi_awid_reg;
status_fifo_wr_tag = tag_table_tag[pcie_tag_reg];
status_fifo_wr_last = 1'b1;
status_fifo_wr_completion = 1'b1;
m_axi_awaddr_next = m_axi_awaddr_reg; m_axi_awaddr_next = m_axi_awaddr_reg;
m_axi_awlen_next = m_axi_awlen_reg; m_axi_awlen_next = m_axi_awlen_reg;
m_axi_awvalid_next = m_axi_awvalid_reg && !m_axi_awready; m_axi_awvalid_next = m_axi_awvalid_reg && !m_axi_awready;
@ -612,6 +686,16 @@ always @* begin
status_error_cor_next = 1'b0; status_error_cor_next = 1'b0;
status_error_uncor_next = 1'b0; status_error_uncor_next = 1'b0;
op_table_finish_ptr = m_axi_bid;
op_table_finish_en = 1'b0;
op_table_read_finish_ptr = op_tag_reg;
op_table_read_finish_en = 1'b0;
op_table_write_start_ptr = op_tag_reg;
op_table_write_start_commit = 1'b0;
op_table_write_start_en = 1'b0;
op_table_write_finish_ptr = m_axi_bid;
op_table_write_finish_en = 1'b0;
// TLP response handling and AXI operation generation // TLP response handling and AXI operation generation
case (tlp_state_reg) case (tlp_state_reg)
TLP_STATE_IDLE: begin TLP_STATE_IDLE: begin
@ -666,6 +750,8 @@ always @* begin
op_count_leq_axi_max_burst_next = op_count_next <= AXI_MAX_BURST_SIZE-axi_addr_next[1:0]; op_count_leq_axi_max_burst_next = op_count_next <= AXI_MAX_BURST_SIZE-axi_addr_next[1:0];
op_tag_next = tag_table_op_tag[pcie_tag_next];
if (active_tags[pcie_tag_next] && error_code_next == RC_ERROR_NORMAL_TERMINATION) begin if (active_tags[pcie_tag_next] && error_code_next == RC_ERROR_NORMAL_TERMINATION) begin
// no error // no error
axi_addr_valid_next = !final_cpl_next; axi_addr_valid_next = !final_cpl_next;
@ -701,18 +787,11 @@ always @* begin
RC_ERROR_FLR: status_error_cor_next = 1'b1; // unexpected completion (2.3.2), advisory non-fatal (6.2.3.2.4.5) RC_ERROR_FLR: status_error_cor_next = 1'b1; // unexpected completion (2.3.2), advisory non-fatal (6.2.3.2.4.5)
default: status_error_uncor_next = 1'b1; default: status_error_uncor_next = 1'b1;
endcase endcase
// release tag
finish_tag = 1'b1;
// last request in current transfer // last request in current transfer
axi_addr_valid_next = 1'b0; axi_addr_valid_next = 1'b0;
// enqueue status FIFO entry
status_fifo_we = 1'b1;
status_fifo_wr_tag = tag_table_tag[pcie_tag_next];
status_fifo_wr_last = 1'b1;
status_fifo_wr_completion = 1'b0;
// drop TLP // drop TLP
s_axis_rc_tready_next = 1'b1; s_axis_rc_tready_next = 1'b1;
tlp_state_next = TLP_STATE_WAIT_END; tlp_state_next = TLP_STATE_DROP_TAG;
end end
end else begin end else begin
s_axis_rc_tready_next = 1'b0; s_axis_rc_tready_next = 1'b0;
@ -787,6 +866,8 @@ always @* begin
op_count_leq_axi_max_burst_next = op_count_next <= AXI_MAX_BURST_SIZE-axi_addr_next[1:0]; op_count_leq_axi_max_burst_next = op_count_next <= AXI_MAX_BURST_SIZE-axi_addr_next[1:0];
op_tag_next = tag_table_op_tag[pcie_tag_next];
if (active_tags[pcie_tag_next] && error_code_reg == RC_ERROR_NORMAL_TERMINATION) begin if (active_tags[pcie_tag_next] && error_code_reg == RC_ERROR_NORMAL_TERMINATION) begin
// no error // no error
axi_addr_valid_next = !final_cpl_next; axi_addr_valid_next = !final_cpl_next;
@ -822,18 +903,11 @@ always @* begin
RC_ERROR_FLR: status_error_cor_next = 1'b1; // unexpected completion (2.3.2), advisory non-fatal (6.2.3.2.4.5) RC_ERROR_FLR: status_error_cor_next = 1'b1; // unexpected completion (2.3.2), advisory non-fatal (6.2.3.2.4.5)
default: status_error_uncor_next = 1'b1; default: status_error_uncor_next = 1'b1;
endcase endcase
// release tag
finish_tag = 1'b1;
// last request in current transfer // last request in current transfer
axi_addr_valid_next = 1'b0; axi_addr_valid_next = 1'b0;
// enqueue status FIFO entry
status_fifo_we = 1'b1;
status_fifo_wr_tag = tag_table_tag[pcie_tag_next];
status_fifo_wr_last = 1'b1;
status_fifo_wr_completion = 1'b0;
// drop TLP // drop TLP
s_axis_rc_tready_next = 1'b1; s_axis_rc_tready_next = 1'b1;
tlp_state_next = TLP_STATE_WAIT_END; tlp_state_next = TLP_STATE_DROP_TAG;
end end
end else begin end else begin
tlp_state_next = TLP_STATE_HEADER; tlp_state_next = TLP_STATE_HEADER;
@ -875,6 +949,7 @@ always @* begin
last_cycle_next = output_cycle_count_next == 0; last_cycle_next = output_cycle_count_next == 0;
input_active_next = 1'b1; input_active_next = 1'b1;
m_axi_awid_next = op_tag_reg;
m_axi_awaddr_next = axi_addr_reg; m_axi_awaddr_next = axi_addr_reg;
m_axi_awlen_next = output_cycle_count_next; m_axi_awlen_next = output_cycle_count_next;
m_axi_awvalid_next = 1'b1; m_axi_awvalid_next = 1'b1;
@ -884,6 +959,10 @@ always @* begin
op_count_leq_axi_max_burst_next = op_count_next <= AXI_MAX_BURST_SIZE-axi_addr_next[1:0]; op_count_leq_axi_max_burst_next = op_count_next <= AXI_MAX_BURST_SIZE-axi_addr_next[1:0];
op_table_write_start_ptr = op_tag_reg;
op_table_write_start_commit = op_count_next == 0 && final_cpl_reg && op_table_read_commit[op_table_write_start_ptr] && (op_table_read_count_start[op_table_write_start_ptr] == op_table_read_count_finish[op_table_write_start_ptr]);
op_table_write_start_en = 1'b1;
input_active_next = input_cycle_count_next != 0; input_active_next = input_cycle_count_next != 0;
input_cycle_count_next = input_cycle_count_next - 1; input_cycle_count_next = input_cycle_count_next - 1;
s_axis_rc_tready_next = m_axi_wready_int_early && input_active_next && bubble_cycle_reg && (!last_cycle_next || op_count_next == 0 || !m_axi_awvalid || m_axi_awready); s_axis_rc_tready_next = m_axi_wready_int_early && input_active_next && bubble_cycle_reg && (!last_cycle_next || op_count_next == 0 || !m_axi_awvalid || m_axi_awready);
@ -971,30 +1050,26 @@ always @* begin
op_count_leq_axi_max_burst_next = op_count_next <= AXI_MAX_BURST_SIZE-axi_addr_next[1:0]; op_count_leq_axi_max_burst_next = op_count_next <= AXI_MAX_BURST_SIZE-axi_addr_next[1:0];
// enqueue status FIFO entry for write completion
status_fifo_we = 1'b1;
status_fifo_wr_tag = tag_table_tag[pcie_tag_reg];
status_fifo_wr_last = 1'b0;
status_fifo_wr_completion = 1'b1;
m_axi_awvalid_next = 1'b1; m_axi_awvalid_next = 1'b1;
op_table_write_start_ptr = op_tag_reg;
op_table_write_start_commit = op_count_next == 0 && final_cpl_reg && op_table_read_commit[op_table_write_start_ptr] && (op_table_read_count_start[op_table_write_start_ptr] == op_table_read_count_finish[op_table_write_start_ptr]);
op_table_write_start_en = 1'b1;
s_axis_rc_tready_next = m_axi_wready_int_early && input_active_next && (!last_cycle_next || op_count_next == 0 || !m_axi_awvalid || m_axi_awready); s_axis_rc_tready_next = m_axi_wready_int_early && input_active_next && (!last_cycle_next || op_count_next == 0 || !m_axi_awvalid || m_axi_awready);
tlp_state_next = TLP_STATE_TRANSFER; tlp_state_next = TLP_STATE_TRANSFER;
end else begin end else begin
if (final_cpl_reg) begin if (final_cpl_reg) begin
// last completion in current read request (PCIe tag) // last completion in current read request (PCIe tag)
finish_tag = 1'b1; // release tag finish_tag = 1'b1; // release tag
// mark done
op_table_read_finish_ptr = op_tag_reg;
op_table_read_finish_en = 1'b1;
end else begin end else begin
// more completions to come, store current address // more completions to come, store current address
tag_table_we_tlp_next = 1'b1; tag_table_we_tlp_next = 1'b1;
end end
// enqueue status FIFO entry for write completion
status_fifo_we = 1'b1;
status_fifo_wr_tag = tag_table_tag[pcie_tag_reg];
status_fifo_wr_last = final_cpl_reg && tag_table_last[pcie_tag_reg];
status_fifo_wr_completion = 1'b1;
if (AXIS_PCIE_DATA_WIDTH > 64) begin if (AXIS_PCIE_DATA_WIDTH > 64) begin
s_axis_rc_tready_next = 1'b0; s_axis_rc_tready_next = 1'b0;
end else begin end else begin
@ -1006,6 +1081,36 @@ always @* begin
tlp_state_next = TLP_STATE_TRANSFER; tlp_state_next = TLP_STATE_TRANSFER;
end end
end end
TLP_STATE_DROP_TAG: begin
// drop tag and TLP
s_axis_rc_tready_next = 1'b1;
// release tag
finish_tag = 1'b1;
// mark done
op_table_read_finish_ptr = op_tag_reg;
op_table_read_finish_en = 1'b1;
// commit writes if we're done
op_table_write_start_ptr = op_tag_reg;
op_table_write_start_commit = op_table_read_commit[op_table_write_start_ptr] && (op_table_read_count_start[op_table_write_start_ptr] == op_table_read_count_finish[op_table_write_start_ptr]);
if (s_axis_rc_tready & s_axis_rc_tvalid) begin
if (s_axis_rc_tlast) begin
if (AXIS_PCIE_DATA_WIDTH > 64) begin
s_axis_rc_tready_next = 1'b0;
end else begin
s_axis_rc_tready_next = 1'b1;
end
tlp_state_next = TLP_STATE_IDLE;
end else begin
tlp_state_next = TLP_STATE_WAIT_END;
end
end else begin
tlp_state_next = TLP_STATE_WAIT_END;
end
end
TLP_STATE_WAIT_END: begin TLP_STATE_WAIT_END: begin
// wait end state, wait for end of TLP // wait end state, wait for end of TLP
s_axis_rc_tready_next = 1'b1; s_axis_rc_tready_next = 1'b1;
@ -1027,25 +1132,19 @@ always @* begin
end end
endcase endcase
if (status_fifo_rd_ptr_reg != status_fifo_wr_ptr_reg) begin m_axi_bready_next = op_table_active != 0;
// status FIFO not empty
if (status_fifo_completion[status_fifo_rd_ptr_reg[STATUS_FIFO_ADDR_WIDTH-1:0]]) begin
// completion entry
if (m_axi_bready && m_axi_bvalid) begin if (m_axi_bready && m_axi_bvalid) begin
// got write completion, pop and return status op_table_finish_ptr = m_axi_bid;
m_axis_read_desc_status_tag_next = status_fifo_tag[status_fifo_rd_ptr_reg[STATUS_FIFO_ADDR_WIDTH-1:0]];
m_axis_read_desc_status_valid_next = status_fifo_last[status_fifo_rd_ptr_reg[STATUS_FIFO_ADDR_WIDTH-1:0]]; op_table_write_finish_ptr = m_axi_bid;
status_fifo_rd_ptr_next = status_fifo_rd_ptr_reg + 1; op_table_write_finish_en = 1'b1;
m_axi_bready_next = 1'b0;
end else begin m_axis_read_desc_status_tag_next = op_table_tag[op_table_write_finish_ptr];
// wait for write completion
m_axi_bready_next = 1'b1; if (op_table_write_commit[op_table_write_finish_ptr] && (op_table_write_count_start[op_table_write_finish_ptr] == op_table_write_count_finish[op_table_write_finish_ptr])) begin
end op_table_finish_en = 1'b1;
end else begin m_axis_read_desc_status_valid_next = 1'b1;
// non-completion entry, pop and return status
m_axis_read_desc_status_tag_next = status_fifo_tag[status_fifo_rd_ptr_reg[STATUS_FIFO_ADDR_WIDTH-1:0]];
m_axis_read_desc_status_valid_next = status_fifo_last[status_fifo_rd_ptr_reg[STATUS_FIFO_ADDR_WIDTH-1:0]];
status_fifo_rd_ptr_next = status_fifo_rd_ptr_reg + 1;
end end
end end
end end
@ -1076,12 +1175,10 @@ always @(posedge clk) begin
m_axi_bready_reg <= 1'b0; m_axi_bready_reg <= 1'b0;
tag_table_we_tlp_reg <= 1'b0; tag_table_we_tlp_reg <= 1'b0;
op_table_active <= 0;
status_error_cor_reg <= 1'b0; status_error_cor_reg <= 1'b0;
status_error_uncor_reg <= 1'b0; status_error_uncor_reg <= 1'b0;
status_fifo_wr_ptr_reg <= 0;
status_fifo_rd_ptr_reg <= 0;
end else begin end else begin
req_state_reg <= req_state_next; req_state_reg <= req_state_next;
tlp_state_reg <= tlp_state_next; tlp_state_reg <= tlp_state_next;
@ -1095,13 +1192,15 @@ always @(posedge clk) begin
tag_table_we_tlp_reg <= tag_table_we_tlp_next; tag_table_we_tlp_reg <= tag_table_we_tlp_next;
if (op_table_start_en) begin
op_table_active[op_table_start_ptr] <= 1'b1;
end
if (op_table_finish_en) begin
op_table_active[op_table_finish_ptr] <= 1'b0;
end
status_error_cor_reg <= status_error_cor_next; status_error_cor_reg <= status_error_cor_next;
status_error_uncor_reg <= status_error_uncor_next; status_error_uncor_reg <= status_error_uncor_next;
if (status_fifo_we) begin
status_fifo_wr_ptr_reg <= status_fifo_wr_ptr_reg + 1;
end
status_fifo_rd_ptr_reg <= status_fifo_rd_ptr_next;
end end
req_pcie_addr_reg <= req_pcie_addr_next; req_pcie_addr_reg <= req_pcie_addr_next;
@ -1124,6 +1223,7 @@ always @(posedge clk) begin
first_cycle_reg <= first_cycle_next; first_cycle_reg <= first_cycle_next;
last_cycle_reg <= last_cycle_next; last_cycle_reg <= last_cycle_next;
pcie_tag_reg <= pcie_tag_next; pcie_tag_reg <= pcie_tag_next;
op_tag_reg <= op_tag_next;
final_cpl_reg <= final_cpl_next; final_cpl_reg <= final_cpl_next;
offset_reg <= offset_next; offset_reg <= offset_next;
@ -1131,12 +1231,14 @@ always @(posedge clk) begin
last_cycle_offset_reg <= last_cycle_offset_next; last_cycle_offset_reg <= last_cycle_offset_next;
tlp_cmd_addr_reg <= tlp_cmd_addr_next; tlp_cmd_addr_reg <= tlp_cmd_addr_next;
tlp_cmd_op_tag_reg <= tlp_cmd_op_tag_next;
tlp_cmd_tag_reg <= tlp_cmd_tag_next; tlp_cmd_tag_reg <= tlp_cmd_tag_next;
tlp_cmd_pcie_tag_reg <= tlp_cmd_pcie_tag_next; tlp_cmd_pcie_tag_reg <= tlp_cmd_pcie_tag_next;
tlp_cmd_last_reg <= tlp_cmd_last_next; tlp_cmd_last_reg <= tlp_cmd_last_next;
m_axis_read_desc_status_tag_reg <= m_axis_read_desc_status_tag_next; m_axis_read_desc_status_tag_reg <= m_axis_read_desc_status_tag_next;
m_axi_awid_reg <= m_axi_awid_next;
m_axi_awaddr_reg <= m_axi_awaddr_next; m_axi_awaddr_reg <= m_axi_awaddr_next;
m_axi_awlen_reg <= m_axi_awlen_next; m_axi_awlen_reg <= m_axi_awlen_next;
@ -1150,15 +1252,46 @@ always @(posedge clk) begin
tag_table_axi_addr[pcie_tag_reg] <= axi_addr_reg; tag_table_axi_addr[pcie_tag_reg] <= axi_addr_reg;
end else if (tlp_cmd_valid_reg && tag_table_we_req) begin end else if (tlp_cmd_valid_reg && tag_table_we_req) begin
tag_table_axi_addr[tlp_cmd_pcie_tag_reg] <= tlp_cmd_addr_reg; tag_table_axi_addr[tlp_cmd_pcie_tag_reg] <= tlp_cmd_addr_reg;
tag_table_tag[tlp_cmd_pcie_tag_reg] <= tlp_cmd_tag_reg; tag_table_op_tag[tlp_cmd_pcie_tag_reg] <= tlp_cmd_op_tag_reg;
tag_table_last[tlp_cmd_pcie_tag_reg] <= tlp_cmd_last_reg;
end end
if (status_fifo_we) begin if (op_table_start_en) begin
status_fifo_tag[status_fifo_wr_ptr_reg[STATUS_FIFO_ADDR_WIDTH-1:0]] <= status_fifo_wr_tag; op_table_tag[op_table_start_ptr] <= op_table_start_tag;
status_fifo_last[status_fifo_wr_ptr_reg[STATUS_FIFO_ADDR_WIDTH-1:0]] <= status_fifo_wr_last; op_table_init[op_table_start_ptr] <= !op_table_init[op_table_start_ptr];
status_fifo_completion[status_fifo_wr_ptr_reg[STATUS_FIFO_ADDR_WIDTH-1:0]] <= status_fifo_wr_completion; //op_table_read_init[op_table_start_ptr] <= 1'b1;
status_fifo_wr_ptr_reg <= status_fifo_wr_ptr_reg + 1; //op_table_write_init[op_table_start_ptr] <= 1'b1;
end
if (op_table_read_start_en) begin
op_table_read_init[op_table_read_start_ptr] <= op_table_init[op_table_read_start_ptr];
op_table_read_commit[op_table_read_start_ptr] <= op_table_read_start_commit;
if (op_table_read_init[op_table_read_start_ptr] != op_table_init[op_table_read_start_ptr]) begin
op_table_read_count_start[op_table_read_start_ptr] <= op_table_read_count_finish[op_table_read_start_ptr];
end else begin
op_table_read_count_start[op_table_read_start_ptr] <= op_table_read_count_start[op_table_read_start_ptr] + 1;
end
end else if (op_table_read_start_commit) begin
op_table_read_commit[op_table_read_start_ptr] <= op_table_read_start_commit;
end
if (op_table_read_finish_en) begin
op_table_read_count_finish[op_table_read_finish_ptr] <= op_table_read_count_finish[op_table_read_finish_ptr] + 1;
end
if (op_table_write_start_en) begin
op_table_write_init[op_table_write_start_ptr] <= op_table_init[op_table_write_start_ptr];
op_table_write_commit[op_table_write_start_ptr] <= op_table_write_start_commit;
if (op_table_write_init[op_table_write_start_ptr] != op_table_init[op_table_write_start_ptr]) begin
op_table_write_count_start[op_table_write_start_ptr] <= op_table_write_count_finish[op_table_write_start_ptr];
end else begin
op_table_write_count_start[op_table_write_start_ptr] <= op_table_write_count_start[op_table_write_start_ptr] + 1;
end
end else if (op_table_write_start_commit) begin
op_table_write_commit[op_table_write_start_ptr] <= op_table_write_start_commit;
end
if (op_table_write_finish_en) begin
op_table_write_count_finish[op_table_write_finish_ptr] <= op_table_write_count_finish[op_table_write_finish_ptr] + 1;
end end
end end