diff options
author | Finn Thain <fthain@telegraphics.com.au> | 2007-05-01 22:55:56 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@woody.linux-foundation.org> | 2007-05-04 20:59:08 -0400 |
commit | f4d86754f956ab5ea73aa91759a0d89a2f0e3f2a (patch) | |
tree | 6afaeefe24fa98385a5f3de6557a31689456071a /drivers/net | |
parent | d74472f0b2553e59eafb7feee0ff9558136a17e0 (diff) |
SONIC interrupt handling
Install the built-in macsonic interrupt handler on both IRQs when using
via_alt_mapping. Otherwise the rare interrupt that still comes from the
nubus slot will wedge the nubus.
$ cat /proc/interrupts
auto 2: 89176 via2
auto 3: 744367 sonic
auto 4: 0 scc
auto 6: 318363 via1
auto 7: 0 NMI
mac 9: 119413 framebuffer vbl
mac 10: 1971 ADB
mac 14: 198517 timer
mac 17: 89104 nubus
mac 19: 72 Mac ESP SCSI
mac 56: 629 sonic
mac 62: 1142593 ide0
Version 1 of this patch had a bug where a nubus sonic card would register
two interrupt handlers. Only a built-in sonic needs both.
Versions 2 and 3 needed some cleanups, as Raylynn Knight and Christoph
Hellwig pointed out (thanks).
Signed-off-by: Finn Thain <fthain@telegraphics.com.au>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'drivers/net')
-rw-r--r-- | drivers/net/jazzsonic.c | 23 | ||||
-rw-r--r-- | drivers/net/macsonic.c | 46 | ||||
-rw-r--r-- | drivers/net/sonic.c | 25 |
3 files changed, 61 insertions, 33 deletions
diff --git a/drivers/net/jazzsonic.c b/drivers/net/jazzsonic.c index 1a73630fd7d5..75f6f441e876 100644 --- a/drivers/net/jazzsonic.c +++ b/drivers/net/jazzsonic.c | |||
@@ -88,6 +88,23 @@ static unsigned short known_revisions[] = | |||
88 | 0xffff /* end of list */ | 88 | 0xffff /* end of list */ |
89 | }; | 89 | }; |
90 | 90 | ||
91 | static int jazzsonic_open(struct net_device* dev) | ||
92 | { | ||
93 | if (request_irq(dev->irq, &sonic_interrupt, IRQF_DISABLED, "sonic", dev)) { | ||
94 | printk(KERN_ERR "%s: unable to get IRQ %d.\n", dev->name, dev->irq); | ||
95 | return -EAGAIN; | ||
96 | } | ||
97 | return sonic_open(dev); | ||
98 | } | ||
99 | |||
100 | static int jazzsonic_close(struct net_device* dev) | ||
101 | { | ||
102 | int err; | ||
103 | err = sonic_close(dev); | ||
104 | free_irq(dev->irq, dev); | ||
105 | return err; | ||
106 | } | ||
107 | |||
91 | static int __init sonic_probe1(struct net_device *dev) | 108 | static int __init sonic_probe1(struct net_device *dev) |
92 | { | 109 | { |
93 | static unsigned version_printed; | 110 | static unsigned version_printed; |
@@ -169,8 +186,8 @@ static int __init sonic_probe1(struct net_device *dev) | |||
169 | lp->rra_laddr = lp->rda_laddr + (SIZEOF_SONIC_RD * SONIC_NUM_RDS | 186 | lp->rra_laddr = lp->rda_laddr + (SIZEOF_SONIC_RD * SONIC_NUM_RDS |
170 | * SONIC_BUS_SCALE(lp->dma_bitmode)); | 187 | * SONIC_BUS_SCALE(lp->dma_bitmode)); |
171 | 188 | ||
172 | dev->open = sonic_open; | 189 | dev->open = jazzsonic_open; |
173 | dev->stop = sonic_close; | 190 | dev->stop = jazzsonic_close; |
174 | dev->hard_start_xmit = sonic_send_packet; | 191 | dev->hard_start_xmit = sonic_send_packet; |
175 | dev->get_stats = sonic_get_stats; | 192 | dev->get_stats = sonic_get_stats; |
176 | dev->set_multicast_list = &sonic_multicast_list; | 193 | dev->set_multicast_list = &sonic_multicast_list; |
@@ -260,8 +277,6 @@ MODULE_DESCRIPTION("Jazz SONIC ethernet driver"); | |||
260 | module_param(sonic_debug, int, 0); | 277 | module_param(sonic_debug, int, 0); |
261 | MODULE_PARM_DESC(sonic_debug, "jazzsonic debug level (1-4)"); | 278 | MODULE_PARM_DESC(sonic_debug, "jazzsonic debug level (1-4)"); |
262 | 279 | ||
263 | #define SONIC_IRQ_FLAG IRQF_DISABLED | ||
264 | |||
265 | #include "sonic.c" | 280 | #include "sonic.c" |
266 | 281 | ||
267 | static int __devexit jazz_sonic_device_remove (struct platform_device *pdev) | 282 | static int __devexit jazz_sonic_device_remove (struct platform_device *pdev) |
diff --git a/drivers/net/macsonic.c b/drivers/net/macsonic.c index c78a7e2e8ba8..e9ecdbf352ae 100644 --- a/drivers/net/macsonic.c +++ b/drivers/net/macsonic.c | |||
@@ -130,6 +130,46 @@ static inline void bit_reverse_addr(unsigned char addr[6]) | |||
130 | addr[i] = bitrev8(addr[i]); | 130 | addr[i] = bitrev8(addr[i]); |
131 | } | 131 | } |
132 | 132 | ||
133 | static irqreturn_t macsonic_interrupt(int irq, void *dev_id) | ||
134 | { | ||
135 | irqreturn_t result; | ||
136 | unsigned long flags; | ||
137 | |||
138 | local_irq_save(flags); | ||
139 | result = sonic_interrupt(irq, dev_id); | ||
140 | local_irq_restore(flags); | ||
141 | return result; | ||
142 | } | ||
143 | |||
144 | static int macsonic_open(struct net_device* dev) | ||
145 | { | ||
146 | if (request_irq(dev->irq, &sonic_interrupt, IRQ_FLG_FAST, "sonic", dev)) { | ||
147 | printk(KERN_ERR "%s: unable to get IRQ %d.\n", dev->name, dev->irq); | ||
148 | return -EAGAIN; | ||
149 | } | ||
150 | /* Under the A/UX interrupt scheme, the onboard SONIC interrupt comes | ||
151 | * in at priority level 3. However, we sometimes get the level 2 inter- | ||
152 | * rupt as well, which must prevent re-entrance of the sonic handler. | ||
153 | */ | ||
154 | if (dev->irq == IRQ_AUTO_3) | ||
155 | if (request_irq(IRQ_NUBUS_9, &macsonic_interrupt, IRQ_FLG_FAST, "sonic", dev)) { | ||
156 | printk(KERN_ERR "%s: unable to get IRQ %d.\n", dev->name, IRQ_NUBUS_9); | ||
157 | free_irq(dev->irq, dev); | ||
158 | return -EAGAIN; | ||
159 | } | ||
160 | return sonic_open(dev); | ||
161 | } | ||
162 | |||
163 | static int macsonic_close(struct net_device* dev) | ||
164 | { | ||
165 | int err; | ||
166 | err = sonic_close(dev); | ||
167 | free_irq(dev->irq, dev); | ||
168 | if (dev->irq == IRQ_AUTO_3) | ||
169 | free_irq(IRQ_NUBUS_9, dev); | ||
170 | return err; | ||
171 | } | ||
172 | |||
133 | int __init macsonic_init(struct net_device* dev) | 173 | int __init macsonic_init(struct net_device* dev) |
134 | { | 174 | { |
135 | struct sonic_local* lp = netdev_priv(dev); | 175 | struct sonic_local* lp = netdev_priv(dev); |
@@ -160,8 +200,8 @@ int __init macsonic_init(struct net_device* dev) | |||
160 | lp->rra_laddr = lp->rda_laddr + (SIZEOF_SONIC_RD * SONIC_NUM_RDS | 200 | lp->rra_laddr = lp->rda_laddr + (SIZEOF_SONIC_RD * SONIC_NUM_RDS |
161 | * SONIC_BUS_SCALE(lp->dma_bitmode)); | 201 | * SONIC_BUS_SCALE(lp->dma_bitmode)); |
162 | 202 | ||
163 | dev->open = sonic_open; | 203 | dev->open = macsonic_open; |
164 | dev->stop = sonic_close; | 204 | dev->stop = macsonic_close; |
165 | dev->hard_start_xmit = sonic_send_packet; | 205 | dev->hard_start_xmit = sonic_send_packet; |
166 | dev->get_stats = sonic_get_stats; | 206 | dev->get_stats = sonic_get_stats; |
167 | dev->set_multicast_list = &sonic_multicast_list; | 207 | dev->set_multicast_list = &sonic_multicast_list; |
@@ -572,8 +612,6 @@ MODULE_DESCRIPTION("Macintosh SONIC ethernet driver"); | |||
572 | module_param(sonic_debug, int, 0); | 612 | module_param(sonic_debug, int, 0); |
573 | MODULE_PARM_DESC(sonic_debug, "macsonic debug level (1-4)"); | 613 | MODULE_PARM_DESC(sonic_debug, "macsonic debug level (1-4)"); |
574 | 614 | ||
575 | #define SONIC_IRQ_FLAG IRQ_FLG_FAST | ||
576 | |||
577 | #include "sonic.c" | 615 | #include "sonic.c" |
578 | 616 | ||
579 | static int __devexit mac_sonic_device_remove (struct platform_device *pdev) | 617 | static int __devexit mac_sonic_device_remove (struct platform_device *pdev) |
diff --git a/drivers/net/sonic.c b/drivers/net/sonic.c index 687c1413c4ee..8069f3e32d83 100644 --- a/drivers/net/sonic.c +++ b/drivers/net/sonic.c | |||
@@ -50,29 +50,6 @@ static int sonic_open(struct net_device *dev) | |||
50 | if (sonic_debug > 2) | 50 | if (sonic_debug > 2) |
51 | printk("sonic_open: initializing sonic driver.\n"); | 51 | printk("sonic_open: initializing sonic driver.\n"); |
52 | 52 | ||
53 | /* | ||
54 | * We don't need to deal with auto-irq stuff since we | ||
55 | * hardwire the sonic interrupt. | ||
56 | */ | ||
57 | /* | ||
58 | * XXX Horrible work around: We install sonic_interrupt as fast interrupt. | ||
59 | * This means that during execution of the handler interrupt are disabled | ||
60 | * covering another bug otherwise corrupting data. This doesn't mean | ||
61 | * this glue works ok under all situations. | ||
62 | * | ||
63 | * Note (dhd): this also appears to prevent lockups on the Macintrash | ||
64 | * when more than one Ethernet card is installed (knock on wood) | ||
65 | * | ||
66 | * Note (fthain): whether the above is still true is anyones guess. Certainly | ||
67 | * the buffer handling algorithms will not tolerate re-entrance without some | ||
68 | * mutual exclusion added. Anyway, the memcpy has now been eliminated from the | ||
69 | * rx code to make this a faster "fast interrupt". | ||
70 | */ | ||
71 | if (request_irq(dev->irq, &sonic_interrupt, SONIC_IRQ_FLAG, "sonic", dev)) { | ||
72 | printk(KERN_ERR "\n%s: unable to get IRQ %d .\n", dev->name, dev->irq); | ||
73 | return -EAGAIN; | ||
74 | } | ||
75 | |||
76 | for (i = 0; i < SONIC_NUM_RRS; i++) { | 53 | for (i = 0; i < SONIC_NUM_RRS; i++) { |
77 | struct sk_buff *skb = dev_alloc_skb(SONIC_RBSIZE + 2); | 54 | struct sk_buff *skb = dev_alloc_skb(SONIC_RBSIZE + 2); |
78 | if (skb == NULL) { | 55 | if (skb == NULL) { |
@@ -169,8 +146,6 @@ static int sonic_close(struct net_device *dev) | |||
169 | } | 146 | } |
170 | } | 147 | } |
171 | 148 | ||
172 | free_irq(dev->irq, dev); /* release the IRQ */ | ||
173 | |||
174 | return 0; | 149 | return 0; |
175 | } | 150 | } |
176 | 151 | ||