1
0
mirror of https://github.com/pConst/basic_verilog.git synced 2025-01-14 06:42:54 +08:00
basic_verilog/debounce_v1.v
Konstantin Pavlov 0475ea0398 Fix typo
2023-05-23 11:12:22 +03:00

119 lines
2.7 KiB
Verilog

//------------------------------------------------------------------------------
// debounce_v1.v
// published as part of https://github.com/pConst/basic_verilog
// Konstantin Pavlov, pavlovconst@gmail.com
//------------------------------------------------------------------------------
// INFO ------------------------------------------------------------------------
// Button debounce v1
//
// - sampling inputs using configurable divided clock (this is the
// simplest form of low-pass filter)
// - switching output only when both samples have equal level
// (this gives some form of hysteresis in case we sample unstable data)
//
/* --- INSTANTIATION TEMPLATE BEGIN ---
debounce_v1 #(
.WIDTH( 4 ),
.SAMPLING_FACTOR( 16 )
) DB1 (
.clk( clk ),
.nrst( 1'b1 ),
.ena( 1'b1 ),
.in( btn[3:0] ),
.out( btn_db[3:0] )
);
--- INSTANTIATION TEMPLATE END ---*/
module debounce_v1 #( parameter
WIDTH = 1,
SAMPLING_FACTOR = 16 // 0 - sampling every clk
// 1 - sampling on clk/2
// 2 - sampling on clk/4 etc....
// only one or none should be enabled
TREAT_UNSTABLE_AS_HIGH = 0,
TREAT_UNSTABLE_AS_LOW = 0
)(
input clk,
input nrst,
input ena,
input [WIDTH-1:0] in,
output reg [WIDTH-1:0] out = 0
);
localparam SAMPLING_RANGE = 32;
wire [SAMPLING_RANGE-1:0] s_clk;
clk_divider #(
.WIDTH( SAMPLING_RANGE )
) clk_div (
.clk( clk ),
.nrst( nrst ),
.ena( 1'b1 ),
.out( s_clk[SAMPLING_RANGE-1:0] )
);
wire [SAMPLING_RANGE-1:0] s_clk_rise;
edge_detect #(
.WIDTH( SAMPLING_RANGE )
) clk_div_ed (
.clk( clk ),
.anrst( nrst ),
.in( s_clk[SAMPLING_RANGE-1:0] ),
.rising( s_clk_rise[SAMPLING_RANGE-1:0] )
);
wire do_sample;
assign do_sample = s_clk_rise[SAMPLING_FACTOR];
reg [WIDTH-1:0] in_d1 = 0;
reg [WIDTH-1:0] in_d2 = 0;
always @(posedge clk) begin
if (~nrst) begin
in_d1[WIDTH-1:0] <= 0;
in_d2[WIDTH-1:0] <= 0;
end else if (ena && do_sample) begin
in_d1[WIDTH-1:0] <= in_d2[WIDTH-1:0];
in_d2[WIDTH-1:0] <= in[WIDTH-1:0];
end // if
end
integer i;
always @(posedge clk) begin
if( ~nrst ) begin
out[WIDTH-1:0] <= 0;
end else begin
// every input has its own state
for (i = 0; i < WIDTH; i=i+1) begin
case ( {in_d2[i],in_d1[i]} )
2'b00: out[i] <= 1'b0;
2'b11: out[i] <= 1'b1;
default: begin
if (TREAT_UNSTABLE_AS_HIGH) begin
out[i] <= 1'b1;
end else if (TREAT_UNSTABLE_AS_LOW) begin
out[i] <= 1'b0;
end
end
endcase
end // for
end
end
endmodule