mirror of
https://github.com/Fishwaldo/Star64_linux.git
synced 2025-06-20 05:31:15 +00:00
rapidio: use device lists handling on per-net basis
Modify handling of device lists to resolve issues caused by using single global list of RIO devices during enumeration/discovery. The most common sign of existing issue is incorrect contents of switch routing tables in systems with multiple mport controllers while single-port configuration performs as expected. Signed-off-by: Alexandre Bounine <alexandre.bounine@idt.com> Cc: Matt Porter <mporter@kernel.crashing.org> Cc: Li Yang <leoli@freescale.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
parent
fa3dbaa010
commit
a7071efc20
2 changed files with 32 additions and 29 deletions
|
@ -38,7 +38,6 @@
|
||||||
#include "rio.h"
|
#include "rio.h"
|
||||||
|
|
||||||
LIST_HEAD(rio_devices);
|
LIST_HEAD(rio_devices);
|
||||||
static LIST_HEAD(rio_switches);
|
|
||||||
|
|
||||||
static void rio_init_em(struct rio_dev *rdev);
|
static void rio_init_em(struct rio_dev *rdev);
|
||||||
|
|
||||||
|
@ -104,14 +103,15 @@ static void rio_local_set_device_id(struct rio_mport *port, u16 did)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* rio_clear_locks- Release all host locks and signal enumeration complete
|
* rio_clear_locks- Release all host locks and signal enumeration complete
|
||||||
* @port: Master port to issue transaction
|
* @net: RIO network to run on
|
||||||
*
|
*
|
||||||
* Marks the component tag CSR on each device with the enumeration
|
* Marks the component tag CSR on each device with the enumeration
|
||||||
* complete flag. When complete, it then release the host locks on
|
* complete flag. When complete, it then release the host locks on
|
||||||
* each device. Returns 0 on success or %-EINVAL on failure.
|
* each device. Returns 0 on success or %-EINVAL on failure.
|
||||||
*/
|
*/
|
||||||
static int rio_clear_locks(struct rio_mport *port)
|
static int rio_clear_locks(struct rio_net *net)
|
||||||
{
|
{
|
||||||
|
struct rio_mport *port = net->hport;
|
||||||
struct rio_dev *rdev;
|
struct rio_dev *rdev;
|
||||||
u32 result;
|
u32 result;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
@ -126,7 +126,7 @@ static int rio_clear_locks(struct rio_mport *port)
|
||||||
result);
|
result);
|
||||||
ret = -EINVAL;
|
ret = -EINVAL;
|
||||||
}
|
}
|
||||||
list_for_each_entry(rdev, &rio_devices, global_list) {
|
list_for_each_entry(rdev, &net->devices, net_list) {
|
||||||
rio_write_config_32(rdev, RIO_HOST_DID_LOCK_CSR,
|
rio_write_config_32(rdev, RIO_HOST_DID_LOCK_CSR,
|
||||||
port->host_deviceid);
|
port->host_deviceid);
|
||||||
rio_read_config_32(rdev, RIO_HOST_DID_LOCK_CSR, &result);
|
rio_read_config_32(rdev, RIO_HOST_DID_LOCK_CSR, &result);
|
||||||
|
@ -479,7 +479,7 @@ static struct rio_dev __devinit *rio_setup_device(struct rio_net *net,
|
||||||
rswitch->clr_table(port, destid, hopcount,
|
rswitch->clr_table(port, destid, hopcount,
|
||||||
RIO_GLOBAL_TABLE);
|
RIO_GLOBAL_TABLE);
|
||||||
|
|
||||||
list_add_tail(&rswitch->node, &rio_switches);
|
list_add_tail(&rswitch->node, &net->switches);
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
if (do_enum)
|
if (do_enum)
|
||||||
|
@ -1058,6 +1058,7 @@ static struct rio_net __devinit *rio_alloc_net(struct rio_mport *port)
|
||||||
if (net) {
|
if (net) {
|
||||||
INIT_LIST_HEAD(&net->node);
|
INIT_LIST_HEAD(&net->node);
|
||||||
INIT_LIST_HEAD(&net->devices);
|
INIT_LIST_HEAD(&net->devices);
|
||||||
|
INIT_LIST_HEAD(&net->switches);
|
||||||
INIT_LIST_HEAD(&net->mports);
|
INIT_LIST_HEAD(&net->mports);
|
||||||
list_add_tail(&port->nnode, &net->mports);
|
list_add_tail(&port->nnode, &net->mports);
|
||||||
net->hport = port;
|
net->hport = port;
|
||||||
|
@ -1068,24 +1069,24 @@ static struct rio_net __devinit *rio_alloc_net(struct rio_mport *port)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* rio_update_route_tables- Updates route tables in switches
|
* rio_update_route_tables- Updates route tables in switches
|
||||||
* @port: Master port associated with the RIO network
|
* @net: RIO network to run update on
|
||||||
*
|
*
|
||||||
* For each enumerated device, ensure that each switch in a system
|
* For each enumerated device, ensure that each switch in a system
|
||||||
* has correct routing entries. Add routes for devices that where
|
* has correct routing entries. Add routes for devices that where
|
||||||
* unknown dirung the first enumeration pass through the switch.
|
* unknown dirung the first enumeration pass through the switch.
|
||||||
*/
|
*/
|
||||||
static void rio_update_route_tables(struct rio_mport *port)
|
static void rio_update_route_tables(struct rio_net *net)
|
||||||
{
|
{
|
||||||
struct rio_dev *rdev, *swrdev;
|
struct rio_dev *rdev, *swrdev;
|
||||||
struct rio_switch *rswitch;
|
struct rio_switch *rswitch;
|
||||||
u8 sport;
|
u8 sport;
|
||||||
u16 destid;
|
u16 destid;
|
||||||
|
|
||||||
list_for_each_entry(rdev, &rio_devices, global_list) {
|
list_for_each_entry(rdev, &net->devices, net_list) {
|
||||||
|
|
||||||
destid = rdev->destid;
|
destid = rdev->destid;
|
||||||
|
|
||||||
list_for_each_entry(rswitch, &rio_switches, node) {
|
list_for_each_entry(rswitch, &net->switches, node) {
|
||||||
|
|
||||||
if (rio_is_switch(rdev) && (rdev->rswitch == rswitch))
|
if (rio_is_switch(rdev) && (rdev->rswitch == rswitch))
|
||||||
continue;
|
continue;
|
||||||
|
@ -1181,12 +1182,12 @@ int __devinit rio_enum_mport(struct rio_mport *mport)
|
||||||
printk(KERN_INFO
|
printk(KERN_INFO
|
||||||
"RIO: master port %d device has lost enumeration to a remote host\n",
|
"RIO: master port %d device has lost enumeration to a remote host\n",
|
||||||
mport->id);
|
mport->id);
|
||||||
rio_clear_locks(mport);
|
rio_clear_locks(net);
|
||||||
rc = -EBUSY;
|
rc = -EBUSY;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
rio_update_route_tables(mport);
|
rio_update_route_tables(net);
|
||||||
rio_clear_locks(mport);
|
rio_clear_locks(net);
|
||||||
rio_pw_enable(mport, 1);
|
rio_pw_enable(mport, 1);
|
||||||
} else {
|
} else {
|
||||||
printk(KERN_INFO "RIO: master port %d link inactive\n",
|
printk(KERN_INFO "RIO: master port %d link inactive\n",
|
||||||
|
@ -1200,32 +1201,33 @@ int __devinit rio_enum_mport(struct rio_mport *mport)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* rio_build_route_tables- Generate route tables from switch route entries
|
* rio_build_route_tables- Generate route tables from switch route entries
|
||||||
|
* @net: RIO network to run route tables scan on
|
||||||
*
|
*
|
||||||
* For each switch device, generate a route table by copying existing
|
* For each switch device, generate a route table by copying existing
|
||||||
* route entries from the switch.
|
* route entries from the switch.
|
||||||
*/
|
*/
|
||||||
static void rio_build_route_tables(void)
|
static void rio_build_route_tables(struct rio_net *net)
|
||||||
{
|
{
|
||||||
|
struct rio_switch *rswitch;
|
||||||
struct rio_dev *rdev;
|
struct rio_dev *rdev;
|
||||||
int i;
|
int i;
|
||||||
u8 sport;
|
u8 sport;
|
||||||
|
|
||||||
list_for_each_entry(rdev, &rio_devices, global_list)
|
list_for_each_entry(rswitch, &net->switches, node) {
|
||||||
if (rio_is_switch(rdev)) {
|
rdev = sw_to_rio_dev(rswitch);
|
||||||
rio_lock_device(rdev->net->hport, rdev->destid,
|
|
||||||
|
rio_lock_device(net->hport, rdev->destid,
|
||||||
rdev->hopcount, 1000);
|
rdev->hopcount, 1000);
|
||||||
for (i = 0;
|
for (i = 0;
|
||||||
i < RIO_MAX_ROUTE_ENTRIES(rdev->net->hport->sys_size);
|
i < RIO_MAX_ROUTE_ENTRIES(net->hport->sys_size);
|
||||||
i++) {
|
i++) {
|
||||||
if (rio_route_get_entry(rdev,
|
if (rio_route_get_entry(rdev, RIO_GLOBAL_TABLE,
|
||||||
RIO_GLOBAL_TABLE, i, &sport, 0) < 0)
|
i, &sport, 0) < 0)
|
||||||
continue;
|
continue;
|
||||||
rdev->rswitch->route_table[i] = sport;
|
rswitch->route_table[i] = sport;
|
||||||
}
|
}
|
||||||
|
|
||||||
rio_unlock_device(rdev->net->hport,
|
rio_unlock_device(net->hport, rdev->destid, rdev->hopcount);
|
||||||
rdev->destid,
|
|
||||||
rdev->hopcount);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1284,7 +1286,7 @@ enum_done:
|
||||||
goto bail;
|
goto bail;
|
||||||
}
|
}
|
||||||
|
|
||||||
rio_build_route_tables();
|
rio_build_route_tables(net);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
|
@ -275,6 +275,7 @@ struct rio_mport {
|
||||||
struct rio_net {
|
struct rio_net {
|
||||||
struct list_head node; /* node in list of networks */
|
struct list_head node; /* node in list of networks */
|
||||||
struct list_head devices; /* list of devices in this net */
|
struct list_head devices; /* list of devices in this net */
|
||||||
|
struct list_head switches; /* list of switches in this net */
|
||||||
struct list_head mports; /* list of ports accessing net */
|
struct list_head mports; /* list of ports accessing net */
|
||||||
struct rio_mport *hport; /* primary port for accessing net */
|
struct rio_mport *hport; /* primary port for accessing net */
|
||||||
unsigned char id; /* RIO network ID */
|
unsigned char id; /* RIO network ID */
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue