2018-01-26 14:22:04 -06:00
// SPDX-License-Identifier: GPL-2.0+
2014-08-20 21:56:02 +05:30
/*
* PCIe host controller driver for Xilinx AXI PCIe Bridge
*
* Copyright ( c ) 2012 - 2014 Xilinx , Inc .
*
* Based on the Tegra PCIe driver
*
2017-09-01 16:35:50 -05:00
* Bits taken from Synopsys DesignWare Host controller driver and
2014-08-20 21:56:02 +05:30
* ARM PCI Host generic driver .
*/
# include <linux/interrupt.h>
# include <linux/irq.h>
# include <linux/irqdomain.h>
# include <linux/kernel.h>
2016-08-24 16:57:49 -04:00
# include <linux/init.h>
2014-08-20 21:56:02 +05:30
# include <linux/msi.h>
# include <linux/of_address.h>
# include <linux/of_pci.h>
# include <linux/of_platform.h>
# include <linux/of_irq.h>
# include <linux/pci.h>
2020-11-29 23:07:39 +00:00
# include <linux/pci-ecam.h>
2014-08-20 21:56:02 +05:30
# include <linux/platform_device.h>
2018-05-11 12:15:30 -05:00
# include "../pci.h"
2014-08-20 21:56:02 +05:30
/* Register definitions */
# define XILINX_PCIE_REG_BIR 0x00000130
# define XILINX_PCIE_REG_IDR 0x00000138
# define XILINX_PCIE_REG_IMR 0x0000013c
# define XILINX_PCIE_REG_PSCR 0x00000144
# define XILINX_PCIE_REG_RPSC 0x00000148
# define XILINX_PCIE_REG_MSIBASE1 0x0000014c
# define XILINX_PCIE_REG_MSIBASE2 0x00000150
# define XILINX_PCIE_REG_RPEFR 0x00000154
# define XILINX_PCIE_REG_RPIFR1 0x00000158
# define XILINX_PCIE_REG_RPIFR2 0x0000015c
/* Interrupt registers definitions */
# define XILINX_PCIE_INTR_LINK_DOWN BIT(0)
# define XILINX_PCIE_INTR_ECRC_ERR BIT(1)
# define XILINX_PCIE_INTR_STR_ERR BIT(2)
# define XILINX_PCIE_INTR_HOT_RESET BIT(3)
# define XILINX_PCIE_INTR_CFG_TIMEOUT BIT(8)
# define XILINX_PCIE_INTR_CORRECTABLE BIT(9)
# define XILINX_PCIE_INTR_NONFATAL BIT(10)
# define XILINX_PCIE_INTR_FATAL BIT(11)
# define XILINX_PCIE_INTR_INTX BIT(16)
# define XILINX_PCIE_INTR_MSI BIT(17)
# define XILINX_PCIE_INTR_SLV_UNSUPP BIT(20)
# define XILINX_PCIE_INTR_SLV_UNEXP BIT(21)
# define XILINX_PCIE_INTR_SLV_COMPL BIT(22)
# define XILINX_PCIE_INTR_SLV_ERRP BIT(23)
# define XILINX_PCIE_INTR_SLV_CMPABT BIT(24)
# define XILINX_PCIE_INTR_SLV_ILLBUR BIT(25)
# define XILINX_PCIE_INTR_MST_DECERR BIT(26)
# define XILINX_PCIE_INTR_MST_SLVERR BIT(27)
# define XILINX_PCIE_INTR_MST_ERRP BIT(28)
# define XILINX_PCIE_IMR_ALL_MASK 0x1FF30FED
2017-08-15 16:25:25 -05:00
# define XILINX_PCIE_IMR_ENABLE_MASK 0x1FF30F0D
2014-08-20 21:56:02 +05:30
# define XILINX_PCIE_IDR_ALL_MASK 0xFFFFFFFF
/* Root Port Error FIFO Read Register definitions */
# define XILINX_PCIE_RPEFR_ERR_VALID BIT(18)
# define XILINX_PCIE_RPEFR_REQ_ID GENMASK(15, 0)
# define XILINX_PCIE_RPEFR_ALL_MASK 0xFFFFFFFF
/* Root Port Interrupt FIFO Read Register 1 definitions */
# define XILINX_PCIE_RPIFR1_INTR_VALID BIT(31)
# define XILINX_PCIE_RPIFR1_MSI_INTR BIT(30)
# define XILINX_PCIE_RPIFR1_INTR_MASK GENMASK(28, 27)
# define XILINX_PCIE_RPIFR1_ALL_MASK 0xFFFFFFFF
# define XILINX_PCIE_RPIFR1_INTR_SHIFT 27
/* Bridge Info Register definitions */
# define XILINX_PCIE_BIR_ECAM_SZ_MASK GENMASK(18, 16)
# define XILINX_PCIE_BIR_ECAM_SZ_SHIFT 16
/* Root Port Interrupt FIFO Read Register 2 definitions */
# define XILINX_PCIE_RPIFR2_MSG_DATA GENMASK(15, 0)
/* Root Port Status/control Register definitions */
# define XILINX_PCIE_REG_RPSC_BEN BIT(0)
/* Phy Status/Control Register definitions */
# define XILINX_PCIE_REG_PSCR_LNKUP BIT(11)
/* Number of MSI IRQs */
# define XILINX_NUM_MSI_IRQS 128
/**
* struct xilinx_pcie_port - PCIe port information
* @ reg_base : IO Mapped Register Base
* @ dev : Device pointer
2021-03-30 16:11:36 +01:00
* @ msi_map : Bitmap of allocated MSIs
* @ map_lock : Mutex protecting the MSI allocation
2016-09-01 15:44:41 +05:30
* @ msi_domain : MSI IRQ domain pointer
* @ leg_domain : Legacy IRQ domain pointer
2014-08-20 21:56:02 +05:30
* @ resources : Bus Resources
*/
struct xilinx_pcie_port {
void __iomem * reg_base ;
struct device * dev ;
2021-03-30 16:11:36 +01:00
unsigned long msi_map [ BITS_TO_LONGS ( XILINX_NUM_MSI_IRQS ) ] ;
struct mutex map_lock ;
2016-09-01 15:44:41 +05:30
struct irq_domain * msi_domain ;
struct irq_domain * leg_domain ;
2014-08-20 21:56:02 +05:30
struct list_head resources ;
} ;
static inline u32 pcie_read ( struct xilinx_pcie_port * port , u32 reg )
{
return readl ( port - > reg_base + reg ) ;
}
static inline void pcie_write ( struct xilinx_pcie_port * port , u32 val , u32 reg )
{
writel ( val , port - > reg_base + reg ) ;
}
2017-11-09 16:19:47 -06:00
static inline bool xilinx_pcie_link_up ( struct xilinx_pcie_port * port )
2014-08-20 21:56:02 +05:30
{
return ( pcie_read ( port , XILINX_PCIE_REG_PSCR ) &
XILINX_PCIE_REG_PSCR_LNKUP ) ? 1 : 0 ;
}
/**
* xilinx_pcie_clear_err_interrupts - Clear Error Interrupts
* @ port : PCIe port information
*/
static void xilinx_pcie_clear_err_interrupts ( struct xilinx_pcie_port * port )
{
2016-10-06 13:44:42 -05:00
struct device * dev = port - > dev ;
2015-01-13 15:20:05 +01:00
unsigned long val = pcie_read ( port , XILINX_PCIE_REG_RPEFR ) ;
2014-08-20 21:56:02 +05:30
if ( val & XILINX_PCIE_RPEFR_ERR_VALID ) {
2016-10-06 13:44:42 -05:00
dev_dbg ( dev , " Requester ID %lu \n " ,
2014-08-20 21:56:02 +05:30
val & XILINX_PCIE_RPEFR_REQ_ID ) ;
pcie_write ( port , XILINX_PCIE_RPEFR_ALL_MASK ,
XILINX_PCIE_REG_RPEFR ) ;
}
}
/**
* xilinx_pcie_valid_device - Check if a valid device is present on bus
* @ bus : PCI Bus structure
* @ devfn : device / function
*
* Return : ' true ' on success and ' false ' if invalid device is found
*/
static bool xilinx_pcie_valid_device ( struct pci_bus * bus , unsigned int devfn )
{
2016-02-11 21:58:08 +05:30
struct xilinx_pcie_port * port = bus - > sysdata ;
2014-08-20 21:56:02 +05:30
/* Check if link is up when trying to access downstream ports */
2020-07-21 20:25:03 -06:00
if ( ! pci_is_root_bus ( bus ) ) {
2017-11-09 16:19:47 -06:00
if ( ! xilinx_pcie_link_up ( port ) )
2014-08-20 21:56:02 +05:30
return false ;
2020-07-21 20:25:03 -06:00
} else if ( devfn > 0 ) {
/* Only one device down on each root port */
2014-08-20 21:56:02 +05:30
return false ;
2020-07-21 20:25:03 -06:00
}
2014-08-20 21:56:02 +05:30
return true ;
}
/**
2015-01-09 20:34:50 -06:00
* xilinx_pcie_map_bus - Get configuration base
2014-08-20 21:56:02 +05:30
* @ bus : PCI Bus structure
* @ devfn : Device / function
* @ where : Offset from base
*
* Return : Base address of the configuration space needed to be
* accessed .
*/
2015-01-09 20:34:50 -06:00
static void __iomem * xilinx_pcie_map_bus ( struct pci_bus * bus ,
unsigned int devfn , int where )
2014-08-20 21:56:02 +05:30
{
2016-02-11 21:58:08 +05:30
struct xilinx_pcie_port * port = bus - > sysdata ;
2014-08-20 21:56:02 +05:30
2015-01-09 20:34:50 -06:00
if ( ! xilinx_pcie_valid_device ( bus , devfn ) )
return NULL ;
2020-11-29 23:07:39 +00:00
return port - > reg_base + PCIE_ECAM_OFFSET ( bus - > number , devfn , where ) ;
2014-08-20 21:56:02 +05:30
}
/* PCIe operations */
static struct pci_ops xilinx_pcie_ops = {
2015-01-09 20:34:50 -06:00
. map_bus = xilinx_pcie_map_bus ,
. read = pci_generic_config_read ,
. write = pci_generic_config_write ,
2014-08-20 21:56:02 +05:30
} ;
/* MSI functions */
2021-03-30 16:11:36 +01:00
static void xilinx_msi_top_irq_ack ( struct irq_data * d )
2014-08-20 21:56:02 +05:30
{
2021-03-30 16:11:36 +01:00
/*
* xilinx_pcie_intr_handler ( ) will have performed the Ack .
* Eventually , this should be fixed and the Ack be moved in
* the respective callbacks for INTx and MSI .
*/
2014-08-20 21:56:02 +05:30
}
2021-03-30 16:11:36 +01:00
static struct irq_chip xilinx_msi_top_chip = {
. name = " PCIe MSI " ,
. irq_ack = xilinx_msi_top_irq_ack ,
} ;
2014-08-20 21:56:02 +05:30
2021-03-30 16:11:36 +01:00
static int xilinx_msi_set_affinity ( struct irq_data * d , const struct cpumask * mask , bool force )
{
return - EINVAL ;
2014-08-20 21:56:02 +05:30
}
2021-03-30 16:11:36 +01:00
static void xilinx_compose_msi_msg ( struct irq_data * data , struct msi_msg * msg )
2014-08-20 21:56:02 +05:30
{
2021-03-30 16:11:36 +01:00
struct xilinx_pcie_port * pcie = irq_data_get_irq_chip_data ( data ) ;
phys_addr_t pa = ALIGN_DOWN ( virt_to_phys ( pcie ) , SZ_4K ) ;
msg - > address_lo = lower_32_bits ( pa ) ;
msg - > address_hi = upper_32_bits ( pa ) ;
msg - > data = data - > hwirq ;
2014-08-20 21:56:02 +05:30
}
2021-03-30 16:11:36 +01:00
static struct irq_chip xilinx_msi_bottom_chip = {
. name = " Xilinx MSI " ,
. irq_set_affinity = xilinx_msi_set_affinity ,
. irq_compose_msi_msg = xilinx_compose_msi_msg ,
} ;
static int xilinx_msi_domain_alloc ( struct irq_domain * domain , unsigned int virq ,
unsigned int nr_irqs , void * args )
2014-08-20 21:56:02 +05:30
{
2021-03-30 16:11:36 +01:00
struct xilinx_pcie_port * port = domain - > host_data ;
int hwirq , i ;
mutex_lock ( & port - > map_lock ) ;
hwirq = bitmap_find_free_region ( port - > msi_map , XILINX_NUM_MSI_IRQS , order_base_2 ( nr_irqs ) ) ;
mutex_unlock ( & port - > map_lock ) ;
2014-08-20 21:56:02 +05:30
2014-09-09 15:11:50 +03:00
if ( hwirq < 0 )
2021-03-30 16:11:36 +01:00
return - ENOSPC ;
2014-08-20 21:56:02 +05:30
2021-03-30 16:11:36 +01:00
for ( i = 0 ; i < nr_irqs ; i + + )
irq_domain_set_info ( domain , virq + i , hwirq + i ,
& xilinx_msi_bottom_chip , domain - > host_data ,
handle_edge_irq , NULL , NULL ) ;
2014-08-20 21:56:02 +05:30
2021-03-30 16:11:36 +01:00
return 0 ;
}
2014-08-20 21:56:02 +05:30
2021-03-30 16:11:36 +01:00
static void xilinx_msi_domain_free ( struct irq_domain * domain , unsigned int virq ,
unsigned int nr_irqs )
{
struct irq_data * d = irq_domain_get_irq_data ( domain , virq ) ;
struct xilinx_pcie_port * port = domain - > host_data ;
2014-08-20 21:56:02 +05:30
2021-03-30 16:11:36 +01:00
mutex_lock ( & port - > map_lock ) ;
2014-08-20 21:56:02 +05:30
2021-03-30 16:11:36 +01:00
bitmap_release_region ( port - > msi_map , d - > hwirq , order_base_2 ( nr_irqs ) ) ;
2014-08-20 21:56:02 +05:30
2021-03-30 16:11:36 +01:00
mutex_unlock ( & port - > map_lock ) ;
2014-08-20 21:56:02 +05:30
}
2021-03-30 16:11:36 +01:00
static const struct irq_domain_ops xilinx_msi_domain_ops = {
. alloc = xilinx_msi_domain_alloc ,
. free = xilinx_msi_domain_free ,
2014-08-20 21:56:02 +05:30
} ;
2021-03-30 16:11:36 +01:00
static struct msi_domain_info xilinx_msi_info = {
. flags = ( MSI_FLAG_USE_DEF_DOM_OPS | MSI_FLAG_USE_DEF_CHIP_OPS ) ,
. chip = & xilinx_msi_top_chip ,
2014-08-20 21:56:02 +05:30
} ;
2021-03-30 16:11:36 +01:00
static int xilinx_allocate_msi_domains ( struct xilinx_pcie_port * pcie )
2014-08-20 21:56:02 +05:30
{
2021-03-30 16:11:36 +01:00
struct fwnode_handle * fwnode = dev_fwnode ( pcie - > dev ) ;
struct irq_domain * parent ;
parent = irq_domain_create_linear ( fwnode , XILINX_NUM_MSI_IRQS ,
& xilinx_msi_domain_ops , pcie ) ;
if ( ! parent ) {
dev_err ( pcie - > dev , " failed to create IRQ domain \n " ) ;
return - ENOMEM ;
}
irq_domain_update_bus_token ( parent , DOMAIN_BUS_NEXUS ) ;
pcie - > msi_domain = pci_msi_create_irq_domain ( fwnode , & xilinx_msi_info , parent ) ;
if ( ! pcie - > msi_domain ) {
dev_err ( pcie - > dev , " failed to create MSI domain \n " ) ;
irq_domain_remove ( parent ) ;
return - ENOMEM ;
}
2014-08-20 21:56:02 +05:30
return 0 ;
}
2021-03-30 16:11:36 +01:00
static void xilinx_free_msi_domains ( struct xilinx_pcie_port * pcie )
2014-08-20 21:56:02 +05:30
{
2021-03-30 16:11:36 +01:00
struct irq_domain * parent = pcie - > msi_domain - > parent ;
2019-03-25 17:19:09 -05:00
2021-03-30 16:11:36 +01:00
irq_domain_remove ( pcie - > msi_domain ) ;
irq_domain_remove ( parent ) ;
2014-08-20 21:56:02 +05:30
}
/* INTx Functions */
/**
* xilinx_pcie_intx_map - Set the handler for the INTx and mark IRQ as valid
* @ domain : IRQ domain
* @ irq : Virtual IRQ number
* @ hwirq : HW interrupt number
*
* Return : Always returns 0.
*/
static int xilinx_pcie_intx_map ( struct irq_domain * domain , unsigned int irq ,
irq_hw_number_t hwirq )
{
irq_set_chip_and_handler ( irq , & dummy_irq_chip , handle_simple_irq ) ;
irq_set_chip_data ( irq , domain - > host_data ) ;
return 0 ;
}
/* INTx IRQ Domain operations */
static const struct irq_domain_ops intx_domain_ops = {
. map = xilinx_pcie_intx_map ,
2017-08-15 16:25:08 -05:00
. xlate = pci_irqd_intx_xlate ,
2014-08-20 21:56:02 +05:30
} ;
/* PCIe HW Functions */
/**
* xilinx_pcie_intr_handler - Interrupt Service Handler
* @ irq : IRQ number
* @ data : PCIe port information
*
* Return : IRQ_HANDLED on success and IRQ_NONE on failure
*/
static irqreturn_t xilinx_pcie_intr_handler ( int irq , void * data )
{
struct xilinx_pcie_port * port = ( struct xilinx_pcie_port * ) data ;
2016-10-06 13:44:42 -05:00
struct device * dev = port - > dev ;
2017-08-15 16:25:21 -05:00
u32 val , mask , status ;
2014-08-20 21:56:02 +05:30
/* Read interrupt decode and mask registers */
val = pcie_read ( port , XILINX_PCIE_REG_IDR ) ;
mask = pcie_read ( port , XILINX_PCIE_REG_IMR ) ;
status = val & mask ;
if ( ! status )
return IRQ_NONE ;
if ( status & XILINX_PCIE_INTR_LINK_DOWN )
2016-10-06 13:44:42 -05:00
dev_warn ( dev , " Link Down \n " ) ;
2014-08-20 21:56:02 +05:30
if ( status & XILINX_PCIE_INTR_ECRC_ERR )
2016-10-06 13:44:42 -05:00
dev_warn ( dev , " ECRC failed \n " ) ;
2014-08-20 21:56:02 +05:30
if ( status & XILINX_PCIE_INTR_STR_ERR )
2016-10-06 13:44:42 -05:00
dev_warn ( dev , " Streaming error \n " ) ;
2014-08-20 21:56:02 +05:30
if ( status & XILINX_PCIE_INTR_HOT_RESET )
2016-10-06 13:44:42 -05:00
dev_info ( dev , " Hot reset \n " ) ;
2014-08-20 21:56:02 +05:30
if ( status & XILINX_PCIE_INTR_CFG_TIMEOUT )
2016-10-06 13:44:42 -05:00
dev_warn ( dev , " ECAM access timeout \n " ) ;
2014-08-20 21:56:02 +05:30
if ( status & XILINX_PCIE_INTR_CORRECTABLE ) {
2016-10-06 13:44:42 -05:00
dev_warn ( dev , " Correctable error message \n " ) ;
2014-08-20 21:56:02 +05:30
xilinx_pcie_clear_err_interrupts ( port ) ;
}
if ( status & XILINX_PCIE_INTR_NONFATAL ) {
2016-10-06 13:44:42 -05:00
dev_warn ( dev , " Non fatal error message \n " ) ;
2014-08-20 21:56:02 +05:30
xilinx_pcie_clear_err_interrupts ( port ) ;
}
if ( status & XILINX_PCIE_INTR_FATAL ) {
2016-10-06 13:44:42 -05:00
dev_warn ( dev , " Fatal error message \n " ) ;
2014-08-20 21:56:02 +05:30
xilinx_pcie_clear_err_interrupts ( port ) ;
}
2017-08-15 16:25:21 -05:00
if ( status & ( XILINX_PCIE_INTR_INTX | XILINX_PCIE_INTR_MSI ) ) {
2021-08-02 17:26:19 +01:00
struct irq_domain * domain ;
2021-03-30 16:11:36 +01:00
2014-08-20 21:56:02 +05:30
val = pcie_read ( port , XILINX_PCIE_REG_RPIFR1 ) ;
/* Check whether interrupt valid */
if ( ! ( val & XILINX_PCIE_RPIFR1_INTR_VALID ) ) {
2016-10-06 13:44:42 -05:00
dev_warn ( dev , " RP Intr FIFO1 read error \n " ) ;
2016-09-01 15:44:42 +05:30
goto error ;
2014-08-20 21:56:02 +05:30
}
2017-08-15 16:25:21 -05:00
/* Decode the IRQ number */
if ( val & XILINX_PCIE_RPIFR1_MSI_INTR ) {
val = pcie_read ( port , XILINX_PCIE_REG_RPIFR2 ) &
XILINX_PCIE_RPIFR2_MSG_DATA ;
2021-08-02 17:26:19 +01:00
domain = port - > msi_domain - > parent ;
2017-08-15 16:25:21 -05:00
} else {
2017-08-15 16:25:08 -05:00
val = ( val & XILINX_PCIE_RPIFR1_INTR_MASK ) > >
XILINX_PCIE_RPIFR1_INTR_SHIFT ;
2021-08-02 17:26:19 +01:00
domain = port - > leg_domain ;
2015-07-07 17:54:19 +01:00
}
2014-08-20 21:56:02 +05:30
2017-08-15 16:25:21 -05:00
/* Clear interrupt FIFO register 1 */
pcie_write ( port , XILINX_PCIE_RPIFR1_ALL_MASK ,
XILINX_PCIE_REG_RPIFR1 ) ;
2014-08-20 21:56:02 +05:30
2021-08-02 17:26:19 +01:00
generic_handle_domain_irq ( domain , val ) ;
2014-08-20 21:56:02 +05:30
}
if ( status & XILINX_PCIE_INTR_SLV_UNSUPP )
2016-10-06 13:44:42 -05:00
dev_warn ( dev , " Slave unsupported request \n " ) ;
2014-08-20 21:56:02 +05:30
if ( status & XILINX_PCIE_INTR_SLV_UNEXP )
2016-10-06 13:44:42 -05:00
dev_warn ( dev , " Slave unexpected completion \n " ) ;
2014-08-20 21:56:02 +05:30
if ( status & XILINX_PCIE_INTR_SLV_COMPL )
2016-10-06 13:44:42 -05:00
dev_warn ( dev , " Slave completion timeout \n " ) ;
2014-08-20 21:56:02 +05:30
if ( status & XILINX_PCIE_INTR_SLV_ERRP )
2016-10-06 13:44:42 -05:00
dev_warn ( dev , " Slave Error Poison \n " ) ;
2014-08-20 21:56:02 +05:30
if ( status & XILINX_PCIE_INTR_SLV_CMPABT )
2016-10-06 13:44:42 -05:00
dev_warn ( dev , " Slave Completer Abort \n " ) ;
2014-08-20 21:56:02 +05:30
if ( status & XILINX_PCIE_INTR_SLV_ILLBUR )
2016-10-06 13:44:42 -05:00
dev_warn ( dev , " Slave Illegal Burst \n " ) ;
2014-08-20 21:56:02 +05:30
if ( status & XILINX_PCIE_INTR_MST_DECERR )
2016-10-06 13:44:42 -05:00
dev_warn ( dev , " Master decode error \n " ) ;
2014-08-20 21:56:02 +05:30
if ( status & XILINX_PCIE_INTR_MST_SLVERR )
2016-10-06 13:44:42 -05:00
dev_warn ( dev , " Master slave error \n " ) ;
2014-08-20 21:56:02 +05:30
if ( status & XILINX_PCIE_INTR_MST_ERRP )
2016-10-06 13:44:42 -05:00
dev_warn ( dev , " Master error poison \n " ) ;
2014-08-20 21:56:02 +05:30
2016-09-01 15:44:42 +05:30
error :
2014-08-20 21:56:02 +05:30
/* Clear the Interrupt Decode register */
pcie_write ( port , status , XILINX_PCIE_REG_IDR ) ;
return IRQ_HANDLED ;
}
/**
* xilinx_pcie_init_irq_domain - Initialize IRQ domain
* @ port : PCIe port information
*
* Return : ' 0 ' on success and error value on failure
*/
static int xilinx_pcie_init_irq_domain ( struct xilinx_pcie_port * port )
{
struct device * dev = port - > dev ;
struct device_node * pcie_intc_node ;
2019-03-25 17:19:09 -05:00
int ret ;
2014-08-20 21:56:02 +05:30
/* Setup INTx */
2021-03-30 16:11:36 +01:00
pcie_intc_node = of_get_next_child ( dev - > of_node , NULL ) ;
2014-08-20 21:56:02 +05:30
if ( ! pcie_intc_node ) {
dev_err ( dev , " No PCIe Intc node found \n " ) ;
2016-07-14 12:10:46 +02:00
return - ENODEV ;
2014-08-20 21:56:02 +05:30
}
2017-08-15 16:25:08 -05:00
port - > leg_domain = irq_domain_add_linear ( pcie_intc_node , PCI_NUM_INTX ,
2014-08-20 21:56:02 +05:30
& intx_domain_ops ,
port ) ;
2018-06-29 13:49:54 -05:00
of_node_put ( pcie_intc_node ) ;
2016-09-01 15:44:41 +05:30
if ( ! port - > leg_domain ) {
2014-08-20 21:56:02 +05:30
dev_err ( dev , " Failed to get a INTx IRQ domain \n " ) ;
2016-07-14 12:10:46 +02:00
return - ENODEV ;
2014-08-20 21:56:02 +05:30
}
/* Setup MSI */
if ( IS_ENABLED ( CONFIG_PCI_MSI ) ) {
2021-03-30 16:11:36 +01:00
phys_addr_t pa = ALIGN_DOWN ( virt_to_phys ( port ) , SZ_4K ) ;
2014-08-20 21:56:02 +05:30
2021-03-30 16:11:36 +01:00
ret = xilinx_allocate_msi_domains ( port ) ;
2019-03-25 17:19:09 -05:00
if ( ret )
return ret ;
2021-03-30 16:11:36 +01:00
pcie_write ( port , upper_32_bits ( pa ) , XILINX_PCIE_REG_MSIBASE1 ) ;
pcie_write ( port , lower_32_bits ( pa ) , XILINX_PCIE_REG_MSIBASE2 ) ;
2014-08-20 21:56:02 +05:30
}
return 0 ;
}
/**
* xilinx_pcie_init_port - Initialize hardware
* @ port : PCIe port information
*/
static void xilinx_pcie_init_port ( struct xilinx_pcie_port * port )
{
2016-10-06 13:44:42 -05:00
struct device * dev = port - > dev ;
2017-11-09 16:19:47 -06:00
if ( xilinx_pcie_link_up ( port ) )
2016-10-06 13:44:42 -05:00
dev_info ( dev , " PCIe Link is UP \n " ) ;
2014-08-20 21:56:02 +05:30
else
2016-10-06 13:44:42 -05:00
dev_info ( dev , " PCIe Link is DOWN \n " ) ;
2014-08-20 21:56:02 +05:30
/* Disable all interrupts */
pcie_write ( port , ~ XILINX_PCIE_IDR_ALL_MASK ,
XILINX_PCIE_REG_IMR ) ;
/* Clear pending interrupts */
pcie_write ( port , pcie_read ( port , XILINX_PCIE_REG_IDR ) &
XILINX_PCIE_IMR_ALL_MASK ,
XILINX_PCIE_REG_IDR ) ;
2017-08-15 16:25:25 -05:00
/* Enable all interrupts we handle */
pcie_write ( port , XILINX_PCIE_IMR_ENABLE_MASK , XILINX_PCIE_REG_IMR ) ;
2014-08-20 21:56:02 +05:30
/* Enable the Bridge enable bit */
pcie_write ( port , pcie_read ( port , XILINX_PCIE_REG_RPSC ) |
XILINX_PCIE_REG_RPSC_BEN ,
XILINX_PCIE_REG_RPSC ) ;
}
/**
* xilinx_pcie_parse_dt - Parse Device tree
* @ port : PCIe port information
*
* Return : ' 0 ' on success and error value on failure
*/
static int xilinx_pcie_parse_dt ( struct xilinx_pcie_port * port )
{
struct device * dev = port - > dev ;
struct device_node * node = dev - > of_node ;
struct resource regs ;
2021-03-30 16:11:36 +01:00
unsigned int irq ;
2014-08-20 21:56:02 +05:30
int err ;
err = of_address_to_resource ( node , 0 , & regs ) ;
if ( err ) {
dev_err ( dev , " missing \" reg \" property \n " ) ;
return err ;
}
2017-04-19 17:48:57 +01:00
port - > reg_base = devm_pci_remap_cfg_resource ( dev , & regs ) ;
2014-08-20 21:56:02 +05:30
if ( IS_ERR ( port - > reg_base ) )
return PTR_ERR ( port - > reg_base ) ;
2021-03-30 16:11:36 +01:00
irq = irq_of_parse_and_map ( node , 0 ) ;
err = devm_request_irq ( dev , irq , xilinx_pcie_intr_handler ,
2015-12-10 21:18:20 +02:00
IRQF_SHARED | IRQF_NO_THREAD ,
" xilinx-pcie " , port ) ;
2014-08-20 21:56:02 +05:30
if ( err ) {
2021-03-30 16:11:36 +01:00
dev_err ( dev , " unable to request irq %d \n " , irq ) ;
2014-08-20 21:56:02 +05:30
return err ;
}
return 0 ;
}
/**
* xilinx_pcie_probe - Probe function
* @ pdev : Platform device pointer
*
* Return : ' 0 ' on success and error value on failure
*/
static int xilinx_pcie_probe ( struct platform_device * pdev )
{
struct device * dev = & pdev - > dev ;
2016-10-06 13:44:42 -05:00
struct xilinx_pcie_port * port ;
2017-06-28 15:13:59 -05:00
struct pci_host_bridge * bridge ;
2014-08-20 21:56:02 +05:30
int err ;
if ( ! dev - > of_node )
return - ENODEV ;
2017-06-28 15:13:59 -05:00
bridge = devm_pci_alloc_host_bridge ( dev , sizeof ( * port ) ) ;
if ( ! bridge )
return - ENODEV ;
port = pci_host_bridge_priv ( bridge ) ;
2021-03-30 16:11:36 +01:00
mutex_init ( & port - > map_lock ) ;
2014-08-20 21:56:02 +05:30
port - > dev = dev ;
err = xilinx_pcie_parse_dt ( port ) ;
if ( err ) {
dev_err ( dev , " Parsing DT failed \n " ) ;
return err ;
}
xilinx_pcie_init_port ( port ) ;
err = xilinx_pcie_init_irq_domain ( port ) ;
if ( err ) {
dev_err ( dev , " Failed creating IRQ Domain \n " ) ;
return err ;
}
2017-06-28 15:13:59 -05:00
bridge - > sysdata = port ;
bridge - > ops = & xilinx_pcie_ops ;
2014-11-11 15:45:31 -07:00
2021-03-30 16:11:36 +01:00
err = pci_host_probe ( bridge ) ;
if ( err )
xilinx_free_msi_domains ( port ) ;
return err ;
2014-08-20 21:56:02 +05:30
}
2017-06-20 11:17:48 +05:30
static const struct of_device_id xilinx_pcie_of_match [ ] = {
2014-08-20 21:56:02 +05:30
{ . compatible = " xlnx,axi-pcie-host-1.00.a " , } ,
{ }
} ;
static struct platform_driver xilinx_pcie_driver = {
. driver = {
. name = " xilinx-pcie " ,
. of_match_table = xilinx_pcie_of_match ,
. suppress_bind_attrs = true ,
} ,
. probe = xilinx_pcie_probe ,
} ;
2016-08-24 16:57:49 -04:00
builtin_platform_driver ( xilinx_pcie_driver ) ;