diff options
Diffstat (limited to 'fpga/usrp3/top/x400/rfdc_timing_control.v')
-rw-r--r-- | fpga/usrp3/top/x400/rfdc_timing_control.v | 289 |
1 files changed, 289 insertions, 0 deletions
diff --git a/fpga/usrp3/top/x400/rfdc_timing_control.v b/fpga/usrp3/top/x400/rfdc_timing_control.v new file mode 100644 index 000000000..f1f1cfad1 --- /dev/null +++ b/fpga/usrp3/top/x400/rfdc_timing_control.v @@ -0,0 +1,289 @@ +// +// Copyright 2021 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: LGPL-3.0-or-later +// +// Module: rfdc_timing_control +// +// Description: +// +// This module handles timed register writes for the RFDC, such as NCO reset +// control. It takes the CtrlPort master from each radio block and splits it +// into a CtrlPort bus for the associated daughter board and another CtrlPort +// bus for the RFDC timing control. Timed commands on the RF timing bus are +// handled by the ctrlport_timer module. +// +// Parameters: +// +// NUM_DBOARDS : Number of daughter boards to support +// + +`default_nettype none + + +module rfdc_timing_control #( + parameter NUM_DBOARDS = 2 +) ( + // Clocks and resets + input wire clk, + input wire rst, + + // Time + input wire [63:0] time_now, + input wire time_now_stb, + input wire [ 3:0] time_ignore_bits, + + // CtrlPort Slave (from RFNoC Radio Block) + input wire [ 1*NUM_DBOARDS-1:0] s_ctrlport_req_wr, + input wire [ 1*NUM_DBOARDS-1:0] s_ctrlport_req_rd, + input wire [ 20*NUM_DBOARDS-1:0] s_ctrlport_req_addr, + input wire [ 32*NUM_DBOARDS-1:0] s_ctrlport_req_data, + input wire [ 4*NUM_DBOARDS-1:0] s_ctrlport_req_byte_en, + input wire [ 1*NUM_DBOARDS-1:0] s_ctrlport_req_has_time, + input wire [ 64*NUM_DBOARDS-1:0] s_ctrlport_req_time, + output wire [ 1*NUM_DBOARDS-1:0] s_ctrlport_resp_ack, + output wire [ 2*NUM_DBOARDS-1:0] s_ctrlport_resp_status, + output wire [ 32*NUM_DBOARDS-1:0] s_ctrlport_resp_data, + + // CtrlPort Master (to Daughter Boards) + output wire [ 1*NUM_DBOARDS-1:0] m_ctrlport_req_wr, + output wire [ 1*NUM_DBOARDS-1:0] m_ctrlport_req_rd, + output wire [ 20*NUM_DBOARDS-1:0] m_ctrlport_req_addr, + output wire [ 32*NUM_DBOARDS-1:0] m_ctrlport_req_data, + output wire [ 4*NUM_DBOARDS-1:0] m_ctrlport_req_byte_en, + output wire [ 1*NUM_DBOARDS-1:0] m_ctrlport_req_has_time, + output wire [ 64*NUM_DBOARDS-1:0] m_ctrlport_req_time, + input wire [ 1*NUM_DBOARDS-1:0] m_ctrlport_resp_ack, + input wire [ 2*NUM_DBOARDS-1:0] m_ctrlport_resp_status, + input wire [ 32*NUM_DBOARDS-1:0] m_ctrlport_resp_data, + + // RF Reset Control + output reg start_nco_reset, + input wire nco_reset_done, + output reg adc_reset_pulse, + output reg dac_reset_pulse +); + + `include "regmap/radio_ctrlport_regmap_utils.vh" + `include "regmap/rfdc_timing_regmap_utils.vh" + + // Reset registers + reg [NUM_DBOARDS-1:0] reg_nco_reset_start = 0; + reg [NUM_DBOARDS-1:0] reg_adc_reset_pulse = 0; + reg [NUM_DBOARDS-1:0] reg_dac_reset_pulse = 0; + + genvar db; + generate + for (db = 0; db < NUM_DBOARDS; db = db+1) begin : gen_db_ctrlport + + //----------------------------------------------------------------------- + // CtrlPort Splitter + //----------------------------------------------------------------------- + + wire [ 1-1:0] rf_ctrlport_req_wr; + wire [ 1-1:0] rf_ctrlport_req_rd; + wire [ 20-1:0] rf_ctrlport_req_addr; + wire [ 32-1:0] rf_ctrlport_req_data; + wire [ 4-1:0] rf_ctrlport_req_byte_en; + wire [ 1-1:0] rf_ctrlport_req_has_time; + wire [ 64-1:0] rf_ctrlport_req_time; + wire [ 1-1:0] rf_ctrlport_resp_ack; + wire [ 2-1:0] rf_ctrlport_resp_status; + wire [ 32-1:0] rf_ctrlport_resp_data; + + localparam [31:0] RFDC_TIMING_WINDOW_SIZE_W = $clog2(RFDC_TIMING_WINDOW_SIZE); + localparam [31:0] DB_WINDOW_SIZE_W = $clog2(DB_WINDOW_SIZE); + + ctrlport_decoder_param #( + .NUM_SLAVES (2), + .PORT_BASE ({ RFDC_TIMING_WINDOW[19:0], DB_WINDOW[19:0] }), + .PORT_ADDR_W ({ RFDC_TIMING_WINDOW_SIZE_W, DB_WINDOW_SIZE_W }) + ) ctrlport_decoder_param_i ( + .ctrlport_clk (clk), + .ctrlport_rst (rst), + .s_ctrlport_req_wr (s_ctrlport_req_wr [ 1*db+: 1]), + .s_ctrlport_req_rd (s_ctrlport_req_rd [ 1*db+: 1]), + .s_ctrlport_req_addr (s_ctrlport_req_addr [20*db+:20]), + .s_ctrlport_req_data (s_ctrlport_req_data [32*db+:32]), + .s_ctrlport_req_byte_en (s_ctrlport_req_byte_en [ 4*db+: 4]), + .s_ctrlport_req_has_time (s_ctrlport_req_has_time [ 1*db+: 1]), + .s_ctrlport_req_time (s_ctrlport_req_time [64*db+:64]), + .s_ctrlport_resp_ack (s_ctrlport_resp_ack [ 1*db+: 1]), + .s_ctrlport_resp_status (s_ctrlport_resp_status [ 2*db+: 2]), + .s_ctrlport_resp_data (s_ctrlport_resp_data [32*db+:32]), + .m_ctrlport_req_wr ({ rf_ctrlport_req_wr, m_ctrlport_req_wr [ 1*db+: 1] }), + .m_ctrlport_req_rd ({ rf_ctrlport_req_rd, m_ctrlport_req_rd [ 1*db+: 1] }), + .m_ctrlport_req_addr ({ rf_ctrlport_req_addr, m_ctrlport_req_addr [20*db+:20] }), + .m_ctrlport_req_data ({ rf_ctrlport_req_data, m_ctrlport_req_data [32*db+:32] }), + .m_ctrlport_req_byte_en ({ rf_ctrlport_req_byte_en, m_ctrlport_req_byte_en [ 4*db+: 4] }), + .m_ctrlport_req_has_time ({ rf_ctrlport_req_has_time, m_ctrlport_req_has_time [ 1*db+: 1] }), + .m_ctrlport_req_time ({ rf_ctrlport_req_time, m_ctrlport_req_time [64*db+:64] }), + .m_ctrlport_resp_ack ({ rf_ctrlport_resp_ack, m_ctrlport_resp_ack [ 1*db+: 1] }), + .m_ctrlport_resp_status ({ rf_ctrlport_resp_status, m_ctrlport_resp_status [ 2*db+: 2] }), + .m_ctrlport_resp_data ({ rf_ctrlport_resp_data, m_ctrlport_resp_data [32*db+:32] }) + ); + + + //----------------------------------------------------------------------- + // RF Reset Control + //----------------------------------------------------------------------- + + wire [ 1-1:0] nco_ctrlport_req_wr; + wire [ 1-1:0] nco_ctrlport_req_rd; + wire [ 20-1:0] nco_ctrlport_req_addr; + wire [ 32-1:0] nco_ctrlport_req_data; + reg [ 1-1:0] nco_ctrlport_resp_ack; + reg [ 32-1:0] nco_ctrlport_resp_data; + + ctrlport_timer #( + .EXEC_LATE_CMDS (1) + ) ctrlport_timer_nco ( + .clk (clk), + .rst (rst), + .time_now (time_now), + .time_now_stb (time_now_stb), + .time_ignore_bits (time_ignore_bits), + .s_ctrlport_req_wr (rf_ctrlport_req_wr), + .s_ctrlport_req_rd (rf_ctrlport_req_rd), + .s_ctrlport_req_addr (rf_ctrlport_req_addr), + .s_ctrlport_req_data (rf_ctrlport_req_data), + .s_ctrlport_req_byte_en (rf_ctrlport_req_byte_en), + .s_ctrlport_req_has_time (rf_ctrlport_req_has_time), + .s_ctrlport_req_time (rf_ctrlport_req_time), + .s_ctrlport_resp_ack (rf_ctrlport_resp_ack), + .s_ctrlport_resp_status (rf_ctrlport_resp_status), + .s_ctrlport_resp_data (rf_ctrlport_resp_data), + .m_ctrlport_req_wr (nco_ctrlport_req_wr), + .m_ctrlport_req_rd (nco_ctrlport_req_rd), + .m_ctrlport_req_addr (nco_ctrlport_req_addr), + .m_ctrlport_req_data (nco_ctrlport_req_data), + .m_ctrlport_req_byte_en (), + .m_ctrlport_resp_ack (nco_ctrlport_resp_ack), + .m_ctrlport_resp_status (2'b0), + .m_ctrlport_resp_data (nco_ctrlport_resp_data) + ); + + always @(posedge clk) begin + if (rst) begin + nco_ctrlport_resp_ack <= 0; + reg_nco_reset_start[db] <= 0; + nco_ctrlport_resp_data <= 'bX; + end else begin + // Default assignments + reg_nco_reset_start[db] <= 0; + reg_adc_reset_pulse[db] <= 0; + reg_dac_reset_pulse[db] <= 0; + nco_ctrlport_resp_ack <= 0; + nco_ctrlport_resp_data <= 0; + + // Handle register reads + if (nco_ctrlport_req_rd) begin + case (nco_ctrlport_req_addr) + NCO_RESET_REG: begin + nco_ctrlport_resp_ack <= 1; + nco_ctrlport_resp_data[NCO_RESET_DONE] <= nco_reset_done; + end + GEARBOX_RESET_REG: begin + nco_ctrlport_resp_ack <= 1; + end + endcase + end + + // Handle register writes + if (nco_ctrlport_req_wr) begin + case (nco_ctrlport_req_addr) + NCO_RESET_REG: begin + nco_ctrlport_resp_ack <= 1; + reg_nco_reset_start[db] <= nco_ctrlport_req_data[NCO_RESET_START]; + end + GEARBOX_RESET_REG: begin + nco_ctrlport_resp_ack <= 1; + reg_adc_reset_pulse[db] <= nco_ctrlport_req_data[ADC_RESET]; + reg_dac_reset_pulse[db] <= nco_ctrlport_req_data[DAC_RESET]; + end + endcase + end + end + end + + end + endgenerate + + + //--------------------------------------------------------------------------- + // Merge Resets + //--------------------------------------------------------------------------- + // + // There are multiple DBs but only one reset signal for each RF component. + // Since the reset is simply a single cycle pulse, we OR the reset registers + // for each daughter board together. + // + //--------------------------------------------------------------------------- + + always @(posedge clk) begin + if (rst) begin + start_nco_reset <= 0; + adc_reset_pulse <= 0; + dac_reset_pulse <= 0; + end else begin + start_nco_reset <= |reg_nco_reset_start; + adc_reset_pulse <= |reg_adc_reset_pulse; + dac_reset_pulse <= |reg_dac_reset_pulse; + end + end + +endmodule + + +`default_nettype wire + + +//XmlParse xml_on +// +//<regmap name="RADIO_CTRLPORT_REGMAP" readablestrobes="false" generatevhdl="true" ettusguidelines="true"> +// <group name="RADIO_CTRLPORT_WINDOWS"> +// <info>Each radio's CtrlPort peripheral interface is divided into the +// following memory spaces. Note that the CtrlPort peripheral interface +// starts at offset 0x80000 in the RFNoC Radio block's register space.</info> +// <window name="DB_WINDOW" offset="0x00000" size="0x08000"> +// <info>Daughterboard GPIO interface. Register access within this space +// is directed to the associated daughterboard CPLD.</info> +// </window> +// <window name="RFDC_TIMING_WINDOW" offset="0x08000" size="0x08000" targetregmap="RFDC_TIMING_REGMAP"> +// <info>RFDC timing control interface.</info> +// </window> +// </group> +//</regmap> +// +//<regmap name="RFDC_TIMING_REGMAP" readablestrobes="false" generatevhdl="true" ettusguidelines="true"> +// <group name="RFDC_TIMING_REGS"> +// <register name="NCO_RESET_REG" offset="0x00" size="32" readable="true" writable="true"> +// <info>NCO reset control register.</info> +// <bitfield name="NCO_RESET_START" range="0" readable="false" strobe="true"> +// <info>Write a 1 to this bit to start a reset the RFDC's NCO.</info> +// </bitfield> +// <bitfield name="NCO_RESET_DONE" range="1" writable="false"> +// <info>When 1, indicates that the NCO reset has completed.</info> +// </bitfield> +// </register> +// <register name="GEARBOX_RESET_REG" offset="0x04" size="32" readable="true" writable="true"> +// <info>Gearbox reset control register.</info> +// <bitfield name="ADC_RESET" range="0" readable="false" strobe="true"> +// <info> +// This reset is for the gearbox on the ADC data path that is used to +// move data from one clock domain to another outside the RFDC. Write +// a 1 to this bit to send a reset pulse to the ADC gearbox. +// </info> +// </bitfield> +// <bitfield name="DAC_RESET" range="1" readable="false" strobe="true"> +// <info> +// This reset is for the gearbox on the DAC data path that is used to +// move data from one clock domain to another outside the RFDC. Write +// a 1 to this bit to send a reset pulse to the DAC gearbox. +// </info> +// </bitfield> +// </register> +// </group> +//</regmap> +// +//XmlParse xml_off |