mirror of
https://github.com/jhshi/openofdm.git
synced 2025-01-01 19:47:08 +00:00
164 lines
3.7 KiB
Verilog
164 lines
3.7 KiB
Verilog
`include "common_defs.v"
|
|
|
|
module rotate
|
|
(
|
|
input clock,
|
|
input enable,
|
|
input reset,
|
|
|
|
input [15:0] in_i,
|
|
input [15:0] in_q,
|
|
// [-PI, PI]
|
|
// scaled up by ATAN_LUT_SCALE_SHIFT
|
|
input signed [31:0] phase,
|
|
input input_strobe,
|
|
|
|
output [`ROTATE_LUT_LEN_SHIFT-1:0] rot_addr,
|
|
input [31:0] rot_data,
|
|
|
|
output signed [15:0] out_i,
|
|
output signed [15:0] out_q,
|
|
output output_strobe
|
|
);
|
|
`include "common_params.v"
|
|
|
|
reg [31:0] phase_delayed;
|
|
reg [31:0] phase_abs;
|
|
|
|
reg [2:0] quadrant;
|
|
reg [2:0] quadrant_delayed;
|
|
wire [15:0] in_i_delayed;
|
|
wire [15:0] in_q_delayed;
|
|
|
|
reg [31:0] actual_phase;
|
|
|
|
wire [15:0] raw_rot_i;
|
|
wire [15:0] raw_rot_q;
|
|
reg [15:0] rot_i;
|
|
reg [15:0] rot_q;
|
|
|
|
wire mult_in_stb;
|
|
|
|
wire [31:0] p_i;
|
|
wire [31:0] p_q;
|
|
|
|
assign out_i = p_i[`ROTATE_LUT_SCALE_SHIFT+15:`ROTATE_LUT_SCALE_SHIFT];
|
|
assign out_q = p_q[`ROTATE_LUT_SCALE_SHIFT+15:`ROTATE_LUT_SCALE_SHIFT];
|
|
|
|
assign rot_addr = actual_phase[`ROTATE_LUT_LEN_SHIFT-1:0];
|
|
assign raw_rot_i = rot_data[31:16];
|
|
assign raw_rot_q = rot_data[15:0];
|
|
|
|
|
|
delayT #(.DATA_WIDTH(32), .DELAY(4)) in_delay_inst (
|
|
.clock(clock),
|
|
.reset(reset),
|
|
|
|
.data_in({in_i, in_q}),
|
|
.data_out({in_i_delayed, in_q_delayed})
|
|
);
|
|
|
|
delayT #(.DATA_WIDTH(1), .DELAY(4)) mult_delay_inst (
|
|
.clock(clock),
|
|
.reset(reset),
|
|
|
|
.data_in(input_strobe),
|
|
.data_out(mult_in_stb)
|
|
);
|
|
|
|
|
|
complex_mult mult_inst (
|
|
.clock(clock),
|
|
.enable(enable),
|
|
.reset(reset),
|
|
.a_i(in_i_delayed),
|
|
.a_q(in_q_delayed),
|
|
.b_i(rot_i),
|
|
.b_q(rot_q),
|
|
.input_strobe(mult_in_stb),
|
|
.p_i(p_i),
|
|
.p_q(p_q),
|
|
.output_strobe(output_strobe)
|
|
);
|
|
|
|
|
|
integer i;
|
|
always @(posedge clock) begin
|
|
if (reset) begin
|
|
actual_phase <= 0;
|
|
|
|
rot_i <= 0;
|
|
rot_q <= 0;
|
|
phase_abs <= 0;
|
|
phase_delayed <= 0;
|
|
|
|
end else if (enable) begin
|
|
`ifdef DEBUG_PRINT
|
|
if (phase > PI || phase < -PI) begin
|
|
$display("[WARN] phase overflow: %d\n", phase);
|
|
end
|
|
`endif
|
|
|
|
// cycle 1
|
|
phase_abs <= phase[31]? ~phase+1: phase;
|
|
phase_delayed <= phase;
|
|
|
|
// cycle 2
|
|
if (phase_abs <= PI_4) begin
|
|
quadrant <= {phase_delayed[31], 2'b00};
|
|
actual_phase <= phase_abs;
|
|
end else if (phase_abs <= PI_2) begin
|
|
quadrant <= {phase_delayed[31], 2'b01};
|
|
actual_phase <= PI_2 - phase_abs;
|
|
end else if (phase_abs <= PI_3_4) begin
|
|
quadrant <= {phase_delayed[31], 2'b10};
|
|
actual_phase <= phase_abs - PI_2;
|
|
end else begin
|
|
quadrant <= {phase_delayed[31], 2'b11};
|
|
actual_phase <= PI - phase_abs;
|
|
end
|
|
|
|
// cycle 3
|
|
// wait for raw_rot_i
|
|
quadrant_delayed <= quadrant;
|
|
|
|
// cycle 4
|
|
case(quadrant_delayed)
|
|
3'b000: begin
|
|
rot_i <= raw_rot_i;
|
|
rot_q <= raw_rot_q;
|
|
end
|
|
3'b001: begin
|
|
rot_i <= raw_rot_q;
|
|
rot_q <= raw_rot_i;
|
|
end
|
|
3'b010: begin
|
|
rot_i <= ~raw_rot_q+1;
|
|
rot_q <= raw_rot_i;
|
|
end
|
|
3'b011: begin
|
|
rot_i <= ~raw_rot_i+1;
|
|
rot_q <= raw_rot_q;
|
|
end
|
|
3'b100: begin
|
|
rot_i <= raw_rot_i;
|
|
rot_q <= ~raw_rot_q+1;
|
|
end
|
|
3'b101: begin
|
|
rot_i <= raw_rot_q;
|
|
rot_q <= ~raw_rot_i+1;
|
|
end
|
|
3'b110: begin
|
|
rot_i <= ~raw_rot_q+1;
|
|
rot_q <= ~raw_rot_i+1;
|
|
end
|
|
3'b111: begin
|
|
rot_i <= ~raw_rot_i+1;
|
|
rot_q <= ~raw_rot_q+1;
|
|
end
|
|
endcase
|
|
end
|
|
end
|
|
|
|
endmodule
|