2005-11-07 12:00:17 +03:00
/*
* RapidIO enumeration and discovery support
*
* Copyright 2005 MontaVista Software , Inc .
* Matt Porter < mporter @ kernel . crashing . org >
*
2010-05-27 01:43:59 +04:00
* Copyright 2009 Integrated Device Technology , Inc .
* Alex Bounine < alexandre . bounine @ idt . com >
* - Added Port - Write / Error Management initialization and handling
*
2010-05-27 01:44:01 +04:00
* Copyright 2009 Sysgo AG
* Thomas Moll < thomas . moll @ sysgo . com >
* - Added Input - Output - enable functionality , to allow full communication
*
2005-11-07 12:00:17 +03:00
* This program is free software ; you can redistribute it and / or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation ; either version 2 of the License , or ( at your
* option ) any later version .
*/
# include <linux/types.h>
# include <linux/kernel.h>
# include <linux/delay.h>
2005-11-07 12:00:18 +03:00
# include <linux/dma-mapping.h>
2005-11-07 12:00:17 +03:00
# include <linux/init.h>
# include <linux/rio.h>
# include <linux/rio_drv.h>
# include <linux/rio_ids.h>
# include <linux/rio_regs.h>
# include <linux/module.h>
# include <linux/spinlock.h>
# include <linux/timer.h>
2012-10-05 04:16:04 +04:00
# include <linux/sched.h>
2006-01-08 12:02:05 +03:00
# include <linux/jiffies.h>
# include <linux/slab.h>
2005-11-07 12:00:17 +03:00
# include "rio.h"
2010-05-27 01:43:59 +04:00
static void rio_init_em ( struct rio_dev * rdev ) ;
2005-11-07 12:00:17 +03:00
static int next_destid = 0 ;
2010-10-28 02:34:34 +04:00
static int next_comptag = 1 ;
2005-11-07 12:00:17 +03:00
static int rio_mport_phys_table [ ] = {
RIO_EFB_PAR_EP_ID ,
RIO_EFB_PAR_EP_REC_ID ,
RIO_EFB_SER_EP_ID ,
RIO_EFB_SER_EP_REC_ID ,
- 1 ,
} ;
2012-10-05 04:16:13 +04:00
2012-10-11 02:54:01 +04:00
/**
2012-10-05 04:16:13 +04:00
* rio_destid_alloc - Allocate next available destID for given network
2012-10-11 02:54:01 +04:00
* @ net : RIO network
2012-10-05 04:16:13 +04:00
*
* Returns next available device destination ID for the specified RIO network .
* Marks allocated ID as one in use .
* Returns RIO_INVALID_DESTID if new destID is not available .
*/
static u16 rio_destid_alloc ( struct rio_net * net )
{
int destid ;
struct rio_id_table * idtab = & net - > destid_table ;
spin_lock ( & idtab - > lock ) ;
2012-10-11 02:54:01 +04:00
destid = find_first_zero_bit ( idtab - > table , idtab - > max ) ;
2012-10-05 04:16:13 +04:00
if ( destid < idtab - > max ) {
set_bit ( destid , idtab - > table ) ;
destid + = idtab - > start ;
} else
destid = RIO_INVALID_DESTID ;
spin_unlock ( & idtab - > lock ) ;
return ( u16 ) destid ;
}
2012-10-11 02:54:01 +04:00
/**
2012-10-05 04:16:13 +04:00
* rio_destid_reserve - Reserve the specivied destID
2012-10-11 02:54:01 +04:00
* @ net : RIO network
* @ destid : destID to reserve
2012-10-05 04:16:13 +04:00
*
* Tries to reserve the specified destID .
2015-05-07 18:02:27 +03:00
* Returns 0 if successful .
2012-10-05 04:16:13 +04:00
*/
static int rio_destid_reserve ( struct rio_net * net , u16 destid )
{
int oldbit ;
struct rio_id_table * idtab = & net - > destid_table ;
destid - = idtab - > start ;
spin_lock ( & idtab - > lock ) ;
oldbit = test_and_set_bit ( destid , idtab - > table ) ;
spin_unlock ( & idtab - > lock ) ;
return oldbit ;
}
2012-10-11 02:54:01 +04:00
/**
2012-10-05 04:16:13 +04:00
* rio_destid_free - free a previously allocated destID
2012-10-11 02:54:01 +04:00
* @ net : RIO network
* @ destid : destID to free
2012-10-05 04:16:13 +04:00
*
* Makes the specified destID available for use .
*/
static void rio_destid_free ( struct rio_net * net , u16 destid )
{
struct rio_id_table * idtab = & net - > destid_table ;
destid - = idtab - > start ;
spin_lock ( & idtab - > lock ) ;
clear_bit ( destid , idtab - > table ) ;
spin_unlock ( & idtab - > lock ) ;
}
2012-10-11 02:54:01 +04:00
/**
2012-10-05 04:16:13 +04:00
* rio_destid_first - return first destID in use
2012-10-11 02:54:01 +04:00
* @ net : RIO network
2012-10-05 04:16:13 +04:00
*/
static u16 rio_destid_first ( struct rio_net * net )
{
int destid ;
struct rio_id_table * idtab = & net - > destid_table ;
spin_lock ( & idtab - > lock ) ;
destid = find_first_bit ( idtab - > table , idtab - > max ) ;
if ( destid > = idtab - > max )
destid = RIO_INVALID_DESTID ;
else
destid + = idtab - > start ;
spin_unlock ( & idtab - > lock ) ;
return ( u16 ) destid ;
}
2012-10-11 02:54:01 +04:00
/**
2012-10-05 04:16:13 +04:00
* rio_destid_next - return next destID in use
2012-10-11 02:54:01 +04:00
* @ net : RIO network
* @ from : destination ID from which search shall continue
2012-10-05 04:16:13 +04:00
*/
static u16 rio_destid_next ( struct rio_net * net , u16 from )
{
int destid ;
struct rio_id_table * idtab = & net - > destid_table ;
spin_lock ( & idtab - > lock ) ;
destid = find_next_bit ( idtab - > table , idtab - > max , from ) ;
if ( destid > = idtab - > max )
destid = RIO_INVALID_DESTID ;
else
destid + = idtab - > start ;
spin_unlock ( & idtab - > lock ) ;
return ( u16 ) destid ;
}
2005-11-07 12:00:17 +03:00
/**
* rio_get_device_id - Get the base / extended device id for a device
* @ port : RIO master port
* @ destid : Destination ID of device
* @ hopcount : Hopcount to device
*
* Reads the base / extended device id from a device . Returns the
* 8 / 16 - bit device ID .
*/
static u16 rio_get_device_id ( struct rio_mport * port , u16 destid , u8 hopcount )
{
u32 result ;
rio_mport_read_config_32 ( port , destid , hopcount , RIO_DID_CSR , & result ) ;
2008-04-19 00:33:42 +04:00
return RIO_GET_DID ( port - > sys_size , result ) ;
2005-11-07 12:00:17 +03:00
}
/**
* rio_set_device_id - Set the base / extended device id for a device
* @ port : RIO master port
* @ destid : Destination ID of device
* @ hopcount : Hopcount to device
* @ did : Device ID value to be written
*
* Writes the base / extended device id from a device .
*/
2005-11-07 12:00:18 +03:00
static void rio_set_device_id ( struct rio_mport * port , u16 destid , u8 hopcount , u16 did )
2005-11-07 12:00:17 +03:00
{
rio_mport_write_config_32 ( port , destid , hopcount , RIO_DID_CSR ,
2008-04-19 00:33:42 +04:00
RIO_SET_DID ( port - > sys_size , did ) ) ;
2005-11-07 12:00:17 +03:00
}
/**
* rio_local_set_device_id - Set the base / extended device id for a port
* @ port : RIO master port
* @ did : Device ID value to be written
*
* Writes the base / extended device id from a device .
*/
static void rio_local_set_device_id ( struct rio_mport * port , u16 did )
{
2008-04-19 00:33:42 +04:00
rio_local_write_config_32 ( port , RIO_DID_CSR , RIO_SET_DID ( port - > sys_size ,
did ) ) ;
2005-11-07 12:00:17 +03:00
}
/**
* rio_clear_locks - Release all host locks and signal enumeration complete
2012-10-05 04:16:05 +04:00
* @ net : RIO network to run on
2005-11-07 12:00:17 +03:00
*
* Marks the component tag CSR on each device with the enumeration
* complete flag . When complete , it then release the host locks on
* each device . Returns 0 on success or % - EINVAL on failure .
*/
2012-10-05 04:16:05 +04:00
static int rio_clear_locks ( struct rio_net * net )
2005-11-07 12:00:17 +03:00
{
2012-10-05 04:16:05 +04:00
struct rio_mport * port = net - > hport ;
2005-11-07 12:00:17 +03:00
struct rio_dev * rdev ;
u32 result ;
int ret = 0 ;
/* Release host device id locks */
rio_local_write_config_32 ( port , RIO_HOST_DID_LOCK_CSR ,
port - > host_deviceid ) ;
rio_local_read_config_32 ( port , RIO_HOST_DID_LOCK_CSR , & result ) ;
if ( ( result & 0xffff ) ! = 0xffff ) {
printk ( KERN_INFO
" RIO: badness when releasing host lock on master port, result %8.8x \n " ,
result ) ;
ret = - EINVAL ;
}
2012-10-05 04:16:05 +04:00
list_for_each_entry ( rdev , & net - > devices , net_list ) {
2005-11-07 12:00:17 +03:00
rio_write_config_32 ( rdev , RIO_HOST_DID_LOCK_CSR ,
port - > host_deviceid ) ;
rio_read_config_32 ( rdev , RIO_HOST_DID_LOCK_CSR , & result ) ;
if ( ( result & 0xffff ) ! = 0xffff ) {
printk ( KERN_INFO
" RIO: badness when releasing host lock on vid %4.4x did %4.4x \n " ,
rdev - > vid , rdev - > did ) ;
ret = - EINVAL ;
}
2010-10-28 02:34:34 +04:00
/* Mark device as discovered and enable master */
rio_read_config_32 ( rdev ,
rdev - > phys_efptr + RIO_PORT_GEN_CTL_CSR ,
& result ) ;
result | = RIO_PORT_GEN_DISCOVERED | RIO_PORT_GEN_MASTER ;
rio_write_config_32 ( rdev ,
rdev - > phys_efptr + RIO_PORT_GEN_CTL_CSR ,
result ) ;
2005-11-07 12:00:17 +03:00
}
return ret ;
}
/**
* rio_enum_host - Set host lock and initialize host destination ID
* @ port : Master port to issue transaction
*
* Sets the local host master port lock and destination ID register
* with the host device ID value . The host device ID value is provided
* by the platform . Returns % 0 on success or % - 1 on failure .
*/
static int rio_enum_host ( struct rio_mport * port )
{
u32 result ;
/* Set master port host device id lock */
rio_local_write_config_32 ( port , RIO_HOST_DID_LOCK_CSR ,
port - > host_deviceid ) ;
rio_local_read_config_32 ( port , RIO_HOST_DID_LOCK_CSR , & result ) ;
if ( ( result & 0xffff ) ! = port - > host_deviceid )
return - 1 ;
/* Set master port destid and init destid ctr */
rio_local_set_device_id ( port , port - > host_deviceid ) ;
return 0 ;
}
/**
* rio_device_has_destid - Test if a device contains a destination ID register
* @ port : Master port to issue transaction
* @ src_ops : RIO device source operations
* @ dst_ops : RIO device destination operations
*
* Checks the provided @ src_ops and @ dst_ops for the necessary transaction
* capabilities that indicate whether or not a device will implement a
* destination ID register . Returns 1 if true or 0 if false .
*/
static int rio_device_has_destid ( struct rio_mport * port , int src_ops ,
int dst_ops )
{
2005-11-07 12:00:18 +03:00
u32 mask = RIO_OPS_READ | RIO_OPS_WRITE | RIO_OPS_ATOMIC_TST_SWP | RIO_OPS_ATOMIC_INC | RIO_OPS_ATOMIC_DEC | RIO_OPS_ATOMIC_SET | RIO_OPS_ATOMIC_CLR ;
return ! ! ( ( src_ops | dst_ops ) & mask ) ;
2005-11-07 12:00:17 +03:00
}
/**
* rio_release_dev - Frees a RIO device struct
* @ dev : LDM device associated with a RIO device struct
*
* Gets the RIO device struct associated a RIO device struct .
* The RIO device struct is freed .
*/
static void rio_release_dev ( struct device * dev )
{
struct rio_dev * rdev ;
rdev = to_rio_dev ( dev ) ;
kfree ( rdev ) ;
}
/**
* rio_is_switch - Tests if a RIO device has switch capabilities
* @ rdev : RIO device
*
* Gets the RIO device Processing Element Features register
* contents and tests for switch capabilities . Returns 1 if
* the device is a switch or 0 if it is not a switch .
* The RIO device struct is freed .
*/
static int rio_is_switch ( struct rio_dev * rdev )
{
if ( rdev - > pef & RIO_PEF_SWITCH )
return 1 ;
return 0 ;
}
/**
* rio_setup_device - Allocates and sets up a RIO device
* @ net : RIO network
* @ port : Master port to send transactions
* @ destid : Current destination ID
* @ hopcount : Current hopcount
* @ do_enum : Enumeration / Discovery mode flag
*
* Allocates a RIO device and configures fields based on configuration
* space contents . If device has a destination ID register , a destination
* ID is either assigned in enumeration mode or read from configuration
* space in discovery mode . If the device has switch capabilities , then
* a switch is allocated and configured appropriately . Returns a pointer
* to a RIO device on success or NULL on failure .
*
*/
2012-11-19 22:23:25 +04:00
static struct rio_dev * rio_setup_device ( struct rio_net * net ,
2005-11-07 12:00:17 +03:00
struct rio_mport * port , u16 destid ,
u8 hopcount , int do_enum )
{
2009-05-12 02:36:02 +04:00
int ret = 0 ;
2005-11-07 12:00:17 +03:00
struct rio_dev * rdev ;
2009-05-12 02:36:02 +04:00
struct rio_switch * rswitch = NULL ;
2005-11-07 12:00:17 +03:00
int result , rdid ;
2011-01-13 04:00:39 +03:00
size_t size ;
u32 swpinfo = 0 ;
2005-11-07 12:00:17 +03:00
2011-01-13 04:00:39 +03:00
size = sizeof ( struct rio_dev ) ;
if ( rio_mport_read_config_32 ( port , destid , hopcount ,
RIO_PEF_CAR , & result ) )
return NULL ;
if ( result & ( RIO_PEF_SWITCH | RIO_PEF_MULTIPORT ) ) {
rio_mport_read_config_32 ( port , destid , hopcount ,
RIO_SWP_INFO_CAR , & swpinfo ) ;
if ( result & RIO_PEF_SWITCH ) {
size + = ( RIO_GET_TOTAL_PORTS ( swpinfo ) *
sizeof ( rswitch - > nextdev [ 0 ] ) ) + sizeof ( * rswitch ) ;
}
}
rdev = kzalloc ( size , GFP_KERNEL ) ;
2005-11-07 12:00:17 +03:00
if ( ! rdev )
2009-05-12 02:36:02 +04:00
return NULL ;
2005-11-07 12:00:17 +03:00
rdev - > net = net ;
2011-01-13 04:00:39 +03:00
rdev - > pef = result ;
rdev - > swpinfo = swpinfo ;
2005-11-07 12:00:17 +03:00
rio_mport_read_config_32 ( port , destid , hopcount , RIO_DEV_ID_CAR ,
& result ) ;
rdev - > did = result > > 16 ;
rdev - > vid = result & 0xffff ;
rio_mport_read_config_32 ( port , destid , hopcount , RIO_DEV_INFO_CAR ,
& rdev - > device_rev ) ;
rio_mport_read_config_32 ( port , destid , hopcount , RIO_ASM_ID_CAR ,
& result ) ;
rdev - > asm_did = result > > 16 ;
rdev - > asm_vid = result & 0xffff ;
rio_mport_read_config_32 ( port , destid , hopcount , RIO_ASM_INFO_CAR ,
& result ) ;
rdev - > asm_rev = result > > 16 ;
2010-05-27 01:43:59 +04:00
if ( rdev - > pef & RIO_PEF_EXT_FEATURES ) {
2005-11-07 12:00:17 +03:00
rdev - > efptr = result & 0xffff ;
2010-05-27 01:43:59 +04:00
rdev - > phys_efptr = rio_mport_get_physefb ( port , 0 , destid ,
hopcount ) ;
rdev - > em_efptr = rio_mport_get_feature ( port , 0 , destid ,
hopcount , RIO_EFB_ERR_MGMNT ) ;
}
2005-11-07 12:00:17 +03:00
rio_mport_read_config_32 ( port , destid , hopcount , RIO_SRC_OPS_CAR ,
& rdev - > src_ops ) ;
rio_mport_read_config_32 ( port , destid , hopcount , RIO_DST_OPS_CAR ,
& rdev - > dst_ops ) ;
2010-10-28 02:34:34 +04:00
if ( do_enum ) {
/* Assign component tag to device */
if ( next_comptag > = 0x10000 ) {
pr_err ( " RIO: Component Tag Counter Overflow \n " ) ;
goto cleanup ;
}
rio_mport_write_config_32 ( port , destid , hopcount ,
RIO_COMPONENT_TAG_CSR , next_comptag ) ;
rdev - > comp_tag = next_comptag + + ;
2013-07-04 02:08:50 +04:00
rdev - > do_enum = true ;
2011-01-13 04:00:42 +03:00
} else {
rio_mport_read_config_32 ( port , destid , hopcount ,
RIO_COMPONENT_TAG_CSR ,
& rdev - > comp_tag ) ;
2010-10-28 02:34:34 +04:00
}
2007-02-10 12:46:47 +03:00
if ( rio_device_has_destid ( port , rdev - > src_ops , rdev - > dst_ops ) ) {
if ( do_enum ) {
rio_set_device_id ( port , destid , hopcount , next_destid ) ;
2012-10-05 04:16:13 +04:00
rdev - > destid = next_destid ;
next_destid = rio_destid_alloc ( net ) ;
2007-02-10 12:46:47 +03:00
} else
rdev - > destid = rio_get_device_id ( port , destid , hopcount ) ;
2011-01-13 04:00:38 +03:00
rdev - > hopcount = 0xff ;
} else {
/* Switch device has an associated destID which
* will be adjusted later
*/
rdev - > destid = destid ;
rdev - > hopcount = hopcount ;
}
2005-11-07 12:00:17 +03:00
/* If a PE has both switch and other functions, show it as a switch */
if ( rio_is_switch ( rdev ) ) {
2011-01-13 04:00:39 +03:00
rswitch = rdev - > rswitch ;
2010-05-27 01:43:59 +04:00
rswitch - > port_ok = 0 ;
2013-07-04 02:08:50 +04:00
spin_lock_init ( & rswitch - > lock ) ;
2008-04-19 00:33:42 +04:00
rswitch - > route_table = kzalloc ( sizeof ( u8 ) *
RIO_MAX_ROUTE_ENTRIES ( port - > sys_size ) ,
GFP_KERNEL ) ;
2009-05-12 02:36:02 +04:00
if ( ! rswitch - > route_table )
goto cleanup ;
2005-11-07 12:00:17 +03:00
/* Initialize switch route table */
2008-04-19 00:33:42 +04:00
for ( rdid = 0 ; rdid < RIO_MAX_ROUTE_ENTRIES ( port - > sys_size ) ;
rdid + + )
2005-11-07 12:00:17 +03:00
rswitch - > route_table [ rdid ] = RIO_INVALID_ROUTE ;
2008-12-04 21:01:52 +03:00
dev_set_name ( & rdev - > dev , " %02x:s:%04x " , rdev - > net - > id ,
2013-07-04 02:09:01 +04:00
rdev - > comp_tag & RIO_CTAG_UDEVID ) ;
2005-11-07 12:00:17 +03:00
2013-07-04 02:08:50 +04:00
if ( do_enum )
rio_route_clr_table ( rdev , RIO_GLOBAL_TABLE , 0 ) ;
2010-05-27 01:43:57 +04:00
2012-10-05 04:16:05 +04:00
list_add_tail ( & rswitch - > node , & net - > switches ) ;
2005-11-07 12:00:17 +03:00
2010-05-27 01:44:01 +04:00
} else {
if ( do_enum )
/*Enable Input Output Port (transmitter reviever)*/
rio_enable_rx_tx_port ( port , 0 , destid , hopcount , 0 ) ;
2008-12-04 21:01:52 +03:00
dev_set_name ( & rdev - > dev , " %02x:e:%04x " , rdev - > net - > id ,
2013-07-04 02:09:01 +04:00
rdev - > comp_tag & RIO_CTAG_UDEVID ) ;
2010-05-27 01:44:01 +04:00
}
2005-11-07 12:00:17 +03:00
2014-04-08 02:38:56 +04:00
rdev - > dev . parent = & port - > dev ;
2013-05-25 02:55:05 +04:00
rio_attach_device ( rdev ) ;
2005-11-07 12:00:17 +03:00
device_initialize ( & rdev - > dev ) ;
rdev - > dev . release = rio_release_dev ;
rio_dev_get ( rdev ) ;
2009-04-07 06:01:15 +04:00
rdev - > dma_mask = DMA_BIT_MASK ( 32 ) ;
2005-11-07 12:00:18 +03:00
rdev - > dev . dma_mask = & rdev - > dma_mask ;
2009-04-07 06:01:15 +04:00
rdev - > dev . coherent_dma_mask = DMA_BIT_MASK ( 32 ) ;
2005-11-07 12:00:17 +03:00
2011-08-26 02:59:13 +04:00
if ( rdev - > dst_ops & RIO_DST_OPS_DOORBELL )
2005-11-07 12:00:17 +03:00
rio_init_dbell_res ( & rdev - > riores [ RIO_DOORBELL_RESOURCE ] ,
0 , 0xffff ) ;
2009-05-12 02:36:02 +04:00
ret = rio_add_device ( rdev ) ;
if ( ret )
goto cleanup ;
2005-11-07 12:00:17 +03:00
return rdev ;
2009-05-12 02:36:02 +04:00
cleanup :
2011-11-03 00:39:11 +04:00
if ( rswitch )
2009-05-12 02:36:02 +04:00
kfree ( rswitch - > route_table ) ;
2011-01-13 04:00:39 +03:00
2009-05-12 02:36:02 +04:00
kfree ( rdev ) ;
return NULL ;
2005-11-07 12:00:17 +03:00
}
/**
* rio_sport_is_active - Tests if a switch port has an active connection .
* @ port : Master port to send transaction
* @ destid : Associated destination ID for switch
* @ hopcount : Hopcount to reach switch
* @ sport : Switch port number
*
* Reads the port error status CSR for a particular switch port to
* determine if the port has an active link . Returns
2010-05-27 01:43:59 +04:00
* % RIO_PORT_N_ERR_STS_PORT_OK if the port is active or % 0 if it is
2005-11-07 12:00:17 +03:00
* inactive .
*/
static int
rio_sport_is_active ( struct rio_mport * port , u16 destid , u8 hopcount , int sport )
{
2010-05-27 01:43:59 +04:00
u32 result = 0 ;
2005-11-07 12:00:17 +03:00
u32 ext_ftr_ptr ;
2010-05-27 01:43:59 +04:00
ext_ftr_ptr = rio_mport_get_efb ( port , 0 , destid , hopcount , 0 ) ;
2005-11-07 12:00:17 +03:00
2010-05-27 01:43:59 +04:00
while ( ext_ftr_ptr ) {
rio_mport_read_config_32 ( port , destid , hopcount ,
ext_ftr_ptr , & result ) ;
result = RIO_GET_BLOCK_ID ( result ) ;
if ( ( result = = RIO_EFB_SER_EP_FREE_ID ) | |
( result = = RIO_EFB_SER_EP_FREE_ID_V13P ) | |
( result = = RIO_EFB_SER_EP_FREC_ID ) )
2005-11-07 12:00:17 +03:00
break ;
2010-05-27 01:43:59 +04:00
ext_ftr_ptr = rio_mport_get_efb ( port , 0 , destid , hopcount ,
ext_ftr_ptr ) ;
}
2005-11-07 12:00:17 +03:00
if ( ext_ftr_ptr )
rio_mport_read_config_32 ( port , destid , hopcount ,
ext_ftr_ptr +
RIO_PORT_N_ERR_STS_CSR ( sport ) ,
& result ) ;
2010-05-27 01:43:59 +04:00
return result & RIO_PORT_N_ERR_STS_PORT_OK ;
2005-11-07 12:00:17 +03:00
}
/**
* rio_get_host_deviceid_lock - Reads the Host Device ID Lock CSR on a device
* @ port : Master port to send transaction
* @ hopcount : Number of hops to the device
*
* Used during enumeration to read the Host Device ID Lock CSR on a
* RIO device . Returns the value of the lock register .
*/
static u16 rio_get_host_deviceid_lock ( struct rio_mport * port , u8 hopcount )
{
u32 result ;
2008-04-19 00:33:42 +04:00
rio_mport_read_config_32 ( port , RIO_ANY_DESTID ( port - > sys_size ) , hopcount ,
2005-11-07 12:00:17 +03:00
RIO_HOST_DID_LOCK_CSR , & result ) ;
return ( u16 ) ( result & 0xffff ) ;
}
/**
* rio_enum_peer - Recursively enumerate a RIO network through a master port
* @ net : RIO network being enumerated
* @ port : Master port to send transactions
* @ hopcount : Number of hops into the network
2010-10-28 02:34:29 +04:00
* @ prev : Previous RIO device connected to the enumerated one
* @ prev_port : Port on previous RIO device
2005-11-07 12:00:17 +03:00
*
* Recursively enumerates a RIO network . Transactions are sent via the
* master port passed in @ port .
*/
2012-11-19 22:23:25 +04:00
static int rio_enum_peer ( struct rio_net * net , struct rio_mport * port ,
2010-10-28 02:34:29 +04:00
u8 hopcount , struct rio_dev * prev , int prev_port )
2005-11-07 12:00:17 +03:00
{
struct rio_dev * rdev ;
2010-10-28 02:34:34 +04:00
u32 regval ;
2005-11-07 12:00:17 +03:00
int tmp ;
2010-10-28 02:34:32 +04:00
if ( rio_mport_chk_dev_access ( port ,
RIO_ANY_DESTID ( port - > sys_size ) , hopcount ) ) {
pr_debug ( " RIO: device access check failed \n " ) ;
return - 1 ;
}
2005-11-07 12:00:17 +03:00
if ( rio_get_host_deviceid_lock ( port , hopcount ) = = port - > host_deviceid ) {
pr_debug ( " RIO: PE already discovered by this host \n " ) ;
/*
* Already discovered by this host . Add it as another
2010-10-28 02:34:34 +04:00
* link to the existing device .
2005-11-07 12:00:17 +03:00
*/
2010-10-28 02:34:34 +04:00
rio_mport_read_config_32 ( port , RIO_ANY_DESTID ( port - > sys_size ) ,
hopcount , RIO_COMPONENT_TAG_CSR , & regval ) ;
if ( regval ) {
rdev = rio_get_comptag ( ( regval & 0xffff ) , NULL ) ;
if ( rdev & & prev & & rio_is_switch ( prev ) ) {
pr_debug ( " RIO: redundant path to %s \n " ,
rio_name ( rdev ) ) ;
prev - > rswitch - > nextdev [ prev_port ] = rdev ;
}
}
2005-11-07 12:00:17 +03:00
return 0 ;
}
/* Attempt to acquire device lock */
2008-04-19 00:33:42 +04:00
rio_mport_write_config_32 ( port , RIO_ANY_DESTID ( port - > sys_size ) ,
hopcount ,
2005-11-07 12:00:17 +03:00
RIO_HOST_DID_LOCK_CSR , port - > host_deviceid ) ;
while ( ( tmp = rio_get_host_deviceid_lock ( port , hopcount ) )
< port - > host_deviceid ) {
/* Delay a bit */
mdelay ( 1 ) ;
/* Attempt to acquire device lock again */
2008-04-19 00:33:42 +04:00
rio_mport_write_config_32 ( port , RIO_ANY_DESTID ( port - > sys_size ) ,
hopcount ,
2005-11-07 12:00:17 +03:00
RIO_HOST_DID_LOCK_CSR ,
port - > host_deviceid ) ;
}
if ( rio_get_host_deviceid_lock ( port , hopcount ) > port - > host_deviceid ) {
pr_debug (
" RIO: PE locked by a higher priority host...retreating \n " ) ;
return - 1 ;
}
/* Setup new RIO device */
2008-04-19 00:33:42 +04:00
rdev = rio_setup_device ( net , port , RIO_ANY_DESTID ( port - > sys_size ) ,
hopcount , 1 ) ;
if ( rdev ) {
2005-11-07 12:00:17 +03:00
/* Add device to the global and bus/net specific list. */
list_add_tail ( & rdev - > net_list , & net - > devices ) ;
2010-10-28 02:34:29 +04:00
rdev - > prev = prev ;
if ( prev & & rio_is_switch ( prev ) )
prev - > rswitch - > nextdev [ prev_port ] = rdev ;
2005-11-07 12:00:17 +03:00
} else
return - 1 ;
if ( rio_is_switch ( rdev ) ) {
2012-10-05 04:16:13 +04:00
int sw_destid ;
int cur_destid ;
int sw_inport ;
u16 destid ;
int port_num ;
2010-10-28 02:34:29 +04:00
sw_inport = RIO_GET_PORT_NUM ( rdev - > swpinfo ) ;
2011-01-13 04:00:38 +03:00
rio_route_add_entry ( rdev , RIO_GLOBAL_TABLE ,
2010-05-27 01:43:58 +04:00
port - > host_deviceid , sw_inport , 0 ) ;
2007-02-10 12:46:47 +03:00
rdev - > rswitch - > route_table [ port - > host_deviceid ] = sw_inport ;
2005-11-07 12:00:17 +03:00
2012-10-05 04:16:13 +04:00
destid = rio_destid_first ( net ) ;
while ( destid ! = RIO_INVALID_DESTID & & destid < next_destid ) {
if ( destid ! = port - > host_deviceid ) {
rio_route_add_entry ( rdev , RIO_GLOBAL_TABLE ,
destid , sw_inport , 0 ) ;
rdev - > rswitch - > route_table [ destid ] = sw_inport ;
}
destid = rio_destid_next ( net , destid + 1 ) ;
2005-11-07 12:00:17 +03:00
}
pr_debug (
" RIO: found %s (vid %4.4x did %4.4x) with %d ports \n " ,
2010-10-28 02:34:29 +04:00
rio_name ( rdev ) , rdev - > vid , rdev - > did ,
RIO_GET_TOTAL_PORTS ( rdev - > swpinfo ) ) ;
2007-02-10 12:46:47 +03:00
sw_destid = next_destid ;
2010-10-28 02:34:29 +04:00
for ( port_num = 0 ;
port_num < RIO_GET_TOTAL_PORTS ( rdev - > swpinfo ) ;
port_num + + ) {
2012-10-05 04:16:02 +04:00
if ( sw_inport = = port_num ) {
rio_enable_rx_tx_port ( port , 0 ,
2010-05-27 01:44:01 +04:00
RIO_ANY_DESTID ( port - > sys_size ) ,
hopcount , port_num ) ;
2010-05-27 01:43:59 +04:00
rdev - > rswitch - > port_ok | = ( 1 < < port_num ) ;
2005-11-07 12:00:17 +03:00
continue ;
2010-05-27 01:43:59 +04:00
}
2005-11-07 12:00:17 +03:00
cur_destid = next_destid ;
if ( rio_sport_is_active
2008-04-19 00:33:42 +04:00
( port , RIO_ANY_DESTID ( port - > sys_size ) , hopcount ,
port_num ) ) {
2005-11-07 12:00:17 +03:00
pr_debug (
" RIO: scanning device on port %d \n " ,
port_num ) ;
2012-10-05 04:16:02 +04:00
rio_enable_rx_tx_port ( port , 0 ,
RIO_ANY_DESTID ( port - > sys_size ) ,
hopcount , port_num ) ;
2010-05-27 01:43:59 +04:00
rdev - > rswitch - > port_ok | = ( 1 < < port_num ) ;
2011-01-13 04:00:38 +03:00
rio_route_add_entry ( rdev , RIO_GLOBAL_TABLE ,
2008-04-19 00:33:42 +04:00
RIO_ANY_DESTID ( port - > sys_size ) ,
2010-05-27 01:43:58 +04:00
port_num , 0 ) ;
2005-11-07 12:00:17 +03:00
2010-10-28 02:34:29 +04:00
if ( rio_enum_peer ( net , port , hopcount + 1 ,
rdev , port_num ) < 0 )
2005-11-07 12:00:17 +03:00
return - 1 ;
/* Update routing tables */
2012-10-05 04:16:13 +04:00
destid = rio_destid_next ( net , cur_destid + 1 ) ;
if ( destid ! = RIO_INVALID_DESTID ) {
2005-11-07 12:00:17 +03:00
for ( destid = cur_destid ;
2012-10-05 04:16:13 +04:00
destid < next_destid ; ) {
if ( destid ! = port - > host_deviceid ) {
rio_route_add_entry ( rdev ,
2005-11-07 12:00:17 +03:00
RIO_GLOBAL_TABLE ,
destid ,
2010-05-27 01:43:58 +04:00
port_num ,
0 ) ;
2012-10-05 04:16:13 +04:00
rdev - > rswitch - >
route_table [ destid ] =
port_num ;
}
destid = rio_destid_next ( net ,
destid + 1 ) ;
2005-11-07 12:00:17 +03:00
}
}
2010-05-27 01:43:59 +04:00
} else {
/* If switch supports Error Management,
* set PORT_LOCKOUT bit for unused port
*/
if ( rdev - > em_efptr )
rio_set_port_lockout ( rdev , port_num , 1 ) ;
rdev - > rswitch - > port_ok & = ~ ( 1 < < port_num ) ;
2005-11-07 12:00:17 +03:00
}
}
2007-02-10 12:46:47 +03:00
2010-05-27 01:43:59 +04:00
/* Direct Port-write messages to the enumeratiing host */
if ( ( rdev - > src_ops & RIO_SRC_OPS_PORT_WRITE ) & &
( rdev - > em_efptr ) ) {
rio_write_config_32 ( rdev ,
rdev - > em_efptr + RIO_EM_PW_TGT_DEVID ,
( port - > host_deviceid < < 16 ) |
( port - > sys_size < < 15 ) ) ;
}
rio_init_em ( rdev ) ;
2007-02-10 12:46:47 +03:00
/* Check for empty switch */
2012-10-05 04:16:13 +04:00
if ( next_destid = = sw_destid )
next_destid = rio_destid_alloc ( net ) ;
2007-02-10 12:46:47 +03:00
2011-01-13 04:00:38 +03:00
rdev - > destid = sw_destid ;
2005-11-07 12:00:17 +03:00
} else
pr_debug ( " RIO: found %s (vid %4.4x did %4.4x) \n " ,
rio_name ( rdev ) , rdev - > vid , rdev - > did ) ;
return 0 ;
}
/**
* rio_enum_complete - Tests if enumeration of a network is complete
* @ port : Master port to send transaction
*
2011-11-03 00:39:05 +04:00
* Tests the PGCCSR discovered bit for non - zero value ( enumeration
2010-05-27 01:43:59 +04:00
* complete flag ) . Return % 1 if enumeration is complete or % 0 if
2005-11-07 12:00:17 +03:00
* enumeration is incomplete .
*/
static int rio_enum_complete ( struct rio_mport * port )
{
2010-10-28 02:34:34 +04:00
u32 regval ;
2005-11-07 12:00:17 +03:00
2010-10-28 02:34:34 +04:00
rio_local_read_config_32 ( port , port - > phys_efptr + RIO_PORT_GEN_CTL_CSR ,
& regval ) ;
2011-11-03 00:39:05 +04:00
return ( regval & RIO_PORT_GEN_DISCOVERED ) ? 1 : 0 ;
2005-11-07 12:00:17 +03:00
}
/**
* rio_disc_peer - Recursively discovers a RIO network through a master port
* @ net : RIO network being discovered
* @ port : Master port to send transactions
* @ destid : Current destination ID in network
* @ hopcount : Number of hops into the network
2011-01-23 07:16:12 +03:00
* @ prev : previous rio_dev
* @ prev_port : previous port number
2005-11-07 12:00:17 +03:00
*
* Recursively discovers a RIO network . Transactions are sent via the
* master port passed in @ port .
*/
2012-11-19 22:23:25 +04:00
static int
2005-11-07 12:00:17 +03:00
rio_disc_peer ( struct rio_net * net , struct rio_mport * port , u16 destid ,
2011-01-13 04:00:41 +03:00
u8 hopcount , struct rio_dev * prev , int prev_port )
2005-11-07 12:00:17 +03:00
{
u8 port_num , route_port ;
struct rio_dev * rdev ;
u16 ndestid ;
/* Setup new RIO device */
if ( ( rdev = rio_setup_device ( net , port , destid , hopcount , 0 ) ) ) {
/* Add device to the global and bus/net specific list. */
list_add_tail ( & rdev - > net_list , & net - > devices ) ;
2011-01-13 04:00:41 +03:00
rdev - > prev = prev ;
if ( prev & & rio_is_switch ( prev ) )
prev - > rswitch - > nextdev [ prev_port ] = rdev ;
2005-11-07 12:00:17 +03:00
} else
return - 1 ;
if ( rio_is_switch ( rdev ) ) {
/* Associated destid is how we accessed this switch */
2011-01-13 04:00:38 +03:00
rdev - > destid = destid ;
2005-11-07 12:00:17 +03:00
pr_debug (
" RIO: found %s (vid %4.4x did %4.4x) with %d ports \n " ,
2010-10-28 02:34:29 +04:00
rio_name ( rdev ) , rdev - > vid , rdev - > did ,
RIO_GET_TOTAL_PORTS ( rdev - > swpinfo ) ) ;
for ( port_num = 0 ;
port_num < RIO_GET_TOTAL_PORTS ( rdev - > swpinfo ) ;
port_num + + ) {
2010-10-28 02:34:29 +04:00
if ( RIO_GET_PORT_NUM ( rdev - > swpinfo ) = = port_num )
2005-11-07 12:00:17 +03:00
continue ;
if ( rio_sport_is_active
( port , destid , hopcount , port_num ) ) {
pr_debug (
" RIO: scanning device on port %d \n " ,
port_num ) ;
2010-05-27 01:43:58 +04:00
rio_lock_device ( port , destid , hopcount , 1000 ) ;
2008-04-19 00:33:42 +04:00
for ( ndestid = 0 ;
ndestid < RIO_ANY_DESTID ( port - > sys_size ) ;
2005-11-07 12:00:17 +03:00
ndestid + + ) {
2011-01-13 04:00:38 +03:00
rio_route_get_entry ( rdev ,
2005-11-07 12:00:17 +03:00
RIO_GLOBAL_TABLE ,
ndestid ,
2010-05-27 01:43:58 +04:00
& route_port , 0 ) ;
2005-11-07 12:00:17 +03:00
if ( route_port = = port_num )
break ;
}
2010-10-28 02:34:34 +04:00
if ( ndestid = = RIO_ANY_DESTID ( port - > sys_size ) )
continue ;
2010-05-27 01:43:58 +04:00
rio_unlock_device ( port , destid , hopcount ) ;
2011-01-13 04:00:41 +03:00
if ( rio_disc_peer ( net , port , ndestid ,
hopcount + 1 , rdev , port_num ) < 0 )
2005-11-07 12:00:17 +03:00
return - 1 ;
}
}
} else
pr_debug ( " RIO: found %s (vid %4.4x did %4.4x) \n " ,
rio_name ( rdev ) , rdev - > vid , rdev - > did ) ;
return 0 ;
}
/**
* rio_mport_is_active - Tests if master port link is active
* @ port : Master port to test
*
* Reads the port error status CSR for the master port to
* determine if the port has an active link . Returns
2010-05-27 01:43:59 +04:00
* % RIO_PORT_N_ERR_STS_PORT_OK if the master port is active
2005-11-07 12:00:17 +03:00
* or % 0 if it is inactive .
*/
static int rio_mport_is_active ( struct rio_mport * port )
{
u32 result = 0 ;
u32 ext_ftr_ptr ;
int * entry = rio_mport_phys_table ;
do {
if ( ( ext_ftr_ptr =
rio_mport_get_feature ( port , 1 , 0 , 0 , * entry ) ) )
break ;
} while ( * + + entry > = 0 ) ;
if ( ext_ftr_ptr )
rio_local_read_config_32 ( port ,
ext_ftr_ptr +
RIO_PORT_N_ERR_STS_CSR ( port - > index ) ,
& result ) ;
2010-05-27 01:43:59 +04:00
return result & RIO_PORT_N_ERR_STS_PORT_OK ;
2005-11-07 12:00:17 +03:00
}
/**
* rio_alloc_net - Allocate and configure a new RIO network
* @ port : Master port associated with the RIO network
2012-10-05 04:16:13 +04:00
* @ do_enum : Enumeration / Discovery mode flag
* @ start : logical minimal start id for new net
2005-11-07 12:00:17 +03:00
*
* Allocates a RIO network structure , initializes per - network
* list heads , and adds the associated master port to the
* network list of associated master ports . Returns a
* RIO network pointer on success or % NULL on failure .
*/
2012-11-19 22:23:25 +04:00
static struct rio_net * rio_alloc_net ( struct rio_mport * port ,
2012-10-05 04:16:13 +04:00
int do_enum , u16 start )
2005-11-07 12:00:17 +03:00
{
struct rio_net * net ;
some kmalloc/memset ->kzalloc (tree wide)
Transform some calls to kmalloc/memset to a single kzalloc (or kcalloc).
Here is a short excerpt of the semantic patch performing
this transformation:
@@
type T2;
expression x;
identifier f,fld;
expression E;
expression E1,E2;
expression e1,e2,e3,y;
statement S;
@@
x =
- kmalloc
+ kzalloc
(E1,E2)
... when != \(x->fld=E;\|y=f(...,x,...);\|f(...,x,...);\|x=E;\|while(...) S\|for(e1;e2;e3) S\)
- memset((T2)x,0,E1);
@@
expression E1,E2,E3;
@@
- kzalloc(E1 * E2,E3)
+ kcalloc(E1,E2,E3)
[akpm@linux-foundation.org: get kcalloc args the right way around]
Signed-off-by: Yoann Padioleau <padator@wanadoo.fr>
Cc: Richard Henderson <rth@twiddle.net>
Cc: Ivan Kokshaysky <ink@jurassic.park.msu.ru>
Acked-by: Russell King <rmk@arm.linux.org.uk>
Cc: Bryan Wu <bryan.wu@analog.com>
Acked-by: Jiri Slaby <jirislaby@gmail.com>
Cc: Dave Airlie <airlied@linux.ie>
Acked-by: Roland Dreier <rolandd@cisco.com>
Cc: Jiri Kosina <jkosina@suse.cz>
Acked-by: Dmitry Torokhov <dtor@mail.ru>
Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Acked-by: Mauro Carvalho Chehab <mchehab@infradead.org>
Acked-by: Pierre Ossman <drzeus-list@drzeus.cx>
Cc: Jeff Garzik <jeff@garzik.org>
Cc: "David S. Miller" <davem@davemloft.net>
Acked-by: Greg KH <greg@kroah.com>
Cc: James Bottomley <James.Bottomley@steeleye.com>
Cc: "Antonino A. Daplas" <adaplas@pol.net>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2007-07-19 12:49:03 +04:00
net = kzalloc ( sizeof ( struct rio_net ) , GFP_KERNEL ) ;
2012-10-05 04:16:13 +04:00
if ( net & & do_enum ) {
2012-10-11 02:54:01 +04:00
net - > destid_table . table = kcalloc (
BITS_TO_LONGS ( RIO_MAX_ROUTE_ENTRIES ( port - > sys_size ) ) ,
2012-10-05 04:16:13 +04:00
sizeof ( long ) ,
GFP_KERNEL ) ;
if ( net - > destid_table . table = = NULL ) {
pr_err ( " RIO: failed to allocate destID table \n " ) ;
kfree ( net ) ;
net = NULL ;
} else {
net - > destid_table . start = start ;
net - > destid_table . max =
RIO_MAX_ROUTE_ENTRIES ( port - > sys_size ) ;
spin_lock_init ( & net - > destid_table . lock ) ;
}
}
2005-11-07 12:00:17 +03:00
if ( net ) {
INIT_LIST_HEAD ( & net - > node ) ;
INIT_LIST_HEAD ( & net - > devices ) ;
2012-10-05 04:16:05 +04:00
INIT_LIST_HEAD ( & net - > switches ) ;
2005-11-07 12:00:17 +03:00
INIT_LIST_HEAD ( & net - > mports ) ;
list_add_tail ( & port - > nnode , & net - > mports ) ;
net - > hport = port ;
2012-10-05 04:16:08 +04:00
net - > id = port - > id ;
2005-11-07 12:00:17 +03:00
}
return net ;
}
2007-02-10 12:46:47 +03:00
/**
* rio_update_route_tables - Updates route tables in switches
2012-10-05 04:16:05 +04:00
* @ net : RIO network to run update on
2007-02-10 12:46:47 +03:00
*
* For each enumerated device , ensure that each switch in a system
* has correct routing entries . Add routes for devices that where
* unknown dirung the first enumeration pass through the switch .
*/
2012-10-05 04:16:05 +04:00
static void rio_update_route_tables ( struct rio_net * net )
2007-02-10 12:46:47 +03:00
{
2011-01-13 04:00:39 +03:00
struct rio_dev * rdev , * swrdev ;
2007-02-10 12:46:47 +03:00
struct rio_switch * rswitch ;
u8 sport ;
u16 destid ;
2012-10-05 04:16:05 +04:00
list_for_each_entry ( rdev , & net - > devices , net_list ) {
2007-02-10 12:46:47 +03:00
2011-01-13 04:00:38 +03:00
destid = rdev - > destid ;
2007-02-10 12:46:47 +03:00
2012-10-05 04:16:05 +04:00
list_for_each_entry ( rswitch , & net - > switches , node ) {
2007-02-10 12:46:47 +03:00
if ( rio_is_switch ( rdev ) & & ( rdev - > rswitch = = rswitch ) )
continue ;
if ( RIO_INVALID_ROUTE = = rswitch - > route_table [ destid ] ) {
2011-01-13 04:00:39 +03:00
swrdev = sw_to_rio_dev ( rswitch ) ;
2010-05-27 01:43:57 +04:00
/* Skip if destid ends in empty switch*/
2011-01-13 04:00:39 +03:00
if ( swrdev - > destid = = destid )
2010-05-27 01:43:57 +04:00
continue ;
2007-02-10 12:46:47 +03:00
2011-01-13 04:00:39 +03:00
sport = RIO_GET_PORT_NUM ( swrdev - > swpinfo ) ;
2007-02-10 12:46:47 +03:00
2013-07-04 02:08:50 +04:00
rio_route_add_entry ( swrdev , RIO_GLOBAL_TABLE ,
destid , sport , 0 ) ;
rswitch - > route_table [ destid ] = sport ;
2007-02-10 12:46:47 +03:00
}
}
}
}
2010-05-27 01:43:59 +04:00
/**
* rio_init_em - Initializes RIO Error Management ( for switches )
2010-05-29 02:08:08 +04:00
* @ rdev : RIO device
2010-05-27 01:43:59 +04:00
*
* For each enumerated switch , call device - specific error management
* initialization routine ( if supplied by the switch driver ) .
*/
static void rio_init_em ( struct rio_dev * rdev )
{
if ( rio_is_switch ( rdev ) & & ( rdev - > em_efptr ) & &
2013-07-04 02:08:50 +04:00
rdev - > rswitch - > ops & & rdev - > rswitch - > ops - > em_init ) {
rdev - > rswitch - > ops - > em_init ( rdev ) ;
2010-05-27 01:43:59 +04:00
}
}
/**
* rio_pw_enable - Enables / disables port - write handling by a master port
* @ port : Master port associated with port - write handling
* @ enable : 1 = enable , 0 = disable
*/
static void rio_pw_enable ( struct rio_mport * port , int enable )
{
if ( port - > ops - > pwenable )
port - > ops - > pwenable ( port , enable ) ;
}
2005-11-07 12:00:17 +03:00
/**
* rio_enum_mport - Start enumeration through a master port
* @ mport : Master port to send transactions
2013-05-25 02:55:06 +04:00
* @ flags : Enumeration control flags
2005-11-07 12:00:17 +03:00
*
* Starts the enumeration process . If somebody has enumerated our
* master port device , then give up . If not and we have an active
* link , then start recursive peer enumeration . Returns % 0 if
* enumeration succeeds or % - EBUSY if enumeration fails .
*/
2013-07-04 02:08:49 +04:00
static int rio_enum_mport ( struct rio_mport * mport , u32 flags )
2005-11-07 12:00:17 +03:00
{
struct rio_net * net = NULL ;
int rc = 0 ;
printk ( KERN_INFO " RIO: enumerate master port %d, %s \n " , mport - > id ,
mport - > name ) ;
2013-05-25 02:55:06 +04:00
/*
* To avoid multiple start requests ( repeat enumeration is not supported
* by this method ) check if enumeration / discovery was performed for this
* mport : if mport was added into the list of mports for a net exit
* with error .
*/
if ( mport - > nnode . next | | mport - > nnode . prev )
return - EBUSY ;
2005-11-07 12:00:17 +03:00
/* If somebody else enumerated our master port device, bail. */
if ( rio_enum_host ( mport ) < 0 ) {
printk ( KERN_INFO
" RIO: master port %d device has been enumerated by a remote host \n " ,
mport - > id ) ;
rc = - EBUSY ;
goto out ;
}
/* If master port has an active link, allocate net and enum peers */
if ( rio_mport_is_active ( mport ) ) {
2012-10-05 04:16:13 +04:00
net = rio_alloc_net ( mport , 1 , 0 ) ;
if ( ! net ) {
2005-11-07 12:00:17 +03:00
printk ( KERN_ERR " RIO: failed to allocate new net \n " ) ;
rc = - ENOMEM ;
goto out ;
}
2010-05-27 01:44:01 +04:00
2012-10-05 04:16:13 +04:00
/* reserve mport destID in new net */
rio_destid_reserve ( net , mport - > host_deviceid ) ;
2010-05-27 01:44:01 +04:00
/* Enable Input Output Port (transmitter reviever) */
rio_enable_rx_tx_port ( mport , 1 , 0 , 0 , 0 ) ;
2010-10-28 02:34:34 +04:00
/* Set component tag for host */
rio_local_write_config_32 ( mport , RIO_COMPONENT_TAG_CSR ,
next_comptag + + ) ;
2012-10-05 04:16:13 +04:00
next_destid = rio_destid_alloc ( net ) ;
2010-10-28 02:34:29 +04:00
if ( rio_enum_peer ( net , mport , 0 , NULL , 0 ) < 0 ) {
2005-11-07 12:00:17 +03:00
/* A higher priority host won enumeration, bail. */
printk ( KERN_INFO
" RIO: master port %d device has lost enumeration to a remote host \n " ,
mport - > id ) ;
2012-10-05 04:16:05 +04:00
rio_clear_locks ( net ) ;
2005-11-07 12:00:17 +03:00
rc = - EBUSY ;
goto out ;
}
2012-10-05 04:16:13 +04:00
/* free the last allocated destID (unused) */
rio_destid_free ( net , next_destid ) ;
2012-10-05 04:16:05 +04:00
rio_update_route_tables ( net ) ;
rio_clear_locks ( net ) ;
2010-05-27 01:43:59 +04:00
rio_pw_enable ( mport , 1 ) ;
2005-11-07 12:00:17 +03:00
} else {
printk ( KERN_INFO " RIO: master port %d link inactive \n " ,
mport - > id ) ;
rc = - EINVAL ;
}
out :
return rc ;
}
/**
* rio_build_route_tables - Generate route tables from switch route entries
2012-10-05 04:16:05 +04:00
* @ net : RIO network to run route tables scan on
2005-11-07 12:00:17 +03:00
*
* For each switch device , generate a route table by copying existing
* route entries from the switch .
*/
2012-10-05 04:16:05 +04:00
static void rio_build_route_tables ( struct rio_net * net )
2005-11-07 12:00:17 +03:00
{
2012-10-05 04:16:05 +04:00
struct rio_switch * rswitch ;
2005-11-07 12:00:17 +03:00
struct rio_dev * rdev ;
int i ;
u8 sport ;
2012-10-05 04:16:05 +04:00
list_for_each_entry ( rswitch , & net - > switches , node ) {
rdev = sw_to_rio_dev ( rswitch ) ;
2010-05-27 01:43:58 +04:00
2012-10-05 04:16:05 +04:00
rio_lock_device ( net - > hport , rdev - > destid ,
rdev - > hopcount , 1000 ) ;
for ( i = 0 ;
i < RIO_MAX_ROUTE_ENTRIES ( net - > hport - > sys_size ) ;
i + + ) {
if ( rio_route_get_entry ( rdev , RIO_GLOBAL_TABLE ,
i , & sport , 0 ) < 0 )
continue ;
rswitch - > route_table [ i ] = sport ;
2005-11-07 12:00:17 +03:00
}
2012-10-05 04:16:05 +04:00
rio_unlock_device ( net - > hport , rdev - > destid , rdev - > hopcount ) ;
}
2005-11-07 12:00:17 +03:00
}
/**
* rio_disc_mport - Start discovery through a master port
* @ mport : Master port to send transactions
2013-05-25 02:55:06 +04:00
* @ flags : discovery control flags
2005-11-07 12:00:17 +03:00
*
* Starts the discovery process . If we have an active link ,
2013-05-25 02:55:06 +04:00
* then wait for the signal that enumeration is complete ( if wait
* is allowed ) .
2005-11-07 12:00:17 +03:00
* When enumeration completion is signaled , start recursive
* peer discovery . Returns % 0 if discovery succeeds or % - EBUSY
* on failure .
*/
2013-07-04 02:08:49 +04:00
static int rio_disc_mport ( struct rio_mport * mport , u32 flags )
2005-11-07 12:00:17 +03:00
{
struct rio_net * net = NULL ;
2012-10-05 04:16:04 +04:00
unsigned long to_end ;
2005-11-07 12:00:17 +03:00
printk ( KERN_INFO " RIO: discover master port %d, %s \n " , mport - > id ,
mport - > name ) ;
/* If master port has an active link, allocate net and discover peers */
if ( rio_mport_is_active ( mport ) ) {
2013-05-25 02:55:06 +04:00
if ( rio_enum_complete ( mport ) )
goto enum_done ;
else if ( flags & RIO_SCAN_ENUM_NO_WAIT )
return - EAGAIN ;
2012-10-05 04:16:04 +04:00
pr_debug ( " RIO: wait for enumeration to complete... \n " ) ;
2005-11-07 12:00:17 +03:00
2012-10-05 04:16:04 +04:00
to_end = jiffies + CONFIG_RAPIDIO_DISC_TIMEOUT * HZ ;
while ( time_before ( jiffies , to_end ) ) {
if ( rio_enum_complete ( mport ) )
goto enum_done ;
2012-10-11 02:53:58 +04:00
msleep ( 10 ) ;
2005-11-07 12:00:17 +03:00
}
2012-10-05 04:16:04 +04:00
pr_debug ( " RIO: discovery timeout on mport %d %s \n " ,
mport - > id , mport - > name ) ;
goto bail ;
enum_done :
pr_debug ( " RIO: ... enumeration done \n " ) ;
2012-10-05 04:16:13 +04:00
net = rio_alloc_net ( mport , 0 , 0 ) ;
2012-10-05 04:16:04 +04:00
if ( ! net ) {
printk ( KERN_ERR " RIO: Failed to allocate new net \n " ) ;
goto bail ;
}
2010-05-27 01:43:58 +04:00
/* Read DestID assigned by enumerator */
rio_local_read_config_32 ( mport , RIO_DID_CSR ,
& mport - > host_deviceid ) ;
mport - > host_deviceid = RIO_GET_DID ( mport - > sys_size ,
mport - > host_deviceid ) ;
2008-04-19 00:33:42 +04:00
if ( rio_disc_peer ( net , mport , RIO_ANY_DESTID ( mport - > sys_size ) ,
2011-01-13 04:00:41 +03:00
0 , NULL , 0 ) < 0 ) {
2005-11-07 12:00:17 +03:00
printk ( KERN_INFO
" RIO: master port %d device has failed discovery \n " ,
mport - > id ) ;
goto bail ;
}
2012-10-05 04:16:05 +04:00
rio_build_route_tables ( net ) ;
2005-11-07 12:00:17 +03:00
}
return 0 ;
2012-10-05 04:16:04 +04:00
bail :
2005-11-07 12:00:17 +03:00
return - EBUSY ;
}
2013-05-25 02:55:05 +04:00
static struct rio_scan rio_scan_ops = {
2013-07-04 02:08:53 +04:00
. owner = THIS_MODULE ,
2013-05-25 02:55:05 +04:00
. enumerate = rio_enum_mport ,
. discover = rio_disc_mport ,
} ;
static bool scan ;
module_param ( scan , bool , 0 ) ;
MODULE_PARM_DESC ( scan , " Start RapidIO network enumeration/discovery "
" (default = 0) " ) ;
/**
* rio_basic_attach :
*
* When this enumeration / discovery method is loaded as a module this function
* registers its specific enumeration and discover routines for all available
* RapidIO mport devices . The " scan " command line parameter controls ability of
* the module to start RapidIO enumeration / discovery automatically .
*
* Returns 0 for success or - EIO if unable to register itself .
*
* This enumeration / discovery method cannot be unloaded and therefore does not
* provide a matching cleanup_module routine .
*/
static int __init rio_basic_attach ( void )
{
if ( rio_register_scan ( RIO_MPORT_ANY , & rio_scan_ops ) )
return - EIO ;
if ( scan )
rio_init_mports ( ) ;
return 0 ;
}
late_initcall ( rio_basic_attach ) ;
MODULE_DESCRIPTION ( " Basic RapidIO enumeration/discovery " ) ;
MODULE_LICENSE ( " GPL " ) ;