OpenCores
URL https://opencores.org/ocsvn/turbo8051/turbo8051/trunk

Subversion Repositories turbo8051

[/] [turbo8051/] [trunk/] [rtl/] [gmac/] [mac/] [g_mii_intf.v] - Rev 12

Go to most recent revision | Compare with Previous | Blame | View Log

//////////////////////////////////////////////////////////////////////
////                                                              ////
////  Tubo 8051 cores MAC Interface Module                        ////
////                                                              ////
////  This file is part of the Turbo 8051 cores project           ////
////  http://www.opencores.org/cores/turbo8051/                   ////
////                                                              ////
////  Description                                                 ////
////  Turbo 8051 definitions.                                     ////
////                                                              ////
////  To Do:                                                      ////
////    nothing                                                   ////
////                                                              ////
////  Author(s):                                                  ////
////      - Dinesh Annayya, dinesha@opencores.org                 ////
////                                                              ////
//////////////////////////////////////////////////////////////////////
////                                                              ////
//// Copyright (C) 2000 Authors and OPENCORES.ORG                 ////
////                                                              ////
//// This source file may be used and distributed without         ////
//// restriction provided that this copyright statement is not    ////
//// removed from the file and that any derivative work contains  ////
//// the original copyright notice and the associated disclaimer. ////
////                                                              ////
//// This source file is free software; you can redistribute it   ////
//// and/or modify it under the terms of the GNU Lesser General   ////
//// Public License as published by the Free Software Foundation; ////
//// either version 2.1 of the License, or (at your option) any   ////
//// later version.                                               ////
////                                                              ////
//// This source 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 Lesser General Public License for more ////
//// details.                                                     ////
////                                                              ////
//// You should have received a copy of the GNU Lesser General    ////
//// Public License along with this source; if not, download it   ////
//// from http://www.opencores.org/lgpl.shtml                     ////
////                                                              ////
//////////////////////////////////////////////////////////////////////
//`timescale 1ns/100ps
 
/***************************************************************
 Description:
 
 mii_intf.v: This verilog file is reduced mii interface for ethenet
 The transmit state machine generates a condition to indicate 
 start transmit. This is done using strt_preamble. The transmit
 state machine upon detecting strt_preamble generates pre-amble
 and start of frame de-limiter and then starts accepting the data
 from the transmit block by asserting the transmit nibble ack.
 
 ***************************************************************/
/************** MODULE DECLARATION ****************************/
module g_mii_intf(
                  // Data and Control Signals to tx_fsm and rx_fsm
          mi2rx_strt_rcv,
          mi2rx_rcv_vld,
          mi2rx_rx_byte,
          mi2rx_end_rcv,
          mi2rx_extend,
          mi2rx_frame_err,
          mi2rx_end_frame,
          mi2rx_crs,
          mi2tx_byte_ack,
          mi2tx_slot_vld,
          cfg_uni_mac_mode_change,
 
          // Phy Signals 
          phy_tx_en,
          phy_tx_er,
          phy_txd,
          phy_tx_clk,
          phy_rx_clk,
	  tx_reset_n,
	  rx_reset_n,
          phy_rx_er,
          phy_rx_dv,
          phy_rxd,
          phy_crs,
          // rx_er fix.  need to fwd to mac wrapper, to drop rx_er pkts.  mfilardo.
	  rx_sts_rx_er_reg,
 
                  // Reset signal
          app_reset_n,
 
                  // Signals from Config Management
          cf2mi_loopback_en,
          cf2mi_rmii_en,
          cf_mac_mode,
	  cf_chk_rx_dfl,
          cf_silent_mode,
 
                  // Signal from Application to transmit JAM
          app_send_jam,
          df2rx_dfl_dn,
 
                  // Inputs from Transmit FSM
          tx2mi_strt_preamble,
          tx2mi_end_transmit,
          tx2mi_tx_byte,
		  tx_ch_en
          );
 
 
parameter NO_GMII_PREAMBLE   = 5'b00111;
parameter NO_MII_PREAMBLE    = 5'b01111;
parameter NO_RMII_PREAMBLE   = 5'b11111;
parameter GMII_JAM_COUNT     = 5'b0011;
parameter MII_JAM_COUNT      = 5'b0111;
parameter RMII_JAM_COUNT     = 5'b1111;
 
  /******* INPUT & OUTPUT DECLARATIONS *************************/
 
  output      mi2rx_strt_rcv;     // This is generated by the MII block to indicate start 
                                  // of receive data.
  output      mi2rx_rcv_vld;      // This signal is asserted by MII on reception of valid
                                  // bytes to indicate to the RX block to accept data from PHY
  output[7:0] mi2rx_rx_byte;      // This the receive data from the PHY gathered as bytes 
  output      mi2rx_end_rcv;      // This signal is asserted with the last data assembled
  output      mi2rx_extend;       // This signal is asserted during carrier extension (Receive)
  output      mi2rx_frame_err;    // This signal is asserted during Dibit Error (In RMII Mode)
                                  // or Nibble Error in (MII Mode)
  output      mi2rx_end_frame;    // End of Frame 
  output      mi2rx_crs;          // CRS signal in rx_clk domain
  output      mi2tx_byte_ack;     // MII block acknowledges a byte during transmit
  output      mi2tx_slot_vld;     // MII block acknowledges valid slot during transmit // mfilardo
 
  output      phy_tx_en;          // Enable data on TX
  output      phy_tx_er;          // Transmit Error (Used in Carrier Extension in 1000 Mode)
  output[7:0] phy_txd;            // Transmit data on the line 
 
  input      phy_tx_clk;         // Transmit Clock in 10/100 Mb/s 
 
  output       rx_sts_rx_er_reg;  // rx_er fix.  need to fwd to mac wrapper, to drop rx_er pkts.  mfilardo.
  input       app_reset_n;          // reset from the application interface
 
  input       phy_rx_clk;         // Receive Clock in 10/100/1000 Mb/s 
  input       phy_rx_er;          // Receive Error. Used in Carrier Extension in 1000 Mode 
  input       phy_rx_dv;          // Receive Data Valid from the PHY 
  input[7:0]  phy_rxd;            // Receive Data 
  input       phy_crs;            // Carrier Sense from the line 
 
  input       tx_reset_n;
  input       rx_reset_n;
 
 
  input       cf2mi_loopback_en;           // loop back enable
  input       cf2mi_rmii_en;               // RMII Mode
  input       cf_mac_mode;                 // Mac Mode 0--> 10/100 Mode, 1--> 1000 Mode 
  input       cf_chk_rx_dfl;               // Check for Deferal 
  input       cf_silent_mode;              // PHY Inactive 
  input       app_send_jam;                // Send a Jam Sequence (From the Application)
  input       df2rx_dfl_dn;                // Deferal Done in Rx Clock Domain 
  input       tx2mi_strt_preamble;         // Tx FSM indicates to MII to generate 
                                           // preamble on the line 
  input       tx2mi_end_transmit;          // This is provided by the TX block to 
                                           // indicate end of transmit
  input[7:0]  tx2mi_tx_byte;               // 8 bits of data from Tx block
  input       tx_ch_en;                    // Transmitt Enable 
  input       cfg_uni_mac_mode_change;
 
  /******* WIRE & REG DECLARATION FOR INPUT AND OUTPUTS ********/
  reg       phy_tx_en;
  reg       mi2tx_byte_ack;
  reg       mi2tx_slot_vld;
  reg [7:0] phy_txd;
  wire [7:0] mi2rx_rx_byte;
  reg [7:0] mi2rx_rx_byte_in;
  reg       mi2rx_extend;
  reg       mi2rx_extend_err;
  reg       mi2rx_strt_rcv;
  reg       mi2rx_end_rcv;
  reg       mi2rx_rcv_vld;
  reg       mi2rx_frame_err;
 
  /*** REG & WIRE DECLARATIONS FOR LOCAL SIGNALS ***************/
 
  reg [4:0]  tx_preamble_cnt_val;
  reg [4:0]  jam_count;
  reg [4:0]  jam_count_reg;
 
 
  reg        strt_rcv_in;
  reg        end_rcv_in;
  reg        rx_dv_in;
  reg        rx_er_in;
  reg        rcv_valid_in;
 
 
 
  reg [7:0]  rxd_in;
 
  parameter  mii_rx_idle_st = 3'd0, mii_rx_pre_st = 3'd1,
             mii_rx_byte_st = 3'd2, mii_rx_end_st = 3'd3,
             mii_rx_dibit_st = 3'd4, mii_rx_nibble_st = 3'd5;
 
  reg [2:0]  mii_rx_nxt_st; 
  reg [2:0]  mii_rx_cur_st;
 
  parameter  mii_tx_idle_st =  4'd0, mii_tx_pre_st  =  4'd1,
             mii_tx_byte_st = 4'd2,  mii_tx_end_st = 4'd3,
             mii_tx_jam_st = 4'd4, mii_tx_nibble_st = 4'd5,
	     mii_tx_nibble_end_st = 4'd6, mii_tx_dibit_st = 4'd7,
	     mii_tx_dibit_end_st = 4'd8;
 
  reg [3:0]  mii_tx_cur_st;
  reg [3:0]  mii_tx_nxt_st;
 
  wire       send_jam;
  wire       receive_detect;
  wire       pre_condition;
  wire       sfd_condition;
  wire       tx_en;
  wire       tx_er;
  wire [7:0] txd;
  wire       byte_boundary_rx, byte_boundary_tx;
 
  reg        tx_en_in;
  reg        tx_err_in;
  reg        tx_ext_in;
  reg        tx_pre_in;
  reg        tx_sfd_in;
  reg        tx_jam_in;
  reg        tx_xfr_ack_in;
  reg        inc_preamble_cntr;
  reg        rst_preamble_cntr;
  reg        inc_jam_cntr;
  reg        rst_jam_cntr;
  reg [1:0]  tx_xfr_cnt, rx_xfr_cnt, tx_slot_xfr_cnt;
  reg        rx_dv;
  reg        rx_er;
  reg        rcv_err_in;
  reg	     mi2rx_end_frame_in;	
 
  reg [1:0]  tx_dibit_in;
  reg        mi2rx_extend_in, mi2rx_extend_err_in, mi2rx_end_frame;
  reg        crs_in;
  reg        phy_tx_er;
 
  wire       dibit_check_rx, dibit_check_tx;
  wire       nibble_check_rx, nibble_check_tx;
  wire       pre_condition_gmii, pre_condition_mii, pre_condition_rmii;
  wire       sfd_condition_gmii, sfd_condition_mii, sfd_condition_rmii;
  wire [3:0] tx_nibble_in;
  reg [7:0]  rxd;
  wire       receive_detect_pulse;
  reg        d_receive_detect;
 
 
  reg        lb_tx_en, lb_tx_er;
  reg        rx_dv_del;
  reg  [1:0] rxd_del;
  reg        rx_dfl_dn;
  reg        rx_dfl_dn_reg;
 
 
  /******** SEQUENTIAL LOGIC **********************************/
 
  // This logic generates appropriate receive data valid
  // in case of loop back 
  always @(tx_en or phy_rxd or txd or phy_rx_dv or tx_er or phy_crs
           or cf2mi_loopback_en or phy_tx_en or phy_rx_er or
	   cf2mi_rmii_en or cf_mac_mode)
    begin
      if(cf2mi_loopback_en)
        begin
          rx_dv_in = tx_en;
          rx_er_in = tx_er;
          rxd_in = txd;
          crs_in = (tx_en | tx_er) && cf_mac_mode;
        end // if (mii_loopback_en)
      else
        begin
          rx_dv_in = phy_rx_dv ;
          rx_er_in = phy_rx_er;
          rxd_in = phy_rxd;
	  // *** NOTE ****
	  // phy_crs should be a combination of crs and tx_en
	  // In Full Duplex tx_en determines deferral in half duplex
	  // crs determines deferral
          crs_in = (tx_en | tx_er);
        end // else: !if(mii_loopback_en)
    end
 
 
  // Following state machine is to detect start preamble and
  // transmit preamble and sfd and then the data.
  // This state machine also generates acknowledge to TX block
  // to allow the TX block to update its byte pointers
  always @(posedge phy_tx_clk or negedge tx_reset_n)
    begin
      if(!tx_reset_n)
        begin
          mii_tx_cur_st <= mii_tx_idle_st; 
        end
      else if (tx_ch_en)
        begin
          mii_tx_cur_st <= mii_tx_nxt_st; 
        end
      else 
        begin
          mii_tx_cur_st <= mii_tx_idle_st; 
        end
    end
 
  always @(mii_tx_cur_st or tx2mi_strt_preamble or tx2mi_end_transmit or cf_mac_mode 
           or cf2mi_rmii_en or tx_preamble_cnt_val or byte_boundary_tx  
	   or jam_count or tx_xfr_cnt or send_jam or receive_detect
	   or receive_detect_pulse or jam_count_reg or jam_count or cfg_uni_mac_mode_change)
    begin
 
      mii_tx_nxt_st = mii_tx_cur_st;
      tx_en_in = 1'b0;
      tx_pre_in = 1'b0;
      tx_sfd_in = 1'b0;
      tx_err_in = 1'b0;
      tx_ext_in = 1'b0;
      tx_jam_in = 1'b0;
      inc_preamble_cntr = 1'b0;
      rst_preamble_cntr = 1'b0;
      inc_jam_cntr = 1'b0;
      rst_jam_cntr = 1'b0;
      tx_xfr_ack_in = 1'b0;
 
      casex(mii_tx_cur_st)       // synopsys parallel_case full_case
 
        mii_tx_idle_st:
	// wait from start from transmit state machine
          begin
            if(tx2mi_strt_preamble)
              begin
                inc_preamble_cntr = 1'b1;
                tx_en_in = 1'b1;
                tx_pre_in = 1'b1;
                mii_tx_nxt_st = mii_tx_pre_st;
              end
            else
              mii_tx_nxt_st = mii_tx_idle_st;
          end
 
        mii_tx_pre_st:
        // This state generates the preamble to be transmitted and
        // generates SFD before transitioning the data state
          begin
            if((tx_preamble_cnt_val == NO_GMII_PREAMBLE) && cf_mac_mode)
              begin
                tx_en_in = 1'b1;
                tx_sfd_in = 1'b1;
                tx_xfr_ack_in = 1'b1;
                rst_preamble_cntr = 1'b1;
                mii_tx_nxt_st = mii_tx_byte_st;
              end
            else if((tx_preamble_cnt_val == NO_MII_PREAMBLE) && !cf_mac_mode &&
	            !cf2mi_rmii_en)
              begin
                tx_en_in = 1'b1;
                tx_sfd_in = 1'b1;
                tx_xfr_ack_in = 1'b1;
                rst_preamble_cntr = 1'b1;
                mii_tx_nxt_st = mii_tx_nibble_st;
	      end
            else if((tx_preamble_cnt_val == NO_RMII_PREAMBLE) && !cf_mac_mode &&
	           cf2mi_rmii_en)
              begin
                tx_en_in = 1'b1;
                tx_sfd_in = 1'b1;
                tx_xfr_ack_in = 1'b1;
                rst_preamble_cntr = 1'b1;
                mii_tx_nxt_st = mii_tx_dibit_st;
	      end
            else
              begin
                inc_preamble_cntr = 1'b1;
                tx_en_in = 1'b1;
                tx_pre_in = 1'b1;
                mii_tx_nxt_st = mii_tx_pre_st;
              end
          end
 
        mii_tx_byte_st:
        // This state picks up a byte from the transmit block
        // before transmitting on the line
		  begin  
            if(tx2mi_end_transmit && byte_boundary_tx )
             	begin
              		tx_en_in = 1'b1;
              		tx_xfr_ack_in = 1'b0;
              		mii_tx_nxt_st = mii_tx_end_st;
             	end
				else if (!cf_mac_mode & cfg_uni_mac_mode_change)  // Mandar
             	begin
            		tx_en_in = 1'b1;
            		tx_xfr_ack_in = 1'b1;
            		mii_tx_nxt_st = mii_tx_nibble_st;
             	end
          	else
              	begin
               	tx_en_in = 1'b1;
               	tx_xfr_ack_in = 1'b1;
               	mii_tx_nxt_st = mii_tx_byte_st;
            	end
         end
 
       /*mii_tx_byte_st:
        // This state picks up a byte from the transmit block
        // before transmitting on the line
          begin
            if(tx2mi_end_transmit && byte_boundary_tx )
              begin
                tx_en_in = 1'b1;
                tx_xfr_ack_in = 1'b0;
                mii_tx_nxt_st = mii_tx_end_st;
              end
            else
              begin
                tx_en_in = 1'b1;
                tx_xfr_ack_in = 1'b1;
                mii_tx_nxt_st = mii_tx_byte_st;
              end
          end*/
 
        mii_tx_jam_st:
	   begin
            if(jam_count == jam_count_reg)
	      begin
                tx_en_in = 1'b1;
                tx_jam_in = 1'b1;
		rst_jam_cntr = 1'b1;
                mii_tx_nxt_st = mii_tx_idle_st;
	      end
            else
	      begin
                tx_en_in = 1'b1;
                tx_jam_in = 1'b1;
		inc_jam_cntr = 1'b1;
                mii_tx_nxt_st = mii_tx_jam_st;
	      end
	   end
 
        mii_tx_end_st:
        // This state checks for the end of transfer 
        // and extend for carrier extension
          begin
            if(tx2mi_strt_preamble)
	      begin
                tx_en_in = 1'b1;
                tx_pre_in = 1'b1;
                mii_tx_nxt_st = mii_tx_pre_st;
	      end
            else
              begin
                tx_en_in = 1'b0;
                mii_tx_nxt_st = mii_tx_idle_st;
              end
          end
 
        /*mii_tx_nibble_st:
        // This state picks up a byte from the transmit block
        // before transmitting on the line
          begin
            if(tx2mi_end_transmit && !byte_boundary_tx )
              begin
                tx_en_in = 1'b1;
                tx_xfr_ack_in = 1'b1;
                mii_tx_nxt_st = mii_tx_nibble_end_st;
              end
            else
              begin
                tx_en_in = 1'b1;
                tx_xfr_ack_in = 1'b1;
                mii_tx_nxt_st = mii_tx_nibble_st;
              end
	  		end*/
 
        mii_tx_nibble_st:                             // Mandar
        // This state picks up a byte from the transmit block
        // before transmitting on the line
					begin
            		if(tx2mi_end_transmit && !byte_boundary_tx )
              			begin
                			tx_en_in = 1'b1;
                			tx_xfr_ack_in = 1'b1;
                			mii_tx_nxt_st = mii_tx_nibble_end_st;
              			end
                 else if (cf_mac_mode & cfg_uni_mac_mode_change)  // Mandar
              			begin
                			tx_en_in = 1'b1;
                			tx_xfr_ack_in = 1'b1;
                        mii_tx_nxt_st = mii_tx_byte_st;
              			end
            		else
              			begin
                			tx_en_in = 1'b1;
                			tx_xfr_ack_in = 1'b1;
                			mii_tx_nxt_st = mii_tx_nibble_st;
              			end
					end		
 
        mii_tx_nibble_end_st:
        // This state checks for the end of transfer 
        // and extend for carrier extension
          begin
            if(tx2mi_strt_preamble)
	      begin
                tx_en_in = 1'b1;
                tx_pre_in = 1'b1;
                mii_tx_nxt_st = mii_tx_pre_st;
	      end
            else
              begin
                tx_en_in = 1'b0;
                mii_tx_nxt_st = mii_tx_idle_st;
              end
	  end
 
        mii_tx_dibit_st:
        // This state picks up a byte from the transmit block
        // before transmitting on the line
          begin
            if(tx2mi_end_transmit && (tx_xfr_cnt[0]) && (tx_xfr_cnt[1]) )
              begin
                tx_en_in = 1'b1;
                tx_xfr_ack_in = 1'b1;
                mii_tx_nxt_st = mii_tx_dibit_end_st;
              end
            else
              begin
                tx_en_in = 1'b1;
                tx_xfr_ack_in = 1'b1;
                mii_tx_nxt_st = mii_tx_dibit_st;
              end
	  end
 
        mii_tx_dibit_end_st:
        // This state checks for the end of transfer 
        // and extend for carrier extension
          begin
             if(tx2mi_strt_preamble)
	      begin
                tx_en_in = 1'b1;
                tx_pre_in = 1'b1;
                mii_tx_nxt_st = mii_tx_pre_st;
	      end
            else
              begin
                tx_en_in = 1'b1;
                mii_tx_nxt_st = mii_tx_idle_st;
              end
	  end
       endcase     
     end // end always 
 
  // All the data, enable and ack signals leaving this blocks are
  // registered tx_data and tx_en are outputs going to PHY tx_nibble_ack 
  // is indicating to the TX block of acceptance of data nibble
  always @(posedge phy_tx_clk or negedge tx_reset_n)
    begin
      if(!tx_reset_n)
        begin
	  lb_tx_en <= 1'b0;
	  lb_tx_er <= 1'b0;
          phy_tx_en <= 1'b0;
          phy_tx_er <= 1'b0;
	  mi2tx_byte_ack <= 1'b0;
	  mi2tx_slot_vld <= 1'b0;
        end
      else
        begin
	  lb_tx_en <= tx_en_in;
	  lb_tx_er <= tx_err_in;
          phy_tx_en <= (cf_silent_mode) ? 1'b0 : tx_en_in; 
          phy_tx_er <= (cf_silent_mode) ? 1'b0 : tx_err_in;
	  mi2tx_byte_ack <= (cf_mac_mode || (cf2mi_rmii_en && tx_xfr_cnt[1] && !tx_xfr_cnt[0]) || 
	                     (!cf2mi_rmii_en && nibble_check_tx)) ? tx_xfr_ack_in : 1'b0; 
	  mi2tx_slot_vld <= (cf_mac_mode || (cf2mi_rmii_en && tx_slot_xfr_cnt[1] && !tx_slot_xfr_cnt[0]) || 
	                     (!cf2mi_rmii_en && tx_slot_xfr_cnt[0] )) ? (tx_xfr_ack_in || inc_preamble_cntr) : 1'b0; 
        end
    end
 
  always @(posedge phy_tx_clk or negedge tx_reset_n)
    begin
      if(!tx_reset_n)
        begin
          phy_txd[7:0] <= 8'b00000000;
	end
      else
	begin
	  if (cf_mac_mode) 
             phy_txd[7:0] <= (tx_pre_in) ? 8'b01010101 : ((tx_sfd_in) ? 
	                      8'b11010101 : ((tx_jam_in) ?  8'b11111111 : 
			      ((tx_ext_in) ? 8'b00001111: tx2mi_tx_byte)));
	  else if (!cf_mac_mode && !cf2mi_rmii_en) 
             phy_txd[3:0] <= (tx_pre_in) ? 4'b0101 : ((tx_sfd_in) ? 
	                      4'b1101 : ((tx_jam_in) ?  4'b1111 : tx_nibble_in)) ;
	  else if (!cf_mac_mode && cf2mi_rmii_en) 
             phy_txd[1:0] <= (tx_pre_in) ? 2'b01 : ((tx_sfd_in) ? 
	                      2'b11 : ((tx_jam_in) ?  2'b11 : tx_dibit_in)) ;
	end
    end
  assign receive_detect_pulse = receive_detect && !d_receive_detect;
 
  always @(posedge phy_tx_clk or negedge tx_reset_n)
    begin
      if(!tx_reset_n)
        jam_count_reg <= 5'd0;
      else if(cf_mac_mode) 
        jam_count_reg <= GMII_JAM_COUNT; 
      else if(cf2mi_rmii_en) 
        jam_count_reg <= RMII_JAM_COUNT; 
      else if(!cf2mi_rmii_en) 
        jam_count_reg <= MII_JAM_COUNT; 
    end
 
  always @(posedge phy_tx_clk or negedge tx_reset_n)
    begin
      if(!tx_reset_n)
        d_receive_detect <= 0;
      else 
        d_receive_detect <= receive_detect; 
    end
  // counter for the number transfers 
  always @(posedge phy_tx_clk or negedge tx_reset_n)
    begin
      if(!tx_reset_n)
        tx_xfr_cnt <= 2'd0;
      else if(tx2mi_strt_preamble)
        tx_xfr_cnt <= 2'd0;
      else if(tx_xfr_ack_in)
        tx_xfr_cnt <= tx_xfr_cnt + 1; 
    end
 
  // phy_tx_en_dly
  reg phy_tx_en_dly;
  always @(posedge phy_tx_clk or negedge tx_reset_n)
    begin
      if(!tx_reset_n)
        phy_tx_en_dly <= 1'd0;
      else
        phy_tx_en_dly <= phy_tx_en;
    end
 
   wire phy_tx_en_fall;
   assign phy_tx_en_fall = phy_tx_en_dly && !phy_tx_en;
 
  // counter for the number transfers  ... 
  always @(posedge phy_tx_clk or negedge tx_reset_n)
    begin
      if(!tx_reset_n)
        tx_slot_xfr_cnt <= 2'd0;
      else if(phy_tx_en_fall)
        tx_slot_xfr_cnt <= 2'd0;
      else if(inc_preamble_cntr || tx_xfr_ack_in)
        tx_slot_xfr_cnt <= tx_slot_xfr_cnt + 1; 
    end
 
  assign nibble_check_tx = tx_xfr_cnt[0];
  assign dibit_check_tx= tx_xfr_cnt[0] && tx_xfr_cnt[1];
  assign byte_boundary_tx = (cf_mac_mode) ? 1'b1: 
               ((!cf_mac_mode && !cf2mi_rmii_en) ? nibble_check_tx :  dibit_check_tx);
 
  assign tx_nibble_in = (tx_xfr_cnt[0]) ? tx2mi_tx_byte[3:0] : tx2mi_tx_byte[7:4];
 
