

## **Product Bulletin**

April 2024 FPGA-PB-02001

# Workaround for Lattice ECP5 (LFE5UM) Known Issue with SerDes Interface Connections Due to Unstable Reset Soft Logic

**Product Affected:** All ECP5 and ECP5-5G with SerDes(LFE5UM and LFE5UM-5G) devices are covered by this Product Bulletin.

**Customers Affected:** Only customers using SerDes/PCS-based designs. It potentially affects several applications such as PCIe IP, g8B10B, and other PCS supported protocol. The workaround is organized on a single channel basis, contact your local Lattice representative for other use cases.

**Background**: The Lattice Semiconductor ECP5 FPGA devices may contain SerDes. These parts with SerDes can be implemented through Clarity Designer by using the PCS module. The PCS module comes with a custom soft logic reset, which helps users automatically reset the receiver portion of the PCS.

**Observation**: Lattice found that in specific situations, the original soft logic reset is not as stable as intended. Some unexpected behaviors include the Receiver CDR failing to lock and the receiver data being unstable. This document provides the workaround procedure to resolve these issues.

#### **Detailed Workaround Description:**

This workaround is organized on a single channel basis, contact your local Lattice representative for other use cases.

Lattice developed a workaround procedure that implements 'extended Reset Soft Logic = extRSL' to bring-up the SerDes/PCS at a stable state. The workaround is to remove the current receiver reset signals from the current RSL and add a new module that will drive the receiver reset signals instead.

The new source code contains the necessary functions for a detailed signal monitoring and controlled reset scenario. It is the single point of control for all the receiver resets in the PCS portion of the design. The transmitter reset signals will still be driven by the original RSL.

It supports the controlled reset sequencing for the different blocks, such as:

- Default power-up situation: Enhanced power-up reset sequence for higher stability. (Refer to Figure 10)
- Receiver Clock Data Recovery block is locked and stable. (Refer to Figure 11)
- PCS block is locked and stable. (Refer to Figure 12)

If these settings are detected, the design releases all resets, resulting to a stable design.

If any of the cases below occurs, the reset scenario starts from the beginning (Refer to Figure 13):

- Code violation error the FSM restarts the PCS again.
- RX loss of signal or the TX PLL will lose its lock.

For the cascaded resets of different functions in the SerDes, the user can select default reset periods as minimum step width. These variables are counter timers for specific portions of the reset sequence and optimized for this workaround. The "Tplol" value is the amount of time

transmit PLL to lock, "Tcdr" is the amount of time receiver CDR has to lock and "Tviol" is the amount of time to wait for code violation and disparity error before concluding the PCS is correct. Note that depending on your design, these values might have to be adjusted. For example, if the reference clock used for the Transmit PLL is more jittery, the "Tplol" might have to be adjusted to a higher value. (*Refer to Figure 14*)

#### Note:

The extended Reset Soft Logic is instantiated into the PCS block generated from Clarity Designer. Keep in mind that once "rx\_rsl" is instantiated to the PCS block, every new generation in Clarity overrides the user's modification so make sure to save your work beforehand.

## **Overview Block Diagram of the Workaround:**



#### Note:

- 1. Connected to the user logic. The signal should be added to the PCS module. Example: can be connected to PLL lock.
- 2. Debug signal.

| Signal Name        | Direction | Description                                                                                                                     |  |  |
|--------------------|-----------|---------------------------------------------------------------------------------------------------------------------------------|--|--|
| refclk             | input     | Connected to TX PLL Reference Clock.                                                                                            |  |  |
| pll_lol            | input     | TX PLL Loss of Lock Output.                                                                                                     |  |  |
| cdr_lol            | input     | CDR PLL Loss of Lock Output.                                                                                                    |  |  |
| cv                 | input     | Connect this to $rx\_cv\_err\_ch$ . This is a code violation signal to indicate an error was detected with the associated data. |  |  |
| Ism                | input     | Connected to Ism_status_s.                                                                                                      |  |  |
| los                | input     | Connected to rx_los_low_s.                                                                                                      |  |  |
| disable_rx_pcs_rst | input     | Test signal for debug purposes only.                                                                                            |  |  |
| rx_serdes_rst      | output    | Connected to the net of RX SerDes RSL signal (rsl_rx_serdes_rst_c)                                                              |  |  |
| rx_pcs_rst         | output    | Connected to the net of RX PCS RSL signal (rsl_rx_pcs_rst_c)                                                                    |  |  |

