Skip to content

Commit

Permalink
SPI Engine: fix segmented transfers with echo_sclk
Browse files Browse the repository at this point in the history
Signed-off-by: Laez Barbosa <[email protected]>
  • Loading branch information
LBFFilho committed Dec 6, 2024
1 parent 741e53f commit 6399547
Show file tree
Hide file tree
Showing 2 changed files with 28 additions and 41 deletions.
22 changes: 17 additions & 5 deletions library/spi_engine/spi_engine_execution/spi_engine_execution.v
Original file line number Diff line number Diff line change
Expand Up @@ -114,6 +114,7 @@ module spi_engine_execution #(
reg trigger_next = 1'b0;
reg wait_for_io = 1'b0;
reg transfer_active = 1'b0;
reg transfer_done = 1'b0;

reg last_transfer;
reg [7:0] word_length = DATA_WIDTH;
Expand All @@ -136,6 +137,7 @@ module spi_engine_execution #(
wire sdo_int_s;

wire last_bit;
wire echo_last_bit;
wire first_bit;
wire end_of_word;

Expand Down Expand Up @@ -163,8 +165,6 @@ module spi_engine_execution #(
wire io_ready1;
wire io_ready2;

wire end_of_sdi_latch;

wire sample_sdo;

(* direct_enable = "yes" *) wire cs_gen;
Expand Down Expand Up @@ -196,12 +196,12 @@ module spi_engine_execution #(
.word_length(word_length),
.sample_sdo(sample_sdo),
.sdo_io_ready(sdo_io_ready),
.echo_last_bit(echo_last_bit),
.transfer_active(transfer_active),
.trigger_tx(trigger_tx),
.trigger_rx(trigger_rx),
.first_bit(first_bit),
.cs_activate(cs_activate),
.end_of_sdi_latch(end_of_sdi_latch));
.cs_activate(cs_activate));

assign sample_sdo = sdo_data_valid && ((trigger_tx && last_bit) || (wait_for_io || exec_transfer_cmd));

Expand Down Expand Up @@ -331,7 +331,7 @@ module spi_engine_execution #(
end else begin
case (inst_d1)
CMD_TRANSFER: begin
if (transfer_active == 1'b0 && wait_for_io == 1'b0 && end_of_sdi_latch == 1'b1)
if (transfer_done)
idle <= 1'b1;
end
CMD_CHIPSELECT: begin
Expand Down Expand Up @@ -426,6 +426,18 @@ module spi_engine_execution #(
end
end

always @(posedge clk ) begin
if (resetn == 1'b0) begin
transfer_done <= 1'b0;
end else begin
if (ECHO_SCLK) begin
transfer_done <= echo_last_bit && last_transfer;
end else begin
transfer_done <= (wait_for_io && io_ready1 && last_transfer) || (!wait_for_io && transfer_active && end_of_word && (last_transfer || !io_ready2)); // same conditions that make (!transfer_active && !wait_for_io)
end
end
end

always @(posedge clk) begin
if (transfer_active == 1'b1 || wait_for_io == 1'b1)
begin
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -72,12 +72,12 @@ module spi_engine_execution_shiftreg #(
// timing from main fsm
input sample_sdo,
output reg sdo_io_ready,
output echo_last_bit,
input transfer_active,
input trigger_tx,
input trigger_rx,
input first_bit,
input cs_activate,
output end_of_sdi_latch
input cs_activate
);

reg [ 7:0] sdi_counter = 8'b0;
Expand Down Expand Up @@ -156,9 +156,7 @@ module spi_engine_execution_shiftreg #(
generate
if (ECHO_SCLK == 1) begin : g_echo_sclk_miso_latch

reg [7:0] sdi_counter_d = 8'b0;
reg [7:0] sdi_transfer_counter = 8'b0;
reg [7:0] num_of_transfers = 8'b0;
reg last_sdi_bit_r;
reg [(NUM_OF_SDI * DATA_WIDTH)-1:0] sdi_data_latch = {(NUM_OF_SDI * DATA_WIDTH){1'b0}};

if ((DEFAULT_SPI_CFG[1:0] == 2'b01) || (DEFAULT_SPI_CFG[1:0] == 2'b10)) begin : g_echo_miso_nshift_reg
Expand All @@ -185,10 +183,10 @@ module spi_engine_execution_shiftreg #(
always @(posedge echo_sclk or posedge cs_activate) begin
if (cs_activate) begin
sdi_counter <= 8'b0;
sdi_counter_d <= 8'b0;
last_sdi_bit_r <= 1'b0;
end else begin
sdi_counter <= (sdi_counter == word_length-1) ? 8'b0 : sdi_counter + 1'b1;
sdi_counter_d <= sdi_counter;
last_sdi_bit_r <= (sdi_counter == word_length - 1); // FIXME: potentially unsafe path: what are the guarantees of settling time between changing word_length and first echo_sclk edge?
sdi_counter <= (sdi_counter == word_length - 1) ? 8'b0 : sdi_counter + 1'b1;
end
end

Expand All @@ -214,17 +212,18 @@ module spi_engine_execution_shiftreg #(
always @(posedge echo_sclk or posedge cs_activate) begin
if (cs_activate) begin
sdi_counter <= 8'b0;
sdi_counter_d <= 8'b0;
last_sdi_bit_r <= 1'b0;
end else begin
sdi_counter <= (sdi_counter == word_length-1) ? 8'b0 : sdi_counter + 1'b1;
sdi_counter_d <= sdi_counter;
last_sdi_bit_r <= (sdi_counter == word_length - 1); // FIXME: potentially unsafe path: what are the guarantees of settling time between changing word_length and first echo_sclk edge?
sdi_counter <= (sdi_counter == word_length - 1) ? 8'b0 : sdi_counter + 1'b1;
end
end

end

assign sdi_data = sdi_data_latch;
assign last_sdi_bit = (sdi_counter == 0) && (sdi_counter_d == word_length-1);
assign last_sdi_bit = last_sdi_bit_r;
assign echo_last_bit = !last_sdi_bit_m[3] && last_sdi_bit_m[2];

// sdi_data_valid is synchronous to SPI clock, so synchronize the
// last_sdi_bit to SPI clock
Expand All @@ -250,34 +249,10 @@ module spi_engine_execution_shiftreg #(
end
end

always @(posedge clk) begin
if (cs_activate) begin
num_of_transfers <= 8'b0;
end else begin
if (current_instr == CMD_TRANSFER) begin
// current_cmd contains the NUM_OF_TRANSFERS - 1
num_of_transfers <= current_cmd[7:0] + 1'b1;
end
end
end

always @(posedge clk) begin
if (cs_activate) begin
sdi_transfer_counter <= 0;
end else if (last_sdi_bit_m[2] == 1'b0 &&
last_sdi_bit_m[1] == 1'b1) begin
sdi_transfer_counter <= sdi_transfer_counter + 1'b1;
end
end

assign end_of_sdi_latch = last_sdi_bit_m[2] & (sdi_transfer_counter == num_of_transfers);

end /* g_echo_sclk_miso_latch */
else
begin : g_sclk_miso_latch

assign end_of_sdi_latch = 1'b1;

for (i=0; i<NUM_OF_SDI; i=i+1) begin: g_sdi_shift_reg

reg [DATA_WIDTH-1:0] data_sdi_shift;
Expand Down

0 comments on commit 6399547

Please sign in to comment.