2 * Sealevel Systems 4021 driver.
4 * This program is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU General Public License
6 * as published by the Free Software Foundation; either version
7 * 2 of the License, or (at your option) any later version.
9 * (c) Copyright 1999, 2001 Alan Cox
10 * (c) Copyright 2001 Red Hat Inc.
14 #include <linux/module.h>
15 #include <linux/kernel.h>
17 #include <linux/net.h>
18 #include <linux/skbuff.h>
19 #include <linux/netdevice.h>
20 #include <linux/if_arp.h>
21 #include <linux/delay.h>
22 #include <linux/ioport.h>
23 #include <linux/init.h>
29 #include <asm/byteorder.h>
30 #include <net/syncppp.h>
36 void *if_ptr
; /* General purpose pointer (used by SPPP) */
37 struct z8530_channel
*chan
;
38 struct ppp_device pppdev
;
45 struct slvl_device
*dev
[2];
46 struct z8530_dev board
;
51 * Network driver support routines
55 * Frame receive. Simple for our card as we do sync ppp and there
56 * is no funny garbage involved
59 static void sealevel_input(struct z8530_channel
*c
, struct sk_buff
*skb
)
61 /* Drop the CRC - it's not a good idea to try and negotiate it ;) */
62 skb_trim(skb
, skb
->len
-2);
63 skb
->protocol
=htons(ETH_P_WAN_PPP
);
64 skb
->mac
.raw
=skb
->data
;
65 skb
->dev
=c
->netdevice
;
67 * Send it to the PPP layer. We don't have time to process
71 c
->netdevice
->last_rx
= jiffies
;
75 * We've been placed in the UP state
78 static int sealevel_open(struct net_device
*d
)
80 struct slvl_device
*slvl
=d
->priv
;
82 int unit
= slvl
->channel
;
91 err
=z8530_sync_dma_open(d
, slvl
->chan
);
94 err
=z8530_sync_open(d
, slvl
->chan
);
109 z8530_sync_dma_close(d
, slvl
->chan
);
112 z8530_sync_close(d
, slvl
->chan
);
118 slvl
->chan
->rx_function
=sealevel_input
;
123 netif_start_queue(d
);
127 static int sealevel_close(struct net_device
*d
)
129 struct slvl_device
*slvl
=d
->priv
;
130 int unit
= slvl
->channel
;
136 slvl
->chan
->rx_function
=z8530_null_rx
;
151 z8530_sync_dma_close(d
, slvl
->chan
);
154 z8530_sync_close(d
, slvl
->chan
);
160 static int sealevel_ioctl(struct net_device
*d
, struct ifreq
*ifr
, int cmd
)
162 /* struct slvl_device *slvl=d->priv;
163 z8530_ioctl(d,&slvl->sync.chanA,ifr,cmd) */
164 return sppp_do_ioctl(d
, ifr
,cmd
);
167 static struct net_device_stats
*sealevel_get_stats(struct net_device
*d
)
169 struct slvl_device
*slvl
=d
->priv
;
171 return z8530_get_stats(slvl
->chan
);
177 * Passed PPP frames, fire them downwind.
180 static int sealevel_queue_xmit(struct sk_buff
*skb
, struct net_device
*d
)
182 struct slvl_device
*slvl
=d
->priv
;
183 return z8530_queue_xmit(slvl
->chan
, skb
);
186 static int sealevel_neigh_setup(struct neighbour
*n
)
188 if (n
->nud_state
== NUD_NONE
) {
189 n
->ops
= &arp_broken_ops
;
190 n
->output
= n
->ops
->output
;
195 static int sealevel_neigh_setup_dev(struct net_device
*dev
, struct neigh_parms
*p
)
197 if (p
->tbl
->family
== AF_INET
) {
198 p
->neigh_setup
= sealevel_neigh_setup
;
205 static int sealevel_attach(struct net_device
*dev
)
207 struct slvl_device
*sv
= dev
->priv
;
208 sppp_attach(&sv
->pppdev
);
212 static void sealevel_detach(struct net_device
*dev
)
217 static void slvl_setup(struct net_device
*d
)
219 d
->open
= sealevel_open
;
220 d
->stop
= sealevel_close
;
221 d
->init
= sealevel_attach
;
222 d
->uninit
= sealevel_detach
;
223 d
->hard_start_xmit
= sealevel_queue_xmit
;
224 d
->get_stats
= sealevel_get_stats
;
225 d
->set_multicast_list
= NULL
;
226 d
->do_ioctl
= sealevel_ioctl
;
227 d
->neigh_setup
= sealevel_neigh_setup_dev
;
228 d
->set_mac_address
= NULL
;
232 static inline struct slvl_device
*slvl_alloc(int iobase
, int irq
)
234 struct net_device
*d
;
235 struct slvl_device
*sv
;
237 d
= alloc_netdev(sizeof(struct slvl_device
), "hdlc%d",
244 sv
->if_ptr
= &sv
->pppdev
;
246 d
->base_addr
= iobase
;
254 * Allocate and setup Sealevel board.
257 static __init
struct slvl_board
*slvl_init(int iobase
, int irq
,
258 int txdma
, int rxdma
, int slow
)
260 struct z8530_dev
*dev
;
261 struct slvl_board
*b
;
264 * Get the needed I/O space
267 if(!request_region(iobase
, 8, "Sealevel 4021"))
269 printk(KERN_WARNING
"sealevel: I/O 0x%X already in use.\n", iobase
);
273 b
= kmalloc(sizeof(struct slvl_board
), GFP_KERNEL
);
277 memset(b
, 0, sizeof(*b
));
278 if (!(b
->dev
[0]= slvl_alloc(iobase
, irq
)))
281 b
->dev
[0]->chan
= &b
->board
.chanA
;
282 b
->dev
[0]->channel
= 0;
284 if (!(b
->dev
[1] = slvl_alloc(iobase
, irq
)))
287 b
->dev
[1]->chan
= &b
->board
.chanB
;
288 b
->dev
[1]->channel
= 1;
293 * Stuff in the I/O addressing
301 * Select 8530 delays for the old board
305 iobase
|= Z8530_PORT_SLEEP
;
307 dev
->chanA
.ctrlio
=iobase
+1;
308 dev
->chanA
.dataio
=iobase
;
309 dev
->chanB
.ctrlio
=iobase
+3;
310 dev
->chanB
.dataio
=iobase
+2;
312 dev
->chanA
.irqs
=&z8530_nop
;
313 dev
->chanB
.irqs
=&z8530_nop
;
316 * Assert DTR enable DMA
319 outb(3|(1<<7), b
->iobase
+4);
322 /* We want a fast IRQ for this device. Actually we'd like an even faster
323 IRQ ;) - This is one driver RtLinux is made for */
325 if(request_irq(irq
, &z8530_interrupt
, SA_INTERRUPT
, "SeaLevel", dev
)<0)
327 printk(KERN_WARNING
"sealevel: IRQ %d already in use.\n", irq
);
332 dev
->chanA
.private=&b
->dev
[0];
333 dev
->chanB
.private=&b
->dev
[1];
334 dev
->chanA
.netdevice
=b
->dev
[0]->pppdev
.dev
;
335 dev
->chanB
.netdevice
=b
->dev
[1]->pppdev
.dev
;
341 if(request_dma(dev
->chanA
.txdma
, "SeaLevel (TX)")!=0)
344 if(request_dma(dev
->chanA
.rxdma
, "SeaLevel (RX)")!=0)
350 * Begin normal initialise
353 if(z8530_init(dev
)!=0)
355 printk(KERN_ERR
"Z8530 series device not found.\n");
359 if(dev
->type
==Z85C30
)
361 z8530_channel_load(&dev
->chanA
, z8530_hdlc_kilostream
);
362 z8530_channel_load(&dev
->chanB
, z8530_hdlc_kilostream
);
366 z8530_channel_load(&dev
->chanA
, z8530_hdlc_kilostream_85230
);
367 z8530_channel_load(&dev
->chanB
, z8530_hdlc_kilostream_85230
);
371 * Now we can take the IRQ
376 if (register_netdev(b
->dev
[0]->pppdev
.dev
))
379 if (register_netdev(b
->dev
[1]->pppdev
.dev
))
382 z8530_describe(dev
, "I/O", iobase
);
387 unregister_netdev(b
->dev
[0]->pppdev
.dev
);
390 free_dma(dev
->chanA
.rxdma
);
392 free_dma(dev
->chanA
.txdma
);
396 free_netdev(b
->dev
[1]->pppdev
.dev
);
398 free_netdev(b
->dev
[0]->pppdev
.dev
);
402 release_region(iobase
,8);
406 static void __exit
slvl_shutdown(struct slvl_board
*b
)
410 z8530_shutdown(&b
->board
);
414 struct net_device
*d
= b
->dev
[u
]->pppdev
.dev
;
415 unregister_netdev(d
);
419 free_irq(b
->board
.irq
, &b
->board
);
420 free_dma(b
->board
.chanA
.rxdma
);
421 free_dma(b
->board
.chanA
.txdma
);
422 /* DMA off on the card, drop DTR */
424 release_region(b
->iobase
, 8);
435 module_param(io
, int, 0);
436 MODULE_PARM_DESC(io
, "The I/O base of the Sealevel card");
437 module_param(txdma
, int, 0);
438 MODULE_PARM_DESC(txdma
, "Transmit DMA channel");
439 module_param(rxdma
, int, 0);
440 MODULE_PARM_DESC(rxdma
, "Receive DMA channel");
441 module_param(irq
, int, 0);
442 MODULE_PARM_DESC(irq
, "The interrupt line setting for the SeaLevel card");
443 module_param(slow
, bool, 0);
444 MODULE_PARM_DESC(slow
, "Set this for an older Sealevel card such as the 4012");
446 MODULE_AUTHOR("Alan Cox");
447 MODULE_LICENSE("GPL");
448 MODULE_DESCRIPTION("Modular driver for the SeaLevel 4021");
450 static struct slvl_board
*slvl_unit
;
452 static int __init
slvl_init_module(void)
455 printk(KERN_INFO
"SeaLevel Z85230 Synchronous Driver v 0.02.\n");
456 printk(KERN_INFO
"(c) Copyright 1998, Building Number Three Ltd.\n");
458 slvl_unit
= slvl_init(io
, irq
, txdma
, rxdma
, slow
);
460 return slvl_unit
? 0 : -ENODEV
;
463 static void __exit
slvl_cleanup_module(void)
466 slvl_shutdown(slvl_unit
);
469 module_init(slvl_init_module
);
470 module_exit(slvl_cleanup_module
);