#### Issue Avoidance when using PCS IP:

To implement the new RSL Logic:

1. Start the PCS IP generation. After the generation, you should have the new PCS module added to your design. Then, place the PCS Module in the desired location.



Figure 1: PCS IP assigned to DCU0 at Channel1

2. Add the new RSL logic file (rx\_rsl.vhd) to the existing implementation/input files.



Figure 2: Addition of Source File for Extended RSL

3. Remove the current signals "rsl\_rx\_serdes\_rst\_c" and "rsl\_rx\_pcs\_rst\_c" from the PCS file and add the "rx\_rsl" module instead for the RX SerDes and PCS RSL. Connections are shown below. Refer to the appendix for the source code and instantiation.

```
rx rsl rx rsl (
.rstn (rsl rst),
.refclk (pll refclki),
.pll_lol (pll_lol),
.cdr | lol (rx_cdr_lol_s),
.cv (rx cv_err[0]),
.lsm (lsm_status_s),
.los ((rx_los_low_s| rx_loss)),
.disable_rx_pcs_rst (disable_rx_pcs_rst),
.rx_serdes_rst (rsl_rx_serdes_rst_c),
.rx_pcs_rst (rsl_rx_pcs_rst_c)
);

pcie2_xl_pcsrsl_core_rsl_inst (.rui_rst(rsl_rst), .rui_serdes_rst_dual_c(serdes_rst_dual_c),
.rui_tx_ref_clk(pll_refclki), .rui_rx_serdes_rst_c(tx_serdes_rst_c),
.rui_tx_pcs_rst_c(3*b000, tx_pcs_rst_c)), .rdi_pll_lol(pll_lol),
.rui_rx_pcs_rst_c(3*b000, rx_pcs_rst_c)),
.rui_rx_pcs_rst_c(3*b000, rx_pcs_rst_c),
.rui_rx_pcs_rst_c(3*b000, rx_pcs_rst_c),
.rui_rx_pcs_rst_c(3*b000, rx_pcs_rst_c),
.ruo_serdes_rst_dual_c(rsl_serdes_rst_dual_c), .rdo_rst_dual_c(rsl_rst_dual_c),
.ruo_rx_rdy(rsl_tx_rdy), .rdo_tx_serdes_rst_c(rsl_tx_serdes_rst_c),
.rdo_tx_pcs_rst_c(f(nl01, nl02, nl03, rsl_tx_pcs_rst_c)),
.ruo_rx_rdy(rsl_tx_rdy), .rdo_rx_serdes_rst_c(f(nl04, nl05, nl06/*, rsl_rx_pcs_rst_c*/));

(a) Removed SerDes and PCS RSL
```

Figure 3: (a) Remove Current RSL (b) Add the rx\_rsl to PCS module.



Figure 4: rx\_rsl embedded into the PCS Block Generated from Clarity

#### Issue Avoidance when using PCIe IP:

To implement the new RSL Logic:

4. Start the PCIe IP generation. After the generation, you should have the new PCS module added to your design. Then, place the PCS Module in the desired location.



Figure 5: PCle IP assigned to DCU0 at Channel1

5. Add the new RSL logic file (rx\_rsl.vhd) to the existing implementation/input files.



Figure 6: Addition of Source File for Extended RSL

6. Remove the current signals "rsl\_rx\_serdes\_rst\_c" and "rsl\_rx\_pcs\_rst\_c" from the PCS file and add the "rx\_rsl" module instead for the RX SerDes and PCS RSL. Connections are shown below. Refer to the appendix for the source code and instantiation.

