Documentation/misc-devices/xilinx_sdfec.rst
.. SPDX-License-Identifier: GPL-2.0+
This driver supports SD-FEC Integrated Block for Zynq |Ultrascale+ (TM)| RFSoCs.
.. |Ultrascale+ (TM)| unicode:: Ultrascale+ U+2122 .. with trademark sign
For a full description of SD-FEC core features, see the SD-FEC Product Guide (PG256) <https://www.xilinx.com/cgi-bin/docs/ipdoc?c=sd_fec;v=latest;d=pg256-sdfec-integrated-block.pdf>_
This driver supports the following features:
Missing features, known issues, and limitations of the SD-FEC driver are as follows:
The device tree entry is described in:
linux-xlnx/Documentation/devicetree/bindings/misc/xlnx,sd-fec.txt <https://github.com/Xilinx/linux-xlnx/blob/master/Documentation/devicetree/bindings/misc/xlnx%2Csd-fec.txt>_
The driver works with the SD-FEC core in two modes of operation:
Run-time Configuration
For Run-time configuration the role of driver is to allow the software application to do the following:
- Load the configuration parameters for either Turbo decode or LDPC encode or decode
- Activate the SD-FEC core
- Monitor the SD-FEC core for errors
- Retrieve the status and configuration of the SD-FEC core
Programmable Logic (PL) Initialization
For PL initialization, supporting logic loads configuration parameters for either the Turbo decode or LDPC encode or decode. The role of the driver is to allow the software application to do the following:
- Activate the SD-FEC core
- Monitor the SD-FEC core for errors
- Retrieve the status and configuration of the SD-FEC core
The driver provides a platform device where the probe and remove
operations are provided.
The driver defines the following driver file operations to provide user application interfaces:
open: Implements restriction that only a single file descriptor can be open per SD-FEC instance at any time
release: Allows another file descriptor to be open, that is after current file descriptor is closed
poll: Provides a method to monitor for SD-FEC Error events
unlocked_ioctl: Provides the following ioctl commands that allows the application configure the SD-FEC core:
- :c:macro:`XSDFEC_START_DEV`
- :c:macro:`XSDFEC_STOP_DEV`
- :c:macro:`XSDFEC_GET_STATUS`
- :c:macro:`XSDFEC_SET_IRQ`
- :c:macro:`XSDFEC_SET_TURBO`
- :c:macro:`XSDFEC_ADD_LDPC_CODE_PARAMS`
- :c:macro:`XSDFEC_GET_CONFIG`
- :c:macro:`XSDFEC_SET_ORDER`
- :c:macro:`XSDFEC_SET_BYPASS`
- :c:macro:`XSDFEC_IS_ACTIVE`
- :c:macro:`XSDFEC_CLEAR_STATS`
- :c:macro:`XSDFEC_SET_DEFAULT_CONFIG`
After opening the driver, the user should find out what operations need to be performed to configure and activate the SD-FEC core and determine the configuration of the driver. The following outlines the flow the user should perform:
Note: When monitoring for interrupts if a critical error is detected where a reset is required, the driver will be required to load the default configuration.
Determine the configuration of the SD-FEC core by using the ioctl
:c:macro:XSDFEC_GET_CONFIG.
Setting the order determines how the order of Blocks can change from input to output.
Setting the order is done by using the ioctl :c:macro:XSDFEC_SET_ORDER
Setting the order can only be done if the following restrictions are met:
- The ``state`` member of struct :c:type:`xsdfec_status <xsdfec_status>` filled by the ioctl :c:macro:`XSDFEC_GET_STATUS` indicates the SD-FEC core has not STARTED
The following steps indicate how to add LDPC codes to the SD-FEC core:
- Use the auto-generated parameters to fill the :c:type:`struct xsdfec_ldpc_params <xsdfec_ldpc_params>` for the desired LDPC code.
- Set the SC, QA, and LA table offsets for the LPDC parameters and the parameters in the structure :c:type:`struct xsdfec_ldpc_params <xsdfec_ldpc_params>`
- Set the desired Code Id value in the structure :c:type:`struct xsdfec_ldpc_params <xsdfec_ldpc_params>`
- Add the LPDC Code Parameters using the ioctl :c:macro:`XSDFEC_ADD_LDPC_CODE_PARAMS`
- For the applied LPDC Code Parameter use the function :c:func:`xsdfec_calculate_shared_ldpc_table_entry_size` to calculate the size of shared LPDC code tables. This allows the user to determine the shared table usage so when selecting the table offsets for the next LDPC code parameters unused table areas can be selected.
- Repeat for each LDPC code parameter.
Adding LDPC codes can only be done if the following restrictions are met:
- The ``code`` member of :c:type:`struct xsdfec_config <xsdfec_config>` filled by the ioctl :c:macro:`XSDFEC_GET_CONFIG` indicates the SD-FEC core is configured as LDPC
- The ``code_wr_protect`` of :c:type:`struct xsdfec_config <xsdfec_config>` filled by the ioctl :c:macro:`XSDFEC_GET_CONFIG` indicates that write protection is not enabled
- The ``state`` member of struct :c:type:`xsdfec_status <xsdfec_status>` filled by the ioctl :c:macro:`XSDFEC_GET_STATUS` indicates the SD-FEC core has not started
Configuring the Turbo decode parameters is done by using the ioctl :c:macro:XSDFEC_SET_TURBO using auto-generated parameters to fill the :c:type:struct xsdfec_turbo <xsdfec_turbo> for the desired Turbo code.
Adding Turbo decode can only be done if the following restrictions are met:
- The ``code`` member of :c:type:`struct xsdfec_config <xsdfec_config>` filled by the ioctl :c:macro:`XSDFEC_GET_CONFIG` indicates the SD-FEC core is configured as TURBO
- The ``state`` member of struct :c:type:`xsdfec_status <xsdfec_status>` filled by the ioctl :c:macro:`XSDFEC_GET_STATUS` indicates the SD-FEC core has not STARTED
Enabling or disabling interrupts is done by using the ioctl :c:macro:XSDFEC_SET_IRQ. The members of the parameter passed, :c:type:struct xsdfec_irq <xsdfec_irq>, to the ioctl are used to set and clear different categories of interrupts. The category of interrupt is controlled as following:
enable_isr controls the tlast interruptsenable_ecc_isr controls the ECC interruptsIf the code member of :c:type:struct xsdfec_config <xsdfec_config> filled by the ioctl :c:macro:XSDFEC_GET_CONFIG indicates the SD-FEC core is configured as TURBO then the enabling ECC errors is not required.
Bypassing the SD-FEC is done by using the ioctl :c:macro:XSDFEC_SET_BYPASS
Bypassing the SD-FEC can only be done if the following restrictions are met:
- The ``state`` member of :c:type:`struct xsdfec_status <xsdfec_status>` filled by the ioctl :c:macro:`XSDFEC_GET_STATUS` indicates the SD-FEC core has not STARTED
Start the SD-FEC core by using the ioctl :c:macro:XSDFEC_START_DEV
Get the SD-FEC status of the device by using the ioctl :c:macro:XSDFEC_GET_STATUS, which will fill the :c:type:struct xsdfec_status <xsdfec_status>
- Use the poll system call to monitor for an interrupt. The poll system call waits for an interrupt to wake it up or times out if no interrupt occurs.
- On return Poll ``revents`` will indicate whether stats and/or state have been updated
- ``POLLPRI`` indicates a critical error and the user should use :c:macro:`XSDFEC_GET_STATUS` and :c:macro:`XSDFEC_GET_STATS` to confirm
- ``POLLRDNORM`` indicates a non-critical error has occurred and the user should use :c:macro:`XSDFEC_GET_STATS` to confirm
- Get stats by using the ioctl :c:macro:`XSDFEC_GET_STATS`
- For critical error the ``isr_err_count`` or ``uecc_count`` member of :c:type:`struct xsdfec_stats <xsdfec_stats>` is non-zero
- For non-critical errors the ``cecc_count`` member of :c:type:`struct xsdfec_stats <xsdfec_stats>` is non-zero
- Get state by using the ioctl :c:macro:`XSDFEC_GET_STATUS`
- For a critical error the ``state`` of :c:type:`xsdfec_status <xsdfec_status>` will indicate a Reset Is Required
- Clear stats by using the ioctl :c:macro:`XSDFEC_CLEAR_STATS`
If a critical error is detected where a reset is required. The application is required to call the ioctl :c:macro:XSDFEC_SET_DEFAULT_CONFIG, after the reset and it is not required to call the ioctl :c:macro:XSDFEC_STOP_DEV
Note: Using poll system call prevents busy looping using :c:macro:XSDFEC_GET_STATS and :c:macro:XSDFEC_GET_STATUS
Stop the device by using the ioctl :c:macro:XSDFEC_STOP_DEV
Load default configuration by using the ioctl :c:macro:XSDFEC_SET_DEFAULT_CONFIG to restore the driver.
Users should not duplicate SD-FEC device file handlers, for example fork() or dup() a process that has a created an SD-FEC file handler.
.. c:macro:: XSDFEC_START_DEV .. kernel-doc:: include/uapi/misc/xilinx_sdfec.h :doc: XSDFEC_START_DEV
.. c:macro:: XSDFEC_STOP_DEV .. kernel-doc:: include/uapi/misc/xilinx_sdfec.h :doc: XSDFEC_STOP_DEV
.. c:macro:: XSDFEC_GET_STATUS .. kernel-doc:: include/uapi/misc/xilinx_sdfec.h :doc: XSDFEC_GET_STATUS
.. c:macro:: XSDFEC_SET_IRQ .. kernel-doc:: include/uapi/misc/xilinx_sdfec.h :doc: XSDFEC_SET_IRQ
.. c:macro:: XSDFEC_SET_TURBO .. kernel-doc:: include/uapi/misc/xilinx_sdfec.h :doc: XSDFEC_SET_TURBO
.. c:macro:: XSDFEC_ADD_LDPC_CODE_PARAMS .. kernel-doc:: include/uapi/misc/xilinx_sdfec.h :doc: XSDFEC_ADD_LDPC_CODE_PARAMS
.. c:macro:: XSDFEC_GET_CONFIG .. kernel-doc:: include/uapi/misc/xilinx_sdfec.h :doc: XSDFEC_GET_CONFIG
.. c:macro:: XSDFEC_SET_ORDER .. kernel-doc:: include/uapi/misc/xilinx_sdfec.h :doc: XSDFEC_SET_ORDER
.. c:macro:: XSDFEC_SET_BYPASS .. kernel-doc:: include/uapi/misc/xilinx_sdfec.h :doc: XSDFEC_SET_BYPASS
.. c:macro:: XSDFEC_IS_ACTIVE .. kernel-doc:: include/uapi/misc/xilinx_sdfec.h :doc: XSDFEC_IS_ACTIVE
.. c:macro:: XSDFEC_CLEAR_STATS .. kernel-doc:: include/uapi/misc/xilinx_sdfec.h :doc: XSDFEC_CLEAR_STATS
.. c:macro:: XSDFEC_GET_STATS .. kernel-doc:: include/uapi/misc/xilinx_sdfec.h :doc: XSDFEC_GET_STATS
.. c:macro:: XSDFEC_SET_DEFAULT_CONFIG .. kernel-doc:: include/uapi/misc/xilinx_sdfec.h :doc: XSDFEC_SET_DEFAULT_CONFIG
.. kernel-doc:: include/uapi/misc/xilinx_sdfec.h :internal: