diff options
Diffstat (limited to 'fpga/usrp2/serdes/serdes_rx.v')
-rw-r--r-- | fpga/usrp2/serdes/serdes_rx.v | 310 |
1 files changed, 0 insertions, 310 deletions
diff --git a/fpga/usrp2/serdes/serdes_rx.v b/fpga/usrp2/serdes/serdes_rx.v deleted file mode 100644 index 1950d4e2a..000000000 --- a/fpga/usrp2/serdes/serdes_rx.v +++ /dev/null @@ -1,310 +0,0 @@ -// -// Copyright 2011 Ettus Research LLC -// -// This program is free software: you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation, either version 3 of the License, or -// (at your option) any later version. -// -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with this program. If not, see <http://www.gnu.org/licenses/>. -// - - -// SERDES Interface - -// LS-Byte is sent first, MS-Byte is second -// Invalid K Codes -// K0.0 000-00000 Error detected -// K31.7 111-11111 Loss of input signal - -// Valid K Codes -// K28.0 000-11100 -// K28.1 001-11100 Alternate COMMA? -// K28.2 010-11100 -// K28.3 011-11100 -// K28.4 100-11100 -// K28.5 101-11100 Standard COMMA? -// K28.6 110-11100 -// K28.7 111-11100 Bad COMMA? -// K23.7 111-10111 -// K27.7 111-11011 -// K29.7 111-11101 -// K30.7 111-11110 - -module serdes_rx - #(parameter FIFOSIZE = 9) - (input clk, - input rst, - - // RX HW Interface - input ser_rx_clk, - input [15:0] ser_r, - input ser_rklsb, - input ser_rkmsb, - - output [31:0] wr_dat_o, - output [3:0] wr_flags_o, - input wr_ready_i, - output wr_ready_o, - - output [15:0] fifo_space, - output xon_rcvd, output xoff_rcvd, - - output [15:0] fifo_occupied, output fifo_full, output fifo_empty, - output reg serdes_link_up, - output [31:0] debug - ); - - localparam K_COMMA = 8'b101_11100; // 0xBC K28.5 - localparam K_IDLE = 8'b001_11100; // 0x3C K28.1 - localparam K_PKT_START = 8'b110_11100; // 0xDC K28.6 - localparam K_PKT_END = 8'b100_11100; // 0x9C K28.4 - localparam K_XON = 8'b010_11100; // 0x5C K28.2 - localparam K_XOFF = 8'b011_11100; // 0x7C K28.3 - localparam K_LOS = 8'b111_11111; // 0xFF K31.7 - localparam K_ERROR = 8'b000_00000; // 0x00 K00.0 - localparam D_56 = 8'b110_00101; // 0xC5 D05.6 - - localparam IDLE = 3'd0; - localparam FIRSTLINE1 = 3'd1; - localparam FIRSTLINE2 = 3'd2; - localparam PKT1 = 3'd3; - localparam PKT2 = 3'd4; - localparam CRC_CHECK = 3'd5; - localparam ERROR = 3'd6; - localparam DONE = 3'd7; - - wire [17:0] even_data; - reg [17:0] odd_data; - wire [17:0] chosen_data; - reg odd; - - reg [31:0] line_i; - reg sop_i, eop_i, error_i; - wire error_o, sop_o, eop_o, write; - reg [15:0] halfline; - reg [8:0] holder; - wire [31:0] line_o; - - reg [2:0] state; - - reg [15:0] CRC; - wire [15:0] nextCRC; - reg write_d; - - wire rst_rxclk; - wire have_space; - - oneshot_2clk rst_1s(.clk_in(clk),.in(rst),.clk_out(ser_rx_clk),.out(rst_rxclk)); - - assign even_data = {ser_rkmsb,ser_rklsb,ser_r}; - - always @(posedge ser_rx_clk) - if(rst_rxclk) - holder <= 9'd0; - else - holder <= {even_data[17],even_data[15:8]}; - - always @(posedge ser_rx_clk) - if(rst_rxclk) - odd_data <= 18'd0; - else - odd_data <= {even_data[16],holder[8],even_data[7:0],holder[7:0]}; - - assign chosen_data = odd ? odd_data : even_data; - - // Transfer xon and xoff info to the main system clock for flow control purposes - reg xon_rcvd_rxclk, xoff_rcvd_rxclk; - always @(posedge ser_rx_clk) - xon_rcvd_rxclk = ({1'b1,K_XON} == {ser_rkmsb,ser_r[15:8]}) | ({1'b1,K_XON} == {ser_rklsb,ser_r[7:0]} ); - always @(posedge ser_rx_clk) - xoff_rcvd_rxclk = ({1'b1,K_XOFF} == {ser_rkmsb,ser_r[15:8]}) | ({1'b1,K_XOFF} == {ser_rklsb,ser_r[7:0]} ); - - oneshot_2clk xon_1s(.clk_in(ser_rx_clk),.in(xon_rcvd_rxclk),.clk_out(clk),.out(xon_rcvd)); - oneshot_2clk xoff_1s(.clk_in(ser_rx_clk),.in(xoff_rcvd_rxclk),.clk_out(clk),.out(xoff_rcvd)); - - // If the other side is sending xon or xoff, or is flow controlled (b/c we told them to be), don't fill the fifos - wire wait_here = ((chosen_data == {2'b10,K_COMMA,D_56})|| - (chosen_data == {2'b11,K_XON,K_XON})|| - (chosen_data == {2'b11,K_XOFF,K_XOFF}) ); - - always @(posedge ser_rx_clk) - if(rst_rxclk) sop_i <= 0; - else if(state == FIRSTLINE1) sop_i <= 1; - else if(write_d) sop_i <= 0; - - reg write_pre; - always @(posedge ser_rx_clk) - if(rst_rxclk) - begin - state <= IDLE; - odd <= 0; - halfline <= 0; - line_i <= 0; - eop_i <= 0; - error_i <= 0; - write_pre <= 0; - end - else - case(state) - IDLE : - begin - error_i <= 0; - write_pre <= 0; - if(even_data == {2'b11,K_PKT_START,K_PKT_START}) - begin - state <= FIRSTLINE1; - odd <= 0; - end - else if(odd_data == {2'b11,K_PKT_START,K_PKT_START}) - begin - state <= FIRSTLINE1; - odd <= 1; - end - end - - FIRSTLINE1 : - if(chosen_data[17:16] == 0) - begin - halfline <= chosen_data[15:0]; - state <= FIRSTLINE2; - end - else if(wait_here) - ; // Flow Controlled, so wait here and do nothing - else - state <= ERROR; - - FIRSTLINE2 : - if(chosen_data[17:16] == 0) - begin - line_i <= {chosen_data[15:0],halfline}; - if(~have_space) // No space to write to! Should have been avoided by flow control - state <= ERROR; - else - begin - state <= PKT1; - write_pre <= 1; - end - end // if (chosen_data[17:16] == 0) - else if(wait_here) - ; // Flow Controlled, so wait here and do nothing - else - state <= ERROR; - - PKT1 : - begin - write_pre <= 0; - if(chosen_data[17:16] == 0) - begin - halfline <= chosen_data[15:0]; - state <= PKT2; - end - else if(wait_here) - ; // Flow Controlled - else if(chosen_data == {2'b11,K_PKT_END,K_PKT_END}) - state <= CRC_CHECK; - else - state <= ERROR; - end // case: PKT1 - - PKT2 : - if(chosen_data[17:16] == 0) - begin - line_i <= {1'b0,1'b0,1'b0,chosen_data[15:0],halfline}; - if(~have_space) // No space to write to! - state <= ERROR; - else - begin - state <= PKT1; - write_pre <= 1; - end - end // if (chosen_data[17:16] == 0) - else if(wait_here) - ; // Flow Controlled - else - state <= ERROR; - - CRC_CHECK : - if(chosen_data[17:0] == {2'b00,CRC}) - begin - if(~have_space) - state <= ERROR; - else - begin - eop_i <= 1; - state <= DONE; - end - end - else if(wait_here) - ; - else - state <= ERROR; - - ERROR : - begin - error_i <= 1; - if(have_space) - state <= IDLE; - end - DONE : - begin - state <= IDLE; - eop_i <= 0; - end - - endcase // case(state) - - - always @(posedge ser_rx_clk) - if(rst_rxclk) - CRC <= 16'hFFFF; - else if(state == IDLE) - CRC <= 16'hFFFF; - else if(chosen_data[17:16] == 2'b00) - CRC <= nextCRC; - - CRC16_D16 crc_blk(chosen_data[15:0],CRC,nextCRC); - - always @(posedge ser_rx_clk) - if(rst_rxclk) write_d <= 0; - else write_d <= write_pre; - - // Internal FIFO, size 9 is 2K, size 10 is 4K Bytes - assign write = eop_i | (error_i & have_space) | (write_d & (state != CRC_CHECK)); - wire dummy; // avoid warning on unconnected pin - - fifo_2clock_cascade #(.WIDTH(36),.SIZE(FIFOSIZE)) serdes_rx_fifo - (.arst(rst), - .wclk(ser_rx_clk),.datain({1'b0,error_i,sop_i,eop_i,line_i}), - .src_rdy_i(write), .dst_rdy_o(have_space), .space(fifo_space), - .rclk(clk),.dataout({dummy,error_o,sop_o,eop_o,line_o}), - .src_rdy_o(wr_ready_o), .dst_rdy_i(wr_ready_i), .occupied(fifo_occupied) ); - - assign fifo_full = ~have_space; // Note -- in the wrong clock domain - assign fifo_empty = ~wr_ready_o; - - // Internal FIFO to Buffer interface - assign wr_dat_o = line_o; - assign wr_flags_o = { 2'b00, eop_o | error_o, sop_o | error_o }; - - wire slu = ~(({2'b11,K_ERROR,K_ERROR}=={ser_rkmsb,ser_rklsb,ser_r}) || - ({2'b11,K_LOS,K_LOS}=={ser_rkmsb,ser_rklsb,ser_r})); - - reg [3:0] slu_reg; - - always @(posedge clk) - if(rst) slu_reg <= 0; - else slu_reg <= {slu_reg[2:0],slu}; - - always @(posedge clk) - serdes_link_up <= &slu_reg[3:1]; - - assign debug = { have_space, wr_ready_o, odd, sop_i, eop_i, error_i, state[2:0] }; - -endmodule // serdes_rx |