```
rx rsl rx rsl (
.rstn (*rsl rst),
.refclk (pll_refclki),
.pll_lol (pll_lol),
.dr lol (rx_cdr_lol s),
.cv (rx_cv_err[0]),
.lsm (lsm_status_s),
.los ((rx_los_low_s | rx_loss)),
.disable_rx_pcs_rst (disable_rx_pcs_rst_c)

.rx_serdes_rst (rsl_rx_serdes_rst_c),
.rx_pcs_rst (rsl_rx_pcs_rst_c)

pcie2_xl_pcsrsl_core_rsl_inst (.rui_rst(rsl_rst), .rui_serdes_rst_dual_c(serdes_rst_dual_c),
.rui_rst_dual_c(rst_dual_c), .rui_rsl_disable(rsl_disable),
.rui_rtx_ref_clk(pll_refclki), .rui_tx_serdes_rst_c(tx_serdes_rst_c),
.rui_rx_ref_clk(rxrefclk), .rui_rx_serdes_rst_c(tx_serdes_rst_c),
.rui_rx_ref_clk(rxrefclk), .rui_rx_serdes_rst_c(3^1b000, rx_pcs_rst_c)),
.rui_rx_pcs_rst_c(3^1b000, rx_pcs_rst_c),
.rui_rx_pcs_rst_c(3^1b000, rx_pcs_rst_c),
.rui_rx_pcs_rst_c(3^1b000, rx_pcs_rst_c),
.ruo_serdes_rst_dual_c(rsl_serdes_rst_dual_c), .rdo_rst_dual_c(rsl_rst_dual_c),
.ruo_rx_dv(rsl_tx_rdy), .rdo_rx_serdes_rst_c(rsl_tx_serdes_rst_c),
.ruo_rx_dv(rsl_tx_rdy), .rdo_rx_serdes_rst_c(fill04, nl05,
.ruo_rx_dv(rsl_tx_rdy), .rdo_rx_pcs_rst_c(fill04, nl05,
.ruo_rx_dv(rsl_tx_rdy), .rdo_rx
```

Figure 7: (a) Remove Current RSL (b) Add the rx\_rsl to PCS module.



Figure 8: PCIe core to PCS sample hierarchy



Figure 9: rx\_rsl embedded into the PCS Block of the PCle IP

#### **Customer Acknowledgement:**

By using the affected devices, the customer acknowledges that single-channel SerDes/PCS based designs have connection issues due to unstable reset soft logic and that instructions on how to avoid this scenario has been provided.

#### **Contact:**

If you have any questions or require additional information, contact Lattice Technical Support through <a href="https://www.latticesemi.com/techsupport">www.latticesemi.com/techsupport</a>.

#### Source code:

```
--rx_rsl.vhd content
library ieee;
use ieee.std_logic_1164.all;
use IEEE.std_logic_unsigned.all;
use IEEE.std_logic_arith.all;
entity rx_rsl is
        port
        (
        rstn
                                       : in
                                               std_logic;
                                       : in
       refclk
                                               std_logic;
       pll_lol
                                       : in
                                               std_logic;
                                       : in
       cdr_lol
                                               std_logic;
       CV
                                       : in
                                               std_logic;
                                               std logic;
       Ism
                                       : in
                                       : in
                                               std logic;
       los
       disable_rx_pcs_rst
                                       : in
                                               std_logic;
                            : out std_logic;
: out std_logic
       rx serdes rst
       rx_pcs_rst
       );
end rx_rsl;
architecture rx rsl arc of rx rsl is
attribute syn_keep: boolean;
                      Constants Declaration
constant Tplol
                                       : std logic vector (31 downto 0):=x"00100000";
                                       : std_logic_vector (31 downto 0):=x"00100000";
constant Tcdr
constant Tviol
                                       : std_logic_vector (31 downto 0):=x"00100000";
                                       Internal Variables
signal pll_lol_s
                                       : std_logic;
signal cdr_lol_s
                                       : std_logic;
signal cv_s
                                       : std logic;
signal lsm_s
                                       : std_logic;
signal los_s
                                       : std_logic;
                                       : std_logic_vector (31 downto 0);
signal cnt
```

