mirror of
				https://source.denx.de/u-boot/u-boot.git
				synced 2025-10-31 00:11:51 +01:00 
			
		
		
		
	Move this out of the common header and include it only where needed. In a number of cases this requires adding "struct udevice;" to avoid adding another large header or in other cases replacing / adding missing header files that had been pulled in, very indirectly. Finally, we have a few cases where we did not need to include <asm/global_data.h> at all, so remove that include. Signed-off-by: Simon Glass <sjg@chromium.org> Signed-off-by: Tom Rini <trini@konsulko.com>
		
			
				
	
	
		
			179 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			179 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0+
 | |
| /*
 | |
|  * Copyright (c) 2019
 | |
|  * Written by Ramon Fried <ramon.fried@gmail.com>
 | |
|  */
 | |
| 
 | |
| #include <common.h>
 | |
| #include <dm.h>
 | |
| #include <errno.h>
 | |
| #include <pci_ep.h>
 | |
| #include <asm/global_data.h>
 | |
| #include <linux/sizes.h>
 | |
| #include <linux/log2.h>
 | |
| #include "pcie-cadence.h"
 | |
| 
 | |
| DECLARE_GLOBAL_DATA_PTR;
 | |
| 
 | |
| static int cdns_write_header(struct udevice *dev, uint fn,
 | |
| 			     struct pci_ep_header *hdr)
 | |
| {
 | |
| 	struct cdns_pcie *pcie = dev_get_priv(dev);
 | |
| 
 | |
| 	cdns_pcie_ep_fn_writew(pcie, fn, PCI_DEVICE_ID, hdr->deviceid);
 | |
| 	cdns_pcie_ep_fn_writeb(pcie, fn, PCI_REVISION_ID, hdr->revid);
 | |
| 	cdns_pcie_ep_fn_writeb(pcie, fn, PCI_CLASS_PROG,
 | |
| 			       hdr->progif_code);
 | |
| 	cdns_pcie_ep_fn_writew(pcie, fn, PCI_CLASS_DEVICE,
 | |
| 			       hdr->subclass_code |
 | |
| 			       hdr->baseclass_code << 8);
 | |
| 	cdns_pcie_ep_fn_writeb(pcie, fn, PCI_CACHE_LINE_SIZE,
 | |
| 			       hdr->cache_line_size);
 | |
| 	cdns_pcie_ep_fn_writew(pcie, fn, PCI_SUBSYSTEM_ID,
 | |
| 			       hdr->subsys_id);
 | |
| 	cdns_pcie_ep_fn_writeb(pcie, fn, PCI_INTERRUPT_PIN,
 | |
| 			       hdr->interrupt_pin);
 | |
| 
 | |
| 	/*
 | |
| 	 * Vendor ID can only be modified from function 0, all other functions
 | |
| 	 * use the same vendor ID as function 0.
 | |
| 	 */
 | |
| 	if (fn == 0) {
 | |
| 		/* Update the vendor IDs. */
 | |
| 		u32 id = CDNS_PCIE_LM_ID_VENDOR(hdr->vendorid) |
 | |
| 			 CDNS_PCIE_LM_ID_SUBSYS(hdr->subsys_vendor_id);
 | |
| 
 | |
| 		cdns_pcie_writel(pcie, CDNS_PCIE_LM_ID, id);
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int cdns_set_bar(struct udevice *dev, uint fn, struct pci_bar *ep_bar)
 | |
| {
 | |
| 	struct cdns_pcie *pcie = dev_get_priv(dev);
 | |
| 	dma_addr_t bar_phys = ep_bar->phys_addr;
 | |
| 	enum pci_barno bar = ep_bar->barno;
 | |
| 	int flags = ep_bar->flags;
 | |
| 	u32 addr0, addr1, reg, cfg, b, aperture, ctrl;
 | |
| 	u64 sz;
 | |
| 
 | |
| 	/* BAR size is 2^(aperture + 7) */
 | |
| 	sz = max_t(size_t, ep_bar->size, CDNS_PCIE_EP_MIN_APERTURE);
 | |
| 	/*
 | |
| 	 * roundup_pow_of_two() returns an unsigned long, which is not suited
 | |
| 	 * for 64bit values.
 | |
| 	 */
 | |
| 	sz = 1ULL << fls64(sz - 1);
 | |
| 	aperture = ilog2(sz) - 7; /* 128B -> 0, 256B -> 1, 512B -> 2, ... */
 | |
| 
 | |
| 	if ((flags & PCI_BASE_ADDRESS_SPACE) == PCI_BASE_ADDRESS_SPACE_IO) {
 | |
| 		ctrl = CDNS_PCIE_LM_BAR_CFG_CTRL_IO_32BITS;
 | |
| 	} else {
 | |
| 		bool is_prefetch = !!(flags & PCI_BASE_ADDRESS_MEM_PREFETCH);
 | |
| 		bool is_64bits = (sz > SZ_2G) |
 | |
| 			!!(ep_bar->flags & PCI_BASE_ADDRESS_MEM_TYPE_64);
 | |
| 
 | |
| 		if (is_64bits && (bar & 1))
 | |
| 			return -EINVAL;
 | |
| 
 | |
| 		if (is_64bits && !(flags & PCI_BASE_ADDRESS_MEM_TYPE_64))
 | |
| 			ep_bar->flags |= PCI_BASE_ADDRESS_MEM_TYPE_64;
 | |
| 
 | |
| 		if (is_64bits && is_prefetch)
 | |
| 			ctrl = CDNS_PCIE_LM_BAR_CFG_CTRL_PREFETCH_MEM_64BITS;
 | |
| 		else if (is_prefetch)
 | |
| 			ctrl = CDNS_PCIE_LM_BAR_CFG_CTRL_PREFETCH_MEM_32BITS;
 | |
| 		else if (is_64bits)
 | |
| 			ctrl = CDNS_PCIE_LM_BAR_CFG_CTRL_MEM_64BITS;
 | |
| 		else
 | |
| 			ctrl = CDNS_PCIE_LM_BAR_CFG_CTRL_MEM_32BITS;
 | |
| 	}
 | |
| 
 | |
| 	addr0 = lower_32_bits(bar_phys);
 | |
| 	addr1 = upper_32_bits(bar_phys);
 | |
| 	cdns_pcie_writel(pcie, CDNS_PCIE_AT_IB_EP_FUNC_BAR_ADDR0(fn, bar),
 | |
| 			 addr0);
 | |
| 	cdns_pcie_writel(pcie, CDNS_PCIE_AT_IB_EP_FUNC_BAR_ADDR1(fn, bar),
 | |
| 			 addr1);
 | |
| 
 | |
| 	if (bar < BAR_4) {
 | |
| 		reg = CDNS_PCIE_LM_EP_FUNC_BAR_CFG0(fn);
 | |
| 		b = bar;
 | |
| 	} else {
 | |
| 		reg = CDNS_PCIE_LM_EP_FUNC_BAR_CFG1(fn);
 | |
| 		b = bar - BAR_4;
 | |
| 	}
 | |
| 
 | |
| 	cfg = cdns_pcie_readl(pcie, reg);
 | |
| 	cfg &= ~(CDNS_PCIE_LM_EP_FUNC_BAR_CFG_BAR_APERTURE_MASK(b) |
 | |
| 		 CDNS_PCIE_LM_EP_FUNC_BAR_CFG_BAR_CTRL_MASK(b));
 | |
| 	cfg |= (CDNS_PCIE_LM_EP_FUNC_BAR_CFG_BAR_APERTURE(b, aperture) |
 | |
| 		CDNS_PCIE_LM_EP_FUNC_BAR_CFG_BAR_CTRL(b, ctrl));
 | |
| 	cdns_pcie_writel(pcie, reg, cfg);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int cdns_set_msi(struct udevice *dev, uint fn, uint mmc)
 | |
| {
 | |
| 	struct cdns_pcie *pcie = dev_get_priv(dev);
 | |
| 	u32 cap = CDNS_PCIE_EP_FUNC_MSI_CAP_OFFSET;
 | |
| 
 | |
| 	/*
 | |
| 	 * Set the Multiple Message Capable bitfield into the Message Control
 | |
| 	 * register.
 | |
| 	 */
 | |
| 	u16 flags;
 | |
| 
 | |
| 	flags = cdns_pcie_ep_fn_readw(pcie, fn, cap + PCI_MSI_FLAGS);
 | |
| 	flags = (flags & ~PCI_MSI_FLAGS_QMASK) | (mmc << 1);
 | |
| 	flags |= PCI_MSI_FLAGS_64BIT;
 | |
| 	flags &= ~PCI_MSI_FLAGS_MASKBIT;
 | |
| 	cdns_pcie_ep_fn_writew(pcie, fn, cap + PCI_MSI_FLAGS, flags);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static struct pci_ep_ops cdns_pci_ep_ops = {
 | |
| 	.write_header = cdns_write_header,
 | |
| 	.set_bar = cdns_set_bar,
 | |
| 	.set_msi = cdns_set_msi,
 | |
| };
 | |
| 
 | |
| static int cdns_pci_ep_probe(struct udevice *dev)
 | |
| {
 | |
| 	struct cdns_pcie *pdata = dev_get_priv(dev);
 | |
| 
 | |
| 	pdata->reg_base = dev_read_addr_ptr(dev);
 | |
| 	if (!pdata->reg_base)
 | |
| 		return -ENOMEM;
 | |
| 
 | |
| 	pdata->max_functions = fdtdec_get_int(gd->fdt_blob, dev_of_offset(dev),
 | |
| 					      "max-functions", 1);
 | |
| 	pdata->max_regions = fdtdec_get_int(gd->fdt_blob, dev_of_offset(dev),
 | |
| 					    "cdns,max-outbound-regions", 8);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int cdns_pci_ep_remove(struct udevice *dev)
 | |
| {
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| const struct udevice_id cadence_pci_ep_of_match[] = {
 | |
| 	{ .compatible = "cdns,cdns-pcie-ep" },
 | |
| 	{ }
 | |
| };
 | |
| 
 | |
| U_BOOT_DRIVER(cdns_pcie) = {
 | |
| 	.name	= "cdns,pcie-ep",
 | |
| 	.id	= UCLASS_PCI_EP,
 | |
| 	.of_match = cadence_pci_ep_of_match,
 | |
| 	.ops = &cdns_pci_ep_ops,
 | |
| 	.probe = cdns_pci_ep_probe,
 | |
| 	.remove = cdns_pci_ep_remove,
 | |
| 	.priv_auto	= sizeof(struct cdns_pcie),
 | |
| };
 |