forked from tanchou/Verilog
145 lines
3.7 KiB
Verilog
145 lines
3.7 KiB
Verilog
module uart_rx #(
|
|
parameter CLK_FREQ = 27_000_000,
|
|
parameter BAUD_RATE = 115200
|
|
)(
|
|
input clk, //clock input
|
|
input rst_p, //asynchronous reset input, high active
|
|
input rx_data_ready, //data receiver module ready
|
|
input rx_pin, //serial data input
|
|
|
|
output reg[7:0] rx_data, //received serial data
|
|
output reg rx_data_valid //received serial data is valid
|
|
);
|
|
|
|
localparam CYCLE = CLK_FREQ * / BAUD_RATE;
|
|
|
|
//state machine code
|
|
localparam S_IDLE = 1;
|
|
localparam S_START = 2; //start bit
|
|
localparam S_REC_BYTE = 3; //data bits
|
|
localparam S_STOP = 4; //stop bit
|
|
localparam S_DATA = 5;
|
|
|
|
reg[2:0] state;
|
|
reg[2:0] next_state;
|
|
|
|
reg rx_d0; //delay 1 clock for rx_pin
|
|
reg rx_d1; //delay 1 clock for rx_d0
|
|
wire rx_negedge; //negedge of rx_pin
|
|
reg[7:0] rx_bits; //temporary storage of received data
|
|
reg[15:0] cycle_cnt; //baud counter
|
|
reg[2:0] bit_cnt; //bit counter
|
|
|
|
assign rx_negedge = rx_d1 && ~rx_d0; // Front déscendant
|
|
|
|
always@(posedge clk or negedge rst_p) // Filtrage du signial
|
|
begin
|
|
if(rst_p == 1'b1)begin
|
|
rx_d0 <= 1'b0;
|
|
rx_d1 <= 1'b0;
|
|
|
|
end else begin
|
|
rx_d0 <= rx_pin;
|
|
rx_d1 <= rx_d0;
|
|
end
|
|
end
|
|
|
|
|
|
always@(posedge clk or negedge rst_p)begin // Compteur d'etat
|
|
if(rst_p == 1'b1)
|
|
state <= S_IDLE;
|
|
else
|
|
state <= next_state;
|
|
end
|
|
|
|
always@(*)begin
|
|
case(state)
|
|
S_IDLE:
|
|
if(rx_negedge) // Detection du start bit
|
|
next_state = S_START;
|
|
else
|
|
next_state = S_IDLE;
|
|
|
|
S_START:
|
|
if(cycle_cnt == CYCLE - 1) //one data cycle
|
|
next_state = S_REC_BYTE;
|
|
else
|
|
next_state = S_START;
|
|
|
|
S_REC_BYTE:
|
|
if(cycle_cnt == CYCLE - 1 && bit_cnt == 3'd7) //receive 8bit data
|
|
next_state = S_STOP;
|
|
else
|
|
next_state = S_REC_BYTE;
|
|
|
|
S_STOP:
|
|
if(cycle_cnt == CYCLE/2 - 1) //half bit cycle,to avoid missing the next byte receiver
|
|
next_state = S_DATA;
|
|
else
|
|
next_state = S_STOP;
|
|
|
|
S_DATA:
|
|
if(rx_data_ready) //data receive complete
|
|
next_state = S_IDLE;
|
|
else
|
|
next_state = S_DATA;
|
|
|
|
default:
|
|
next_state = S_IDLE;
|
|
endcase
|
|
end
|
|
|
|
always@(posedge clk or negedge rst_p)
|
|
begin
|
|
if(rst_p == 1'b1)
|
|
rx_data_valid <= 1'b0;
|
|
else if(state == S_STOP && next_state != state)
|
|
rx_data_valid <= 1'b1;
|
|
else if(state == S_DATA && rx_data_ready)
|
|
rx_data_valid <= 1'b0;
|
|
end
|
|
|
|
always@(posedge clk or negedge rst_p)
|
|
begin
|
|
if(rst_p == 1'b1)
|
|
rx_data <= 8'd0;
|
|
else if(state == S_STOP && next_state != state)
|
|
rx_data <= rx_bits;//latch received data
|
|
end
|
|
|
|
always@(posedge clk or negedge rst_p)
|
|
begin
|
|
if(rst_p == 1'b1)
|
|
begin
|
|
bit_cnt <= 3'd0;
|
|
end
|
|
else if(state == S_REC_BYTE)
|
|
if(cycle_cnt == CYCLE - 1)
|
|
bit_cnt <= bit_cnt + 3'd1;
|
|
else
|
|
bit_cnt <= bit_cnt;
|
|
else
|
|
bit_cnt <= 3'd0;
|
|
end
|
|
|
|
|
|
always@(posedge clk or negedge rst_p)
|
|
begin
|
|
if(rst_p == 1'b1)
|
|
cycle_cnt <= 16'd0;
|
|
else if((state == S_REC_BYTE && cycle_cnt == CYCLE - 1) || next_state != state)
|
|
cycle_cnt <= 16'd0;
|
|
else
|
|
cycle_cnt <= cycle_cnt + 16'd1;
|
|
end
|
|
//receive serial data bit data
|
|
always@(posedge clk or negedge rst_p)
|
|
begin
|
|
if(rst_p == 1'b1)
|
|
rx_bits <= 8'd0;
|
|
else if(state == S_REC_BYTE && cycle_cnt == CYCLE/2 - 1)
|
|
rx_bits[bit_cnt] <= rx_pin;
|
|
else
|
|
rx_bits <= rx_bits;
|
|
end
|
|
endmodule
|