diff options
author | Alexandre Bounine <alexandre.bounine@idt.com> | 2010-05-26 17:43:59 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2010-05-27 12:12:50 -0400 |
commit | e5cabeb3d60f9cd3e3950aff071319ae0e2d08d8 (patch) | |
tree | e866f1a9076608630a40f21f0a50c073dedb0e57 /drivers/rapidio | |
parent | 818a04a0bb93643d57dd8935815de2ff307b58a3 (diff) |
rapidio: add Port-Write handling for EM
Add RapidIO Port-Write message handling in the context of Error
Management Extensions Specification Rev.1.3.
Signed-off-by: Alexandre Bounine <alexandre.bounine@idt.com>
Tested-by: Thomas Moll <thomas.moll@sysgo.com>
Cc: Matt Porter <mporter@kernel.crashing.org>
Cc: Li Yang <leoli@freescale.com>
Cc: Kumar Gala <galak@kernel.crashing.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'drivers/rapidio')
-rw-r--r-- | drivers/rapidio/rio-scan.c | 166 | ||||
-rw-r--r-- | drivers/rapidio/rio.c | 328 | ||||
-rw-r--r-- | drivers/rapidio/rio.h | 30 | ||||
-rw-r--r-- | drivers/rapidio/switches/tsi568.c | 24 | ||||
-rw-r--r-- | drivers/rapidio/switches/tsi57x.c | 153 |
5 files changed, 665 insertions, 36 deletions
diff --git a/drivers/rapidio/rio-scan.c b/drivers/rapidio/rio-scan.c index cbf0d5f4fba8..74633cc6b2eb 100644 --- a/drivers/rapidio/rio-scan.c +++ b/drivers/rapidio/rio-scan.c | |||
@@ -4,6 +4,10 @@ | |||
4 | * Copyright 2005 MontaVista Software, Inc. | 4 | * Copyright 2005 MontaVista Software, Inc. |
5 | * Matt Porter <mporter@kernel.crashing.org> | 5 | * Matt Porter <mporter@kernel.crashing.org> |
6 | * | 6 | * |
7 | * Copyright 2009 Integrated Device Technology, Inc. | ||
8 | * Alex Bounine <alexandre.bounine@idt.com> | ||
9 | * - Added Port-Write/Error Management initialization and handling | ||
10 | * | ||
7 | * This program is free software; you can redistribute it and/or modify it | 11 | * This program is free software; you can redistribute it and/or modify it |
8 | * under the terms of the GNU General Public License as published by the | 12 | * under the terms of the GNU General Public License as published by the |
9 | * Free Software Foundation; either version 2 of the License, or (at your | 13 | * Free Software Foundation; either version 2 of the License, or (at your |
@@ -31,15 +35,16 @@ | |||
31 | LIST_HEAD(rio_devices); | 35 | LIST_HEAD(rio_devices); |
32 | static LIST_HEAD(rio_switches); | 36 | static LIST_HEAD(rio_switches); |
33 | 37 | ||
34 | #define RIO_ENUM_CMPL_MAGIC 0xdeadbeef | ||
35 | |||
36 | static void rio_enum_timeout(unsigned long); | 38 | static void rio_enum_timeout(unsigned long); |
37 | 39 | ||
40 | static void rio_init_em(struct rio_dev *rdev); | ||
41 | |||
38 | DEFINE_SPINLOCK(rio_global_list_lock); | 42 | DEFINE_SPINLOCK(rio_global_list_lock); |
39 | 43 | ||
40 | static int next_destid = 0; | 44 | static int next_destid = 0; |
41 | static int next_switchid = 0; | 45 | static int next_switchid = 0; |
42 | static int next_net = 0; | 46 | static int next_net = 0; |
47 | static int next_comptag; | ||
43 | 48 | ||
44 | static struct timer_list rio_enum_timer = | 49 | static struct timer_list rio_enum_timer = |
45 | TIMER_INITIALIZER(rio_enum_timeout, 0, 0); | 50 | TIMER_INITIALIZER(rio_enum_timeout, 0, 0); |
@@ -52,13 +57,6 @@ static int rio_mport_phys_table[] = { | |||
52 | -1, | 57 | -1, |
53 | }; | 58 | }; |
54 | 59 | ||
55 | static int rio_sport_phys_table[] = { | ||
56 | RIO_EFB_PAR_EP_FREE_ID, | ||
57 | RIO_EFB_SER_EP_FREE_ID, | ||
58 | RIO_EFB_SER_EP_FREC_ID, | ||
59 | -1, | ||
60 | }; | ||
61 | |||
62 | /** | 60 | /** |
63 | * rio_get_device_id - Get the base/extended device id for a device | 61 | * rio_get_device_id - Get the base/extended device id for a device |
64 | * @port: RIO master port | 62 | * @port: RIO master port |
@@ -119,12 +117,26 @@ static int rio_clear_locks(struct rio_mport *port) | |||
119 | u32 result; | 117 | u32 result; |
120 | int ret = 0; | 118 | int ret = 0; |
121 | 119 | ||
122 | /* Write component tag CSR magic complete value */ | 120 | /* Assign component tag to all devices */ |
123 | rio_local_write_config_32(port, RIO_COMPONENT_TAG_CSR, | 121 | next_comptag = 1; |
124 | RIO_ENUM_CMPL_MAGIC); | 122 | rio_local_write_config_32(port, RIO_COMPONENT_TAG_CSR, next_comptag++); |
125 | list_for_each_entry(rdev, &rio_devices, global_list) | 123 | |
126 | rio_write_config_32(rdev, RIO_COMPONENT_TAG_CSR, | 124 | list_for_each_entry(rdev, &rio_devices, global_list) { |
127 | RIO_ENUM_CMPL_MAGIC); | 125 | /* Mark device as discovered */ |
126 | rio_read_config_32(rdev, | ||
127 | rdev->phys_efptr + RIO_PORT_GEN_CTL_CSR, | ||
128 | &result); | ||
129 | rio_write_config_32(rdev, | ||
130 | rdev->phys_efptr + RIO_PORT_GEN_CTL_CSR, | ||
131 | result | RIO_PORT_GEN_DISCOVERED); | ||
132 | |||
133 | rio_write_config_32(rdev, RIO_COMPONENT_TAG_CSR, next_comptag); | ||
134 | rdev->comp_tag = next_comptag++; | ||
135 | if (next_comptag >= 0x10000) { | ||
136 | pr_err("RIO: Component Tag Counter Overflow\n"); | ||
137 | break; | ||
138 | } | ||
139 | } | ||
128 | 140 | ||
129 | /* Release host device id locks */ | 141 | /* Release host device id locks */ |
130 | rio_local_write_config_32(port, RIO_HOST_DID_LOCK_CSR, | 142 | rio_local_write_config_32(port, RIO_HOST_DID_LOCK_CSR, |
@@ -267,6 +279,30 @@ static void rio_route_set_ops(struct rio_dev *rdev) | |||
267 | } | 279 | } |
268 | 280 | ||
269 | /** | 281 | /** |
282 | * rio_em_set_ops- Sets Error Managment operations for a particular vendor switch | ||
283 | * @rdev: RIO device | ||
284 | * | ||
285 | * Searches the RIO EM ops table for known switch types. If the vid | ||
286 | * and did match a switch table entry, then set the em_init() and | ||
287 | * em_handle() ops to the table entry values. | ||
288 | */ | ||
289 | static void rio_em_set_ops(struct rio_dev *rdev) | ||
290 | { | ||
291 | struct rio_em_ops *cur = __start_rio_em_ops; | ||
292 | struct rio_em_ops *end = __end_rio_em_ops; | ||
293 | |||
294 | while (cur < end) { | ||
295 | if ((cur->vid == rdev->vid) && (cur->did == rdev->did)) { | ||
296 | pr_debug("RIO: adding EM ops for %s\n", rio_name(rdev)); | ||
297 | rdev->rswitch->em_init = cur->init_hook; | ||
298 | rdev->rswitch->em_handle = cur->handler_hook; | ||
299 | break; | ||
300 | } | ||
301 | cur++; | ||
302 | } | ||
303 | } | ||
304 | |||
305 | /** | ||
270 | * rio_add_device- Adds a RIO device to the device model | 306 | * rio_add_device- Adds a RIO device to the device model |
271 | * @rdev: RIO device | 307 | * @rdev: RIO device |
272 | * | 308 | * |
@@ -336,8 +372,14 @@ static struct rio_dev __devinit *rio_setup_device(struct rio_net *net, | |||
336 | rdev->asm_rev = result >> 16; | 372 | rdev->asm_rev = result >> 16; |
337 | rio_mport_read_config_32(port, destid, hopcount, RIO_PEF_CAR, | 373 | rio_mport_read_config_32(port, destid, hopcount, RIO_PEF_CAR, |
338 | &rdev->pef); | 374 | &rdev->pef); |
339 | if (rdev->pef & RIO_PEF_EXT_FEATURES) | 375 | if (rdev->pef & RIO_PEF_EXT_FEATURES) { |
340 | rdev->efptr = result & 0xffff; | 376 | rdev->efptr = result & 0xffff; |
377 | rdev->phys_efptr = rio_mport_get_physefb(port, 0, destid, | ||
378 | hopcount); | ||
379 | |||
380 | rdev->em_efptr = rio_mport_get_feature(port, 0, destid, | ||
381 | hopcount, RIO_EFB_ERR_MGMNT); | ||
382 | } | ||
341 | 383 | ||
342 | rio_mport_read_config_32(port, destid, hopcount, RIO_SRC_OPS_CAR, | 384 | rio_mport_read_config_32(port, destid, hopcount, RIO_SRC_OPS_CAR, |
343 | &rdev->src_ops); | 385 | &rdev->src_ops); |
@@ -366,6 +408,7 @@ static struct rio_dev __devinit *rio_setup_device(struct rio_net *net, | |||
366 | rswitch->switchid = next_switchid; | 408 | rswitch->switchid = next_switchid; |
367 | rswitch->hopcount = hopcount; | 409 | rswitch->hopcount = hopcount; |
368 | rswitch->destid = destid; | 410 | rswitch->destid = destid; |
411 | rswitch->port_ok = 0; | ||
369 | rswitch->route_table = kzalloc(sizeof(u8)* | 412 | rswitch->route_table = kzalloc(sizeof(u8)* |
370 | RIO_MAX_ROUTE_ENTRIES(port->sys_size), | 413 | RIO_MAX_ROUTE_ENTRIES(port->sys_size), |
371 | GFP_KERNEL); | 414 | GFP_KERNEL); |
@@ -379,6 +422,7 @@ static struct rio_dev __devinit *rio_setup_device(struct rio_net *net, | |||
379 | dev_set_name(&rdev->dev, "%02x:s:%04x", rdev->net->id, | 422 | dev_set_name(&rdev->dev, "%02x:s:%04x", rdev->net->id, |
380 | rdev->rswitch->switchid); | 423 | rdev->rswitch->switchid); |
381 | rio_route_set_ops(rdev); | 424 | rio_route_set_ops(rdev); |
425 | rio_em_set_ops(rdev); | ||
382 | 426 | ||
383 | if (do_enum && rdev->rswitch->clr_table) | 427 | if (do_enum && rdev->rswitch->clr_table) |
384 | rdev->rswitch->clr_table(port, destid, hopcount, | 428 | rdev->rswitch->clr_table(port, destid, hopcount, |
@@ -429,23 +473,29 @@ cleanup: | |||
429 | * | 473 | * |
430 | * Reads the port error status CSR for a particular switch port to | 474 | * Reads the port error status CSR for a particular switch port to |
431 | * determine if the port has an active link. Returns | 475 | * determine if the port has an active link. Returns |
432 | * %PORT_N_ERR_STS_PORT_OK if the port is active or %0 if it is | 476 | * %RIO_PORT_N_ERR_STS_PORT_OK if the port is active or %0 if it is |
433 | * inactive. | 477 | * inactive. |
434 | */ | 478 | */ |
435 | static int | 479 | static int |
436 | rio_sport_is_active(struct rio_mport *port, u16 destid, u8 hopcount, int sport) | 480 | rio_sport_is_active(struct rio_mport *port, u16 destid, u8 hopcount, int sport) |
437 | { | 481 | { |
438 | u32 result; | 482 | u32 result = 0; |
439 | u32 ext_ftr_ptr; | 483 | u32 ext_ftr_ptr; |
440 | 484 | ||
441 | int *entry = rio_sport_phys_table; | 485 | ext_ftr_ptr = rio_mport_get_efb(port, 0, destid, hopcount, 0); |
442 | |||
443 | do { | ||
444 | if ((ext_ftr_ptr = | ||
445 | rio_mport_get_feature(port, 0, destid, hopcount, *entry))) | ||
446 | 486 | ||
487 | while (ext_ftr_ptr) { | ||
488 | rio_mport_read_config_32(port, destid, hopcount, | ||
489 | ext_ftr_ptr, &result); | ||
490 | result = RIO_GET_BLOCK_ID(result); | ||
491 | if ((result == RIO_EFB_SER_EP_FREE_ID) || | ||
492 | (result == RIO_EFB_SER_EP_FREE_ID_V13P) || | ||
493 | (result == RIO_EFB_SER_EP_FREC_ID)) | ||
447 | break; | 494 | break; |
448 | } while (*++entry >= 0); | 495 | |
496 | ext_ftr_ptr = rio_mport_get_efb(port, 0, destid, hopcount, | ||
497 | ext_ftr_ptr); | ||
498 | } | ||
449 | 499 | ||
450 | if (ext_ftr_ptr) | 500 | if (ext_ftr_ptr) |
451 | rio_mport_read_config_32(port, destid, hopcount, | 501 | rio_mport_read_config_32(port, destid, hopcount, |
@@ -453,7 +503,7 @@ rio_sport_is_active(struct rio_mport *port, u16 destid, u8 hopcount, int sport) | |||
453 | RIO_PORT_N_ERR_STS_CSR(sport), | 503 | RIO_PORT_N_ERR_STS_CSR(sport), |
454 | &result); | 504 | &result); |
455 | 505 | ||
456 | return (result & PORT_N_ERR_STS_PORT_OK); | 506 | return result & RIO_PORT_N_ERR_STS_PORT_OK; |
457 | } | 507 | } |
458 | 508 | ||
459 | /** | 509 | /** |
@@ -762,8 +812,10 @@ static int __devinit rio_enum_peer(struct rio_net *net, struct rio_mport *port, | |||
762 | rio_name(rdev), rdev->vid, rdev->did, num_ports); | 812 | rio_name(rdev), rdev->vid, rdev->did, num_ports); |
763 | sw_destid = next_destid; | 813 | sw_destid = next_destid; |
764 | for (port_num = 0; port_num < num_ports; port_num++) { | 814 | for (port_num = 0; port_num < num_ports; port_num++) { |
765 | if (sw_inport == port_num) | 815 | if (sw_inport == port_num) { |
816 | rdev->rswitch->port_ok |= (1 << port_num); | ||
766 | continue; | 817 | continue; |
818 | } | ||
767 | 819 | ||
768 | cur_destid = next_destid; | 820 | cur_destid = next_destid; |
769 | 821 | ||
@@ -773,6 +825,7 @@ static int __devinit rio_enum_peer(struct rio_net *net, struct rio_mport *port, | |||
773 | pr_debug( | 825 | pr_debug( |
774 | "RIO: scanning device on port %d\n", | 826 | "RIO: scanning device on port %d\n", |
775 | port_num); | 827 | port_num); |
828 | rdev->rswitch->port_ok |= (1 << port_num); | ||
776 | rio_route_add_entry(port, rdev->rswitch, | 829 | rio_route_add_entry(port, rdev->rswitch, |
777 | RIO_GLOBAL_TABLE, | 830 | RIO_GLOBAL_TABLE, |
778 | RIO_ANY_DESTID(port->sys_size), | 831 | RIO_ANY_DESTID(port->sys_size), |
@@ -797,9 +850,28 @@ static int __devinit rio_enum_peer(struct rio_net *net, struct rio_mport *port, | |||
797 | port_num; | 850 | port_num; |
798 | } | 851 | } |
799 | } | 852 | } |
853 | } else { | ||
854 | /* If switch supports Error Management, | ||
855 | * set PORT_LOCKOUT bit for unused port | ||
856 | */ | ||
857 | if (rdev->em_efptr) | ||
858 | rio_set_port_lockout(rdev, port_num, 1); | ||
859 | |||
860 | rdev->rswitch->port_ok &= ~(1 << port_num); | ||
800 | } | 861 | } |
801 | } | 862 | } |
802 | 863 | ||
864 | /* Direct Port-write messages to the enumeratiing host */ | ||
865 | if ((rdev->src_ops & RIO_SRC_OPS_PORT_WRITE) && | ||
866 | (rdev->em_efptr)) { | ||
867 | rio_write_config_32(rdev, | ||
868 | rdev->em_efptr + RIO_EM_PW_TGT_DEVID, | ||
869 | (port->host_deviceid << 16) | | ||
870 | (port->sys_size << 15)); | ||
871 | } | ||
872 | |||
873 | rio_init_em(rdev); | ||
874 | |||
803 | /* Check for empty switch */ | 875 | /* Check for empty switch */ |
804 | if (next_destid == sw_destid) { | 876 | if (next_destid == sw_destid) { |
805 | next_destid++; | 877 | next_destid++; |
@@ -819,21 +891,16 @@ static int __devinit rio_enum_peer(struct rio_net *net, struct rio_mport *port, | |||
819 | * rio_enum_complete- Tests if enumeration of a network is complete | 891 | * rio_enum_complete- Tests if enumeration of a network is complete |
820 | * @port: Master port to send transaction | 892 | * @port: Master port to send transaction |
821 | * | 893 | * |
822 | * Tests the Component Tag CSR for presence of the magic enumeration | 894 | * Tests the Component Tag CSR for non-zero value (enumeration |
823 | * complete flag. Return %1 if enumeration is complete or %0 if | 895 | * complete flag). Return %1 if enumeration is complete or %0 if |
824 | * enumeration is incomplete. | 896 | * enumeration is incomplete. |
825 | */ | 897 | */ |
826 | static int rio_enum_complete(struct rio_mport *port) | 898 | static int rio_enum_complete(struct rio_mport *port) |
827 | { | 899 | { |
828 | u32 tag_csr; | 900 | u32 tag_csr; |
829 | int ret = 0; | ||
830 | 901 | ||
831 | rio_local_read_config_32(port, RIO_COMPONENT_TAG_CSR, &tag_csr); | 902 | rio_local_read_config_32(port, RIO_COMPONENT_TAG_CSR, &tag_csr); |
832 | 903 | return (tag_csr & 0xffff) ? 1 : 0; | |
833 | if (tag_csr == RIO_ENUM_CMPL_MAGIC) | ||
834 | ret = 1; | ||
835 | |||
836 | return ret; | ||
837 | } | 904 | } |
838 | 905 | ||
839 | /** | 906 | /** |
@@ -915,7 +982,7 @@ rio_disc_peer(struct rio_net *net, struct rio_mport *port, u16 destid, | |||
915 | * | 982 | * |
916 | * Reads the port error status CSR for the master port to | 983 | * Reads the port error status CSR for the master port to |
917 | * determine if the port has an active link. Returns | 984 | * determine if the port has an active link. Returns |
918 | * %PORT_N_ERR_STS_PORT_OK if the master port is active | 985 | * %RIO_PORT_N_ERR_STS_PORT_OK if the master port is active |
919 | * or %0 if it is inactive. | 986 | * or %0 if it is inactive. |
920 | */ | 987 | */ |
921 | static int rio_mport_is_active(struct rio_mport *port) | 988 | static int rio_mport_is_active(struct rio_mport *port) |
@@ -936,7 +1003,7 @@ static int rio_mport_is_active(struct rio_mport *port) | |||
936 | RIO_PORT_N_ERR_STS_CSR(port->index), | 1003 | RIO_PORT_N_ERR_STS_CSR(port->index), |
937 | &result); | 1004 | &result); |
938 | 1005 | ||
939 | return (result & PORT_N_ERR_STS_PORT_OK); | 1006 | return result & RIO_PORT_N_ERR_STS_PORT_OK; |
940 | } | 1007 | } |
941 | 1008 | ||
942 | /** | 1009 | /** |
@@ -1008,6 +1075,32 @@ static void rio_update_route_tables(struct rio_mport *port) | |||
1008 | } | 1075 | } |
1009 | 1076 | ||
1010 | /** | 1077 | /** |
1078 | * rio_init_em - Initializes RIO Error Management (for switches) | ||
1079 | * @port: Master port associated with the RIO network | ||
1080 | * | ||
1081 | * For each enumerated switch, call device-specific error management | ||
1082 | * initialization routine (if supplied by the switch driver). | ||
1083 | */ | ||
1084 | static void rio_init_em(struct rio_dev *rdev) | ||
1085 | { | ||
1086 | if (rio_is_switch(rdev) && (rdev->em_efptr) && | ||
1087 | (rdev->rswitch->em_init)) { | ||
1088 | rdev->rswitch->em_init(rdev); | ||
1089 | } | ||
1090 | } | ||
1091 | |||
1092 | /** | ||
1093 | * rio_pw_enable - Enables/disables port-write handling by a master port | ||
1094 | * @port: Master port associated with port-write handling | ||
1095 | * @enable: 1=enable, 0=disable | ||
1096 | */ | ||
1097 | static void rio_pw_enable(struct rio_mport *port, int enable) | ||
1098 | { | ||
1099 | if (port->ops->pwenable) | ||
1100 | port->ops->pwenable(port, enable); | ||
1101 | } | ||
1102 | |||
1103 | /** | ||
1011 | * rio_enum_mport- Start enumeration through a master port | 1104 | * rio_enum_mport- Start enumeration through a master port |
1012 | * @mport: Master port to send transactions | 1105 | * @mport: Master port to send transactions |
1013 | * | 1106 | * |
@@ -1050,6 +1143,7 @@ int __devinit rio_enum_mport(struct rio_mport *mport) | |||
1050 | } | 1143 | } |
1051 | rio_update_route_tables(mport); | 1144 | rio_update_route_tables(mport); |
1052 | rio_clear_locks(mport); | 1145 | rio_clear_locks(mport); |
1146 | rio_pw_enable(mport, 1); | ||
1053 | } else { | 1147 | } else { |
1054 | printk(KERN_INFO "RIO: master port %d link inactive\n", | 1148 | printk(KERN_INFO "RIO: master port %d link inactive\n", |
1055 | mport->id); | 1149 | mport->id); |
diff --git a/drivers/rapidio/rio.c b/drivers/rapidio/rio.c index 67a379216959..8fa732e46bf6 100644 --- a/drivers/rapidio/rio.c +++ b/drivers/rapidio/rio.c | |||
@@ -5,6 +5,10 @@ | |||
5 | * Copyright 2005 MontaVista Software, Inc. | 5 | * Copyright 2005 MontaVista Software, Inc. |
6 | * Matt Porter <mporter@kernel.crashing.org> | 6 | * Matt Porter <mporter@kernel.crashing.org> |
7 | * | 7 | * |
8 | * Copyright 2009 Integrated Device Technology, Inc. | ||
9 | * Alex Bounine <alexandre.bounine@idt.com> | ||
10 | * - Added Port-Write/Error Management initialization and handling | ||
11 | * | ||
8 | * This program is free software; you can redistribute it and/or modify it | 12 | * This program is free software; you can redistribute it and/or modify it |
9 | * under the terms of the GNU General Public License as published by the | 13 | * under the terms of the GNU General Public License as published by the |
10 | * Free Software Foundation; either version 2 of the License, or (at your | 14 | * Free Software Foundation; either version 2 of the License, or (at your |
@@ -333,6 +337,329 @@ int rio_release_outb_dbell(struct rio_dev *rdev, struct resource *res) | |||
333 | } | 337 | } |
334 | 338 | ||
335 | /** | 339 | /** |
340 | * rio_request_inb_pwrite - request inbound port-write message service | ||
341 | * @mport: RIO device to which register inbound port-write callback routine | ||
342 | * @pwcback: Callback routine to execute when port-write is received | ||
343 | * | ||
344 | * Binds a port-write callback function to the RapidIO device. | ||
345 | * Returns 0 if the request has been satisfied. | ||
346 | */ | ||
347 | int rio_request_inb_pwrite(struct rio_dev *rdev, | ||
348 | int (*pwcback)(struct rio_dev *rdev, union rio_pw_msg *msg, int step)) | ||
349 | { | ||
350 | int rc = 0; | ||
351 | |||
352 | spin_lock(&rio_global_list_lock); | ||
353 | if (rdev->pwcback != NULL) | ||
354 | rc = -ENOMEM; | ||
355 | else | ||
356 | rdev->pwcback = pwcback; | ||
357 | |||
358 | spin_unlock(&rio_global_list_lock); | ||
359 | return rc; | ||
360 | } | ||
361 | EXPORT_SYMBOL_GPL(rio_request_inb_pwrite); | ||
362 | |||
363 | /** | ||
364 | * rio_release_inb_pwrite - release inbound port-write message service | ||
365 | * @rdev: RIO device which registered for inbound port-write callback | ||
366 | * | ||
367 | * Removes callback from the rio_dev structure. Returns 0 if the request | ||
368 | * has been satisfied. | ||
369 | */ | ||
370 | int rio_release_inb_pwrite(struct rio_dev *rdev) | ||
371 | { | ||
372 | int rc = -ENOMEM; | ||
373 | |||
374 | spin_lock(&rio_global_list_lock); | ||
375 | if (rdev->pwcback) { | ||
376 | rdev->pwcback = NULL; | ||
377 | rc = 0; | ||
378 | } | ||
379 | |||
380 | spin_unlock(&rio_global_list_lock); | ||
381 | return rc; | ||
382 | } | ||
383 | EXPORT_SYMBOL_GPL(rio_release_inb_pwrite); | ||
384 | |||
385 | /** | ||
386 | * rio_mport_get_physefb - Helper function that returns register offset | ||
387 | * for Physical Layer Extended Features Block. | ||
388 | * @rdev: RIO device | ||
389 | */ | ||
390 | u32 | ||
391 | rio_mport_get_physefb(struct rio_mport *port, int local, | ||
392 | u16 destid, u8 hopcount) | ||
393 | { | ||
394 | u32 ext_ftr_ptr; | ||
395 | u32 ftr_header; | ||
396 | |||
397 | ext_ftr_ptr = rio_mport_get_efb(port, local, destid, hopcount, 0); | ||
398 | |||
399 | while (ext_ftr_ptr) { | ||
400 | if (local) | ||
401 | rio_local_read_config_32(port, ext_ftr_ptr, | ||
402 | &ftr_header); | ||
403 | else | ||
404 | rio_mport_read_config_32(port, destid, hopcount, | ||
405 | ext_ftr_ptr, &ftr_header); | ||
406 | |||
407 | ftr_header = RIO_GET_BLOCK_ID(ftr_header); | ||
408 | switch (ftr_header) { | ||
409 | |||
410 | case RIO_EFB_SER_EP_ID_V13P: | ||
411 | case RIO_EFB_SER_EP_REC_ID_V13P: | ||
412 | case RIO_EFB_SER_EP_FREE_ID_V13P: | ||
413 | case RIO_EFB_SER_EP_ID: | ||
414 | case RIO_EFB_SER_EP_REC_ID: | ||
415 | case RIO_EFB_SER_EP_FREE_ID: | ||
416 | case RIO_EFB_SER_EP_FREC_ID: | ||
417 | |||
418 | return ext_ftr_ptr; | ||
419 | |||
420 | default: | ||
421 | break; | ||
422 | } | ||
423 | |||
424 | ext_ftr_ptr = rio_mport_get_efb(port, local, destid, | ||
425 | hopcount, ext_ftr_ptr); | ||
426 | } | ||
427 | |||
428 | return ext_ftr_ptr; | ||
429 | } | ||
430 | |||
431 | /** | ||
432 | * rio_get_comptag - Begin or continue searching for a RIO device by component tag | ||
433 | * @comp_tag: RIO component tad to match | ||
434 | * @from: Previous RIO device found in search, or %NULL for new search | ||
435 | * | ||
436 | * Iterates through the list of known RIO devices. If a RIO device is | ||
437 | * found with a matching @comp_tag, a pointer to its device | ||
438 | * structure is returned. Otherwise, %NULL is returned. A new search | ||
439 | * is initiated by passing %NULL to the @from argument. Otherwise, if | ||
440 | * @from is not %NULL, searches continue from next device on the global | ||
441 | * list. | ||
442 | */ | ||
443 | static struct rio_dev *rio_get_comptag(u32 comp_tag, struct rio_dev *from) | ||
444 | { | ||
445 | struct list_head *n; | ||
446 | struct rio_dev *rdev; | ||
447 | |||
448 | WARN_ON(in_interrupt()); | ||
449 | spin_lock(&rio_global_list_lock); | ||
450 | n = from ? from->global_list.next : rio_devices.next; | ||
451 | |||
452 | while (n && (n != &rio_devices)) { | ||
453 | rdev = rio_dev_g(n); | ||
454 | if (rdev->comp_tag == comp_tag) | ||
455 | goto exit; | ||
456 | n = n->next; | ||
457 | } | ||
458 | rdev = NULL; | ||
459 | exit: | ||
460 | spin_unlock(&rio_global_list_lock); | ||
461 | return rdev; | ||
462 | } | ||
463 | |||
464 | /** | ||
465 | * rio_set_port_lockout - Sets/clears LOCKOUT bit (RIO EM 1.3) for a switch port. | ||
466 | * @rdev: Pointer to RIO device control structure | ||
467 | * @pnum: Switch port number to set LOCKOUT bit | ||
468 | * @lock: Operation : set (=1) or clear (=0) | ||
469 | */ | ||
470 | int rio_set_port_lockout(struct rio_dev *rdev, u32 pnum, int lock) | ||
471 | { | ||
472 | u8 hopcount = 0xff; | ||
473 | u16 destid = rdev->destid; | ||
474 | u32 regval; | ||
475 | |||
476 | if (rdev->rswitch) { | ||
477 | destid = rdev->rswitch->destid; | ||
478 | hopcount = rdev->rswitch->hopcount; | ||
479 | } | ||
480 | |||
481 | rio_mport_read_config_32(rdev->net->hport, destid, hopcount, | ||
482 | rdev->phys_efptr + RIO_PORT_N_CTL_CSR(pnum), | ||
483 | ®val); | ||
484 | if (lock) | ||
485 | regval |= RIO_PORT_N_CTL_LOCKOUT; | ||
486 | else | ||
487 | regval &= ~RIO_PORT_N_CTL_LOCKOUT; | ||
488 | |||
489 | rio_mport_write_config_32(rdev->net->hport, destid, hopcount, | ||
490 | rdev->phys_efptr + RIO_PORT_N_CTL_CSR(pnum), | ||
491 | regval); | ||
492 | return 0; | ||
493 | } | ||
494 | |||
495 | /** | ||
496 | * rio_inb_pwrite_handler - process inbound port-write message | ||
497 | * @pw_msg: pointer to inbound port-write message | ||
498 | * | ||
499 | * Processes an inbound port-write message. Returns 0 if the request | ||
500 | * has been satisfied. | ||
501 | */ | ||
502 | int rio_inb_pwrite_handler(union rio_pw_msg *pw_msg) | ||
503 | { | ||
504 | struct rio_dev *rdev; | ||
505 | struct rio_mport *mport; | ||
506 | u8 hopcount; | ||
507 | u16 destid; | ||
508 | u32 err_status; | ||
509 | int rc, portnum; | ||
510 | |||
511 | rdev = rio_get_comptag(pw_msg->em.comptag, NULL); | ||
512 | if (rdev == NULL) { | ||
513 | /* Someting bad here (probably enumeration error) */ | ||
514 | pr_err("RIO: %s No matching device for CTag 0x%08x\n", | ||
515 | __func__, pw_msg->em.comptag); | ||
516 | return -EIO; | ||
517 | } | ||
518 | |||
519 | pr_debug("RIO: Port-Write message from %s\n", rio_name(rdev)); | ||
520 | |||
521 | #ifdef DEBUG_PW | ||
522 | { | ||
523 | u32 i; | ||
524 | for (i = 0; i < RIO_PW_MSG_SIZE/sizeof(u32);) { | ||
525 | pr_debug("0x%02x: %08x %08x %08x %08x", | ||
526 | i*4, pw_msg->raw[i], pw_msg->raw[i + 1], | ||
527 | pw_msg->raw[i + 2], pw_msg->raw[i + 3]); | ||
528 | i += 4; | ||
529 | } | ||
530 | pr_debug("\n"); | ||
531 | } | ||
532 | #endif | ||
533 | |||
534 | /* Call an external service function (if such is registered | ||
535 | * for this device). This may be the service for endpoints that send | ||
536 | * device-specific port-write messages. End-point messages expected | ||
537 | * to be handled completely by EP specific device driver. | ||
538 | * For switches rc==0 signals that no standard processing required. | ||
539 | */ | ||
540 | if (rdev->pwcback != NULL) { | ||
541 | rc = rdev->pwcback(rdev, pw_msg, 0); | ||
542 | if (rc == 0) | ||
543 | return 0; | ||
544 | } | ||
545 | |||
546 | /* For End-point devices processing stops here */ | ||
547 | if (!(rdev->pef & RIO_PEF_SWITCH)) | ||
548 | return 0; | ||
549 | |||
550 | if (rdev->phys_efptr == 0) { | ||
551 | pr_err("RIO_PW: Bad switch initialization for %s\n", | ||
552 | rio_name(rdev)); | ||
553 | return 0; | ||
554 | } | ||
555 | |||
556 | mport = rdev->net->hport; | ||
557 | destid = rdev->rswitch->destid; | ||
558 | hopcount = rdev->rswitch->hopcount; | ||
559 | |||
560 | /* | ||
561 | * Process the port-write notification from switch | ||
562 | */ | ||
563 | |||
564 | portnum = pw_msg->em.is_port & 0xFF; | ||
565 | |||
566 | if (rdev->rswitch->em_handle) | ||
567 | rdev->rswitch->em_handle(rdev, portnum); | ||
568 | |||
569 | rio_mport_read_config_32(mport, destid, hopcount, | ||
570 | rdev->phys_efptr + RIO_PORT_N_ERR_STS_CSR(portnum), | ||
571 | &err_status); | ||
572 | pr_debug("RIO_PW: SP%d_ERR_STS_CSR=0x%08x\n", portnum, err_status); | ||
573 | |||
574 | if (pw_msg->em.errdetect) { | ||
575 | pr_debug("RIO_PW: RIO_EM_P%d_ERR_DETECT=0x%08x\n", | ||
576 | portnum, pw_msg->em.errdetect); | ||
577 | /* Clear EM Port N Error Detect CSR */ | ||
578 | rio_mport_write_config_32(mport, destid, hopcount, | ||
579 | rdev->em_efptr + RIO_EM_PN_ERR_DETECT(portnum), 0); | ||
580 | } | ||
581 | |||
582 | if (pw_msg->em.ltlerrdet) { | ||
583 | pr_debug("RIO_PW: RIO_EM_LTL_ERR_DETECT=0x%08x\n", | ||
584 | pw_msg->em.ltlerrdet); | ||
585 | /* Clear EM L/T Layer Error Detect CSR */ | ||
586 | rio_mport_write_config_32(mport, destid, hopcount, | ||
587 | rdev->em_efptr + RIO_EM_LTL_ERR_DETECT, 0); | ||
588 | } | ||
589 | |||
590 | /* Clear Port Errors */ | ||
591 | rio_mport_write_config_32(mport, destid, hopcount, | ||
592 | rdev->phys_efptr + RIO_PORT_N_ERR_STS_CSR(portnum), | ||
593 | err_status & RIO_PORT_N_ERR_STS_CLR_MASK); | ||
594 | |||
595 | if (rdev->rswitch->port_ok & (1 << portnum)) { | ||
596 | if (err_status & RIO_PORT_N_ERR_STS_PORT_UNINIT) { | ||
597 | rdev->rswitch->port_ok &= ~(1 << portnum); | ||
598 | rio_set_port_lockout(rdev, portnum, 1); | ||
599 | |||
600 | rio_mport_write_config_32(mport, destid, hopcount, | ||
601 | rdev->phys_efptr + | ||
602 | RIO_PORT_N_ACK_STS_CSR(portnum), | ||
603 | RIO_PORT_N_ACK_CLEAR); | ||
604 | |||
605 | /* Schedule Extraction Service */ | ||
606 | pr_debug("RIO_PW: Device Extraction on [%s]-P%d\n", | ||
607 | rio_name(rdev), portnum); | ||
608 | } | ||
609 | } else { | ||
610 | if (err_status & RIO_PORT_N_ERR_STS_PORT_OK) { | ||
611 | rdev->rswitch->port_ok |= (1 << portnum); | ||
612 | rio_set_port_lockout(rdev, portnum, 0); | ||
613 | |||
614 | /* Schedule Insertion Service */ | ||
615 | pr_debug("RIO_PW: Device Insertion on [%s]-P%d\n", | ||
616 | rio_name(rdev), portnum); | ||
617 | } | ||
618 | } | ||
619 | |||
620 | /* Clear Port-Write Pending bit */ | ||
621 | rio_mport_write_config_32(mport, destid, hopcount, | ||
622 | rdev->phys_efptr + RIO_PORT_N_ERR_STS_CSR(portnum), | ||
623 | RIO_PORT_N_ERR_STS_PW_PEND); | ||
624 | |||
625 | return 0; | ||
626 | } | ||
627 | EXPORT_SYMBOL_GPL(rio_inb_pwrite_handler); | ||
628 | |||
629 | /** | ||
630 | * rio_mport_get_efb - get pointer to next extended features block | ||
631 | * @port: Master port to issue transaction | ||
632 | * @local: Indicate a local master port or remote device access | ||
633 | * @destid: Destination ID of the device | ||
634 | * @hopcount: Number of switch hops to the device | ||
635 | * @from: Offset of current Extended Feature block header (if 0 starts | ||
636 | * from ExtFeaturePtr) | ||
637 | */ | ||
638 | u32 | ||
639 | rio_mport_get_efb(struct rio_mport *port, int local, u16 destid, | ||
640 | u8 hopcount, u32 from) | ||
641 | { | ||
642 | u32 reg_val; | ||
643 | |||
644 | if (from == 0) { | ||
645 | if (local) | ||
646 | rio_local_read_config_32(port, RIO_ASM_INFO_CAR, | ||
647 | ®_val); | ||
648 | else | ||
649 | rio_mport_read_config_32(port, destid, hopcount, | ||
650 | RIO_ASM_INFO_CAR, ®_val); | ||
651 | return reg_val & RIO_EXT_FTR_PTR_MASK; | ||
652 | } else { | ||
653 | if (local) | ||
654 | rio_local_read_config_32(port, from, ®_val); | ||
655 | else | ||
656 | rio_mport_read_config_32(port, destid, hopcount, | ||
657 | from, ®_val); | ||
658 | return RIO_GET_BLOCK_ID(reg_val); | ||
659 | } | ||
660 | } | ||
661 | |||
662 | /** | ||
336 | * rio_mport_get_feature - query for devices' extended features | 663 | * rio_mport_get_feature - query for devices' extended features |
337 | * @port: Master port to issue transaction | 664 | * @port: Master port to issue transaction |
338 | * @local: Indicate a local master port or remote device access | 665 | * @local: Indicate a local master port or remote device access |
@@ -472,6 +799,7 @@ int rio_std_route_add_entry(struct rio_mport *mport, u16 destid, u8 hopcount, | |||
472 | RIO_STD_RTE_CONF_PORT_SEL_CSR, | 799 | RIO_STD_RTE_CONF_PORT_SEL_CSR, |
473 | (u32)route_port); | 800 | (u32)route_port); |
474 | } | 801 | } |
802 | |||
475 | udelay(10); | 803 | udelay(10); |
476 | return 0; | 804 | return 0; |
477 | } | 805 | } |
diff --git a/drivers/rapidio/rio.h b/drivers/rapidio/rio.h index b53c5ec276a5..2f628ce1a1c1 100644 --- a/drivers/rapidio/rio.h +++ b/drivers/rapidio/rio.h | |||
@@ -18,6 +18,10 @@ | |||
18 | 18 | ||
19 | extern u32 rio_mport_get_feature(struct rio_mport *mport, int local, u16 destid, | 19 | extern u32 rio_mport_get_feature(struct rio_mport *mport, int local, u16 destid, |
20 | u8 hopcount, int ftr); | 20 | u8 hopcount, int ftr); |
21 | extern u32 rio_mport_get_physefb(struct rio_mport *port, int local, | ||
22 | u16 destid, u8 hopcount); | ||
23 | extern u32 rio_mport_get_efb(struct rio_mport *port, int local, u16 destid, | ||
24 | u8 hopcount, u32 from); | ||
21 | extern int rio_create_sysfs_dev_files(struct rio_dev *rdev); | 25 | extern int rio_create_sysfs_dev_files(struct rio_dev *rdev); |
22 | extern int rio_enum_mport(struct rio_mport *mport); | 26 | extern int rio_enum_mport(struct rio_mport *mport); |
23 | extern int rio_disc_mport(struct rio_mport *mport); | 27 | extern int rio_disc_mport(struct rio_mport *mport); |
@@ -29,6 +33,7 @@ extern int rio_std_route_get_entry(struct rio_mport *mport, u16 destid, | |||
29 | u8 *route_port); | 33 | u8 *route_port); |
30 | extern int rio_std_route_clr_table(struct rio_mport *mport, u16 destid, | 34 | extern int rio_std_route_clr_table(struct rio_mport *mport, u16 destid, |
31 | u8 hopcount, u16 table); | 35 | u8 hopcount, u16 table); |
36 | extern int rio_set_port_lockout(struct rio_dev *rdev, u32 pnum, int lock); | ||
32 | 37 | ||
33 | /* Structures internal to the RIO core code */ | 38 | /* Structures internal to the RIO core code */ |
34 | extern struct device_attribute rio_dev_attrs[]; | 39 | extern struct device_attribute rio_dev_attrs[]; |
@@ -61,3 +66,28 @@ extern struct rio_route_ops __end_rio_route_ops[]; | |||
61 | 66 | ||
62 | #define RIO_GET_DID(size, x) (size ? (x & 0xffff) : ((x & 0x00ff0000) >> 16)) | 67 | #define RIO_GET_DID(size, x) (size ? (x & 0xffff) : ((x & 0x00ff0000) >> 16)) |
63 | #define RIO_SET_DID(size, x) (size ? (x & 0xffff) : ((x & 0x000000ff) << 16)) | 68 | #define RIO_SET_DID(size, x) (size ? (x & 0xffff) : ((x & 0x000000ff) << 16)) |
69 | |||
70 | /* | ||
71 | * RapidIO Error Management | ||
72 | */ | ||
73 | extern struct rio_em_ops __start_rio_em_ops[]; | ||
74 | extern struct rio_em_ops __end_rio_em_ops[]; | ||
75 | |||
76 | /* Helpers internal to the RIO core code */ | ||
77 | #define DECLARE_RIO_EM_SECTION(section, name, vid, did, init_hook, em_hook) \ | ||
78 | static const struct rio_em_ops __rio_em_##name __used \ | ||
79 | __section(section) = { vid, did, init_hook, em_hook }; | ||
80 | |||
81 | /** | ||
82 | * DECLARE_RIO_EM_OPS - Registers switch EM operations | ||
83 | * @vid: RIO vendor ID | ||
84 | * @did: RIO device ID | ||
85 | * @init_hook: Callback that initializes device specific EM | ||
86 | * @em_hook: Callback that handles device specific EM | ||
87 | * | ||
88 | * A &struct rio_em_ops is initialized with the ops and placed into a | ||
89 | * RIO-specific kernel section. | ||
90 | */ | ||
91 | #define DECLARE_RIO_EM_OPS(vid, did, init_hook, em_hook) \ | ||
92 | DECLARE_RIO_EM_SECTION(.rio_em_ops, vid##did, \ | ||
93 | vid, did, init_hook, em_hook) | ||
diff --git a/drivers/rapidio/switches/tsi568.c b/drivers/rapidio/switches/tsi568.c index bce9112ff0d9..905cf9cb09cc 100644 --- a/drivers/rapidio/switches/tsi568.c +++ b/drivers/rapidio/switches/tsi568.c | |||
@@ -25,6 +25,9 @@ | |||
25 | #define SPP_ROUTE_CFG_DESTID(n) (0x11070 + 0x100*n) | 25 | #define SPP_ROUTE_CFG_DESTID(n) (0x11070 + 0x100*n) |
26 | #define SPP_ROUTE_CFG_PORT(n) (0x11074 + 0x100*n) | 26 | #define SPP_ROUTE_CFG_PORT(n) (0x11074 + 0x100*n) |
27 | 27 | ||
28 | #define TSI568_SP_MODE_BC 0x10004 | ||
29 | #define TSI568_SP_MODE_PW_DIS 0x08000000 | ||
30 | |||
28 | static int | 31 | static int |
29 | tsi568_route_add_entry(struct rio_mport *mport, u16 destid, u8 hopcount, | 32 | tsi568_route_add_entry(struct rio_mport *mport, u16 destid, u8 hopcount, |
30 | u16 table, u16 route_destid, u8 route_port) | 33 | u16 table, u16 route_destid, u8 route_port) |
@@ -104,3 +107,24 @@ tsi568_route_clr_table(struct rio_mport *mport, u16 destid, u8 hopcount, | |||
104 | } | 107 | } |
105 | 108 | ||
106 | DECLARE_RIO_ROUTE_OPS(RIO_VID_TUNDRA, RIO_DID_TSI568, tsi568_route_add_entry, tsi568_route_get_entry, tsi568_route_clr_table); | 109 | DECLARE_RIO_ROUTE_OPS(RIO_VID_TUNDRA, RIO_DID_TSI568, tsi568_route_add_entry, tsi568_route_get_entry, tsi568_route_clr_table); |
110 | |||
111 | static int | ||
112 | tsi568_em_init(struct rio_dev *rdev) | ||
113 | { | ||
114 | struct rio_mport *mport = rdev->net->hport; | ||
115 | u16 destid = rdev->rswitch->destid; | ||
116 | u8 hopcount = rdev->rswitch->hopcount; | ||
117 | u32 regval; | ||
118 | |||
119 | pr_debug("TSI568 %s [%d:%d]\n", __func__, destid, hopcount); | ||
120 | |||
121 | /* Make sure that Port-Writes are disabled (for all ports) */ | ||
122 | rio_mport_read_config_32(mport, destid, hopcount, | ||
123 | TSI568_SP_MODE_BC, ®val); | ||
124 | rio_mport_write_config_32(mport, destid, hopcount, | ||
125 | TSI568_SP_MODE_BC, regval | TSI568_SP_MODE_PW_DIS); | ||
126 | |||
127 | return 0; | ||
128 | } | ||
129 | |||
130 | DECLARE_RIO_EM_OPS(RIO_VID_TUNDRA, RIO_DID_TSI568, tsi568_em_init, NULL); | ||
diff --git a/drivers/rapidio/switches/tsi57x.c b/drivers/rapidio/switches/tsi57x.c index 5ad7880787c9..23040b92ea76 100644 --- a/drivers/rapidio/switches/tsi57x.c +++ b/drivers/rapidio/switches/tsi57x.c | |||
@@ -25,6 +25,14 @@ | |||
25 | #define SPP_ROUTE_CFG_DESTID(n) (0x11070 + 0x100*n) | 25 | #define SPP_ROUTE_CFG_DESTID(n) (0x11070 + 0x100*n) |
26 | #define SPP_ROUTE_CFG_PORT(n) (0x11074 + 0x100*n) | 26 | #define SPP_ROUTE_CFG_PORT(n) (0x11074 + 0x100*n) |
27 | 27 | ||
28 | #define TSI578_SP_MODE(n) (0x11004 + n*0x100) | ||
29 | #define TSI578_SP_MODE_PW_DIS 0x08000000 | ||
30 | |||
31 | #define TSI578_SP_CTL_INDEP(n) (0x13004 + n*0x100) | ||
32 | #define TSI578_SP_LUT_PEINF(n) (0x13010 + n*0x100) | ||
33 | #define TSI578_SP_CS_TX(n) (0x13014 + n*0x100) | ||
34 | #define TSI578_SP_INT_STATUS(n) (0x13018 + n*0x100) | ||
35 | |||
28 | static int | 36 | static int |
29 | tsi57x_route_add_entry(struct rio_mport *mport, u16 destid, u8 hopcount, | 37 | tsi57x_route_add_entry(struct rio_mport *mport, u16 destid, u8 hopcount, |
30 | u16 table, u16 route_destid, u8 route_port) | 38 | u16 table, u16 route_destid, u8 route_port) |
@@ -104,3 +112,148 @@ DECLARE_RIO_ROUTE_OPS(RIO_VID_TUNDRA, RIO_DID_TSI572, tsi57x_route_add_entry, ts | |||
104 | DECLARE_RIO_ROUTE_OPS(RIO_VID_TUNDRA, RIO_DID_TSI574, tsi57x_route_add_entry, tsi57x_route_get_entry, tsi57x_route_clr_table); | 112 | DECLARE_RIO_ROUTE_OPS(RIO_VID_TUNDRA, RIO_DID_TSI574, tsi57x_route_add_entry, tsi57x_route_get_entry, tsi57x_route_clr_table); |
105 | DECLARE_RIO_ROUTE_OPS(RIO_VID_TUNDRA, RIO_DID_TSI577, tsi57x_route_add_entry, tsi57x_route_get_entry, tsi57x_route_clr_table); | 113 | DECLARE_RIO_ROUTE_OPS(RIO_VID_TUNDRA, RIO_DID_TSI577, tsi57x_route_add_entry, tsi57x_route_get_entry, tsi57x_route_clr_table); |
106 | DECLARE_RIO_ROUTE_OPS(RIO_VID_TUNDRA, RIO_DID_TSI578, tsi57x_route_add_entry, tsi57x_route_get_entry, tsi57x_route_clr_table); | 114 | DECLARE_RIO_ROUTE_OPS(RIO_VID_TUNDRA, RIO_DID_TSI578, tsi57x_route_add_entry, tsi57x_route_get_entry, tsi57x_route_clr_table); |
115 | |||
116 | static int | ||
117 | tsi57x_em_init(struct rio_dev *rdev) | ||
118 | { | ||
119 | struct rio_mport *mport = rdev->net->hport; | ||
120 | u16 destid = rdev->rswitch->destid; | ||
121 | u8 hopcount = rdev->rswitch->hopcount; | ||
122 | u32 regval; | ||
123 | int portnum; | ||
124 | |||
125 | pr_debug("TSI578 %s [%d:%d]\n", __func__, destid, hopcount); | ||
126 | |||
127 | for (portnum = 0; portnum < 16; portnum++) { | ||
128 | /* Make sure that Port-Writes are enabled (for all ports) */ | ||
129 | rio_mport_read_config_32(mport, destid, hopcount, | ||
130 | TSI578_SP_MODE(portnum), ®val); | ||
131 | rio_mport_write_config_32(mport, destid, hopcount, | ||
132 | TSI578_SP_MODE(portnum), | ||
133 | regval & ~TSI578_SP_MODE_PW_DIS); | ||
134 | |||
135 | /* Clear all pending interrupts */ | ||
136 | rio_mport_read_config_32(mport, destid, hopcount, | ||
137 | rdev->phys_efptr + | ||
138 | RIO_PORT_N_ERR_STS_CSR(portnum), | ||
139 | ®val); | ||
140 | rio_mport_write_config_32(mport, destid, hopcount, | ||
141 | rdev->phys_efptr + | ||
142 | RIO_PORT_N_ERR_STS_CSR(portnum), | ||
143 | regval & 0x07120214); | ||
144 | |||
145 | rio_mport_read_config_32(mport, destid, hopcount, | ||
146 | TSI578_SP_INT_STATUS(portnum), ®val); | ||
147 | rio_mport_write_config_32(mport, destid, hopcount, | ||
148 | TSI578_SP_INT_STATUS(portnum), | ||
149 | regval & 0x000700bd); | ||
150 | |||
151 | /* Enable all interrupts to allow ports to send a port-write */ | ||
152 | rio_mport_read_config_32(mport, destid, hopcount, | ||
153 | TSI578_SP_CTL_INDEP(portnum), ®val); | ||
154 | rio_mport_write_config_32(mport, destid, hopcount, | ||
155 | TSI578_SP_CTL_INDEP(portnum), | ||
156 | regval | 0x000b0000); | ||
157 | |||
158 | /* Skip next (odd) port if the current port is in x4 mode */ | ||
159 | rio_mport_read_config_32(mport, destid, hopcount, | ||
160 | rdev->phys_efptr + RIO_PORT_N_CTL_CSR(portnum), | ||
161 | ®val); | ||
162 | if ((regval & RIO_PORT_N_CTL_PWIDTH) == RIO_PORT_N_CTL_PWIDTH_4) | ||
163 | portnum++; | ||
164 | } | ||
165 | |||
166 | return 0; | ||
167 | } | ||
168 | |||
169 | static int | ||
170 | tsi57x_em_handler(struct rio_dev *rdev, u8 portnum) | ||
171 | { | ||
172 | struct rio_mport *mport = rdev->net->hport; | ||
173 | u16 destid = rdev->rswitch->destid; | ||
174 | u8 hopcount = rdev->rswitch->hopcount; | ||
175 | u32 intstat, err_status; | ||
176 | int sendcount, checkcount; | ||
177 | u8 route_port; | ||
178 | u32 regval; | ||
179 | |||
180 | rio_mport_read_config_32(mport, destid, hopcount, | ||
181 | rdev->phys_efptr + RIO_PORT_N_ERR_STS_CSR(portnum), | ||
182 | &err_status); | ||
183 | |||
184 | if ((err_status & RIO_PORT_N_ERR_STS_PORT_OK) && | ||
185 | (err_status & (RIO_PORT_N_ERR_STS_PW_OUT_ES | | ||
186 | RIO_PORT_N_ERR_STS_PW_INP_ES))) { | ||
187 | /* Remove any queued packets by locking/unlocking port */ | ||
188 | rio_mport_read_config_32(mport, destid, hopcount, | ||
189 | rdev->phys_efptr + RIO_PORT_N_CTL_CSR(portnum), | ||
190 | ®val); | ||
191 | if (!(regval & RIO_PORT_N_CTL_LOCKOUT)) { | ||
192 | rio_mport_write_config_32(mport, destid, hopcount, | ||
193 | rdev->phys_efptr + RIO_PORT_N_CTL_CSR(portnum), | ||
194 | regval | RIO_PORT_N_CTL_LOCKOUT); | ||
195 | udelay(50); | ||
196 | rio_mport_write_config_32(mport, destid, hopcount, | ||
197 | rdev->phys_efptr + RIO_PORT_N_CTL_CSR(portnum), | ||
198 | regval); | ||
199 | } | ||
200 | |||
201 | /* Read from link maintenance response register to clear | ||
202 | * valid bit | ||
203 | */ | ||
204 | rio_mport_read_config_32(mport, destid, hopcount, | ||
205 | rdev->phys_efptr + RIO_PORT_N_MNT_RSP_CSR(portnum), | ||
206 | ®val); | ||
207 | |||
208 | /* Send a Packet-Not-Accepted/Link-Request-Input-Status control | ||
209 | * symbol to recover from IES/OES | ||
210 | */ | ||
211 | sendcount = 3; | ||
212 | while (sendcount) { | ||
213 | rio_mport_write_config_32(mport, destid, hopcount, | ||
214 | TSI578_SP_CS_TX(portnum), 0x40fc8000); | ||
215 | checkcount = 3; | ||
216 | while (checkcount--) { | ||
217 | udelay(50); | ||
218 | rio_mport_read_config_32( | ||
219 | mport, destid, hopcount, | ||
220 | rdev->phys_efptr + | ||
221 | RIO_PORT_N_MNT_RSP_CSR(portnum), | ||
222 | ®val); | ||
223 | if (regval & RIO_PORT_N_MNT_RSP_RVAL) | ||
224 | goto exit_es; | ||
225 | } | ||
226 | |||
227 | sendcount--; | ||
228 | } | ||
229 | } | ||
230 | |||
231 | exit_es: | ||
232 | /* Clear implementation specific error status bits */ | ||
233 | rio_mport_read_config_32(mport, destid, hopcount, | ||
234 | TSI578_SP_INT_STATUS(portnum), &intstat); | ||
235 | pr_debug("TSI578[%x:%x] SP%d_INT_STATUS=0x%08x\n", | ||
236 | destid, hopcount, portnum, intstat); | ||
237 | |||
238 | if (intstat & 0x10000) { | ||
239 | rio_mport_read_config_32(mport, destid, hopcount, | ||
240 | TSI578_SP_LUT_PEINF(portnum), ®val); | ||
241 | regval = (mport->sys_size) ? (regval >> 16) : (regval >> 24); | ||
242 | route_port = rdev->rswitch->route_table[regval]; | ||
243 | pr_debug("RIO: TSI578[%s] P%d LUT Parity Error (destID=%d)\n", | ||
244 | rio_name(rdev), portnum, regval); | ||
245 | tsi57x_route_add_entry(mport, destid, hopcount, | ||
246 | RIO_GLOBAL_TABLE, regval, route_port); | ||
247 | } | ||
248 | |||
249 | rio_mport_write_config_32(mport, destid, hopcount, | ||
250 | TSI578_SP_INT_STATUS(portnum), | ||
251 | intstat & 0x000700bd); | ||
252 | |||
253 | return 0; | ||
254 | } | ||
255 | |||
256 | DECLARE_RIO_EM_OPS(RIO_VID_TUNDRA, RIO_DID_TSI572, tsi57x_em_init, tsi57x_em_handler); | ||
257 | DECLARE_RIO_EM_OPS(RIO_VID_TUNDRA, RIO_DID_TSI574, tsi57x_em_init, tsi57x_em_handler); | ||
258 | DECLARE_RIO_EM_OPS(RIO_VID_TUNDRA, RIO_DID_TSI577, tsi57x_em_init, tsi57x_em_handler); | ||
259 | DECLARE_RIO_EM_OPS(RIO_VID_TUNDRA, RIO_DID_TSI578, tsi57x_em_init, tsi57x_em_handler); | ||