2011-05-09 18:56:46 +02:00
/*
* Broadcom specific AMBA
* Bus subsystem
*
* Licensed under the GNU / GPL . See COPYING for details .
*/
# include "bcma_private.h"
2011-07-01 16:06:37 -04:00
# include <linux/module.h>
2012-08-10 21:23:53 +02:00
# include <linux/platform_device.h>
2011-05-09 18:56:46 +02:00
# include <linux/bcma/bcma.h>
2011-06-26 10:19:44 +02:00
# include <linux/slab.h>
2014-09-26 00:09:19 +02:00
# include <linux/of_address.h>
2011-05-09 18:56:46 +02:00
MODULE_DESCRIPTION ( " Broadcom's specific AMBA driver " ) ;
MODULE_LICENSE ( " GPL " ) ;
2012-01-31 00:03:36 +01:00
/* contains the number the next bus should get. */
static unsigned int bcma_bus_next_num = 0 ;
/* bcma_buses_mutex locks the bcma_bus_next_num */
static DEFINE_MUTEX ( bcma_buses_mutex ) ;
2011-05-09 18:56:46 +02:00
static int bcma_bus_match ( struct device * dev , struct device_driver * drv ) ;
static int bcma_device_probe ( struct device * dev ) ;
static int bcma_device_remove ( struct device * dev ) ;
2011-08-19 22:14:47 +02:00
static int bcma_device_uevent ( struct device * dev , struct kobj_uevent_env * env ) ;
2011-05-09 18:56:46 +02:00
static ssize_t manuf_show ( struct device * dev , struct device_attribute * attr , char * buf )
{
struct bcma_device * core = container_of ( dev , struct bcma_device , dev ) ;
return sprintf ( buf , " 0x%03X \n " , core - > id . manuf ) ;
}
2013-10-06 23:55:45 -07:00
static DEVICE_ATTR_RO ( manuf ) ;
2011-05-09 18:56:46 +02:00
static ssize_t id_show ( struct device * dev , struct device_attribute * attr , char * buf )
{
struct bcma_device * core = container_of ( dev , struct bcma_device , dev ) ;
return sprintf ( buf , " 0x%03X \n " , core - > id . id ) ;
}
2013-10-06 23:55:45 -07:00
static DEVICE_ATTR_RO ( id ) ;
2011-05-09 18:56:46 +02:00
static ssize_t rev_show ( struct device * dev , struct device_attribute * attr , char * buf )
{
struct bcma_device * core = container_of ( dev , struct bcma_device , dev ) ;
return sprintf ( buf , " 0x%02X \n " , core - > id . rev ) ;
}
2013-10-06 23:55:45 -07:00
static DEVICE_ATTR_RO ( rev ) ;
2011-05-09 18:56:46 +02:00
static ssize_t class_show ( struct device * dev , struct device_attribute * attr , char * buf )
{
struct bcma_device * core = container_of ( dev , struct bcma_device , dev ) ;
return sprintf ( buf , " 0x%X \n " , core - > id . class ) ;
}
2013-10-06 23:55:45 -07:00
static DEVICE_ATTR_RO ( class ) ;
static struct attribute * bcma_device_attrs [ ] = {
& dev_attr_manuf . attr ,
& dev_attr_id . attr ,
& dev_attr_rev . attr ,
& dev_attr_class . attr ,
NULL ,
2011-05-09 18:56:46 +02:00
} ;
2013-10-06 23:55:45 -07:00
ATTRIBUTE_GROUPS ( bcma_device ) ;
2011-05-09 18:56:46 +02:00
static struct bus_type bcma_bus_type = {
. name = " bcma " ,
. match = bcma_bus_match ,
. probe = bcma_device_probe ,
. remove = bcma_device_remove ,
2011-08-19 22:14:47 +02:00
. uevent = bcma_device_uevent ,
2013-10-06 23:55:45 -07:00
. dev_groups = bcma_device_groups ,
2011-05-09 18:56:46 +02:00
} ;
2012-07-10 23:45:49 +02:00
static u16 bcma_cc_core_id ( struct bcma_bus * bus )
{
if ( bus - > chipinfo . id = = BCMA_CHIP_ID_BCM4706 )
return BCMA_CORE_4706_CHIPCOMMON ;
return BCMA_CORE_CHIPCOMMON ;
}
2013-01-04 00:51:20 +01:00
struct bcma_device * bcma_find_core_unit ( struct bcma_bus * bus , u16 coreid ,
u8 unit )
2012-09-29 20:40:18 +02:00
{
struct bcma_device * core ;
list_for_each_entry ( core , & bus - > cores , list ) {
if ( core - > id . id = = coreid & & core - > core_unit = = unit )
return core ;
}
return NULL ;
}
2014-01-05 01:10:43 +01:00
EXPORT_SYMBOL_GPL ( bcma_find_core_unit ) ;
2012-09-29 20:40:18 +02:00
2013-06-26 10:02:11 +02:00
bool bcma_wait_value ( struct bcma_device * core , u16 reg , u32 mask , u32 value ,
int timeout )
{
unsigned long deadline = jiffies + timeout ;
u32 val ;
do {
val = bcma_read32 ( core , reg ) ;
if ( ( val & mask ) = = value )
return true ;
cpu_relax ( ) ;
udelay ( 10 ) ;
} while ( ! time_after_eq ( jiffies , deadline ) ) ;
bcma_warn ( core - > bus , " Timeout waiting for register 0x%04X! \n " , reg ) ;
return false ;
}
2011-05-09 18:56:46 +02:00
static void bcma_release_core_dev ( struct device * dev )
{
struct bcma_device * core = container_of ( dev , struct bcma_device , dev ) ;
2011-07-23 01:20:08 +02:00
if ( core - > io_addr )
iounmap ( core - > io_addr ) ;
if ( core - > io_wrap )
iounmap ( core - > io_wrap ) ;
2011-05-09 18:56:46 +02:00
kfree ( core ) ;
}
2014-09-05 00:18:49 +02:00
static bool bcma_is_core_needed_early ( u16 core_id )
{
switch ( core_id ) {
case BCMA_CORE_NS_NAND :
case BCMA_CORE_NS_QSPI :
return true ;
}
return false ;
}
2014-10-09 23:39:41 +02:00
# if defined(CONFIG_OF) && defined(CONFIG_OF_ADDRESS)
2014-09-26 00:09:19 +02:00
static struct device_node * bcma_of_find_child_device ( struct platform_device * parent ,
struct bcma_device * core )
{
struct device_node * node ;
u64 size ;
const __be32 * reg ;
if ( ! parent | | ! parent - > dev . of_node )
return NULL ;
for_each_child_of_node ( parent - > dev . of_node , node ) {
reg = of_get_address ( node , 0 , & size , NULL ) ;
if ( ! reg )
continue ;
if ( of_translate_address ( node , reg ) = = core - > addr )
return node ;
}
return NULL ;
}
static void bcma_of_fill_device ( struct platform_device * parent ,
struct bcma_device * core )
{
struct device_node * node ;
node = bcma_of_find_child_device ( parent , core ) ;
if ( node )
core - > dev . of_node = node ;
}
# else
static void bcma_of_fill_device ( struct platform_device * parent ,
struct bcma_device * core )
{
}
# endif /* CONFIG_OF */
2014-09-05 00:18:48 +02:00
static void bcma_register_core ( struct bcma_bus * bus , struct bcma_device * core )
{
int err ;
core - > dev . release = bcma_release_core_dev ;
core - > dev . bus = & bcma_bus_type ;
dev_set_name ( & core - > dev , " bcma%d:%d " , bus - > num , core - > core_index ) ;
switch ( bus - > hosttype ) {
case BCMA_HOSTTYPE_PCI :
core - > dev . parent = & bus - > host_pci - > dev ;
core - > dma_dev = & bus - > host_pci - > dev ;
core - > irq = bus - > host_pci - > irq ;
break ;
case BCMA_HOSTTYPE_SOC :
core - > dev . dma_mask = & core - > dev . coherent_dma_mask ;
2014-09-26 00:09:19 +02:00
if ( bus - > host_pdev ) {
core - > dma_dev = & bus - > host_pdev - > dev ;
core - > dev . parent = & bus - > host_pdev - > dev ;
bcma_of_fill_device ( bus - > host_pdev , core ) ;
} else {
core - > dma_dev = & core - > dev ;
}
2014-09-05 00:18:48 +02:00
break ;
case BCMA_HOSTTYPE_SDIO :
break ;
}
err = device_register ( & core - > dev ) ;
if ( err ) {
bcma_err ( bus , " Could not register dev for core 0x%03X \n " ,
core - > id . id ) ;
put_device ( & core - > dev ) ;
return ;
}
core - > dev_registered = true ;
}
static int bcma_register_devices ( struct bcma_bus * bus )
2011-05-09 18:56:46 +02:00
{
struct bcma_device * core ;
2014-09-05 00:18:48 +02:00
int err ;
2011-05-09 18:56:46 +02:00
list_for_each_entry ( core , & bus - > cores , list ) {
/* We support that cores ourself */
switch ( core - > id . id ) {
2012-07-10 23:45:49 +02:00
case BCMA_CORE_4706_CHIPCOMMON :
2011-05-09 18:56:46 +02:00
case BCMA_CORE_CHIPCOMMON :
2014-09-08 22:53:36 +02:00
case BCMA_CORE_NS_CHIPCOMMON_B :
2011-05-09 18:56:46 +02:00
case BCMA_CORE_PCI :
case BCMA_CORE_PCIE :
2014-07-05 01:10:41 +02:00
case BCMA_CORE_PCIE2 :
2011-07-23 01:20:09 +02:00
case BCMA_CORE_MIPS_74K :
2012-07-11 09:23:43 +02:00
case BCMA_CORE_4706_MAC_GBIT_COMMON :
2011-05-09 18:56:46 +02:00
continue ;
}
2014-09-05 00:18:49 +02:00
/* Early cores were already registered */
if ( bcma_is_core_needed_early ( core - > id . id ) )
continue ;
2013-02-19 19:41:42 +01:00
/* Only first GMAC core on BCM4706 is connected and working */
if ( core - > id . id = = BCMA_CORE_4706_MAC_GBIT & &
core - > core_unit > 0 )
continue ;
2014-09-05 00:18:48 +02:00
bcma_register_core ( bus , core ) ;
2011-05-09 18:56:46 +02:00
}
2013-01-25 11:37:26 +01:00
# ifdef CONFIG_BCMA_DRIVER_MIPS
if ( bus - > drv_cc . pflash . present ) {
err = platform_device_register ( & bcma_pflash_dev ) ;
if ( err )
bcma_err ( bus , " Error registering parallel flash \n " ) ;
}
# endif
2012-08-10 21:23:53 +02:00
# ifdef CONFIG_BCMA_SFLASH
if ( bus - > drv_cc . sflash . present ) {
err = platform_device_register ( & bcma_sflash_dev ) ;
if ( err )
bcma_err ( bus , " Error registering serial flash \n " ) ;
}
# endif
2012-08-12 13:08:05 +02:00
# ifdef CONFIG_BCMA_NFLASH
if ( bus - > drv_cc . nflash . present ) {
err = platform_device_register ( & bcma_nflash_dev ) ;
if ( err )
bcma_err ( bus , " Error registering NAND flash \n " ) ;
}
# endif
2012-11-20 22:24:30 +00:00
err = bcma_gpio_init ( & bus - > drv_cc ) ;
if ( err = = - ENOTSUPP )
bcma_debug ( bus , " GPIO driver not activated \n " ) ;
else if ( err )
bcma_err ( bus , " Error registering GPIO driver: %i \n " , err ) ;
2012-08-12 13:08:05 +02:00
2012-12-05 18:46:02 +01:00
if ( bus - > hosttype = = BCMA_HOSTTYPE_SOC ) {
err = bcma_chipco_watchdog_register ( & bus - > drv_cc ) ;
if ( err )
bcma_err ( bus , " Error registering watchdog driver \n " ) ;
}
2011-05-09 18:56:46 +02:00
return 0 ;
}
static void bcma_unregister_cores ( struct bcma_bus * bus )
{
2012-10-11 14:05:15 +02:00
struct bcma_device * core , * tmp ;
2011-05-09 18:56:46 +02:00
2012-10-11 14:05:15 +02:00
list_for_each_entry_safe ( core , tmp , & bus - > cores , list ) {
list_del ( & core - > list ) ;
2011-05-09 18:56:46 +02:00
if ( core - > dev_registered )
device_unregister ( & core - > dev ) ;
}
2012-12-05 18:46:02 +01:00
if ( bus - > hosttype = = BCMA_HOSTTYPE_SOC )
platform_device_unregister ( bus - > drv_cc . watchdog ) ;
2011-05-09 18:56:46 +02:00
}
2012-12-21 15:12:59 -08:00
int bcma_bus_register ( struct bcma_bus * bus )
2011-05-09 18:56:46 +02:00
{
int err ;
struct bcma_device * core ;
2012-01-31 00:03:36 +01:00
mutex_lock ( & bcma_buses_mutex ) ;
bus - > num = bcma_bus_next_num + + ;
mutex_unlock ( & bcma_buses_mutex ) ;
2011-05-09 18:56:46 +02:00
/* Scan for devices (cores) */
err = bcma_bus_scan ( bus ) ;
if ( err ) {
2012-07-05 22:07:32 +02:00
bcma_err ( bus , " Failed to scan: %d \n " , err ) ;
2013-07-15 13:15:07 +02:00
return err ;
2011-05-09 18:56:46 +02:00
}
2012-09-29 20:29:50 +02:00
/* Early init CC core */
core = bcma_find_core ( bus , bcma_cc_core_id ( bus ) ) ;
if ( core ) {
bus - > drv_cc . core = core ;
bcma_core_chipcommon_early_init ( & bus - > drv_cc ) ;
}
2014-09-05 00:18:49 +02:00
/* Cores providing flash access go before SPROM init */
list_for_each_entry ( core , & bus - > cores , list ) {
if ( bcma_is_core_needed_early ( core - > id . id ) )
bcma_register_core ( bus , core ) ;
}
2012-09-29 20:29:50 +02:00
/* Try to get SPROM */
err = bcma_sprom_get ( bus ) ;
if ( err = = - ENOENT ) {
bcma_err ( bus , " No SPROM available \n " ) ;
} else if ( err )
bcma_err ( bus , " Failed to get SPROM: %d \n " , err ) ;
2011-05-09 18:56:46 +02:00
/* Init CC core */
2012-07-10 23:45:49 +02:00
core = bcma_find_core ( bus , bcma_cc_core_id ( bus ) ) ;
2011-05-09 18:56:46 +02:00
if ( core ) {
bus - > drv_cc . core = core ;
bcma_core_chipcommon_init ( & bus - > drv_cc ) ;
}
2014-09-08 22:53:36 +02:00
/* Init CC core */
core = bcma_find_core ( bus , BCMA_CORE_NS_CHIPCOMMON_B ) ;
if ( core ) {
bus - > drv_cc_b . core = core ;
bcma_core_chipcommon_b_init ( & bus - > drv_cc_b ) ;
}
2011-07-23 01:20:09 +02:00
/* Init MIPS core */
core = bcma_find_core ( bus , BCMA_CORE_MIPS_74K ) ;
if ( core ) {
bus - > drv_mips . core = core ;
bcma_core_mips_init ( & bus - > drv_mips ) ;
}
2011-05-09 18:56:46 +02:00
/* Init PCIE core */
2012-09-29 20:40:18 +02:00
core = bcma_find_core_unit ( bus , BCMA_CORE_PCIE , 0 ) ;
if ( core ) {
bus - > drv_pci [ 0 ] . core = core ;
bcma_core_pci_init ( & bus - > drv_pci [ 0 ] ) ;
}
/* Init PCIE core */
core = bcma_find_core_unit ( bus , BCMA_CORE_PCIE , 1 ) ;
2011-05-09 18:56:46 +02:00
if ( core ) {
2012-09-29 20:40:18 +02:00
bus - > drv_pci [ 1 ] . core = core ;
bcma_core_pci_init ( & bus - > drv_pci [ 1 ] ) ;
2011-05-09 18:56:46 +02:00
}
2014-07-05 01:10:41 +02:00
/* Init PCIe Gen 2 core */
core = bcma_find_core_unit ( bus , BCMA_CORE_PCIE2 , 0 ) ;
if ( core ) {
bus - > drv_pcie2 . core = core ;
bcma_core_pcie2_init ( & bus - > drv_pcie2 ) ;
}
2012-07-11 09:23:43 +02:00
/* Init GBIT MAC COMMON core */
core = bcma_find_core ( bus , BCMA_CORE_4706_MAC_GBIT_COMMON ) ;
if ( core ) {
bus - > drv_gmac_cmn . core = core ;
bcma_core_gmac_cmn_init ( & bus - > drv_gmac_cmn ) ;
}
2011-05-09 18:56:46 +02:00
/* Register found cores */
2014-09-05 00:18:48 +02:00
bcma_register_devices ( bus ) ;
2011-05-09 18:56:46 +02:00
2012-07-05 22:07:32 +02:00
bcma_info ( bus , " Bus registered \n " ) ;
2011-05-09 18:56:46 +02:00
return 0 ;
}
void bcma_bus_unregister ( struct bcma_bus * bus )
{
2012-08-16 15:42:30 -05:00
struct bcma_device * cores [ 3 ] ;
2013-02-03 23:25:33 +01:00
int err ;
err = bcma_gpio_unregister ( & bus - > drv_cc ) ;
if ( err = = - EBUSY )
bcma_err ( bus , " Some GPIOs are still in use. \n " ) ;
else if ( err )
bcma_err ( bus , " Can not unregister GPIO driver: %i \n " , err ) ;
2012-08-16 15:42:30 -05:00
2014-09-08 22:53:36 +02:00
bcma_core_chipcommon_b_free ( & bus - > drv_cc_b ) ;
2012-08-16 15:42:30 -05:00
cores [ 0 ] = bcma_find_core ( bus , BCMA_CORE_MIPS_74K ) ;
cores [ 1 ] = bcma_find_core ( bus , BCMA_CORE_PCIE ) ;
cores [ 2 ] = bcma_find_core ( bus , BCMA_CORE_4706_MAC_GBIT_COMMON ) ;
2011-05-09 18:56:46 +02:00
bcma_unregister_cores ( bus ) ;
2012-08-16 15:42:30 -05:00
kfree ( cores [ 2 ] ) ;
kfree ( cores [ 1 ] ) ;
kfree ( cores [ 0 ] ) ;
2011-05-09 18:56:46 +02:00
}
2011-07-23 01:20:07 +02:00
int __init bcma_bus_early_register ( struct bcma_bus * bus ,
struct bcma_device * core_cc ,
struct bcma_device * core_mips )
{
int err ;
struct bcma_device * core ;
struct bcma_device_id match ;
match . manuf = BCMA_MANUF_BCM ;
2012-07-10 23:45:49 +02:00
match . id = bcma_cc_core_id ( bus ) ;
2011-07-23 01:20:07 +02:00
match . class = BCMA_CL_SIM ;
match . rev = BCMA_ANY_REV ;
/* Scan for chip common core */
err = bcma_bus_scan_early ( bus , & match , core_cc ) ;
if ( err ) {
2012-07-05 22:07:32 +02:00
bcma_err ( bus , " Failed to scan for common core: %d \n " , err ) ;
2011-07-23 01:20:07 +02:00
return - 1 ;
}
match . manuf = BCMA_MANUF_MIPS ;
match . id = BCMA_CORE_MIPS_74K ;
match . class = BCMA_CL_SIM ;
match . rev = BCMA_ANY_REV ;
/* Scan for mips core */
err = bcma_bus_scan_early ( bus , & match , core_mips ) ;
if ( err ) {
2012-07-05 22:07:32 +02:00
bcma_err ( bus , " Failed to scan for mips core: %d \n " , err ) ;
2011-07-23 01:20:07 +02:00
return - 1 ;
}
2012-09-29 20:29:49 +02:00
/* Early init CC core */
2012-07-10 23:45:49 +02:00
core = bcma_find_core ( bus , bcma_cc_core_id ( bus ) ) ;
2011-07-23 01:20:07 +02:00
if ( core ) {
bus - > drv_cc . core = core ;
2012-09-29 20:29:49 +02:00
bcma_core_chipcommon_early_init ( & bus - > drv_cc ) ;
2011-07-23 01:20:07 +02:00
}
2012-09-29 20:29:49 +02:00
/* Early init MIPS core */
2011-07-23 01:20:09 +02:00
core = bcma_find_core ( bus , BCMA_CORE_MIPS_74K ) ;
if ( core ) {
bus - > drv_mips . core = core ;
2012-09-29 20:29:49 +02:00
bcma_core_mips_early_init ( & bus - > drv_mips ) ;
2011-07-23 01:20:09 +02:00
}
2012-07-05 22:07:32 +02:00
bcma_info ( bus , " Early bus registered \n " ) ;
2011-07-23 01:20:07 +02:00
return 0 ;
}
2011-12-09 22:16:07 +01:00
# ifdef CONFIG_PM
2012-01-13 23:58:40 +01:00
int bcma_bus_suspend ( struct bcma_bus * bus )
{
2012-01-13 23:58:41 +01:00
struct bcma_device * core ;
list_for_each_entry ( core , & bus - > cores , list ) {
struct device_driver * drv = core - > dev . driver ;
if ( drv ) {
struct bcma_driver * adrv = container_of ( drv , struct bcma_driver , drv ) ;
if ( adrv - > suspend )
adrv - > suspend ( core ) ;
}
}
2012-01-13 23:58:40 +01:00
return 0 ;
}
2011-12-09 22:16:07 +01:00
int bcma_bus_resume ( struct bcma_bus * bus )
{
struct bcma_device * core ;
/* Init CC core */
2012-07-10 23:45:49 +02:00
if ( bus - > drv_cc . core ) {
2011-12-09 22:16:07 +01:00
bus - > drv_cc . setup_done = false ;
bcma_core_chipcommon_init ( & bus - > drv_cc ) ;
}
2012-01-13 23:58:41 +01:00
list_for_each_entry ( core , & bus - > cores , list ) {
struct device_driver * drv = core - > dev . driver ;
if ( drv ) {
struct bcma_driver * adrv = container_of ( drv , struct bcma_driver , drv ) ;
if ( adrv - > resume )
adrv - > resume ( core ) ;
}
}
2011-12-09 22:16:07 +01:00
return 0 ;
}
# endif
2011-05-09 18:56:46 +02:00
int __bcma_driver_register ( struct bcma_driver * drv , struct module * owner )
{
drv - > drv . name = drv - > name ;
drv - > drv . bus = & bcma_bus_type ;
drv - > drv . owner = owner ;
return driver_register ( & drv - > drv ) ;
}
EXPORT_SYMBOL_GPL ( __bcma_driver_register ) ;
void bcma_driver_unregister ( struct bcma_driver * drv )
{
driver_unregister ( & drv - > drv ) ;
}
EXPORT_SYMBOL_GPL ( bcma_driver_unregister ) ;
static int bcma_bus_match ( struct device * dev , struct device_driver * drv )
{
struct bcma_device * core = container_of ( dev , struct bcma_device , dev ) ;
struct bcma_driver * adrv = container_of ( drv , struct bcma_driver , drv ) ;
const struct bcma_device_id * cid = & core - > id ;
const struct bcma_device_id * did ;
for ( did = adrv - > id_table ; did - > manuf | | did - > id | | did - > rev ; did + + ) {
if ( ( did - > manuf = = cid - > manuf | | did - > manuf = = BCMA_ANY_MANUF ) & &
( did - > id = = cid - > id | | did - > id = = BCMA_ANY_ID ) & &
( did - > rev = = cid - > rev | | did - > rev = = BCMA_ANY_REV ) & &
( did - > class = = cid - > class | | did - > class = = BCMA_ANY_CLASS ) )
return 1 ;
}
return 0 ;
}
static int bcma_device_probe ( struct device * dev )
{
struct bcma_device * core = container_of ( dev , struct bcma_device , dev ) ;
struct bcma_driver * adrv = container_of ( dev - > driver , struct bcma_driver ,
drv ) ;
int err = 0 ;
if ( adrv - > probe )
err = adrv - > probe ( core ) ;
return err ;
}
static int bcma_device_remove ( struct device * dev )
{
struct bcma_device * core = container_of ( dev , struct bcma_device , dev ) ;
struct bcma_driver * adrv = container_of ( dev - > driver , struct bcma_driver ,
drv ) ;
if ( adrv - > remove )
adrv - > remove ( core ) ;
return 0 ;
}
2011-08-19 22:14:47 +02:00
static int bcma_device_uevent ( struct device * dev , struct kobj_uevent_env * env )
{
struct bcma_device * core = container_of ( dev , struct bcma_device , dev ) ;
return add_uevent_var ( env ,
" MODALIAS=bcma:m%04Xid%04Xrev%02Xcl%02X " ,
core - > id . manuf , core - > id . id ,
core - > id . rev , core - > id . class ) ;
}
2011-05-09 18:56:46 +02:00
static int __init bcma_modinit ( void )
{
int err ;
err = bus_register ( & bcma_bus_type ) ;
if ( err )
return err ;
2014-09-26 00:09:19 +02:00
err = bcma_host_soc_register_driver ( ) ;
if ( err ) {
pr_err ( " SoC host initialization failed \n " ) ;
err = 0 ;
}
2011-05-09 18:56:46 +02:00
# ifdef CONFIG_BCMA_HOST_PCI
err = bcma_host_pci_init ( ) ;
if ( err ) {
pr_err ( " PCI host initialization failed \n " ) ;
err = 0 ;
}
# endif
return err ;
}
fs_initcall ( bcma_modinit ) ;
static void __exit bcma_modexit ( void )
{
# ifdef CONFIG_BCMA_HOST_PCI
bcma_host_pci_exit ( ) ;
# endif
2014-09-26 00:09:19 +02:00
bcma_host_soc_unregister_driver ( ) ;
2011-05-09 18:56:46 +02:00
bus_unregister ( & bcma_bus_type ) ;
}
module_exit ( bcma_modexit )