```
type rx_sm_state is (powerup, apply_cdr_rst, wait_cdr_lock, test_cdr, apply_rxpcs_rst, wait_rxpcs_lock,
test_rxpcs, idle);
signal rx_sm
                                        : rx_sm_state;
attribute syn keep of rx sm
                                        : signal is true;
begin
                               Begin Of The Design
rx_reset_proc
                                        : process (rstn, refclk)
begin
if rstn = '0' then
                                         <= '1';
        pll_lol_s
                                         <= '1';
        cdr lol s
                                        <= '1';
        CV S
        Ism_s
                                         <= '0';
                                         <= '1';
        los_s
                                         <= '1';
        rx serdes rst
                                        <= '1';
        rx_pcs_rst
                                         <= powerup;
        rx_sm
        cnt
                                         <= (others => '0');
elsif rising_edge(refclk) then
                                         <= pll lol;
        pll_lol_s
        cdr_lol_s
                                        <= cdr_lol;
                                         <= cv;
        CV_S
        Ism s
                                        <= lsm;
                                         <= los;
        los_s
        case rx_sm is
        when powerup =>
                rx_serdes_rst <= '1';
rx_pcs_rst <= '1' AND_NOT(disable_rx_pcs_rst);
                if (pll\_lol\_s = '1') or (los\_s = '1') then
                                       <= (others => '0');
                        cnt
                else
                        if (cnt = Tplol) then
                                 cnt <= (others => '0');
                                 rx_sm <= apply_cdr_rst;</pre>
                        else
                                cnt <= cnt + '1';
                        end if;
                end if;
```

```
when apply cdr rst =>
                                <= '1';
        rx_serdes_rst
        rx_pcs_rst
                                <= '1' AND NOT(disable_rx_pcs_rst);
        if (cnt = x"00000007") then
                                <= (others => '0');
                cnt
                rx_sm
                                <= wait_cdr_lock;
        else
                cnt
                                <= cnt + '1';
        end if;
when wait cdr lock =>
                                <= '0';
        rx_serdes_rst
                                <= '1' AND NOT(disable_rx_pcs_rst);
        rx_pcs_rst
        if (cnt = Tcdr) then
                                <= (others => '0');
                cnt
                                <= test_cdr;
                rx_sm
        else
                cnt
                                <= cnt + '1';
        end if;
when test cdr =>
        rx_serdes_rst
                                <= '0';
                                <= '1' AND NOT(disable_rx_pcs_rst);
        rx_pcs_rst
        if (cdr_lol_s = '1') then
                                <= (others => '0');
                cnt
                                <= apply_cdr_rst;
                rx_sm
        else
                if (cnt = Tcdr) then
                                <= (others => '0');
                        rx_sm <= apply_rxpcs_rst;</pre>
                else
                        cnt
                                 <= cnt + '1';
                end if;
        end if;
when apply_rxpcs_rst =>
        rx_serdes_rst
                                <= '0';
                                <= '1' AND NOT(disable_rx_pcs_rst);
        rx_pcs_rst
        if (cnt = x"00000007") then
                                <= (others => '0');
                cnt
                                <= wait_rxpcs_lock;
                rx_sm
        else
                cnt
                                <= cnt + '1';
        end if;
when wait_rxpcs_lock =>
                                <= '0';
        rx_serdes_rst
                                <= '0';
        rx_pcs_rst
        if (cnt = Tviol) then
                                <= (others => '0');
                cnt
                                <= test_rxpcs;
                rx_sm
        else
                                <= cnt + '1';
                cnt
        end if;
```

```
when test rxpcs =>
                rx_serdes_rst <= '0';
rx_pcs_rst <= '0';
                rx_pcs_rst
                if (lsm_s = '0') or (cv_s = '1') then
                                     <= (others => '0');
                        cnt
                                       <= apply_rxpcs_rst;
                        rx_sm
                else
                        if (cnt = Tviol) then
                                        <= (others => '0');
                                cnt
                                 rx sm <= idle;
                        else
                                cnt <= cnt + '1';
                        end if;
                end if;
        when idle =>
                rx_serdes_rst <= '0';
                                       <= '0';
                rx pcs rst
                if (lsm_s = '0') or (cv_s = '1') then
                        rx_sm <= apply_rxpcs_rst;
cnt <= (others => '0');
                end if;
        end case;
        if (pll\_lol\_s = '1') or (los\_s = '1') then
                rx_sm
                                       <= powerup;
                                         <= (others => '0');
                cnt
        end if;
end if;
end process rx_reset_proc;
end rx_rsl_arc;
```

#### rx\_rsl contents definitions:

