Read flag is set when the sd controller reads response data in from the sd card. When the cpu reads from the controller, the flag is reset. This flag does not trigger an interrupt, it mmust be polled.
168 lines
3.4 KiB
Systemverilog
168 lines
3.4 KiB
Systemverilog
module sd_controller(
|
|
input clk,
|
|
input sd_clk,
|
|
input rst,
|
|
|
|
input [2:0] addr,
|
|
input [7:0] data,
|
|
input cs,
|
|
input rw,
|
|
|
|
input i_sd_cmd,
|
|
output logic o_sd_cmd,
|
|
|
|
input i_sd_data,
|
|
output logic o_sd_data,
|
|
|
|
output logic [7:0] data_out
|
|
);
|
|
|
|
logic [31:0] arg;
|
|
logic [5:0] cmd;
|
|
|
|
logic [47:0] rxcmd_buf;
|
|
logic [31:0] rx_val;
|
|
|
|
assign rx_val = rxcmd_buf[39:8];
|
|
|
|
always_comb begin
|
|
data_out = 'x;
|
|
|
|
if (addr < 4'h4) begin
|
|
data_out = rx_val[8 * addr +: 8];
|
|
end else if (addr == 4'h4) begin
|
|
data_out = read_flag;
|
|
end
|
|
end
|
|
|
|
logic read_flag, next_read_flag;
|
|
|
|
typedef enum bit [2:0] {IDLE, LOAD, CRC, TXCMD, RXCMD} macro_t;
|
|
struct packed {
|
|
macro_t macro;
|
|
logic [5:0] count;
|
|
} state, next_state;
|
|
|
|
always_ff @(posedge clk) begin
|
|
if (rst) begin
|
|
state.macro <= IDLE;
|
|
state.count <= '0;
|
|
read_flag <= '0;
|
|
end else begin
|
|
if (state.macro == TXCMD || state.macro == CRC) begin
|
|
if (sd_clk) begin
|
|
state <= next_state;
|
|
end
|
|
end else if (state.macro == RXCMD) begin
|
|
if (~sd_clk) begin
|
|
state <= next_state;
|
|
end
|
|
end else begin
|
|
state <= next_state;
|
|
end
|
|
end
|
|
|
|
if (sd_clk) begin
|
|
read_flag <= next_read_flag;
|
|
end
|
|
|
|
if (cs & ~rw) begin
|
|
if (addr < 4'h4) begin
|
|
arg[8 * addr +: 8] <= data;
|
|
end else if (addr == 4'h4) begin
|
|
cmd <= data;
|
|
end
|
|
end
|
|
if (state.macro == RXCMD) begin
|
|
rxcmd_buf[6'd46-state.count] <= i_sd_cmd; //we probabily missed bit 47
|
|
end
|
|
end
|
|
|
|
logic [6:0] crc;
|
|
logic load_crc;
|
|
logic crc_valid;
|
|
logic [39:0] _packet;
|
|
assign _packet = {1'b0, 1'b1, cmd, arg};
|
|
logic [47:0] packet_crc;
|
|
assign packet_crc = {_packet, crc, 1'b1};
|
|
|
|
crc7 u_crc7(
|
|
.clk(clk),
|
|
.rst(rst),
|
|
.load(load_crc),
|
|
.data_in(_packet),
|
|
.crc_out(crc),
|
|
.valid(crc_valid)
|
|
);
|
|
|
|
always_comb begin
|
|
next_state = state;
|
|
next_read_flag = read_flag;
|
|
|
|
case (state.macro)
|
|
IDLE: begin
|
|
if (~i_sd_cmd) begin // receive data if sd pulls cmd low
|
|
next_state.macro = RXCMD;
|
|
end
|
|
|
|
if (addr == 4'h4 & cs & ~rw) begin // transmit if cpu writes to cmd
|
|
next_state.macro = LOAD;
|
|
end
|
|
|
|
if (addr == 4'h4 & cs & rw) begin
|
|
next_read_flag = '0;
|
|
end
|
|
end
|
|
|
|
LOAD: begin
|
|
next_state.macro = CRC;
|
|
end
|
|
|
|
CRC: begin
|
|
next_state.macro = TXCMD;
|
|
end
|
|
|
|
TXCMD: begin
|
|
if (state.count < 47) begin
|
|
next_state.count = state.count + 6'b1;
|
|
end else begin
|
|
next_state.macro = IDLE;
|
|
next_state.count = '0;
|
|
end
|
|
end
|
|
|
|
RXCMD: begin
|
|
if (state.count < 47) begin
|
|
next_state.count = state.count + 6'b1;
|
|
end else begin
|
|
next_read_flag = '1;
|
|
next_state.macro = IDLE;
|
|
next_state.count = '0;
|
|
end
|
|
end
|
|
endcase
|
|
end
|
|
|
|
always_comb begin
|
|
o_sd_cmd = '1; //default to 1
|
|
o_sd_data = '1;
|
|
|
|
load_crc = '0;
|
|
|
|
case (state.macro)
|
|
IDLE:;
|
|
|
|
CRC: begin
|
|
load_crc = '1;
|
|
end
|
|
|
|
TXCMD: begin
|
|
o_sd_cmd = packet_crc[6'd47 - state.count];
|
|
end
|
|
|
|
RXCMD:;
|
|
endcase
|
|
end
|
|
|
|
endmodule
|