openofdm/verilog/ofdm_decoder.v
Xianjun Jiao 8ce830c262 Shrink the bits of num_bits_to_decode and deinter_out_count:
and adapt the verilog/ofdm_decoder.v accordingly
2023-01-09 15:24:42 +01:00

217 lines
6.4 KiB
Verilog

module ofdm_decoder
(
input clock,
input enable,
input reset,
input [31:0] sample_in,
input sample_in_strobe,
input soft_decoding,
// decode instructions
input [7:0] rate,
input do_descramble,
input [19:0] num_bits_to_decode, //4bits + ht_len: num_bits_to_decode <= (22+(ht_len<<3));
output [5:0] demod_out,
output [5:0] demod_soft_bits,
output [3:0] demod_soft_bits_pos,
output demod_out_strobe,
output [7:0] deinterleave_erase_out,
output deinterleave_erase_out_strobe,
output conv_decoder_out,
output conv_decoder_out_stb,
output descramble_out,
output descramble_out_strobe,
output [7:0] byte_out,
output byte_out_strobe
);
reg conv_in_stb, conv_in_stb_dly, do_descramble_dly;
reg [2:0] conv_in0, conv_in0_dly;
reg [2:0] conv_in1, conv_in1_dly;
reg [1:0] conv_erase, conv_erase_dly;
wire [15:0] input_i = sample_in[31:16];
wire [15:0] input_q = sample_in[15:0];
// wire vit_ce = reset | (enable & conv_in_stb) | conv_in_stb_dly; //Seems new viter decoder IP core does not need this complicated CE signal
wire vit_ce = 1'b1 ; //Need to be 1 to avoid the viterbi decoder freezing issue on adrv9364z7020 (demod_is_ongoing always high. dot11 stuck at state 3)
wire vit_clr = reset;
reg vit_clr_dly;
wire vit_rdy;
wire [5:0] deinterleave_out;
wire deinterleave_out_strobe;
wire [1:0] erase;
// assign conv_decoder_out_stb = vit_ce & vit_rdy;
assign conv_decoder_out_stb = m_axis_data_tvalid; // vit_rdy was used as data valid in the old version of the core, which is no longer the case
reg [3:0] skip_bit;
reg bit_in;
reg bit_in_stb;
reg [19:0] deinter_out_count; // bitwidth same as num_bits_to_decode
//reg flush;
assign deinterleave_erase_out = {erase,deinterleave_out};
assign deinterleave_erase_out_strobe = deinterleave_out_strobe;
demodulate demod_inst (
.clock(clock),
.reset(reset),
.enable(enable),
.rate(rate),
.cons_i(input_i),
.cons_q(input_q),
.input_strobe(sample_in_strobe),
.bits(demod_out),
.soft_bits(demod_soft_bits),
.soft_bits_pos(demod_soft_bits_pos),
.output_strobe(demod_out_strobe)
);
deinterleave deinterleave_inst (
.clock(clock),
.reset(reset),
.enable(enable),
.rate(rate),
.in_bits(demod_out),
.soft_in_bits(demod_soft_bits),
.soft_in_bits_pos(demod_soft_bits_pos),
.input_strobe(demod_out_strobe),
.soft_decoding(soft_decoding),
.out_bits(deinterleave_out),
.output_strobe(deinterleave_out_strobe),
.erase(erase)
);
/*
viterbi_v7_0 viterbi_inst (
.clk(clock),
.ce(vit_ce),
.sclr(vit_clr),
.data_in0(conv_in0),
.data_in1(conv_in1),
.erase(conv_erase),
.rdy(vit_rdy),
.data_out(conv_decoder_out)
);
*/
wire m_axis_data_tvalid ;
//reg [4:0] idle_wire_5bit ;
//wire [6:0] idle_wire_7bit ;
viterbi_v7_0 viterbi_inst (
.aclk(clock), // input wire aclk
.aresetn(~vit_clr), // input wire aresetn
.aclken(vit_ce), // input wire aclken
.s_axis_data_tdata({5'b0,conv_in1_dly,5'b0,conv_in0_dly}), // input wire [15 : 0] s_axis_data_tdata
.s_axis_data_tuser({6'b0,conv_erase_dly}), // input wire [7 : 0] s_axis_data_tuser
.s_axis_data_tvalid(conv_in_stb_dly), // input wire s_axis_data_tvalid
.s_axis_data_tready(vit_rdy), // output wire s_axis_data_tready
.m_axis_data_tdata({idle_wire_7bit, conv_decoder_out}), // output wire [7 : 0] m_axis_data_tdata
.m_axis_data_tvalid(m_axis_data_tvalid) // output wire m_axis_data_tvalid
);
descramble decramble_inst (
.clock(clock),
.enable(enable),
.reset(reset),
.in_bit(conv_decoder_out),
.input_strobe(conv_decoder_out_stb),
.out_bit(descramble_out),
.output_strobe(descramble_out_strobe)
);
bits_to_bytes byte_inst (
.clock(clock),
.enable(enable),
.reset(reset),
.bit_in(bit_in),
.input_strobe(bit_in_stb),
.byte_out(byte_out),
.output_strobe(byte_out_strobe)
);
always @(posedge clock) begin
if (reset) begin
conv_in_stb <= 0;
conv_in0 <= 0;
conv_in1 <= 0;
conv_erase <= 0;
bit_in <= 0;
// skip the first 9 bits of descramble out (service bits)
skip_bit <= 9;
bit_in_stb <= 0;
//flush <= 0;
deinter_out_count <= 0;
end else if (enable) begin
if (deinterleave_out_strobe) begin
deinter_out_count <= deinter_out_count + 1;
end //else begin
// wait for finishing deinterleaving current symbol
// only do flush for non-DATA bits, such as SIG and HT-SIG, which
// are not scrambled
//if (~do_descramble && deinter_out_count >= num_bits_to_decode) begin
//if (deinter_out_count >= num_bits_to_decode) begin // careful! deinter_out_count is only correct from 6M ~ 48M! under 54M, it should be 2*216, but actual value is 288!
//flush <= 1;
//end
//end
//if (!flush) begin
if (!(deinter_out_count >= num_bits_to_decode)) begin
conv_in_stb <= deinterleave_out_strobe;
conv_in0 <= deinterleave_out[2:0];
conv_in1 <= deinterleave_out[5:3];
conv_erase <= erase;
end else begin
conv_in_stb <= 1;
conv_in0 <= 3'b011;
conv_in1 <= 3'b011;
conv_erase <= 0;
end
if (deinter_out_count > 0) begin
if (~do_descramble_dly) begin
bit_in <= conv_decoder_out;
bit_in_stb <= conv_decoder_out_stb;
end else begin
bit_in <= descramble_out;
if (descramble_out_strobe) begin
if (skip_bit > 0 ) begin
skip_bit <= skip_bit - 1;
bit_in_stb <= 0;
end else begin
bit_in_stb <= 1;
end
end else begin
bit_in_stb <= 0;
end
end
end
end
end
// process used to delay things
// TODO: this is only a temp solution, as tready only rise one clock after ce goes high, delay statically by one clock, in future should take into account tready
always @(posedge clock) begin
conv_in1_dly <= conv_in1;
conv_in0_dly <= conv_in0;
conv_erase_dly <= conv_erase;
conv_in_stb_dly <= conv_in_stb ;
do_descramble_dly <= do_descramble;
end
endmodule