mirror of
https://github.com/jhshi/openofdm.git
synced 2025-01-01 19:47:08 +00:00
873 lines
29 KiB
Verilog
873 lines
29 KiB
Verilog
`include "common_defs.v"
|
|
|
|
module equalizer
|
|
(
|
|
input clock,
|
|
input enable,
|
|
input reset,
|
|
|
|
input [31:0] sample_in,
|
|
input sample_in_strobe,
|
|
input ht_next,
|
|
input pkt_ht,
|
|
input ht_smoothing,
|
|
|
|
output [31:0] phase_in_i,
|
|
output [31:0] phase_in_q,
|
|
output reg phase_in_stb,
|
|
input [15:0] phase_out,
|
|
input phase_out_stb,
|
|
|
|
output [`ROTATE_LUT_LEN_SHIFT-1:0] rot_addr,
|
|
input [31:0] rot_data,
|
|
|
|
output reg [31:0] sample_out,
|
|
output reg sample_out_strobe,
|
|
|
|
output reg [3:0] state,
|
|
|
|
// for side channel
|
|
output wire [31:0] csi,
|
|
output wire csi_valid
|
|
);
|
|
|
|
|
|
// mask[0] is DC, mask[1:26] -> 1,..., 26
|
|
// mask[38:63] -> -26,..., -1
|
|
localparam SUBCARRIER_MASK =
|
|
64'b1111111111111111111111111100000000000111111111111111111111111110;
|
|
|
|
localparam HT_SUBCARRIER_MASK =
|
|
64'b1111111111111111111111111111000000011111111111111111111111111110;
|
|
|
|
// -7, -21, 21, 7
|
|
localparam PILOT_MASK =
|
|
64'b0000001000000000000010000000000000000000001000000000000010000000;
|
|
|
|
localparam DATA_SUBCARRIER_MASK =
|
|
SUBCARRIER_MASK ^ PILOT_MASK;
|
|
|
|
localparam HT_DATA_SUBCARRIER_MASK =
|
|
HT_SUBCARRIER_MASK ^ PILOT_MASK;
|
|
|
|
|
|
// -1,..,-26, 26,..,1
|
|
localparam LTS_REF =
|
|
64'b0000101001100000010100110000000000000000010101100111110101001100;
|
|
|
|
localparam HT_LTS_REF =
|
|
64'b0000101001100000010100110000000000011000010101100111110101001100;
|
|
|
|
|
|
localparam POLARITY =
|
|
127'b1111111000111011000101001011111010101000010110111100111001010110011000001101101011101000110010001000000100100110100111101110000;
|
|
|
|
// 21, 7, -7, -21
|
|
localparam HT_POLARITY = 4'b1000;
|
|
|
|
|
|
localparam IN_BUF_LEN_SHIFT = 6;
|
|
|
|
reg ht;
|
|
reg [5:0] num_data_carrier;
|
|
reg [7:0] num_ofdm_sym;
|
|
|
|
// bit masks
|
|
reg [63:0] lts_ref;
|
|
reg [63:0] ht_lts_ref;
|
|
reg [63:0] subcarrier_mask;
|
|
reg [63:0] data_subcarrier_mask;
|
|
reg [63:0] pilot_mask;
|
|
|
|
reg [126:0] polarity;
|
|
reg [3:0] ht_polarity;
|
|
reg [3:0] current_polarity;
|
|
reg [3:0] pilot_count1, pilot_count2;
|
|
|
|
reg signed [15:0] input_i;
|
|
reg signed [15:0] input_q;
|
|
|
|
reg current_sign;
|
|
|
|
wire signed [15:0] new_lts_i;
|
|
wire signed [15:0] new_lts_q;
|
|
wire new_lts_stb;
|
|
|
|
reg calc_mean_strobe;
|
|
|
|
reg [5:0] lts_waddr;
|
|
reg [6:0] lts_raddr; // one bit wider to detect overflow
|
|
reg [15:0] lts_i_in;
|
|
reg [15:0] lts_q_in;
|
|
reg lts_in_stb;
|
|
wire signed [15:0] lts_i_out;
|
|
wire signed [15:0] lts_q_out;
|
|
wire signed [15:0] lts_q_out_neg = ~lts_q_out + 1;
|
|
|
|
reg [5:0] in_waddr;
|
|
reg [6:0] in_raddr;
|
|
wire [15:0] buf_i_out;
|
|
wire [15:0] buf_q_out;
|
|
|
|
reg pilot_in_stb;
|
|
wire signed [31:0] pilot_i;
|
|
wire signed [31:0] pilot_q;
|
|
reg signed [31:0] pilot_i_reg, pilot_q_reg;
|
|
reg signed [15:0] pilot_iq_phase[0:3];
|
|
|
|
reg signed [31:0] pilot_sum_i;
|
|
reg signed [31:0] pilot_sum_q;
|
|
|
|
assign phase_in_i = pilot_i_reg;
|
|
assign phase_in_q = pilot_q_reg;
|
|
|
|
reg signed [15:0] pilot_phase_err;
|
|
reg signed [15:0] cpe; // common phase error due to RFO
|
|
reg signed [15:0] Sxy;
|
|
localparam Sx2 = 980;
|
|
|
|
// linear varying phase error (LVPE) parameters
|
|
reg signed [7:0] sym_idx;
|
|
reg lvpe_in_stb;
|
|
wire lvpe_out_stb;
|
|
wire signed [31:0] lvpe_dividend, lvpe;
|
|
wire signed [23:0] lvpe_divisor;
|
|
assign lvpe_dividend = (sym_idx <= 33 ? sym_idx*Sxy : (sym_idx-64)*Sxy);
|
|
assign lvpe_divisor = Sx2;
|
|
|
|
|
|
reg signed [15:0] phase_err;
|
|
wire signed [15:0] sym_phase;
|
|
assign sym_phase = (phase_err > 1608) ? (phase_err - 3217) : ((phase_err < -1608) ? (phase_err + 3217) : phase_err);
|
|
|
|
reg rot_in_stb;
|
|
wire signed [15:0] rot_i;
|
|
wire signed [15:0] rot_q;
|
|
|
|
wire [31:0] mag_sq;
|
|
wire [31:0] prod_i;
|
|
wire [31:0] prod_q;
|
|
wire [31:0] prod_i_scaled = prod_i<<(`CONS_SCALE_SHIFT+1);
|
|
wire [31:0] prod_q_scaled = prod_q<<(`CONS_SCALE_SHIFT+1); // +1 to fix the bug threshold for demodulate.v
|
|
wire prod_stb;
|
|
|
|
reg signed [15:0] lts_reg1_i, lts_reg2_i, lts_reg3_i, lts_reg4_i, lts_reg5_i;
|
|
reg signed [15:0] lts_reg1_q, lts_reg2_q, lts_reg3_q, lts_reg4_q, lts_reg5_q;
|
|
wire signed [18:0] lts_sum_1_3_i = lts_reg1_i + lts_reg2_i + lts_reg3_i;
|
|
wire signed [18:0] lts_sum_1_3_q = lts_reg1_q + lts_reg2_q + lts_reg3_q;
|
|
wire signed [18:0] lts_sum_1_4_i = lts_reg1_i + lts_reg2_i + lts_reg3_i + lts_reg4_i;
|
|
wire signed [18:0] lts_sum_1_4_q = lts_reg1_q + lts_reg2_q + lts_reg3_q + lts_reg4_q;
|
|
wire signed [18:0] lts_sum_1_5_i = lts_reg1_i + lts_reg2_i + lts_reg3_i + lts_reg4_i + lts_reg5_i;
|
|
wire signed [18:0] lts_sum_1_5_q = lts_reg1_q + lts_reg2_q + lts_reg3_q + lts_reg4_q + lts_reg5_q;
|
|
wire signed [18:0] lts_sum_2_5_i = lts_reg2_i + lts_reg3_i + lts_reg4_i + lts_reg5_i;
|
|
wire signed [18:0] lts_sum_2_5_q = lts_reg2_q + lts_reg3_q + lts_reg4_q + lts_reg5_q;
|
|
wire signed [18:0] lts_sum_3_5_i = lts_reg3_i + lts_reg4_i + lts_reg5_i;
|
|
wire signed [18:0] lts_sum_3_5_q = lts_reg3_q + lts_reg4_q + lts_reg5_q;
|
|
wire signed [18:0] lts_sum_wo3_i = lts_reg1_i + lts_reg2_i + lts_reg4_i + lts_reg5_i;
|
|
wire signed [18:0] lts_sum_wo3_q = lts_reg1_q + lts_reg2_q + lts_reg4_q + lts_reg5_q;
|
|
reg signed [18:0] lts_sum_i;
|
|
reg signed [18:0] lts_sum_q;
|
|
|
|
reg [2:0] lts_mv_avg_len;
|
|
reg lts_div_in_stb;
|
|
|
|
wire [31:0] dividend_i = (state == S_SMOOTH_CH_DC || state == S_SMOOTH_CH_LTS) ? (lts_sum_i[18] == 0 ? {13'h0,lts_sum_i} : {13'h1FFF,lts_sum_i}) : (state == S_ADJUST_FREQ_and_SAMPL_OFFSET ? prod_i_scaled : 0);
|
|
wire [31:0] dividend_q = (state == S_SMOOTH_CH_DC || state == S_SMOOTH_CH_LTS) ? (lts_sum_q[18] == 0 ? {13'h0,lts_sum_q} : {13'h1FFF,lts_sum_q}) : (state == S_ADJUST_FREQ_and_SAMPL_OFFSET ? prod_q_scaled : 0);
|
|
wire [23:0] divisor_i = (state == S_SMOOTH_CH_DC || state == S_SMOOTH_CH_LTS) ? {21'b0,lts_mv_avg_len} : (state == S_ADJUST_FREQ_and_SAMPL_OFFSET ? mag_sq[23:0] : 1);
|
|
wire [23:0] divisor_q = (state == S_SMOOTH_CH_DC || state == S_SMOOTH_CH_LTS) ? {21'b0,lts_mv_avg_len} : (state == S_ADJUST_FREQ_and_SAMPL_OFFSET ? mag_sq[23:0] : 1);
|
|
wire div_in_stb = (state == S_SMOOTH_CH_DC || state == S_SMOOTH_CH_LTS) ? lts_div_in_stb : (state == S_ADJUST_FREQ_and_SAMPL_OFFSET ? prod_out_strobe : 0);
|
|
|
|
|
|
reg [15:0] num_output;
|
|
wire [31:0] quotient_i;
|
|
wire [31:0] quotient_q;
|
|
wire [31:0] norm_i = quotient_i;
|
|
wire [31:0] norm_q = quotient_q;
|
|
wire [31:0] lts_div_i = quotient_i;
|
|
wire [31:0] lts_div_q = quotient_q;
|
|
|
|
wire div_out_stb;
|
|
wire norm_out_stb = div_out_stb;
|
|
wire lts_div_out_stb = div_out_stb;
|
|
|
|
reg prod_in_strobe;
|
|
wire prod_out_strobe;
|
|
|
|
// for side channel
|
|
reg sample_in_strobe_dly;
|
|
assign csi = {lts_i_out, lts_q_out};
|
|
assign csi_valid = ( (num_ofdm_sym == 1 || (pkt_ht==1 && num_ofdm_sym==5)) && state == S_CALC_FREQ_OFFSET && sample_in_strobe_dly == 1 && enable && (~reset) );
|
|
|
|
/*
|
|
// =============save signal to file for matlab bit-true comparison===========
|
|
integer file_open_trigger = 0;
|
|
integer new_lts_fd, phase_offset_pilot_input_fd, phase_offset_lts_input_fd, phase_offset_pilot_fd, phase_offset_pilot_sum_fd, phase_offset_phase_out_fd, rot_out_fd, equalizer_prod_fd, equalizer_prod_scaled_fd, equalizer_mag_sq_fd, equalizer_out_fd;
|
|
|
|
wire signed [15:0] norm_i_signed, norm_q_signed;
|
|
assign norm_i_signed = sample_out[31:16];
|
|
assign norm_q_signed = sample_out[15:0];
|
|
|
|
wire signed [31:0] prod_i_signed, prod_q_signed, prod_i_scaled_signed, prod_q_scaled_signed;
|
|
wire signed [15:0] phase_out_signed;
|
|
assign prod_i_signed = prod_i;
|
|
assign prod_q_signed = prod_q;
|
|
assign prod_i_scaled_signed = prod_i_scaled;
|
|
assign prod_q_scaled_signed = prod_q_scaled;
|
|
assign phase_out_signed = phase_out;
|
|
|
|
always @(posedge clock) begin
|
|
file_open_trigger = file_open_trigger + 1;
|
|
if (file_open_trigger==1) begin
|
|
new_lts_fd = $fopen("./new_lts.txt", "w");
|
|
phase_offset_pilot_input_fd = $fopen("./phase_offset_pilot_input.txt", "w");
|
|
phase_offset_lts_input_fd = $fopen("./phase_offset_lts_input.txt", "w");
|
|
phase_offset_pilot_fd = $fopen("./phase_offset_pilot.txt", "w");
|
|
phase_offset_pilot_sum_fd = $fopen("./phase_offset_pilot_sum.txt", "w");
|
|
phase_offset_phase_out_fd = $fopen("./phase_offset_phase_out.txt", "w");
|
|
rot_out_fd = $fopen("./rot_out.txt", "w");
|
|
equalizer_prod_fd = $fopen("./equalizer_prod.txt", "w");
|
|
equalizer_prod_scaled_fd = $fopen("./equalizer_prod_scaled.txt", "w");
|
|
equalizer_mag_sq_fd = $fopen("./equalizer_mag_sq.txt", "w");
|
|
equalizer_out_fd = $fopen("./equalizer_out.txt", "w");
|
|
end
|
|
|
|
if ((num_ofdm_sym == 1 || (pkt_ht==1 && num_ofdm_sym==5)) && state == S_CALC_FREQ_OFFSET && sample_in_strobe_dly == 1 && enable && (~reset) ) begin
|
|
$fwrite(new_lts_fd, "%d %d\n", lts_i_out, lts_q_out);
|
|
$fflush(new_lts_fd);
|
|
end
|
|
|
|
if (pilot_in_stb && enable && (~reset) ) begin
|
|
$fwrite(phase_offset_pilot_input_fd, "%d %d\n", input_i, input_q);
|
|
$fflush(phase_offset_pilot_input_fd);
|
|
$fwrite(phase_offset_lts_input_fd, "%d %d\n", lts_i_out, lts_q_out);
|
|
$fflush(phase_offset_lts_input_fd);
|
|
end
|
|
|
|
if (pilot_out_stb && enable && (~reset) ) begin
|
|
$fwrite(phase_offset_pilot_fd, "%d %d\n", pilot_i, pilot_q);
|
|
$fflush(phase_offset_pilot_fd);
|
|
end
|
|
|
|
if (phase_in_stb && enable && (~reset) ) begin
|
|
$fwrite(phase_offset_pilot_sum_fd, "%d %d\n", pilot_sum_i, pilot_sum_q);
|
|
$fflush(phase_offset_pilot_sum_fd);
|
|
end
|
|
|
|
if (phase_out_stb && enable && (~reset) ) begin
|
|
$fwrite(phase_offset_phase_out_fd, "%d\n", phase_out_signed);
|
|
$fflush(phase_offset_phase_out_fd);
|
|
end
|
|
|
|
if (rot_out_stb && enable && (~reset) ) begin
|
|
$fwrite(rot_out_fd, "%d %d\n", rot_i, rot_q);
|
|
$fflush(rot_out_fd);
|
|
end
|
|
|
|
if (prod_out_strobe && enable && (~reset) ) begin
|
|
$fwrite(equalizer_prod_fd, "%d %d\n", prod_i_signed, prod_q_signed);
|
|
$fflush(equalizer_prod_fd);
|
|
$fwrite(equalizer_prod_scaled_fd, "%d %d\n", prod_i_scaled_signed, prod_q_scaled_signed);
|
|
$fflush(equalizer_prod_scaled_fd);
|
|
$fwrite(equalizer_mag_sq_fd, "%d\n", mag_sq);
|
|
$fflush(equalizer_mag_sq_fd);
|
|
end
|
|
|
|
if (sample_out_strobe && enable && (~reset) ) begin
|
|
$fwrite(equalizer_out_fd, "%d %d\n", norm_i_signed, norm_q_signed);
|
|
$fflush(equalizer_out_fd);
|
|
end
|
|
end
|
|
// ==========end of save signal to file for matlab bit-true comparison===========
|
|
*/
|
|
|
|
ram_2port #(.DWIDTH(32), .AWIDTH(6)) lts_inst (
|
|
.clka(clock),
|
|
.ena(1),
|
|
.wea(lts_in_stb),
|
|
.addra(lts_waddr),
|
|
.dia({lts_i_in, lts_q_in}),
|
|
.doa(),
|
|
.clkb(clock),
|
|
.enb(1),
|
|
.web(1'b0),
|
|
.addrb(lts_raddr[5:0]),
|
|
.dib(32'hFFFF),
|
|
.dob({lts_i_out, lts_q_out})
|
|
);
|
|
|
|
calc_mean lts_i_inst (
|
|
.clock(clock),
|
|
.enable(enable),
|
|
.reset(reset),
|
|
|
|
.a(lts_i_out),
|
|
.b(input_i),
|
|
.sign(current_sign),
|
|
.input_strobe(calc_mean_strobe),
|
|
|
|
.c(new_lts_i),
|
|
.output_strobe(new_lts_stb)
|
|
);
|
|
|
|
calc_mean lts_q_inst (
|
|
.clock(clock),
|
|
.enable(enable),
|
|
.reset(reset),
|
|
|
|
.a(lts_q_out),
|
|
.b(input_q),
|
|
.sign(current_sign),
|
|
.input_strobe(calc_mean_strobe),
|
|
|
|
.c(new_lts_q)
|
|
);
|
|
|
|
ram_2port #(.DWIDTH(32), .AWIDTH(6)) in_buf_inst (
|
|
.clka(clock),
|
|
.ena(1),
|
|
.wea(sample_in_strobe),
|
|
.addra(in_waddr),
|
|
.dia(sample_in),
|
|
.doa(),
|
|
.clkb(clock),
|
|
.enb(1),
|
|
.web(1'b0),
|
|
.addrb(in_raddr[5:0]),
|
|
.dib(32'hFFFF),
|
|
.dob({buf_i_out, buf_q_out})
|
|
);
|
|
|
|
complex_mult pilot_inst (
|
|
.clock(clock),
|
|
.enable(enable),
|
|
.reset(reset),
|
|
.a_i(input_i),
|
|
.a_q(input_q),
|
|
.b_i(lts_i_out),
|
|
.b_q(lts_q_out),
|
|
.input_strobe(pilot_in_stb),
|
|
.p_i(pilot_i),
|
|
.p_q(pilot_q),
|
|
.output_strobe(pilot_out_stb)
|
|
);
|
|
|
|
rotate rotate_inst (
|
|
.clock(clock),
|
|
.enable(enable),
|
|
.reset(reset),
|
|
|
|
.in_i(buf_i_out),
|
|
.in_q(buf_q_out),
|
|
.phase(sym_phase),
|
|
.input_strobe(rot_in_stb),
|
|
|
|
.rot_addr(rot_addr),
|
|
.rot_data(rot_data),
|
|
|
|
.out_i(rot_i),
|
|
.out_q(rot_q),
|
|
.output_strobe(rot_out_stb)
|
|
);
|
|
|
|
complex_mult input_lts_prod_inst (
|
|
.clock(clock),
|
|
.enable(enable),
|
|
.reset(reset),
|
|
.a_i(rot_i),
|
|
.a_q(rot_q),
|
|
.b_i(lts_i_out),
|
|
.b_q(lts_q_out_neg),
|
|
.input_strobe(rot_out_stb),
|
|
.p_i(prod_i),
|
|
.p_q(prod_q),
|
|
.output_strobe(prod_out_strobe)
|
|
);
|
|
|
|
complex_mult lts_lts_prod_inst (
|
|
.clock(clock),
|
|
.enable(enable),
|
|
.reset(reset),
|
|
.a_i(lts_i_out),
|
|
.a_q(lts_q_out),
|
|
.b_i(lts_i_out),
|
|
.b_q(lts_q_out_neg),
|
|
.input_strobe(rot_out_stb),
|
|
.p_i(mag_sq)
|
|
);
|
|
|
|
divider norm_i_inst (
|
|
.clock(clock),
|
|
.enable(enable),
|
|
.reset(reset),
|
|
|
|
.dividend(dividend_i),
|
|
.divisor(divisor_i),
|
|
.input_strobe(div_in_stb),
|
|
|
|
.quotient(quotient_i),
|
|
.output_strobe(div_out_stb)
|
|
);
|
|
|
|
divider norm_q_inst (
|
|
.clock(clock),
|
|
.enable(enable),
|
|
.reset(reset),
|
|
|
|
.dividend(dividend_q),
|
|
.divisor(divisor_q),
|
|
.input_strobe(div_in_stb),
|
|
|
|
.quotient(quotient_q)
|
|
);
|
|
|
|
// LVPE calculation to estimate SFO
|
|
divider lvpe_inst (
|
|
.clock(clock),
|
|
.enable(enable),
|
|
.reset(reset),
|
|
|
|
.dividend(lvpe_dividend),
|
|
.divisor(lvpe_divisor),
|
|
.input_strobe(lvpe_in_stb),
|
|
|
|
.quotient(lvpe),
|
|
.output_strobe(lvpe_out_stb)
|
|
);
|
|
|
|
localparam S_FIRST_LTS = 0;
|
|
localparam S_SECOND_LTS = 1;
|
|
localparam S_SMOOTH_CH_DC = 2;
|
|
localparam S_SMOOTH_CH_LTS = 3;
|
|
localparam S_GET_POLARITY = 4;
|
|
localparam S_CALC_FREQ_OFFSET = 5;
|
|
localparam S_CALC_SAMPL_OFFSET = 6;
|
|
localparam S_ADJUST_FREQ_and_SAMPL_OFFSET = 7;
|
|
localparam S_HT_LTS = 8;
|
|
|
|
always @(posedge clock) begin
|
|
if (reset) begin
|
|
sample_out_strobe <= 0;
|
|
lts_raddr <= 0;
|
|
lts_waddr <= 0;
|
|
sample_out <= 0;
|
|
|
|
lts_in_stb <= 0;
|
|
lts_i_in <= 0;
|
|
lts_q_in <= 0;
|
|
|
|
ht <= 0;
|
|
num_data_carrier <= 48;
|
|
num_ofdm_sym <= 0;
|
|
|
|
subcarrier_mask <= SUBCARRIER_MASK;
|
|
data_subcarrier_mask <= DATA_SUBCARRIER_MASK;
|
|
pilot_mask <= PILOT_MASK;
|
|
lts_ref <= LTS_REF;
|
|
ht_lts_ref <= HT_LTS_REF;
|
|
polarity <= POLARITY;
|
|
|
|
ht_polarity <= HT_POLARITY;
|
|
|
|
current_polarity <= 0;
|
|
pilot_count1 <= 0;
|
|
pilot_count2 <= 0;
|
|
|
|
in_waddr <= 0;
|
|
in_raddr <= 0;
|
|
sym_idx <= 0;
|
|
|
|
lts_reg1_i <= 0; lts_reg2_i <= 0; lts_reg3_i <= 0; lts_reg4_i <= 0; lts_reg5_i <= 0;
|
|
lts_reg1_q <= 0; lts_reg2_q <= 0; lts_reg3_q <= 0; lts_reg4_q <= 0; lts_reg5_q <= 0;
|
|
lts_sum_i <= 0;
|
|
lts_sum_q <= 0;
|
|
lts_mv_avg_len <= 0;
|
|
lts_div_in_stb <= 0;
|
|
|
|
phase_in_stb <= 0;
|
|
pilot_sum_i <= 0;
|
|
pilot_sum_q <= 0;
|
|
pilot_phase_err <= 0;
|
|
cpe <= 0;
|
|
Sxy <= 0;
|
|
lvpe_in_stb <= 0;
|
|
phase_err <= 0;
|
|
pilot_in_stb <= 0;
|
|
pilot_i_reg <= 0;
|
|
pilot_q_reg <= 0;
|
|
pilot_iq_phase[0] <= 0; pilot_iq_phase[1] <= 0; pilot_iq_phase[2] <= 0; pilot_iq_phase[3] <= 0;
|
|
|
|
prod_in_strobe <= 0;
|
|
|
|
rot_in_stb <= 0;
|
|
|
|
current_sign <= 0;
|
|
input_i <= 0;
|
|
input_q <= 0;
|
|
calc_mean_strobe <= 0;
|
|
|
|
num_output <= 0;
|
|
|
|
state <= S_FIRST_LTS;
|
|
end else if (enable) begin
|
|
sample_in_strobe_dly <= sample_in_strobe;
|
|
case(state)
|
|
S_FIRST_LTS: begin
|
|
// store first LTS as is
|
|
lts_in_stb <= sample_in_strobe;
|
|
{lts_i_in, lts_q_in} <= sample_in;
|
|
|
|
if (lts_in_stb) begin
|
|
if (lts_waddr == 63) begin
|
|
lts_waddr <= 0;
|
|
lts_raddr <= 0;
|
|
state <= S_SECOND_LTS;
|
|
end else begin
|
|
lts_waddr <= lts_waddr + 1;
|
|
end
|
|
end
|
|
end
|
|
|
|
S_SECOND_LTS: begin
|
|
// calculate and store the mean of the two LTS
|
|
if (sample_in_strobe) begin
|
|
calc_mean_strobe <= sample_in_strobe;
|
|
{input_i, input_q} <= sample_in;
|
|
current_sign <= lts_ref[0];
|
|
lts_ref <= {lts_ref[0], lts_ref[63:1]};
|
|
lts_raddr <= lts_raddr + 1;
|
|
end else begin
|
|
calc_mean_strobe <= 0;
|
|
end
|
|
|
|
lts_in_stb <= new_lts_stb;
|
|
{lts_i_in, lts_q_in} <= {new_lts_i, new_lts_q};
|
|
|
|
if (lts_in_stb) begin
|
|
if (lts_waddr == 63) begin
|
|
lts_waddr <= 0;
|
|
lts_raddr <= 62;
|
|
lts_in_stb <= 0;
|
|
lts_div_in_stb <= 0;
|
|
// Always smooth legacy channel
|
|
state <= S_SMOOTH_CH_DC;
|
|
end else begin
|
|
lts_waddr <= lts_waddr + 1;
|
|
end
|
|
end
|
|
end
|
|
|
|
// 802.11-2012.pdf: 20.3.9.4.3 Table 20-11
|
|
// channel estimate smoothing (averaging length = 5)
|
|
S_SMOOTH_CH_DC: begin
|
|
if(lts_div_in_stb == 1) begin
|
|
lts_div_in_stb <= 0;
|
|
end else if(lts_raddr == 4) begin
|
|
lts_sum_i <= lts_sum_wo3_i;
|
|
lts_sum_q <= lts_sum_wo3_q;
|
|
lts_mv_avg_len <= 4;
|
|
lts_div_in_stb <= 1;
|
|
lts_raddr <= 5;
|
|
end else if(lts_raddr != 5) begin
|
|
// LTS Shift register
|
|
lts_reg1_i <= lts_i_out; lts_reg2_i <= lts_reg1_i; lts_reg3_i <= lts_reg2_i; lts_reg4_i <= lts_reg3_i; lts_reg5_i <= lts_reg4_i;
|
|
lts_reg1_q <= lts_q_out; lts_reg2_q <= lts_reg1_q; lts_reg3_q <= lts_reg2_q; lts_reg4_q <= lts_reg3_q; lts_reg5_q <= lts_reg4_q;
|
|
lts_raddr[5:0] <= lts_raddr[5:0] + 1;
|
|
end else begin
|
|
if(lts_in_stb == 1) begin
|
|
lts_waddr <= 37;
|
|
lts_raddr <= 38;
|
|
lts_in_stb <= 0;
|
|
state <= S_SMOOTH_CH_LTS;
|
|
end else if(lts_div_out_stb == 1) begin
|
|
lts_i_in <= lts_div_i[15:0];
|
|
lts_q_in <= lts_div_q[15:0];
|
|
lts_in_stb <= 1;
|
|
end
|
|
end
|
|
|
|
end
|
|
|
|
// 802.11-2012.pdf: 20.3.9.4.3 Table 20-11
|
|
// channel estimate smoothing (averaging length = 5)
|
|
S_SMOOTH_CH_LTS: begin
|
|
if(lts_raddr == 42) begin
|
|
lts_sum_i <= lts_sum_1_3_i;
|
|
lts_sum_q <= lts_sum_1_3_q;
|
|
lts_mv_avg_len <= 3;
|
|
lts_div_in_stb <= 1;
|
|
end else if(lts_raddr == 43) begin
|
|
lts_sum_i <= lts_sum_1_4_i;
|
|
lts_sum_q <= lts_sum_1_4_q;
|
|
lts_mv_avg_len <= 4;
|
|
lts_div_in_stb <= 1;
|
|
end else if(lts_raddr > 43 || lts_raddr < 29) begin
|
|
lts_sum_i <= lts_sum_1_5_i;
|
|
lts_sum_q <= lts_sum_1_5_q;
|
|
lts_mv_avg_len <= 5;
|
|
lts_div_in_stb <= 1;
|
|
end else if(lts_raddr == 29) begin
|
|
lts_sum_i <= lts_sum_2_5_i;
|
|
lts_sum_q <= lts_sum_2_5_q;
|
|
lts_mv_avg_len <= 4;
|
|
lts_div_in_stb <= 1;
|
|
end else if(lts_raddr == 30) begin
|
|
lts_sum_i <= lts_sum_3_5_i;
|
|
lts_sum_q <= lts_sum_3_5_q;
|
|
lts_mv_avg_len <= 3;
|
|
lts_div_in_stb <= 1;
|
|
end else if(lts_raddr == 31) begin
|
|
lts_div_in_stb <= 0;
|
|
end
|
|
|
|
if(lts_raddr >= 38 || lts_raddr <= 30) begin
|
|
// LTS Shift register
|
|
lts_reg1_i <= lts_i_out; lts_reg2_i <= lts_reg1_i; lts_reg3_i <= lts_reg2_i; lts_reg4_i <= lts_reg3_i; lts_reg5_i <= lts_reg4_i;
|
|
lts_reg1_q <= lts_q_out; lts_reg2_q <= lts_reg1_q; lts_reg3_q <= lts_reg2_q; lts_reg4_q <= lts_reg3_q; lts_reg5_q <= lts_reg4_q;
|
|
lts_raddr[5:0] <= lts_raddr[5:0] + 1;
|
|
end
|
|
|
|
if(lts_div_out_stb == 1) begin
|
|
lts_i_in <= lts_div_i[15:0];
|
|
lts_q_in <= lts_div_q[15:0];
|
|
lts_waddr[5:0] <= lts_waddr[5:0] + 1;
|
|
end
|
|
lts_in_stb <= lts_div_out_stb;
|
|
|
|
if(lts_waddr == 26) begin
|
|
state <= S_GET_POLARITY;
|
|
end
|
|
end
|
|
|
|
S_GET_POLARITY: begin
|
|
// obtain the polarity of pilot sub-carriers for next OFDM symbol
|
|
if (ht) begin
|
|
current_polarity <= {
|
|
ht_polarity[1]^polarity[0], // -7
|
|
ht_polarity[0]^polarity[0], // -21
|
|
ht_polarity[3]^polarity[0], // 21
|
|
ht_polarity[2]^polarity[0] // 7
|
|
};
|
|
ht_polarity <= {ht_polarity[0], ht_polarity[3:1]};
|
|
end else begin
|
|
current_polarity <= {
|
|
polarity[0], // -7
|
|
polarity[0], // -21
|
|
~polarity[0], // 21
|
|
polarity[0] // 7
|
|
};
|
|
end
|
|
polarity <= {polarity[0], polarity[126:1]};
|
|
|
|
pilot_sum_i <= 0;
|
|
pilot_sum_q <= 0;
|
|
pilot_count1 <= 0;
|
|
pilot_count2 <= 0;
|
|
cpe <= 0;
|
|
in_waddr <= 0;
|
|
in_raddr <= 0;
|
|
input_i <= 0;
|
|
input_q <= 0;
|
|
lts_raddr <= 0;
|
|
num_ofdm_sym <= num_ofdm_sym + 1;
|
|
state <= S_CALC_FREQ_OFFSET;
|
|
end
|
|
|
|
S_CALC_FREQ_OFFSET: begin
|
|
if (~ht & ht_next) begin
|
|
ht <= 1;
|
|
num_data_carrier <= 52;
|
|
lts_waddr <= 0;
|
|
lts_ref <= HT_LTS_REF;
|
|
subcarrier_mask <= HT_SUBCARRIER_MASK;
|
|
data_subcarrier_mask <= HT_DATA_SUBCARRIER_MASK;
|
|
pilot_mask <= PILOT_MASK;
|
|
// reverse this extra shift
|
|
polarity <= {polarity[125:0], polarity[126]};
|
|
state <= S_HT_LTS;
|
|
end
|
|
|
|
// calculate residue freq offset using pilot sub carriers
|
|
if (sample_in_strobe) begin
|
|
in_waddr <= in_waddr + 1;
|
|
lts_raddr <= lts_raddr + 1;
|
|
|
|
pilot_mask <= {pilot_mask[0], pilot_mask[63:1]};
|
|
if (pilot_mask[0]) begin
|
|
pilot_count1 <= pilot_count1 + 1;
|
|
current_polarity <= {current_polarity[0],
|
|
current_polarity[3:1]};
|
|
// obtain the conjugate of current pilot sub carrier
|
|
if (current_polarity[0] == 0) begin
|
|
input_i <= sample_in[31:16];
|
|
input_q <= ~sample_in[15:0] + 1;
|
|
end else begin
|
|
input_i <= ~sample_in[31:16] + 1;
|
|
input_q <= sample_in[15:0];
|
|
end
|
|
pilot_in_stb <= 1;
|
|
end else begin
|
|
pilot_in_stb <= 0;
|
|
end
|
|
end else begin
|
|
pilot_in_stb <= 0;
|
|
end
|
|
|
|
if (pilot_out_stb) begin
|
|
pilot_sum_i <= pilot_sum_i + pilot_i;
|
|
pilot_sum_q <= pilot_sum_q + pilot_q;
|
|
pilot_i_reg <= pilot_i;
|
|
pilot_q_reg <= pilot_q;
|
|
phase_in_stb <= 1;
|
|
end else begin
|
|
phase_in_stb <= 0;
|
|
end
|
|
|
|
if (phase_out_stb) begin
|
|
pilot_count2 <= pilot_count2 + 1;
|
|
pilot_iq_phase[pilot_count2] <= phase_out;
|
|
`ifdef DEBUG_PRINT
|
|
$display("[PILOT OFFSET] %d", phase_out);
|
|
`endif
|
|
end else if (pilot_count2 > 3) begin
|
|
pilot_count2 <= pilot_count2 + 1;
|
|
end
|
|
|
|
if (pilot_count2 == 8) begin
|
|
pilot_count1 <= 0;
|
|
pilot_count2 <= 0;
|
|
cpe <= {(cpe[15] == 0 ? 2'b00:2'b11),cpe[15:2]};
|
|
Sxy <= 0;
|
|
state <= S_CALC_SAMPL_OFFSET;
|
|
end else if (pilot_count2 > 3) begin
|
|
// sampling rate offset (SFO) is calculated as pilot phase error
|
|
if(pilot_sum_i < 0 && pilot_sum_q > 0 && pilot_iq_phase[pilot_count2[1:0]] < 0) begin
|
|
cpe = cpe + pilot_iq_phase[pilot_count2[1:0]] + 3217;
|
|
end else if(pilot_sum_i < 0 && pilot_sum_q < 0 && pilot_iq_phase[pilot_count2[1:0]] > 0) begin
|
|
cpe = cpe + pilot_iq_phase[pilot_count2[1:0]] - 3217;
|
|
end else begin
|
|
cpe = cpe + pilot_iq_phase[pilot_count2[1:0]];
|
|
end
|
|
end
|
|
end
|
|
|
|
S_CALC_SAMPL_OFFSET: begin
|
|
if (pilot_count1 < 4) begin
|
|
// sampling rate offset (SFO) is calculated as pilot phase error
|
|
if(cpe > 804 && pilot_iq_phase[pilot_count1] < 0) begin
|
|
pilot_phase_err <= pilot_iq_phase[pilot_count1] - cpe + 3217;
|
|
end else if(cpe < -804 && pilot_iq_phase[pilot_count1] > 0) begin
|
|
pilot_phase_err <= pilot_iq_phase[pilot_count1] - cpe - 3217;
|
|
end else begin
|
|
pilot_phase_err <= pilot_iq_phase[pilot_count1] - cpe;
|
|
end
|
|
|
|
pilot_count1 <= pilot_count1 + 1;
|
|
end
|
|
|
|
if(pilot_count1 == 1) begin
|
|
Sxy <= Sxy + 7*pilot_phase_err;
|
|
end else if(pilot_count1 == 2) begin
|
|
Sxy <= Sxy + 21*pilot_phase_err;
|
|
end else if(pilot_count1 == 3) begin
|
|
Sxy <= Sxy + -21*pilot_phase_err;
|
|
end else if(pilot_count1 == 4) begin
|
|
Sxy <= Sxy + -7*pilot_phase_err;
|
|
|
|
in_raddr <= 0;
|
|
sym_idx <= 0;
|
|
lvpe_in_stb <= 0;
|
|
// compensate for RAM read delay
|
|
lts_raddr <= 1;
|
|
rot_in_stb <= 0;
|
|
num_output <= 0;
|
|
state <= S_ADJUST_FREQ_and_SAMPL_OFFSET;
|
|
end
|
|
// Sx² = ∑(x-x̄)*(x-x̄) = ∑x² = (7² + 21² + (-21)² + (-7)²) = 980
|
|
// phase error gradient (PEG) = Sxy/Sx²
|
|
end
|
|
|
|
S_ADJUST_FREQ_and_SAMPL_OFFSET: begin
|
|
if (sym_idx < 64) begin
|
|
sym_idx <= sym_idx + 1;
|
|
lvpe_in_stb <= 1;
|
|
end else begin
|
|
lvpe_in_stb <= 0;
|
|
end
|
|
|
|
// first rotate, then normalize by avg LTS
|
|
if (lvpe_out_stb) begin
|
|
phase_err <= cpe + lvpe[15:0];
|
|
rot_in_stb <= 1;
|
|
in_raddr <= in_raddr + 1;
|
|
end else begin
|
|
rot_in_stb <= 0;
|
|
end
|
|
|
|
if (rot_out_stb) begin
|
|
lts_raddr <= lts_raddr + 1;
|
|
end
|
|
|
|
if (norm_out_stb) begin
|
|
data_subcarrier_mask <= {data_subcarrier_mask[0],
|
|
data_subcarrier_mask[63:1]};
|
|
if (data_subcarrier_mask[0]) begin
|
|
sample_out_strobe <= 1;
|
|
sample_out <= {norm_i[31], norm_i[14:0],
|
|
norm_q[31], norm_q[14:0]};
|
|
num_output <= num_output + 1;
|
|
end else begin
|
|
sample_out_strobe <= 0;
|
|
end
|
|
end else begin
|
|
sample_out_strobe <= 0;
|
|
end
|
|
|
|
if (num_output == num_data_carrier) begin
|
|
state <= S_GET_POLARITY;
|
|
end
|
|
end
|
|
|
|
S_HT_LTS: begin
|
|
if (sample_in_strobe) begin
|
|
lts_in_stb <= 1;
|
|
ht_lts_ref <= {ht_lts_ref[0], ht_lts_ref[63:1]};
|
|
if (ht_lts_ref[0] == 0) begin
|
|
{lts_i_in, lts_q_in} <= sample_in;
|
|
end else begin
|
|
lts_i_in <= ~sample_in[31:16]+1;
|
|
lts_q_in <= ~sample_in[15:0]+1;
|
|
end
|
|
end else begin
|
|
lts_in_stb <= 0;
|
|
end
|
|
|
|
if (lts_in_stb) begin
|
|
if (lts_waddr == 63) begin
|
|
lts_waddr <= 0;
|
|
lts_raddr <= 62;
|
|
lts_in_stb <= 0;
|
|
lts_div_in_stb <= 0;
|
|
// Depending on smoothing bit in HT-SIG, smooth the channel
|
|
if(ht_smoothing) begin
|
|
state <= S_SMOOTH_CH_DC;
|
|
end else begin
|
|
state <= S_GET_POLARITY;
|
|
end
|
|
end else begin
|
|
lts_waddr <= lts_waddr + 1;
|
|
end
|
|
end
|
|
|
|
end
|
|
|
|
default: begin
|
|
state <= S_FIRST_LTS;
|
|
end
|
|
endcase
|
|
end else begin
|
|
sample_out_strobe <= 0;
|
|
end
|
|
end
|
|
|
|
endmodule
|