```
proc
                    : process (rstn, refclk)
begin

if rstn = '0' then
     pll_lol_s
cdr_lol_s
                                                                 rx_rsl.vhd
     cv s
                                                                            The controlled reset sequencing for the different
                                                                            blocks such as:
                                                                                       Default power up situation:
     rx serdes_rst
     rx_pcs_rst
                                                                                       (PLL is stable and RX loss of signal is
                                                                                       detected)
cnt <= (oth
                                                                                       Clock data recovery block is locked and
                         <= pll_lol;
<= cdr_lol;
<= cv;
     pll_lol_s
cdr_lol_s
                                                                                       stable.
     cv s
                                                                                       PCS block is locked and stable.
                         <= lsm;
<= los;
                                                                            If these settings are detected, the design releases
     case rx_sm is
when powerup =>
                                                                            all resets and becomes stable.
         rx_serdes_rst <= '1';
rx pcs rst <= '1';
                                                                            However, in case of any code violation error, the
         rx_sstues_t:
rx_pcs_rst <= '1';
if (pll_lol_s = '1') or (los_s = '1') then
cnt <= (others => '0');
                                                                             FSM restarts the PCS again; and
                                                                            If any RX loss of signal or the TX_PLL will lose its
             if (cnt = Tplo1) then
    cnt <= (others => '0');
                                                                            lock, then the reset scenario starts from the
             rx_sm <= (others => '0'
rx_sm <= apply_cdr_rst;
else
                                                                            beginning.
                         <= cnt + '1';
                 cnt
          end if;
end if;
```

Figure 10: Default Power-up Situations

```
when apply cdr rst =>
   rx_serdes_rst <= '1';
rx_pcs_rst <= '1';</pre>
   if (cnt = x"00000007") then
                 <= (others
       cnt
                   <= wait_cdr_lock;
       rx_sm
   else
                   <= cnt + '1';
       cnt
                                              rx rsl.vhd
   end if;
                                                      The controlled reset sequencing for the different
when wait_cdr_lock =>
   rx_serdes_rst <= '0';
                                                      blocks such as:
                   <= '1';
   rx_pcs_rst
                                                               Default power up situation:
   if (cnt = Tcdr) then
                                                               (PLL is stable and RX loss of signal is
                   <= (others => '0');
       cnt
                                                               detected)
       rx_sm
                   <= test_cdr;</pre>
   else
                                                               Clock data recovery block is locked and
                   <= cnt + '1':
       cnt
                                                               stable.
   end if;
when test_cdr =>
                                                               PCS block is locked and stable.
   rx_serdes_rst <= '0';
                                                      If these settings are detected, the design releases
                  <= '1';
   rx_pcs_rst
   if (cdr_lol_s = '1') then
                                                      all resets and becomes stable.
                   <= (others => '0');
       cnt
                                                      However, in case of any code violation error, the
                  <= apply_cdr_rst;</pre>
       rx_sm
                                                      FSM restarts the PCS again; and
    else
                                                      If any RX loss of signal or the TX_PLL I its lock,
       if (cnt = Tcdr) then
          cnt <= (others => '0');
rx_sm <= apply_rxpcs_rst;</pre>
                                                      then the reset scenario starts from the beginning.
        else
           cnt
                   <= cnt + '1';
        end if;
    end if:
```

Figure 11: Clock Data Recovery Block is Locked and Stable

```
rx_serdes_rst <= '0
                   <= '1';
   rx pcs rst
   if (cnt = x"00000007") then
                 <= (others =>
                  <= wait_rxpcs_look;
       rx sm
   else
                                                rx rsl.vhd
                   <= cnt + '1':
       cnt
                                                         The controlled reset sequencing for the different
   end if;
 nen wait_rxpcs_lock =>
                                                         blocks such as:
   rx_serdes_rst <= '0';
                  <= '0';
                                                                  Default power up situation:
   rx_pcs_rst
   if (cnt = Tviol) then
                                                                  (PLL is stable and RX loss of signal is
                   <= (others => '0');
       cnt
                                                                  detected)
                  <= test_rxpcs;
       rx sm
                                                                  Clock data recovery block is locked and
                  <= cnt + '1':
       cnt
                                                                  stable.
   end if:
                                                                  PCS block is locked and stable.
when test rxpcs =>
   rx serdes rst
                   <= '0':
                                                         If these settings are detected, the design releases
   rx_pcs_rst
                   <= '0';
   if (lsm_s = '0') or (cv_s = '1') then
cnt <= (others => '0');
                                                         all resets and becomes stable.
                                                         However, in case of any code violation error, the
                   <= apply_rxpcs_rst;</pre>
       rx sm
                                                         FSM restarts the PCS again; and
       if (cnt = Tviol) then
                                                         If any RX loss of signal or the TX_PLL loses its
                   <= (others => '0');
           cnt
                                                         lock, then the reset scenario start from the
           rx_sm <= idle;
       else
                                                         beginning.
                   <= cnt + '1';
          cnt
       end if;
   end if;
```

