| // SPDX-License-Identifier: (GPL-2.0+ OR BSD-3-Clause) |
| /* Copyright 2021 NXP |
| * |
| * The Integrated Endpoint Register Block (IERB) is configured by pre-boot |
| * software and is supposed to be to ENETC what a NVRAM is to a 'real' PCIe |
| * card. Upon FLR, values from the IERB are transferred to the ENETC PFs, and |
| * are read-only in the PF memory space. |
| * |
| * This driver fixes up the power-on reset values for the ENETC shared FIFO, |
| * such that the TX and RX allocations are sufficient for jumbo frames, and |
| * that intelligent FIFO dropping is enabled before the internal data |
| * structures are corrupted. |
| * |
| * Even though not all ports might be used on a given board, we are not |
| * concerned with partitioning the FIFO, because the default values configure |
| * no strict reservations, so the entire FIFO can be used by the RX of a single |
| * port, or the TX of a single port. |
| */ |
| |
| #include <linux/io.h> |
| #include <linux/module.h> |
| #include <linux/of_device.h> |
| #include <linux/pci.h> |
| #include <linux/platform_device.h> |
| #include "enetc.h" |
| #include "enetc_ierb.h" |
| |
| /* IERB registers */ |
| #define ENETC_IERB_TXMBAR(port) (((port) * 0x100) + 0x8080) |
| #define ENETC_IERB_RXMBER(port) (((port) * 0x100) + 0x8090) |
| #define ENETC_IERB_RXMBLR(port) (((port) * 0x100) + 0x8094) |
| #define ENETC_IERB_RXBCR(port) (((port) * 0x100) + 0x80a0) |
| #define ENETC_IERB_TXBCR(port) (((port) * 0x100) + 0x80a8) |
| #define ENETC_IERB_FMBDTR 0xa000 |
| |
| #define ENETC_RESERVED_FOR_ICM 1024 |
| |
| struct enetc_ierb { |
| void __iomem *regs; |
| }; |
| |
| static void enetc_ierb_write(struct enetc_ierb *ierb, u32 offset, u32 val) |
| { |
| iowrite32(val, ierb->regs + offset); |
| } |
| |
| int enetc_ierb_register_pf(struct platform_device *pdev, |
| struct pci_dev *pf_pdev) |
| { |
| struct enetc_ierb *ierb = platform_get_drvdata(pdev); |
| int port = enetc_pf_to_port(pf_pdev); |
| u16 tx_credit, rx_credit, tx_alloc; |
| |
| if (port < 0) |
| return -ENODEV; |
| |
| if (!ierb) |
| return -EPROBE_DEFER; |
| |
| /* By default, it is recommended to set the Host Transfer Agent |
| * per port transmit byte credit to "1000 + max_frame_size/2". |
| * The power-on reset value (1800 bytes) is rounded up to the nearest |
| * 100 assuming a maximum frame size of 1536 bytes. |
| */ |
| tx_credit = roundup(1000 + ENETC_MAC_MAXFRM_SIZE / 2, 100); |
| |
| /* Internal memory allocated for transmit buffering is guaranteed but |
| * not reserved; i.e. if the total transmit allocation is not used, |
| * then the unused portion is not left idle, it can be used for receive |
| * buffering but it will be reclaimed, if required, from receive by |
| * intelligently dropping already stored receive frames in the internal |
| * memory to ensure that the transmit allocation is respected. |
| * |
| * PaTXMBAR must be set to a value larger than |
| * PaTXBCR + 2 * max_frame_size + 32 |
| * if frame preemption is not enabled, or to |
| * 2 * PaTXBCR + 2 * p_max_frame_size (pMAC maximum frame size) + |
| * 2 * np_max_frame_size (eMAC maximum frame size) + 64 |
| * if frame preemption is enabled. |
| */ |
| tx_alloc = roundup(2 * tx_credit + 4 * ENETC_MAC_MAXFRM_SIZE + 64, 16); |
| |
| /* Initial credits, in units of 8 bytes, to the Ingress Congestion |
| * Manager for the maximum amount of bytes the port is allocated for |
| * pending traffic. |
| * It is recommended to set the initial credits to 2 times the maximum |
| * frame size (2 frames of maximum size). |
| */ |
| rx_credit = DIV_ROUND_UP(ENETC_MAC_MAXFRM_SIZE * 2, 8); |
| |
| enetc_ierb_write(ierb, ENETC_IERB_TXBCR(port), tx_credit); |
| enetc_ierb_write(ierb, ENETC_IERB_TXMBAR(port), tx_alloc); |
| enetc_ierb_write(ierb, ENETC_IERB_RXBCR(port), rx_credit); |
| |
| return 0; |
| } |
| EXPORT_SYMBOL(enetc_ierb_register_pf); |
| |
| static int enetc_ierb_probe(struct platform_device *pdev) |
| { |
| struct enetc_ierb *ierb; |
| void __iomem *regs; |
| |
| ierb = devm_kzalloc(&pdev->dev, sizeof(*ierb), GFP_KERNEL); |
| if (!ierb) |
| return -ENOMEM; |
| |
| regs = devm_platform_get_and_ioremap_resource(pdev, 0, NULL); |
| if (IS_ERR(regs)) |
| return PTR_ERR(regs); |
| |
| ierb->regs = regs; |
| |
| /* Free buffer depletion threshold in bytes. |
| * This sets the minimum amount of free buffer memory that should be |
| * maintained in the datapath sub system, and when the amount of free |
| * buffer memory falls below this threshold, a depletion indication is |
| * asserted, which may trigger "intelligent drop" frame releases from |
| * the ingress queues in the ICM. |
| * It is recommended to set the free buffer depletion threshold to 1024 |
| * bytes, since the ICM needs some FIFO memory for its own use. |
| */ |
| enetc_ierb_write(ierb, ENETC_IERB_FMBDTR, ENETC_RESERVED_FOR_ICM); |
| |
| platform_set_drvdata(pdev, ierb); |
| |
| return 0; |
| } |
| |
| static int enetc_ierb_remove(struct platform_device *pdev) |
| { |
| return 0; |
| } |
| |
| static const struct of_device_id enetc_ierb_match[] = { |
| { .compatible = "fsl,ls1028a-enetc-ierb", }, |
| {}, |
| }; |
| MODULE_DEVICE_TABLE(of, enetc_ierb_match); |
| |
| static struct platform_driver enetc_ierb_driver = { |
| .driver = { |
| .name = "fsl-enetc-ierb", |
| .of_match_table = enetc_ierb_match, |
| }, |
| .probe = enetc_ierb_probe, |
| .remove = enetc_ierb_remove, |
| }; |
| |
| module_platform_driver(enetc_ierb_driver); |
| |
| MODULE_DESCRIPTION("NXP ENETC IERB"); |
| MODULE_LICENSE("Dual BSD/GPL"); |