2018-01-26 21:50:27 +03:00
// SPDX-License-Identifier: GPL-2.0
2015-05-13 00:23:01 +03:00
/*
* Copyright ( C ) 2015 Broadcom Corporation
* Copyright ( C ) 2015 Hauke Mehrtens < hauke @ hauke - m . de >
*/
# include <linux/kernel.h>
# include <linux/pci.h>
# include <linux/module.h>
# include <linux/slab.h>
# include <linux/phy/phy.h>
# include <linux/bcma/bcma.h>
# include <linux/ioport.h>
# include "pcie-iproc.h"
/* NS: CLASS field is R/O, and set to wrong 0x200 value */
static void bcma_pcie2_fixup_class ( struct pci_dev * dev )
{
dev - > class = PCI_CLASS_BRIDGE_PCI < < 8 ;
}
DECLARE_PCI_FIXUP_EARLY ( PCI_VENDOR_ID_BROADCOM , 0x8011 , bcma_pcie2_fixup_class ) ;
DECLARE_PCI_FIXUP_EARLY ( PCI_VENDOR_ID_BROADCOM , 0x8012 , bcma_pcie2_fixup_class ) ;
static int iproc_pcie_bcma_map_irq ( const struct pci_dev * dev , u8 slot , u8 pin )
{
2018-03-07 18:42:36 +03:00
struct iproc_pcie * pcie = dev - > sysdata ;
2015-05-13 00:23:01 +03:00
struct bcma_device * bdev = container_of ( pcie - > dev , struct bcma_device , dev ) ;
return bcma_core_irq ( bdev , 5 ) ;
}
static int iproc_pcie_bcma_probe ( struct bcma_device * bdev )
{
2016-10-06 21:36:08 +03:00
struct device * dev = & bdev - > dev ;
2015-05-13 00:23:01 +03:00
struct iproc_pcie * pcie ;
2017-03-09 20:27:07 +03:00
LIST_HEAD ( resources ) ;
2017-06-28 23:13:57 +03:00
struct pci_host_bridge * bridge ;
2015-05-13 00:23:01 +03:00
int ret ;
2017-06-28 23:13:57 +03:00
bridge = devm_pci_alloc_host_bridge ( dev , sizeof ( * pcie ) ) ;
if ( ! bridge )
2015-05-13 00:23:01 +03:00
return - ENOMEM ;
2017-06-28 23:13:57 +03:00
pcie = pci_host_bridge_priv ( bridge ) ;
2016-10-06 21:36:08 +03:00
pcie - > dev = dev ;
2015-05-13 00:23:01 +03:00
2016-11-01 03:38:32 +03:00
pcie - > type = IPROC_PCIE_PAXB_BCMA ;
2015-05-13 00:23:01 +03:00
pcie - > base = bdev - > io_addr ;
2016-10-06 21:36:07 +03:00
if ( ! pcie - > base ) {
dev_err ( dev , " no controller registers \n " ) ;
return - ENOMEM ;
}
2016-01-07 03:04:35 +03:00
pcie - > base_addr = bdev - > addr ;
2015-05-13 00:23:01 +03:00
2017-03-09 20:27:07 +03:00
pcie - > mem . start = bdev - > addr_s [ 0 ] ;
pcie - > mem . end = bdev - > addr_s [ 0 ] + SZ_128M - 1 ;
pcie - > mem . name = " PCIe MEM space " ;
pcie - > mem . flags = IORESOURCE_MEM ;
pci_add_resource ( & resources , & pcie - > mem ) ;
2015-05-13 00:23:01 +03:00
pcie - > map_irq = iproc_pcie_bcma_map_irq ;
2017-03-09 20:27:07 +03:00
ret = iproc_pcie_setup ( pcie , & resources ) ;
if ( ret ) {
2016-10-06 21:36:08 +03:00
dev_err ( dev , " PCIe controller setup failed \n " ) ;
2017-03-09 20:27:07 +03:00
pci_free_resource_list ( & resources ) ;
return ret ;
}
2015-05-24 23:37:03 +03:00
2016-10-06 21:36:08 +03:00
bcma_set_drvdata ( bdev , pcie ) ;
2017-03-09 20:27:07 +03:00
return 0 ;
2015-05-13 00:23:01 +03:00
}
static void iproc_pcie_bcma_remove ( struct bcma_device * bdev )
{
struct iproc_pcie * pcie = bcma_get_drvdata ( bdev ) ;
iproc_pcie_remove ( pcie ) ;
}
static const struct bcma_device_id iproc_pcie_bcma_table [ ] = {
BCMA_CORE ( BCMA_MANUF_BCM , BCMA_CORE_NS_PCIEG2 , BCMA_ANY_REV , BCMA_ANY_CLASS ) ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( bcma , iproc_pcie_bcma_table ) ;
static struct bcma_driver iproc_pcie_bcma_driver = {
. name = KBUILD_MODNAME ,
. id_table = iproc_pcie_bcma_table ,
. probe = iproc_pcie_bcma_probe ,
. remove = iproc_pcie_bcma_remove ,
} ;
static int __init iproc_pcie_bcma_init ( void )
{
return bcma_driver_register ( & iproc_pcie_bcma_driver ) ;
}
module_init ( iproc_pcie_bcma_init ) ;
static void __exit iproc_pcie_bcma_exit ( void )
{
bcma_driver_unregister ( & iproc_pcie_bcma_driver ) ;
}
module_exit ( iproc_pcie_bcma_exit ) ;
MODULE_AUTHOR ( " Hauke Mehrtens " ) ;
MODULE_DESCRIPTION ( " Broadcom iProc PCIe BCMA driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;