/*  always @(tx_xfr_cnt or tx2mi_tx_byte or phy_tx_clk)
	begin
	  if (!tx_xfr_cnt[1] && !tx_xfr_cnt[0])
	    tx_dibit_in <= tx2mi_tx_byte[1:0];
	  else if (!tx_xfr_cnt[1] && tx_xfr_cnt[0])
	    tx_dibit_in <= tx2mi_tx_byte[3:2];
	  else if (tx_xfr_cnt[1] && !tx_xfr_cnt[0])
	    tx_dibit_in <= tx2mi_tx_byte[5:4];
	  else if (tx_xfr_cnt[1] && tx_xfr_cnt[0])
	    tx_dibit_in <= tx2mi_tx_byte[7:6];
	  else
	    tx_dibit_in <= tx2mi_tx_byte[1:0];
	end
*/
  always @(posedge phy_tx_clk or negedge tx_reset_n)
    begin
      if(!tx_reset_n)
	tx_dibit_in <= 2'b0;
      else if(tx2mi_strt_preamble)
	tx_dibit_in <= 2'b0;
      else
	begin
	  if (!tx_xfr_cnt[1] && !tx_xfr_cnt[0])
	    tx_dibit_in <= tx2mi_tx_byte[1:0];
	  else if (!tx_xfr_cnt[1] && tx_xfr_cnt[0])
	    tx_dibit_in <= tx2mi_tx_byte[3:2];
	  else if (tx_xfr_cnt[1] && !tx_xfr_cnt[0])
	    tx_dibit_in <= tx2mi_tx_byte[5:4];
	  else if (tx_xfr_cnt[1] && tx_xfr_cnt[0])
	    tx_dibit_in <= tx2mi_tx_byte[7:6];
	end
    end
 
  // counter for the number of preamble to be sent
  // before transmitting the sfd
  always @(posedge phy_tx_clk or negedge tx_reset_n)
    begin
      if(!tx_reset_n)
        tx_preamble_cnt_val <= 5'd0;
      else if(rst_preamble_cntr)
        tx_preamble_cnt_val <= 5'd0;
      else if(inc_preamble_cntr)
        tx_preamble_cnt_val <= tx_preamble_cnt_val + 1;
    end
 
  // Jam Counter 
    always @(posedge phy_tx_clk or negedge tx_reset_n)
      begin
        if(!tx_reset_n)
          jam_count <= 5'd0;
        else if(rst_jam_cntr)
          jam_count <= 5'd0;
        else if(inc_jam_cntr)
          jam_count <= jam_count + 1;
      end
 
 
    always @(posedge phy_rx_clk or negedge rx_reset_n)
      begin
        if(!rx_reset_n)
	  rx_dfl_dn_reg <= 1'b0;
	else if (df2rx_dfl_dn)
	  rx_dfl_dn_reg <= 1'b1;
	else 
	  begin
	    if (!phy_rx_dv && !phy_rx_er)
	      rx_dfl_dn_reg <= 1'b0;
	  end
      end
 
  assign mi2rx_rx_byte = mi2rx_rx_byte_in;
  //assign rxd = (!cf_mac_mode && cf2mi_rmii_en)? mi2rx_rx_byte : mi2rx_rx_byte_in;
 
  assign pre_condition_gmii = (!rxd[7] && rxd[6] && !rxd[5] && rxd[4] 
                               && !rxd[3] && rxd[2] && !rxd[1] && rxd[0] && rx_dv
			       && !(!rx_dfl_dn_reg && cf_chk_rx_dfl));
 
  assign pre_condition_mii = (!rxd[3] && rxd[2] && !rxd[1] && rxd[0] && rx_dv
		              && !(!rx_dfl_dn_reg && cf_chk_rx_dfl));
 
  assign pre_condition_rmii = (!rxd[1] && rxd[0] && rx_dv
                               && !rxd_del[1] && rxd_del[0] && rx_dv_del 
		               && !(!rx_dfl_dn_reg && cf_chk_rx_dfl));
 
  assign sfd_condition_gmii = (rxd[7] && rxd[6] && !rxd[5] && rxd[4] 
                               && !rxd[3] && rxd[2] && !rxd[1] && rxd[0] && rx_dv
			       && !(!rx_dfl_dn_reg && cf_chk_rx_dfl));
 
  assign sfd_condition_mii = (rxd[3] && rxd[2] && !rxd[1] && rxd[0] && rx_dv
		              && !(!rx_dfl_dn_reg && cf_chk_rx_dfl));
 
  assign sfd_condition_rmii = (rxd[1] && rxd[0] && rx_dv
                               && !rxd_del[1] && rxd_del[0] && rx_dv_del 
		               && !(!rx_dfl_dn_reg && cf_chk_rx_dfl));
 
 
 
  assign pre_condition = (cf_mac_mode) ? pre_condition_gmii : 
                         ((cf2mi_rmii_en) ? pre_condition_rmii : pre_condition_mii);
 
  assign sfd_condition = (cf_mac_mode) ? sfd_condition_gmii : 
                         ((cf2mi_rmii_en) ? sfd_condition_rmii : sfd_condition_mii);
 
  // Following state machine is to detect strt preamble and
  // receive preamble and sfd and then the data.
  always @(posedge phy_rx_clk or negedge rx_reset_n)
    begin
      if(!rx_reset_n)
        begin
          mii_rx_cur_st <= mii_rx_idle_st; 
	end
      else
        begin
          mii_rx_cur_st <= mii_rx_nxt_st; 
        end
    end
 
  always @(mii_rx_cur_st or rx_dv or rx_er or pre_condition or sfd_condition 
           or byte_boundary_rx or rxd or rx_xfr_cnt or cf2mi_rmii_en or cf_mac_mode)
    begin
      mii_rx_nxt_st = mii_rx_cur_st;
      strt_rcv_in = 1'b0;
      end_rcv_in = 1'b0;
      rcv_valid_in = 1'b0;
      rcv_err_in = 1'b0;
      mi2rx_extend_in = 1'b0;
      mi2rx_extend_err_in = 1'b0;
      mi2rx_end_frame_in = 1'b0;
 
      casex(mii_rx_cur_st)       // synopsys parallel_case full_case
 
        mii_rx_idle_st:
        // This state is waiting for pre-amble to
        // appear on the line in Receive mode
        begin
          if(pre_condition)
            mii_rx_nxt_st = mii_rx_pre_st;
          else
            mii_rx_nxt_st = mii_rx_idle_st;
        end
 
        mii_rx_pre_st:
        // This state checks the pre-amble and Waits for SFD on the line 
        begin
          if(sfd_condition)
            begin
              strt_rcv_in = 1'b1;
	      if(cf_mac_mode)
                mii_rx_nxt_st = mii_rx_byte_st;
	      else if(!cf_mac_mode && !cf2mi_rmii_en)
                mii_rx_nxt_st = mii_rx_nibble_st;
	      else if(!cf_mac_mode && cf2mi_rmii_en)
                mii_rx_nxt_st = mii_rx_dibit_st;
            end
          else if(pre_condition)
	    begin
              mii_rx_nxt_st = mii_rx_pre_st;
	    end
          else if(!rx_dv && rx_er && cf_mac_mode)
	    begin
	      mi2rx_extend_in = (rxd == 8'h0F) ? 1'b1: 1'b0;
	      mi2rx_extend_err_in = (rxd == 8'h1F) ? 1'b1: 1'b0;
              mii_rx_nxt_st = mii_rx_pre_st;
	    end
          else
            begin
	      mi2rx_end_frame_in = 1'b1;
              mii_rx_nxt_st = mii_rx_idle_st;
            end
        end
 
        mii_rx_byte_st:
        // This state looks for data validity and latches di-bit2 and
        // sends it to the receiver
        begin
          if(rx_dv)
            begin
              rcv_valid_in = 1'b1;
              mii_rx_nxt_st = mii_rx_byte_st;
            end
	  else if (!rx_dv && rx_er && cf_mac_mode)
            begin
	      mi2rx_extend_in = (rxd == 8'h0F) ? 1'b1: 1'b0;
	      mi2rx_extend_err_in = (rxd == 8'h1F) ? 1'b1: 1'b0;
              end_rcv_in = 1'b1;
              mii_rx_nxt_st = mii_rx_pre_st;
            end
          else
            begin
              end_rcv_in = 1'b1;
              mii_rx_nxt_st = mii_rx_end_st;
            end
        end
 
        mii_rx_end_st:
        // This state looks for data validity and latches di-bit2 and
        // sends it to the receiver
           mii_rx_nxt_st = mii_rx_idle_st;
 
        mii_rx_nibble_st:
        begin
          if(rx_dv)
            begin
              rcv_valid_in = 1'b1;
              mii_rx_nxt_st = mii_rx_nibble_st;
            end
          else
            begin
              end_rcv_in = 1'b1;
              mii_rx_nxt_st = mii_rx_end_st;
	      if(rx_xfr_cnt[0])
	        rcv_err_in = 1'b1;
            end
        end
 
        mii_rx_dibit_st:
        begin
          if(rx_dv)
            begin
              rcv_valid_in = 1'b1;
              mii_rx_nxt_st = mii_rx_dibit_st;
            end
          else
            begin
              end_rcv_in = 1'b1;
              mii_rx_nxt_st = mii_rx_end_st;
	      if(!(!rx_xfr_cnt[0] && !rx_xfr_cnt[1]))
	        rcv_err_in = 1'b1;
            end
        end
 
      endcase
    end // always @ (mii_rx_cur_st...
 
  // counter for the number receives 
  always @(posedge phy_rx_clk or negedge rx_reset_n)
    begin
      if(!rx_reset_n)
        rx_xfr_cnt <= 2'd0;
      else if(mi2rx_end_rcv)
        rx_xfr_cnt <= 2'd0; 
      else if(rcv_valid_in)
        rx_xfr_cnt <= rx_xfr_cnt + 1; 
    end
 
  always @(posedge phy_rx_clk or negedge rx_reset_n)
    begin
      if(!rx_reset_n)
        begin
	  mi2rx_rx_byte_in <= 8'b0;
	  rxd <= 8'b0;
	  rxd_del <= 2'b0;
	end
      else
	begin
	  rxd <= rxd_in;
//	  rxd_del <= rxd_in[1:0];
	  rxd_del <= rxd[1:0];
 
	  if (cf_mac_mode)
	    mi2rx_rx_byte_in <= rxd;
	  else if (!cf_mac_mode && !cf2mi_rmii_en)
	    begin
	      if(!rx_xfr_cnt[0])
	        mi2rx_rx_byte_in[3:0] <= rxd[3:0];
	      else
	        mi2rx_rx_byte_in[7:4] <= rxd[3:0];
	    end
	  else if(!cf_mac_mode && cf2mi_rmii_en)
	    begin
	      if(!rx_xfr_cnt[1] && !rx_xfr_cnt[0])
	        mi2rx_rx_byte_in[1:0] <= rxd[1:0];
	      else if(!rx_xfr_cnt[1] && rx_xfr_cnt[0])
	        mi2rx_rx_byte_in[3:2] <= rxd[1:0];
	      else if(rx_xfr_cnt[1] && !rx_xfr_cnt[0])
	        mi2rx_rx_byte_in[5:4] <= rxd[1:0];
	      else if(rx_xfr_cnt[1] && rx_xfr_cnt[0])
	        mi2rx_rx_byte_in[7:6] <= rxd[1:0];
	    end
	end
    end
 
  reg  rx_sts_rx_er_reg;
  always @(posedge phy_rx_clk or negedge rx_reset_n) begin
      if(!rx_reset_n) begin
	  rx_sts_rx_er_reg <= 1'b0;
      end
      else if (mi2rx_strt_rcv) begin
	  rx_sts_rx_er_reg <= 1'b0;
      end
      else if(phy_rx_dv && phy_rx_er) begin
	  rx_sts_rx_er_reg <= 1'b1;
      end
  end
 
  always @(posedge phy_rx_clk 
       or negedge rx_reset_n)
    begin
      if(!rx_reset_n)
        begin
          mi2rx_rcv_vld <= 1'b0;
	end
      else if(cf_mac_mode)
          mi2rx_rcv_vld <= rcv_valid_in;
      else if(!cf_mac_mode && cf2mi_rmii_en && rx_xfr_cnt[0] && rx_xfr_cnt[1])
          mi2rx_rcv_vld <= rcv_valid_in;
      else if(!cf_mac_mode && !cf2mi_rmii_en && rx_xfr_cnt[0])
          mi2rx_rcv_vld <= rcv_valid_in;
      else
          mi2rx_rcv_vld <= 1'b0;
    end
  // All the data, enable and ack signals out of RX block are
  // registered 
  always @(posedge phy_rx_clk or negedge rx_reset_n)
    begin
      if(!rx_reset_n)
        begin
          mi2rx_strt_rcv <= 1'b0;
          mi2rx_end_rcv <= 1'b0;
          //mi2rx_rcv_vld <= 1'b0;
          mi2rx_frame_err <= 1'b0;
          mi2rx_extend <= 1'b0;
          mi2rx_extend_err <= 1'b0;
	  mi2rx_end_frame <= 1'b0; 
	  rx_dv <= 1'b0;
	  rx_er <= 1'b0;
	  rx_dv_del <= 1'b0;
        end
      else
        begin
          mi2rx_strt_rcv <= strt_rcv_in;
          mi2rx_end_rcv <= end_rcv_in; 
          //mi2rx_rcv_vld <= rcv_valid_in;
          mi2rx_frame_err <= rcv_err_in;
          mi2rx_extend <= mi2rx_extend_in; 
          mi2rx_extend_err <= mi2rx_extend_err_in;
	  mi2rx_end_frame <= mi2rx_end_frame_in;
	  rx_dv <= rx_dv_in;
	  rx_er <= rx_er_in;
	  rx_dv_del <= rx_dv;
        end
    end
 
 
 half_dup_dble_reg U_dble_reg0 (
                 //outputs
                 .sync_out_pulse(send_jam),
                 //inputs
                 .in_pulse(app_send_jam),
                 .dest_clk(phy_tx_clk),
                 .reset_n(tx_reset_n)
             );
 
 half_dup_dble_reg U_dble_reg1 (
                 //outputs
                 .sync_out_pulse(receive_detect),
                 //inputs
                 .in_pulse(rx_dv_in),
                 .dest_clk(phy_tx_clk),
                 .reset_n(tx_reset_n)
             );
 
 
wire test;
 
 half_dup_dble_reg U_dble_reg5 (
                 //outputs
                 .sync_out_pulse(tx_en),
                 //inputs
                 .in_pulse(lb_tx_en),
                 .dest_clk(phy_rx_clk),
                 .reset_n(rx_reset_n)
             );
 
 half_dup_dble_reg U_dble_reg6 (
                 //outputs
                 .sync_out_pulse(txd[0]),
                 //inputs
                 .in_pulse(phy_txd[0]),
                 .dest_clk(phy_rx_clk),
                 .reset_n(rx_reset_n)
             );
 
 half_dup_dble_reg U_dble_reg7 (
                 //outputs
                 .sync_out_pulse(txd[1]),
                 //inputs
                 .in_pulse(phy_txd[1]),
                 .dest_clk(phy_rx_clk),
                 .reset_n(rx_reset_n)
             );
 
 half_dup_dble_reg U_dble_reg8 (
                 //outputs
                 .sync_out_pulse(txd[2]),
                 //inputs
                 .in_pulse(phy_txd[2]),
                 .dest_clk(phy_rx_clk),
                 .reset_n(rx_reset_n)
             );
 
 half_dup_dble_reg U_dble_reg9 (
                 //outputs
                 .sync_out_pulse(txd[3]),
                 //inputs
                 .in_pulse(phy_txd[3]),
                 .dest_clk(phy_rx_clk),
                 .reset_n(rx_reset_n)
             );
 
 half_dup_dble_reg U_dble_reg10 (
                 //outputs
                 .sync_out_pulse(txd[4]),
                 //inputs
                 .in_pulse(phy_txd[4]),
                 .dest_clk(phy_rx_clk),
                 .reset_n(rx_reset_n)
             );
 
 
 half_dup_dble_reg U_dble_reg11 (
                 //outputs
                 .sync_out_pulse(txd[5]),
                 //inputs
                 .in_pulse(phy_txd[5]),
                 .dest_clk(phy_rx_clk),
                 .reset_n(rx_reset_n)
             );
 
 
 half_dup_dble_reg U_dble_reg12 (
                 //outputs
                 .sync_out_pulse(txd[6]),
                 //inputs
                 .in_pulse(phy_txd[6]),
                 .dest_clk(phy_rx_clk),
                 .reset_n(rx_reset_n)
             );
 
 
 half_dup_dble_reg U_dble_reg13 (
                 //outputs
                 .sync_out_pulse(txd[7]),
                 //inputs
                 .in_pulse(phy_txd[7]),
                 .dest_clk(phy_rx_clk),
                 .reset_n(rx_reset_n)
             );
 
 
 half_dup_dble_reg U_dble_reg14 (
                 //outputs
                 .sync_out_pulse(tx_er),
                 //inputs
                 .in_pulse(lb_tx_er),
                 .dest_clk(phy_rx_clk),
                 .reset_n(rx_reset_n)
             );
 
 
 half_dup_dble_reg U_dble_reg15 (
                 //outputs
                 .sync_out_pulse(mi2rx_crs),
                 //inputs
                 .in_pulse(crs_in),
                 .dest_clk(phy_rx_clk),
                 .reset_n(rx_reset_n)
             );
 
endmodule
 
 

Go to most recent revision | Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.