Figure 12: PCS Block is Locked and Stable

```
when idle =>
                                                         rx rsl.vhd
        rx_serdes_rst <= '0';
rx pcs rst <= '0';</pre>
                                                                  The controlled reset sequencing for the different
        if (1sm \ s = '0') or (cv \ s = '1') then
                                                                  blocks such as:
                     <= apply_rxpcs_rst;
<= (others => '0');
            rx sm
                                                                           Default power up situation:
            cnt
                                                                           (PLL is stable and RX loss of signal is
        end if:
    end case:
                                                                           Clock data recovery block is locked and stable.
                                                                           PCS block is locked and stable.
    if (pll_lol_s = '1') or (los_s = '1') then
                                                                  If these settings are detected, the design releases all
                         <= powerup;</pre>
        rx sm
                                                                  resets and becomes stable.
                         <= (others => '0');
        cnt
                                                                  However, in case of any code violation error, the FSM
    end if;
                                                                  restarts the PCS again; and
end if:
                                                                  If any RX loss of signal or the TX PLL loses its lock,
end process rx_reset_proc;
                                                                  then the reset scenario starts from the beginning.
```

Figure 13: Use in Case of Any Code Violation Error

```
-- Constants Declaration
-- R125MHz different values
-- aux reset is short
-- cdr synch takes longer
-- pcs_rst is short again

constant Tplol : std_logic_vector (31 downto 0):=x"00100000";
constant Todr : std_logic_vector (31 downto 0):=x"00100000";
constant Tviol : std_logic_vector (31 downto 0):=x"00100000";
width.
```

Figure 14: Reset Periods as Minimum Step Width

### rx\_rsl Instantiation:

```
rx_rsl rx_rsl (
  .rstn
                         (~rsl_rst),
  .refclk
                         (pll_refclki),
  .pll_lol
                         (pll_lol),
  .cdr_lol
                         (rx_cdr_lol_s),
  .cv
                         (rx_cv_err[0]),
                         (lsm_status_s),
  .lsm
                         ((rx_los_low_s | rx_loss)),
  .los
  .disable_rx_pcs_rst (disable_rx_pcs_rst),
  .rx_serdes_rst
                         (rsl_rx_serdes_rst_c),
                         (rsl_rx_pcs_rst_c)
  .rx_pcs_rst
);
```

# **Revision History**

#### Revision 1.1, April 2024

| Section | Change Summary           |                                                                      |
|---------|--------------------------|----------------------------------------------------------------------|
| All     | • Upda                   | ated the affected customers, parameter names, and some terminologies |
|         | • Adde                   | ed the block diagram of the workaround                               |
|         | <ul> <li>Sepa</li> </ul> | rated the steps for PCS IP & PCIe IP                                 |

#### Revision 1.0, June 2021

| Section | Change Summary  |
|---------|-----------------|
| All     | Initial release |

#### **Disclaimers**

Lattice makes no warranty, representation, or guarantee regarding the accuracy of information contained in this document or the suitability of its products for any particular purpose. All information herein is provided AS IS and with all faults, and all risk associated with such information is entirely with Buyer. Buyer shall not rely on any data and performance specifications or parameters provided herein. Products sold by Lattice have been subject to limited testing and it is the Buyer's responsibility to independently determine the suitability of any products and to test and verify the same. No Lattice products should be used in conjunction with mission- or safety-critical or any other application in which the failure of Lattice's product could create a situation where personal injury, death, severe property or environmental damage may occur. The information provided in this document is proprietary to Lattice Semiconductor, and Lattice reserves the right to make any changes to the information in this document or to any products at any time without notice.