| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1 | /* [xirc2ps_cs.c wk 03.11.99] (1.40 1999/11/18 00:06:03) | 
|  | 2 | * Xircom CreditCard Ethernet Adapter IIps driver | 
|  | 3 | * Xircom Realport 10/100 (RE-100) driver | 
|  | 4 | * | 
|  | 5 | * This driver supports various Xircom CreditCard Ethernet adapters | 
|  | 6 | * including the CE2, CE IIps, RE-10, CEM28, CEM33, CE33, CEM56, | 
|  | 7 | * CE3-100, CE3B, RE-100, REM10BT, and REM56G-100. | 
|  | 8 | * | 
|  | 9 | * 2000-09-24 <psheer@icon.co.za> The Xircom CE3B-100 may not | 
|  | 10 | * autodetect the media properly. In this case use the | 
|  | 11 | * if_port=1 (for 10BaseT) or if_port=4 (for 100BaseT) options | 
|  | 12 | * to force the media type. | 
|  | 13 | * | 
|  | 14 | * Written originally by Werner Koch based on David Hinds' skeleton of the | 
|  | 15 | * PCMCIA driver. | 
|  | 16 | * | 
|  | 17 | * Copyright (c) 1997,1998 Werner Koch (dd9jn) | 
|  | 18 | * | 
|  | 19 | * This driver is free software; you can redistribute it and/or modify | 
|  | 20 | * it under the terms of the GNU General Public License as published by | 
|  | 21 | * the Free Software Foundation; either version 2 of the License, or | 
|  | 22 | * (at your option) any later version. | 
|  | 23 | * | 
|  | 24 | * It is distributed in the hope that it will be useful, | 
|  | 25 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | 26 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
|  | 27 | * GNU General Public License for more details. | 
|  | 28 | * | 
|  | 29 | * You should have received a copy of the GNU General Public License | 
|  | 30 | * along with this program; if not, write to the Free Software | 
|  | 31 | * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA | 
|  | 32 | * | 
|  | 33 | * | 
|  | 34 | * ALTERNATIVELY, this driver may be distributed under the terms of | 
|  | 35 | * the following license, in which case the provisions of this license | 
|  | 36 | * are required INSTEAD OF the GNU General Public License.  (This clause | 
|  | 37 | * is necessary due to a potential bad interaction between the GPL and | 
|  | 38 | * the restrictions contained in a BSD-style copyright.) | 
|  | 39 | * | 
|  | 40 | * Redistribution and use in source and binary forms, with or without | 
|  | 41 | * modification, are permitted provided that the following conditions | 
|  | 42 | * are met: | 
|  | 43 | * 1. Redistributions of source code must retain the above copyright | 
|  | 44 | *    notice, and the entire permission notice in its entirety, | 
|  | 45 | *    including the disclaimer of warranties. | 
|  | 46 | * 2. Redistributions in binary form must reproduce the above copyright | 
|  | 47 | *    notice, this list of conditions and the following disclaimer in the | 
|  | 48 | *    documentation and/or other materials provided with the distribution. | 
|  | 49 | * 3. The name of the author may not be used to endorse or promote | 
|  | 50 | *    products derived from this software without specific prior | 
|  | 51 | *    written permission. | 
|  | 52 | * | 
|  | 53 | * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED | 
|  | 54 | * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES | 
|  | 55 | * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | 
|  | 56 | * DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, | 
|  | 57 | * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | 
|  | 58 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | 
|  | 59 | * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) | 
|  | 60 | * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, | 
|  | 61 | * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) | 
|  | 62 | * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED | 
|  | 63 | * OF THE POSSIBILITY OF SUCH DAMAGE. | 
|  | 64 | */ | 
|  | 65 |  | 
|  | 66 | #include <linux/module.h> | 
|  | 67 | #include <linux/kernel.h> | 
|  | 68 | #include <linux/init.h> | 
|  | 69 | #include <linux/ptrace.h> | 
|  | 70 | #include <linux/slab.h> | 
|  | 71 | #include <linux/string.h> | 
|  | 72 | #include <linux/timer.h> | 
|  | 73 | #include <linux/interrupt.h> | 
|  | 74 | #include <linux/in.h> | 
|  | 75 | #include <linux/delay.h> | 
|  | 76 | #include <linux/ethtool.h> | 
|  | 77 | #include <linux/netdevice.h> | 
|  | 78 | #include <linux/etherdevice.h> | 
|  | 79 | #include <linux/skbuff.h> | 
|  | 80 | #include <linux/if_arp.h> | 
|  | 81 | #include <linux/ioport.h> | 
|  | 82 | #include <linux/bitops.h> | 
| Ben Hutchings | 0fa0ee05 | 2009-09-03 10:41:17 +0000 | [diff] [blame] | 83 | #include <linux/mii.h> | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 84 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 85 | #include <pcmcia/cs.h> | 
|  | 86 | #include <pcmcia/cistpl.h> | 
|  | 87 | #include <pcmcia/cisreg.h> | 
|  | 88 | #include <pcmcia/ciscode.h> | 
|  | 89 |  | 
|  | 90 | #include <asm/io.h> | 
|  | 91 | #include <asm/system.h> | 
|  | 92 | #include <asm/uaccess.h> | 
|  | 93 |  | 
|  | 94 | #ifndef MANFID_COMPAQ | 
|  | 95 | #define MANFID_COMPAQ 	   0x0138 | 
|  | 96 | #define MANFID_COMPAQ2	   0x0183  /* is this correct? */ | 
|  | 97 | #endif | 
|  | 98 |  | 
|  | 99 | #include <pcmcia/ds.h> | 
|  | 100 |  | 
|  | 101 | /* Time in jiffies before concluding Tx hung */ | 
|  | 102 | #define TX_TIMEOUT	((400*HZ)/1000) | 
|  | 103 |  | 
|  | 104 | /**************** | 
|  | 105 | * Some constants used to access the hardware | 
|  | 106 | */ | 
|  | 107 |  | 
|  | 108 | /* Register offsets and value constans */ | 
|  | 109 | #define XIRCREG_CR  0	/* Command register (wr) */ | 
|  | 110 | enum xirc_cr { | 
|  | 111 | TransmitPacket = 0x01, | 
|  | 112 | SoftReset = 0x02, | 
|  | 113 | EnableIntr = 0x04, | 
|  | 114 | ForceIntr  = 0x08, | 
|  | 115 | ClearTxFIFO = 0x10, | 
|  | 116 | ClearRxOvrun = 0x20, | 
|  | 117 | RestartTx	 = 0x40 | 
|  | 118 | }; | 
|  | 119 | #define XIRCREG_ESR 0	/* Ethernet status register (rd) */ | 
|  | 120 | enum xirc_esr { | 
|  | 121 | FullPktRcvd = 0x01, /* full packet in receive buffer */ | 
|  | 122 | PktRejected = 0x04, /* a packet has been rejected */ | 
|  | 123 | TxPktPend = 0x08,	/* TX Packet Pending */ | 
|  | 124 | IncorPolarity = 0x10, | 
|  | 125 | MediaSelect = 0x20	/* set if TP, clear if AUI */ | 
|  | 126 | }; | 
|  | 127 | #define XIRCREG_PR  1	/* Page Register select */ | 
|  | 128 | #define XIRCREG_EDP 4	/* Ethernet Data Port Register */ | 
|  | 129 | #define XIRCREG_ISR 6	/* Ethernet Interrupt Status Register */ | 
|  | 130 | enum xirc_isr { | 
|  | 131 | TxBufOvr = 0x01,	/* TX Buffer Overflow */ | 
|  | 132 | PktTxed  = 0x02,	/* Packet Transmitted */ | 
|  | 133 | MACIntr  = 0x04,	/* MAC Interrupt occurred */ | 
|  | 134 | TxResGrant = 0x08,	/* Tx Reservation Granted */ | 
|  | 135 | RxFullPkt = 0x20,	/* Rx Full Packet */ | 
|  | 136 | RxPktRej  = 0x40,	/* Rx Packet Rejected */ | 
|  | 137 | ForcedIntr= 0x80	/* Forced Interrupt */ | 
|  | 138 | }; | 
|  | 139 | #define XIRCREG1_IMR0 12 /* Ethernet Interrupt Mask Register (on page 1)*/ | 
|  | 140 | #define XIRCREG1_IMR1 13 | 
|  | 141 | #define XIRCREG0_TSO  8  /* Transmit Space Open Register (on page 0)*/ | 
|  | 142 | #define XIRCREG0_TRS  10 /* Transmit reservation Size Register (page 0)*/ | 
|  | 143 | #define XIRCREG0_DO   12 /* Data Offset Register (page 0) (wr) */ | 
|  | 144 | #define XIRCREG0_RSR  12 /* Receive Status Register (page 0) (rd) */ | 
|  | 145 | enum xirc_rsr { | 
|  | 146 | PhyPkt = 0x01,	/* set:physical packet, clear: multicast packet */ | 
|  | 147 | BrdcstPkt = 0x02,	/* set if it is a broadcast packet */ | 
|  | 148 | PktTooLong = 0x04,	/* set if packet length > 1518 */ | 
|  | 149 | AlignErr = 0x10,	/* incorrect CRC and last octet not complete */ | 
|  | 150 | CRCErr = 0x20,	/* incorrect CRC and last octet is complete */ | 
|  | 151 | PktRxOk = 0x80	/* received ok */ | 
|  | 152 | }; | 
|  | 153 | #define XIRCREG0_PTR 13 /* packets transmitted register (rd) */ | 
|  | 154 | #define XIRCREG0_RBC 14 /* receive byte count regsister (rd) */ | 
|  | 155 | #define XIRCREG1_ECR 14 /* ethernet configurationn register */ | 
|  | 156 | enum xirc_ecr { | 
|  | 157 | FullDuplex = 0x04,	/* enable full duplex mode */ | 
|  | 158 | LongTPMode = 0x08,	/* adjust for longer lengths of TP cable */ | 
|  | 159 | DisablePolCor = 0x10,/* disable auto polarity correction */ | 
|  | 160 | DisableLinkPulse = 0x20, /* disable link pulse generation */ | 
|  | 161 | DisableAutoTx = 0x40, /* disable auto-transmit */ | 
|  | 162 | }; | 
|  | 163 | #define XIRCREG2_RBS 8	/* receive buffer start register */ | 
|  | 164 | #define XIRCREG2_LED 10 /* LED Configuration register */ | 
|  | 165 | /* values for the leds:    Bits 2-0 for led 1 | 
|  | 166 | *  0 disabled		   Bits 5-3 for led 2 | 
|  | 167 | *  1 collision | 
|  | 168 | *  2 noncollision | 
|  | 169 | *  3 link_detected | 
|  | 170 | *  4 incor_polarity | 
|  | 171 | *  5 jabber | 
|  | 172 | *  6 auto_assertion | 
|  | 173 | *  7 rx_tx_activity | 
|  | 174 | */ | 
|  | 175 | #define XIRCREG2_MSR 12 /* Mohawk specific register */ | 
|  | 176 |  | 
|  | 177 | #define XIRCREG4_GPR0 8 /* General Purpose Register 0 */ | 
|  | 178 | #define XIRCREG4_GPR1 9 /* General Purpose Register 1 */ | 
|  | 179 | #define XIRCREG2_GPR2 13 /* General Purpose Register 2 (page2!)*/ | 
|  | 180 | #define XIRCREG4_BOV 10 /* Bonding Version Register */ | 
|  | 181 | #define XIRCREG4_LMA 12 /* Local Memory Address Register */ | 
|  | 182 | #define XIRCREG4_LMD 14 /* Local Memory Data Port */ | 
|  | 183 | /* MAC register can only by accessed with 8 bit operations */ | 
|  | 184 | #define XIRCREG40_CMD0 8    /* Command Register (wr) */ | 
|  | 185 | enum xirc_cmd { 	    /* Commands */ | 
|  | 186 | Transmit = 0x01, | 
|  | 187 | EnableRecv = 0x04, | 
|  | 188 | DisableRecv = 0x08, | 
|  | 189 | Abort = 0x10, | 
|  | 190 | Online = 0x20, | 
|  | 191 | IntrAck = 0x40, | 
|  | 192 | Offline = 0x80 | 
|  | 193 | }; | 
|  | 194 | #define XIRCREG5_RHSA0	10  /* Rx Host Start Address */ | 
|  | 195 | #define XIRCREG40_RXST0 9   /* Receive Status Register */ | 
|  | 196 | #define XIRCREG40_TXST0 11  /* Transmit Status Register 0 */ | 
|  | 197 | #define XIRCREG40_TXST1 12  /* Transmit Status Register 10 */ | 
|  | 198 | #define XIRCREG40_RMASK0 13  /* Receive Mask Register */ | 
|  | 199 | #define XIRCREG40_TMASK0 14  /* Transmit Mask Register 0 */ | 
|  | 200 | #define XIRCREG40_TMASK1 15  /* Transmit Mask Register 0 */ | 
|  | 201 | #define XIRCREG42_SWC0	8   /* Software Configuration 0 */ | 
|  | 202 | #define XIRCREG42_SWC1	9   /* Software Configuration 1 */ | 
|  | 203 | #define XIRCREG42_BOC	10  /* Back-Off Configuration */ | 
|  | 204 | #define XIRCREG44_TDR0	8   /* Time Domain Reflectometry 0 */ | 
|  | 205 | #define XIRCREG44_TDR1	9   /* Time Domain Reflectometry 1 */ | 
|  | 206 | #define XIRCREG44_RXBC_LO 10 /* Rx Byte Count 0 (rd) */ | 
|  | 207 | #define XIRCREG44_RXBC_HI 11 /* Rx Byte Count 1 (rd) */ | 
|  | 208 | #define XIRCREG45_REV	 15 /* Revision Register (rd) */ | 
|  | 209 | #define XIRCREG50_IA	8   /* Individual Address (8-13) */ | 
|  | 210 |  | 
| Arjan van de Ven | f71e130 | 2006-03-03 21:33:57 -0500 | [diff] [blame] | 211 | static const char *if_names[] = { "Auto", "10BaseT", "10Base2", "AUI", "100BaseT" }; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 212 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 213 |  | 
|  | 214 | #define KDBG_XIRC KERN_DEBUG   "xirc2ps_cs: " | 
|  | 215 | #define KERR_XIRC KERN_ERR     "xirc2ps_cs: " | 
|  | 216 | #define KWRN_XIRC KERN_WARNING "xirc2ps_cs: " | 
|  | 217 | #define KNOT_XIRC KERN_NOTICE  "xirc2ps_cs: " | 
|  | 218 | #define KINF_XIRC KERN_INFO    "xirc2ps_cs: " | 
|  | 219 |  | 
|  | 220 | /* card types */ | 
|  | 221 | #define XIR_UNKNOWN  0	/* unknown: not supported */ | 
|  | 222 | #define XIR_CE	     1	/* (prodid 1) different hardware: not supported */ | 
|  | 223 | #define XIR_CE2      2	/* (prodid 2) */ | 
|  | 224 | #define XIR_CE3      3	/* (prodid 3) */ | 
|  | 225 | #define XIR_CEM      4	/* (prodid 1) different hardware: not supported */ | 
|  | 226 | #define XIR_CEM2     5	/* (prodid 2) */ | 
|  | 227 | #define XIR_CEM3     6	/* (prodid 3) */ | 
|  | 228 | #define XIR_CEM33    7	/* (prodid 4) */ | 
|  | 229 | #define XIR_CEM56M   8	/* (prodid 5) */ | 
|  | 230 | #define XIR_CEM56    9	/* (prodid 6) */ | 
|  | 231 | #define XIR_CM28    10	/* (prodid 3) modem only: not supported here */ | 
|  | 232 | #define XIR_CM33    11	/* (prodid 4) modem only: not supported here */ | 
|  | 233 | #define XIR_CM56    12	/* (prodid 5) modem only: not supported here */ | 
|  | 234 | #define XIR_CG	    13	/* (prodid 1) GSM modem only: not supported */ | 
|  | 235 | #define XIR_CBE     14	/* (prodid 1) cardbus ethernet: not supported */ | 
|  | 236 | /*====================================================================*/ | 
|  | 237 |  | 
|  | 238 | /* Module parameters */ | 
|  | 239 |  | 
|  | 240 | MODULE_DESCRIPTION("Xircom PCMCIA ethernet driver"); | 
|  | 241 | MODULE_LICENSE("Dual MPL/GPL"); | 
|  | 242 |  | 
|  | 243 | #define INT_MODULE_PARM(n, v) static int n = v; module_param(n, int, 0) | 
|  | 244 |  | 
|  | 245 | INT_MODULE_PARM(if_port,	0); | 
|  | 246 | INT_MODULE_PARM(full_duplex,	0); | 
|  | 247 | INT_MODULE_PARM(do_sound, 	1); | 
|  | 248 | INT_MODULE_PARM(lockup_hack,	0);  /* anti lockup hack */ | 
|  | 249 |  | 
|  | 250 | /*====================================================================*/ | 
|  | 251 |  | 
|  | 252 | /* We do not process more than these number of bytes during one | 
|  | 253 | * interrupt. (Of course we receive complete packets, so this is not | 
|  | 254 | * an exact value). | 
|  | 255 | * Something between 2000..22000; first value gives best interrupt latency, | 
|  | 256 | * the second enables the usage of the complete on-chip buffer. We use the | 
|  | 257 | * high value as the initial value. | 
|  | 258 | */ | 
|  | 259 | static unsigned maxrx_bytes = 22000; | 
|  | 260 |  | 
|  | 261 | /* MII management prototypes */ | 
| Olof Johansson | 906da80 | 2008-02-04 22:27:35 -0800 | [diff] [blame] | 262 | static void mii_idle(unsigned int ioaddr); | 
|  | 263 | static void mii_putbit(unsigned int ioaddr, unsigned data); | 
|  | 264 | static int  mii_getbit(unsigned int ioaddr); | 
|  | 265 | static void mii_wbits(unsigned int ioaddr, unsigned data, int len); | 
|  | 266 | static unsigned mii_rd(unsigned int ioaddr, u_char phyaddr, u_char phyreg); | 
|  | 267 | static void mii_wr(unsigned int ioaddr, u_char phyaddr, u_char phyreg, | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 268 | unsigned data, int len); | 
|  | 269 |  | 
|  | 270 | /* | 
|  | 271 | * The event() function is this driver's Card Services event handler. | 
|  | 272 | * It will be called by Card Services when an appropriate card status | 
|  | 273 | * event is received.  The config() and release() entry points are | 
|  | 274 | * used to configure or release a socket, in response to card insertion | 
|  | 275 | * and ejection events.  They are invoked from the event handler. | 
|  | 276 | */ | 
|  | 277 |  | 
| Dominik Brodowski | fba395e | 2006-03-31 17:21:06 +0200 | [diff] [blame] | 278 | static int has_ce2_string(struct pcmcia_device * link); | 
| Dominik Brodowski | 15b99ac | 2006-03-31 17:26:06 +0200 | [diff] [blame] | 279 | static int xirc2ps_config(struct pcmcia_device * link); | 
| Dominik Brodowski | fba395e | 2006-03-31 17:21:06 +0200 | [diff] [blame] | 280 | static void xirc2ps_release(struct pcmcia_device * link); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 281 |  | 
|  | 282 | /**************** | 
|  | 283 | * The attach() and detach() entry points are used to create and destroy | 
|  | 284 | * "instances" of the driver, where each instance represents everything | 
|  | 285 | * needed to manage one actual PCMCIA card. | 
|  | 286 | */ | 
|  | 287 |  | 
| Dominik Brodowski | cc3b486 | 2005-11-14 21:23:14 +0100 | [diff] [blame] | 288 | static void xirc2ps_detach(struct pcmcia_device *p_dev); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 289 |  | 
|  | 290 | /**************** | 
|  | 291 | * You'll also need to prototype all the functions that will actually | 
|  | 292 | * be used to talk to your device.  See 'pcmem_cs' for a good example | 
|  | 293 | * of a fully self-sufficient driver; the other drivers rely more or | 
|  | 294 | * less on other parts of the kernel. | 
|  | 295 | */ | 
|  | 296 |  | 
| David Howells | 7d12e78 | 2006-10-05 14:55:46 +0100 | [diff] [blame] | 297 | static irqreturn_t xirc2ps_interrupt(int irq, void *dev_id); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 298 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 299 | typedef struct local_info_t { | 
| David Howells | c402895 | 2006-11-22 14:57:56 +0000 | [diff] [blame] | 300 | struct net_device	*dev; | 
| Dominik Brodowski | fd23823 | 2006-03-05 10:45:09 +0100 | [diff] [blame] | 301 | struct pcmcia_device	*p_dev; | 
| Stephen Hemminger | 6394d7c | 2009-03-20 19:36:07 +0000 | [diff] [blame] | 302 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 303 | int card_type; | 
|  | 304 | int probe_port; | 
|  | 305 | int silicon; /* silicon revision. 0=old CE2, 1=Scipper, 4=Mohawk */ | 
|  | 306 | int mohawk;  /* a CE3 type card */ | 
|  | 307 | int dingo;	 /* a CEM56 type card */ | 
|  | 308 | int new_mii; /* has full 10baseT/100baseT MII */ | 
|  | 309 | int modem;	 /* is a multi function card (i.e with a modem) */ | 
|  | 310 | void __iomem *dingo_ccr; /* only used for CEM56 cards */ | 
|  | 311 | unsigned last_ptr_value; /* last packets transmitted value */ | 
|  | 312 | const char *manf_str; | 
| Joerg Ahrens | 9a469ab | 2006-08-20 21:51:57 +0100 | [diff] [blame] | 313 | struct work_struct tx_timeout_task; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 314 | } local_info_t; | 
|  | 315 |  | 
|  | 316 | /**************** | 
|  | 317 | * Some more prototypes | 
|  | 318 | */ | 
| Stephen Hemminger | dbf02fa | 2009-08-31 19:50:49 +0000 | [diff] [blame] | 319 | static netdev_tx_t do_start_xmit(struct sk_buff *skb, | 
|  | 320 | struct net_device *dev); | 
| Arjan van de Ven | ed4cb13 | 2008-10-05 07:35:05 +0000 | [diff] [blame] | 321 | static void xirc_tx_timeout(struct net_device *dev); | 
| David Howells | c402895 | 2006-11-22 14:57:56 +0000 | [diff] [blame] | 322 | static void xirc2ps_tx_timeout_task(struct work_struct *work); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 323 | static void set_addresses(struct net_device *dev); | 
|  | 324 | static void set_multicast_list(struct net_device *dev); | 
| Dominik Brodowski | dddfbd8 | 2009-10-18 23:54:24 +0200 | [diff] [blame] | 325 | static int set_card_type(struct pcmcia_device *link); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 326 | static int do_config(struct net_device *dev, struct ifmap *map); | 
|  | 327 | static int do_open(struct net_device *dev); | 
|  | 328 | static int do_ioctl(struct net_device *dev, struct ifreq *rq, int cmd); | 
| Jeff Garzik | 7282d49 | 2006-09-13 14:30:00 -0400 | [diff] [blame] | 329 | static const struct ethtool_ops netdev_ethtool_ops; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 330 | static void hardreset(struct net_device *dev); | 
|  | 331 | static void do_reset(struct net_device *dev, int full); | 
|  | 332 | static int init_mii(struct net_device *dev); | 
|  | 333 | static void do_powerdown(struct net_device *dev); | 
|  | 334 | static int do_stop(struct net_device *dev); | 
|  | 335 |  | 
|  | 336 | /*=============== Helper functions =========================*/ | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 337 | #define SelectPage(pgnr)   outb((pgnr), ioaddr + XIRCREG_PR) | 
|  | 338 | #define GetByte(reg)	   ((unsigned)inb(ioaddr + (reg))) | 
|  | 339 | #define GetWord(reg)	   ((unsigned)inw(ioaddr + (reg))) | 
|  | 340 | #define PutByte(reg,value) outb((value), ioaddr+(reg)) | 
|  | 341 | #define PutWord(reg,value) outw((value), ioaddr+(reg)) | 
|  | 342 |  | 
|  | 343 | /*====== Functions used for debugging =================================*/ | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 344 | #if 0 /* reading regs may change system status */ | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 345 | static void | 
|  | 346 | PrintRegisters(struct net_device *dev) | 
|  | 347 | { | 
| Olof Johansson | 906da80 | 2008-02-04 22:27:35 -0800 | [diff] [blame] | 348 | unsigned int ioaddr = dev->base_addr; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 349 |  | 
|  | 350 | if (pc_debug > 1) { | 
|  | 351 | int i, page; | 
|  | 352 |  | 
|  | 353 | printk(KDBG_XIRC "Register  common: "); | 
|  | 354 | for (i = 0; i < 8; i++) | 
|  | 355 | printk(" %2.2x", GetByte(i)); | 
|  | 356 | printk("\n"); | 
|  | 357 | for (page = 0; page <= 8; page++) { | 
|  | 358 | printk(KDBG_XIRC "Register page %2x: ", page); | 
|  | 359 | SelectPage(page); | 
|  | 360 | for (i = 8; i < 16; i++) | 
|  | 361 | printk(" %2.2x", GetByte(i)); | 
|  | 362 | printk("\n"); | 
|  | 363 | } | 
|  | 364 | for (page=0x40 ; page <= 0x5f; page++) { | 
| Joe Perches | 8e95a20 | 2009-12-03 07:58:21 +0000 | [diff] [blame] | 365 | if (page == 0x43 || (page >= 0x46 && page <= 0x4f) || | 
|  | 366 | (page >= 0x51 && page <=0x5e)) | 
|  | 367 | continue; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 368 | printk(KDBG_XIRC "Register page %2x: ", page); | 
|  | 369 | SelectPage(page); | 
|  | 370 | for (i = 8; i < 16; i++) | 
|  | 371 | printk(" %2.2x", GetByte(i)); | 
|  | 372 | printk("\n"); | 
|  | 373 | } | 
|  | 374 | } | 
|  | 375 | } | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 376 | #endif /* 0 */ | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 377 |  | 
|  | 378 | /*============== MII Management functions ===============*/ | 
|  | 379 |  | 
|  | 380 | /**************** | 
|  | 381 | * Turn around for read | 
|  | 382 | */ | 
|  | 383 | static void | 
| Olof Johansson | 906da80 | 2008-02-04 22:27:35 -0800 | [diff] [blame] | 384 | mii_idle(unsigned int ioaddr) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 385 | { | 
|  | 386 | PutByte(XIRCREG2_GPR2, 0x04|0); /* drive MDCK low */ | 
|  | 387 | udelay(1); | 
|  | 388 | PutByte(XIRCREG2_GPR2, 0x04|1); /* and drive MDCK high */ | 
|  | 389 | udelay(1); | 
|  | 390 | } | 
|  | 391 |  | 
|  | 392 | /**************** | 
|  | 393 | * Write a bit to MDI/O | 
|  | 394 | */ | 
|  | 395 | static void | 
| Olof Johansson | 906da80 | 2008-02-04 22:27:35 -0800 | [diff] [blame] | 396 | mii_putbit(unsigned int ioaddr, unsigned data) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 397 | { | 
|  | 398 | #if 1 | 
|  | 399 | if (data) { | 
|  | 400 | PutByte(XIRCREG2_GPR2, 0x0c|2|0); /* set MDIO */ | 
|  | 401 | udelay(1); | 
|  | 402 | PutByte(XIRCREG2_GPR2, 0x0c|2|1); /* and drive MDCK high */ | 
|  | 403 | udelay(1); | 
|  | 404 | } else { | 
|  | 405 | PutByte(XIRCREG2_GPR2, 0x0c|0|0); /* clear MDIO */ | 
|  | 406 | udelay(1); | 
|  | 407 | PutByte(XIRCREG2_GPR2, 0x0c|0|1); /* and drive MDCK high */ | 
|  | 408 | udelay(1); | 
|  | 409 | } | 
|  | 410 | #else | 
|  | 411 | if (data) { | 
|  | 412 | PutWord(XIRCREG2_GPR2-1, 0x0e0e); | 
|  | 413 | udelay(1); | 
|  | 414 | PutWord(XIRCREG2_GPR2-1, 0x0f0f); | 
|  | 415 | udelay(1); | 
|  | 416 | } else { | 
|  | 417 | PutWord(XIRCREG2_GPR2-1, 0x0c0c); | 
|  | 418 | udelay(1); | 
|  | 419 | PutWord(XIRCREG2_GPR2-1, 0x0d0d); | 
|  | 420 | udelay(1); | 
|  | 421 | } | 
|  | 422 | #endif | 
|  | 423 | } | 
|  | 424 |  | 
|  | 425 | /**************** | 
|  | 426 | * Get a bit from MDI/O | 
|  | 427 | */ | 
|  | 428 | static int | 
| Olof Johansson | 906da80 | 2008-02-04 22:27:35 -0800 | [diff] [blame] | 429 | mii_getbit(unsigned int ioaddr) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 430 | { | 
|  | 431 | unsigned d; | 
|  | 432 |  | 
|  | 433 | PutByte(XIRCREG2_GPR2, 4|0); /* drive MDCK low */ | 
|  | 434 | udelay(1); | 
|  | 435 | d = GetByte(XIRCREG2_GPR2); /* read MDIO */ | 
|  | 436 | PutByte(XIRCREG2_GPR2, 4|1); /* drive MDCK high again */ | 
|  | 437 | udelay(1); | 
|  | 438 | return d & 0x20; /* read MDIO */ | 
|  | 439 | } | 
|  | 440 |  | 
|  | 441 | static void | 
| Olof Johansson | 906da80 | 2008-02-04 22:27:35 -0800 | [diff] [blame] | 442 | mii_wbits(unsigned int ioaddr, unsigned data, int len) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 443 | { | 
|  | 444 | unsigned m = 1 << (len-1); | 
|  | 445 | for (; m; m >>= 1) | 
|  | 446 | mii_putbit(ioaddr, data & m); | 
|  | 447 | } | 
|  | 448 |  | 
|  | 449 | static unsigned | 
| Olof Johansson | 906da80 | 2008-02-04 22:27:35 -0800 | [diff] [blame] | 450 | mii_rd(unsigned int ioaddr,	u_char phyaddr, u_char phyreg) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 451 | { | 
|  | 452 | int i; | 
|  | 453 | unsigned data=0, m; | 
|  | 454 |  | 
|  | 455 | SelectPage(2); | 
|  | 456 | for (i=0; i < 32; i++)		/* 32 bit preamble */ | 
|  | 457 | mii_putbit(ioaddr, 1); | 
|  | 458 | mii_wbits(ioaddr, 0x06, 4); 	/* Start and opcode for read */ | 
|  | 459 | mii_wbits(ioaddr, phyaddr, 5);	/* PHY address to be accessed */ | 
|  | 460 | mii_wbits(ioaddr, phyreg, 5);	/* PHY register to read */ | 
|  | 461 | mii_idle(ioaddr);			/* turn around */ | 
|  | 462 | mii_getbit(ioaddr); | 
|  | 463 |  | 
|  | 464 | for (m = 1<<15; m; m >>= 1) | 
|  | 465 | if (mii_getbit(ioaddr)) | 
|  | 466 | data |= m; | 
|  | 467 | mii_idle(ioaddr); | 
|  | 468 | return data; | 
|  | 469 | } | 
|  | 470 |  | 
|  | 471 | static void | 
| Olof Johansson | 906da80 | 2008-02-04 22:27:35 -0800 | [diff] [blame] | 472 | mii_wr(unsigned int ioaddr, u_char phyaddr, u_char phyreg, unsigned data, | 
|  | 473 | int len) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 474 | { | 
|  | 475 | int i; | 
|  | 476 |  | 
|  | 477 | SelectPage(2); | 
|  | 478 | for (i=0; i < 32; i++)		/* 32 bit preamble */ | 
|  | 479 | mii_putbit(ioaddr, 1); | 
|  | 480 | mii_wbits(ioaddr, 0x05, 4); 	/* Start and opcode for write */ | 
|  | 481 | mii_wbits(ioaddr, phyaddr, 5);	/* PHY address to be accessed */ | 
|  | 482 | mii_wbits(ioaddr, phyreg, 5);	/* PHY Register to write */ | 
|  | 483 | mii_putbit(ioaddr, 1);		/* turn around */ | 
|  | 484 | mii_putbit(ioaddr, 0); | 
|  | 485 | mii_wbits(ioaddr, data, len);	/* And write the data */ | 
|  | 486 | mii_idle(ioaddr); | 
|  | 487 | } | 
|  | 488 |  | 
|  | 489 | /*============= Main bulk of functions	=========================*/ | 
|  | 490 |  | 
| Stephen Hemminger | 0cd6e82 | 2009-03-20 19:36:08 +0000 | [diff] [blame] | 491 | static const struct net_device_ops netdev_ops = { | 
|  | 492 | .ndo_open		= do_open, | 
|  | 493 | .ndo_stop		= do_stop, | 
|  | 494 | .ndo_start_xmit		= do_start_xmit, | 
|  | 495 | .ndo_tx_timeout 	= xirc_tx_timeout, | 
|  | 496 | .ndo_set_config		= do_config, | 
|  | 497 | .ndo_do_ioctl		= do_ioctl, | 
|  | 498 | .ndo_set_multicast_list	= set_multicast_list, | 
|  | 499 | .ndo_change_mtu		= eth_change_mtu, | 
|  | 500 | .ndo_set_mac_address 	= eth_mac_addr, | 
|  | 501 | .ndo_validate_addr	= eth_validate_addr, | 
|  | 502 | }; | 
|  | 503 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 504 | /**************** | 
|  | 505 | * xirc2ps_attach() creates an "instance" of the driver, allocating | 
|  | 506 | * local data structures for one device.  The device is registered | 
|  | 507 | * with Card Services. | 
|  | 508 | * | 
|  | 509 | * The dev_link structure is initialized, but we don't actually | 
|  | 510 | * configure the card at this point -- we wait until we receive a | 
|  | 511 | * card insertion event. | 
|  | 512 | */ | 
|  | 513 |  | 
| Dominik Brodowski | f8cfa61 | 2005-11-14 21:25:51 +0100 | [diff] [blame] | 514 | static int | 
| Dominik Brodowski | 15b99ac | 2006-03-31 17:26:06 +0200 | [diff] [blame] | 515 | xirc2ps_probe(struct pcmcia_device *link) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 516 | { | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 517 | struct net_device *dev; | 
|  | 518 | local_info_t *local; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 519 |  | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 520 | dev_dbg(&link->dev, "attach()\n"); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 521 |  | 
|  | 522 | /* Allocate the device structure */ | 
|  | 523 | dev = alloc_etherdev(sizeof(local_info_t)); | 
|  | 524 | if (!dev) | 
| Dominik Brodowski | f8cfa61 | 2005-11-14 21:25:51 +0100 | [diff] [blame] | 525 | return -ENOMEM; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 526 | local = netdev_priv(dev); | 
| David Howells | c402895 | 2006-11-22 14:57:56 +0000 | [diff] [blame] | 527 | local->dev = dev; | 
| Dominik Brodowski | fba395e | 2006-03-31 17:21:06 +0200 | [diff] [blame] | 528 | local->p_dev = link; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 529 | link->priv = dev; | 
|  | 530 |  | 
|  | 531 | /* General socket configuration */ | 
|  | 532 | link->conf.Attributes = CONF_ENABLE_IRQ; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 533 | link->conf.IntType = INT_MEMORY_AND_IO; | 
|  | 534 | link->conf.ConfigIndex = 1; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 535 |  | 
|  | 536 | /* Fill in card specific entries */ | 
| Stephen Hemminger | 0cd6e82 | 2009-03-20 19:36:08 +0000 | [diff] [blame] | 537 | dev->netdev_ops = &netdev_ops; | 
|  | 538 | dev->ethtool_ops = &netdev_ethtool_ops; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 539 | dev->watchdog_timeo = TX_TIMEOUT; | 
| David Howells | c402895 | 2006-11-22 14:57:56 +0000 | [diff] [blame] | 540 | INIT_WORK(&local->tx_timeout_task, xirc2ps_tx_timeout_task); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 541 |  | 
| Dominik Brodowski | 15b99ac | 2006-03-31 17:26:06 +0200 | [diff] [blame] | 542 | return xirc2ps_config(link); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 543 | } /* xirc2ps_attach */ | 
|  | 544 |  | 
|  | 545 | /**************** | 
|  | 546 | *  This deletes a driver "instance".  The device is de-registered | 
|  | 547 | *  with Card Services.  If it has been released, all local data | 
|  | 548 | *  structures are freed.  Otherwise, the structures will be freed | 
|  | 549 | *  when the device is released. | 
|  | 550 | */ | 
|  | 551 |  | 
|  | 552 | static void | 
| Dominik Brodowski | fba395e | 2006-03-31 17:21:06 +0200 | [diff] [blame] | 553 | xirc2ps_detach(struct pcmcia_device *link) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 554 | { | 
|  | 555 | struct net_device *dev = link->priv; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 556 |  | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 557 | dev_dbg(&link->dev, "detach\n"); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 558 |  | 
| Dominik Brodowski | c7c2fa0 | 2010-03-20 19:39:26 +0100 | [diff] [blame] | 559 | unregister_netdev(dev); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 560 |  | 
| Dominik Brodowski | e2d4096 | 2006-03-02 00:09:29 +0100 | [diff] [blame] | 561 | xirc2ps_release(link); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 562 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 563 | free_netdev(dev); | 
|  | 564 | } /* xirc2ps_detach */ | 
|  | 565 |  | 
|  | 566 | /**************** | 
|  | 567 | * Detect the type of the card. s is the buffer with the data of tuple 0x20 | 
|  | 568 | * Returns: 0 := not supported | 
|  | 569 | *		       mediaid=11 and prodid=47 | 
|  | 570 | * Media-Id bits: | 
|  | 571 | *  Ethernet	    0x01 | 
|  | 572 | *  Tokenring	    0x02 | 
|  | 573 | *  Arcnet	    0x04 | 
|  | 574 | *  Wireless	    0x08 | 
|  | 575 | *  Modem	    0x10 | 
|  | 576 | *  GSM only	    0x20 | 
|  | 577 | * Prod-Id bits: | 
|  | 578 | *  Pocket	    0x10 | 
|  | 579 | *  External	    0x20 | 
|  | 580 | *  Creditcard	    0x40 | 
|  | 581 | *  Cardbus	    0x80 | 
|  | 582 | * | 
|  | 583 | */ | 
|  | 584 | static int | 
| Dominik Brodowski | dddfbd8 | 2009-10-18 23:54:24 +0200 | [diff] [blame] | 585 | set_card_type(struct pcmcia_device *link) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 586 | { | 
|  | 587 | struct net_device *dev = link->priv; | 
|  | 588 | local_info_t *local = netdev_priv(dev); | 
| Dominik Brodowski | dddfbd8 | 2009-10-18 23:54:24 +0200 | [diff] [blame] | 589 | u8 *buf; | 
|  | 590 | unsigned int cisrev, mediaid, prodid; | 
|  | 591 | size_t len; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 592 |  | 
| Dominik Brodowski | dddfbd8 | 2009-10-18 23:54:24 +0200 | [diff] [blame] | 593 | len = pcmcia_get_tuple(link, CISTPL_MANFID, &buf); | 
|  | 594 | if (len < 5) { | 
|  | 595 | dev_err(&link->dev, "invalid CIS -- sorry\n"); | 
|  | 596 | return 0; | 
|  | 597 | } | 
|  | 598 |  | 
|  | 599 | cisrev = buf[2]; | 
|  | 600 | mediaid = buf[3]; | 
|  | 601 | prodid = buf[4]; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 602 |  | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 603 | dev_dbg(&link->dev, "cisrev=%02x mediaid=%02x prodid=%02x\n", | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 604 | cisrev, mediaid, prodid); | 
|  | 605 |  | 
|  | 606 | local->mohawk = 0; | 
|  | 607 | local->dingo = 0; | 
|  | 608 | local->modem = 0; | 
|  | 609 | local->card_type = XIR_UNKNOWN; | 
|  | 610 | if (!(prodid & 0x40)) { | 
|  | 611 | printk(KNOT_XIRC "Ooops: Not a creditcard\n"); | 
|  | 612 | return 0; | 
|  | 613 | } | 
|  | 614 | if (!(mediaid & 0x01)) { | 
|  | 615 | printk(KNOT_XIRC "Not an Ethernet card\n"); | 
|  | 616 | return 0; | 
|  | 617 | } | 
|  | 618 | if (mediaid & 0x10) { | 
|  | 619 | local->modem = 1; | 
|  | 620 | switch(prodid & 15) { | 
|  | 621 | case 1: local->card_type = XIR_CEM   ; break; | 
|  | 622 | case 2: local->card_type = XIR_CEM2  ; break; | 
|  | 623 | case 3: local->card_type = XIR_CEM3  ; break; | 
|  | 624 | case 4: local->card_type = XIR_CEM33 ; break; | 
|  | 625 | case 5: local->card_type = XIR_CEM56M; | 
|  | 626 | local->mohawk = 1; | 
|  | 627 | break; | 
|  | 628 | case 6: | 
|  | 629 | case 7: /* 7 is the RealPort 10/56 */ | 
|  | 630 | local->card_type = XIR_CEM56 ; | 
|  | 631 | local->mohawk = 1; | 
|  | 632 | local->dingo = 1; | 
|  | 633 | break; | 
|  | 634 | } | 
|  | 635 | } else { | 
|  | 636 | switch(prodid & 15) { | 
|  | 637 | case 1: local->card_type = has_ce2_string(link)? XIR_CE2 : XIR_CE ; | 
|  | 638 | break; | 
|  | 639 | case 2: local->card_type = XIR_CE2; break; | 
|  | 640 | case 3: local->card_type = XIR_CE3; | 
|  | 641 | local->mohawk = 1; | 
|  | 642 | break; | 
|  | 643 | } | 
|  | 644 | } | 
|  | 645 | if (local->card_type == XIR_CE || local->card_type == XIR_CEM) { | 
|  | 646 | printk(KNOT_XIRC "Sorry, this is an old CE card\n"); | 
|  | 647 | return 0; | 
|  | 648 | } | 
|  | 649 | if (local->card_type == XIR_UNKNOWN) | 
|  | 650 | printk(KNOT_XIRC "unknown card (mediaid=%02x prodid=%02x)\n", | 
|  | 651 | mediaid, prodid); | 
|  | 652 |  | 
|  | 653 | return 1; | 
|  | 654 | } | 
|  | 655 |  | 
|  | 656 | /**************** | 
|  | 657 | * There are some CE2 cards out which claim to be a CE card. | 
|  | 658 | * This function looks for a "CE2" in the 3rd version field. | 
|  | 659 | * Returns: true if this is a CE2 | 
|  | 660 | */ | 
|  | 661 | static int | 
| Dominik Brodowski | a9606fd | 2006-06-04 18:06:13 +0200 | [diff] [blame] | 662 | has_ce2_string(struct pcmcia_device * p_dev) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 663 | { | 
| Dominik Brodowski | a9606fd | 2006-06-04 18:06:13 +0200 | [diff] [blame] | 664 | if (p_dev->prod_id[2] && strstr(p_dev->prod_id[2], "CE2")) | 
|  | 665 | return 1; | 
|  | 666 | return 0; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 667 | } | 
|  | 668 |  | 
| Dominik Brodowski | b54bf94 | 2008-08-02 14:28:43 +0200 | [diff] [blame] | 669 | static int | 
|  | 670 | xirc2ps_config_modem(struct pcmcia_device *p_dev, | 
|  | 671 | cistpl_cftable_entry_t *cf, | 
| Dominik Brodowski | 8e2fc39 | 2008-08-02 15:30:31 +0200 | [diff] [blame] | 672 | cistpl_cftable_entry_t *dflt, | 
| Dominik Brodowski | ad913c1 | 2008-08-02 16:12:00 +0200 | [diff] [blame] | 673 | unsigned int vcc, | 
| Dominik Brodowski | b54bf94 | 2008-08-02 14:28:43 +0200 | [diff] [blame] | 674 | void *priv_data) | 
|  | 675 | { | 
|  | 676 | unsigned int ioaddr; | 
|  | 677 |  | 
|  | 678 | if (cf->io.nwin > 0  &&  (cf->io.win[0].base & 0xf) == 8) { | 
|  | 679 | for (ioaddr = 0x300; ioaddr < 0x400; ioaddr += 0x10) { | 
| Dominik Brodowski | 90abdc3 | 2010-07-24 17:23:51 +0200 | [diff] [blame] | 680 | p_dev->resource[1]->start = cf->io.win[0].base; | 
|  | 681 | p_dev->resource[0]->start = ioaddr; | 
|  | 682 | if (!pcmcia_request_io(p_dev)) | 
| Dominik Brodowski | b54bf94 | 2008-08-02 14:28:43 +0200 | [diff] [blame] | 683 | return 0; | 
|  | 684 | } | 
|  | 685 | } | 
|  | 686 | return -ENODEV; | 
|  | 687 | } | 
|  | 688 |  | 
|  | 689 | static int | 
|  | 690 | xirc2ps_config_check(struct pcmcia_device *p_dev, | 
|  | 691 | cistpl_cftable_entry_t *cf, | 
| Dominik Brodowski | 8e2fc39 | 2008-08-02 15:30:31 +0200 | [diff] [blame] | 692 | cistpl_cftable_entry_t *dflt, | 
| Dominik Brodowski | ad913c1 | 2008-08-02 16:12:00 +0200 | [diff] [blame] | 693 | unsigned int vcc, | 
| Dominik Brodowski | b54bf94 | 2008-08-02 14:28:43 +0200 | [diff] [blame] | 694 | void *priv_data) | 
|  | 695 | { | 
|  | 696 | int *pass = priv_data; | 
|  | 697 |  | 
|  | 698 | if (cf->io.nwin > 0 && (cf->io.win[0].base & 0xf) == 8) { | 
| Dominik Brodowski | 90abdc3 | 2010-07-24 17:23:51 +0200 | [diff] [blame] | 699 | p_dev->resource[1]->start = cf->io.win[0].base; | 
|  | 700 | p_dev->resource[0]->start = p_dev->resource[1]->start | 
| Dominik Brodowski | b54bf94 | 2008-08-02 14:28:43 +0200 | [diff] [blame] | 701 | + (*pass ? (cf->index & 0x20 ? -24:8) | 
|  | 702 | : (cf->index & 0x20 ?   8:-24)); | 
| Dominik Brodowski | 90abdc3 | 2010-07-24 17:23:51 +0200 | [diff] [blame] | 703 | if (!pcmcia_request_io(p_dev)) | 
| Dominik Brodowski | b54bf94 | 2008-08-02 14:28:43 +0200 | [diff] [blame] | 704 | return 0; | 
|  | 705 | } | 
|  | 706 | return -ENODEV; | 
|  | 707 |  | 
|  | 708 | } | 
|  | 709 |  | 
| Dominik Brodowski | dddfbd8 | 2009-10-18 23:54:24 +0200 | [diff] [blame] | 710 |  | 
|  | 711 | static int pcmcia_get_mac_ce(struct pcmcia_device *p_dev, | 
|  | 712 | tuple_t *tuple, | 
|  | 713 | void *priv) | 
|  | 714 | { | 
|  | 715 | struct net_device *dev = priv; | 
|  | 716 | int i; | 
|  | 717 |  | 
|  | 718 | if (tuple->TupleDataLen != 13) | 
|  | 719 | return -EINVAL; | 
|  | 720 | if ((tuple->TupleData[0] != 2) || (tuple->TupleData[1] != 1) || | 
|  | 721 | (tuple->TupleData[2] != 6)) | 
|  | 722 | return -EINVAL; | 
|  | 723 | /* another try	(James Lehmer's CE2 version 4.1)*/ | 
|  | 724 | for (i = 2; i < 6; i++) | 
|  | 725 | dev->dev_addr[i] = tuple->TupleData[i+2]; | 
|  | 726 | return 0; | 
|  | 727 | }; | 
|  | 728 |  | 
|  | 729 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 730 | /**************** | 
|  | 731 | * xirc2ps_config() is scheduled to run after a CARD_INSERTION event | 
|  | 732 | * is received, to configure the PCMCIA socket, and to make the | 
|  | 733 | * ethernet device available to the system. | 
|  | 734 | */ | 
| Dominik Brodowski | 15b99ac | 2006-03-31 17:26:06 +0200 | [diff] [blame] | 735 | static int | 
| Dominik Brodowski | fba395e | 2006-03-31 17:21:06 +0200 | [diff] [blame] | 736 | xirc2ps_config(struct pcmcia_device * link) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 737 | { | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 738 | struct net_device *dev = link->priv; | 
|  | 739 | local_info_t *local = netdev_priv(dev); | 
| Dominik Brodowski | b54bf94 | 2008-08-02 14:28:43 +0200 | [diff] [blame] | 740 | unsigned int ioaddr; | 
| Dominik Brodowski | dddfbd8 | 2009-10-18 23:54:24 +0200 | [diff] [blame] | 741 | int err; | 
|  | 742 | u8 *buf; | 
|  | 743 | size_t len; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 744 |  | 
|  | 745 | local->dingo_ccr = NULL; | 
|  | 746 |  | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 747 | dev_dbg(&link->dev, "config\n"); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 748 |  | 
|  | 749 | /* Is this a valid	card */ | 
| Dominik Brodowski | 7d2e8d0 | 2009-10-18 18:22:32 +0200 | [diff] [blame] | 750 | if (link->has_manf_id == 0) { | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 751 | printk(KNOT_XIRC "manfid not found in CIS\n"); | 
|  | 752 | goto failure; | 
|  | 753 | } | 
|  | 754 |  | 
| Dominik Brodowski | 7d2e8d0 | 2009-10-18 18:22:32 +0200 | [diff] [blame] | 755 | switch (link->manf_id) { | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 756 | case MANFID_XIRCOM: | 
|  | 757 | local->manf_str = "Xircom"; | 
|  | 758 | break; | 
|  | 759 | case MANFID_ACCTON: | 
|  | 760 | local->manf_str = "Accton"; | 
|  | 761 | break; | 
|  | 762 | case MANFID_COMPAQ: | 
|  | 763 | case MANFID_COMPAQ2: | 
|  | 764 | local->manf_str = "Compaq"; | 
|  | 765 | break; | 
|  | 766 | case MANFID_INTEL: | 
|  | 767 | local->manf_str = "Intel"; | 
|  | 768 | break; | 
|  | 769 | case MANFID_TOSHIBA: | 
|  | 770 | local->manf_str = "Toshiba"; | 
|  | 771 | break; | 
|  | 772 | default: | 
|  | 773 | printk(KNOT_XIRC "Unknown Card Manufacturer ID: 0x%04x\n", | 
| Dominik Brodowski | dddfbd8 | 2009-10-18 23:54:24 +0200 | [diff] [blame] | 774 | (unsigned)link->manf_id); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 775 | goto failure; | 
|  | 776 | } | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 777 | dev_dbg(&link->dev, "found %s card\n", local->manf_str); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 778 |  | 
| Dominik Brodowski | dddfbd8 | 2009-10-18 23:54:24 +0200 | [diff] [blame] | 779 | if (!set_card_type(link)) { | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 780 | printk(KNOT_XIRC "this card is not supported\n"); | 
|  | 781 | goto failure; | 
|  | 782 | } | 
|  | 783 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 784 | /* get the ethernet address from the CIS */ | 
| Dominik Brodowski | dddfbd8 | 2009-10-18 23:54:24 +0200 | [diff] [blame] | 785 | err = pcmcia_get_mac_from_cis(link, dev); | 
|  | 786 |  | 
|  | 787 | /* not found: try to get the node-id from tuple 0x89 */ | 
|  | 788 | if (err) { | 
|  | 789 | len = pcmcia_get_tuple(link, 0x89, &buf); | 
|  | 790 | /* data layout looks like tuple 0x22 */ | 
|  | 791 | if (buf && len == 8) { | 
|  | 792 | if (*buf == CISTPL_FUNCE_LAN_NODE_ID) { | 
|  | 793 | int i; | 
|  | 794 | for (i = 2; i < 6; i++) | 
|  | 795 | dev->dev_addr[i] = buf[i+2]; | 
|  | 796 | } else | 
|  | 797 | err = -1; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 798 | } | 
| Dominik Brodowski | dddfbd8 | 2009-10-18 23:54:24 +0200 | [diff] [blame] | 799 | kfree(buf); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 800 | } | 
| Dominik Brodowski | dddfbd8 | 2009-10-18 23:54:24 +0200 | [diff] [blame] | 801 |  | 
|  | 802 | if (err) | 
|  | 803 | err = pcmcia_loop_tuple(link, CISTPL_FUNCE, pcmcia_get_mac_ce, dev); | 
|  | 804 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 805 | if (err) { | 
|  | 806 | printk(KNOT_XIRC "node-id not found in CIS\n"); | 
|  | 807 | goto failure; | 
|  | 808 | } | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 809 |  | 
| Dominik Brodowski | 90abdc3 | 2010-07-24 17:23:51 +0200 | [diff] [blame] | 810 | link->resource[0]->flags |= IO_DATA_PATH_WIDTH_16; | 
| Dominik Brodowski | 4914c7f | 2010-08-13 20:33:34 +0200 | [diff] [blame] | 811 | link->io_lines = 10; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 812 | if (local->modem) { | 
|  | 813 | int pass; | 
|  | 814 |  | 
|  | 815 | if (do_sound) { | 
|  | 816 | link->conf.Attributes |= CONF_ENABLE_SPKR; | 
|  | 817 | link->conf.Status |= CCSR_AUDIO_ENA; | 
|  | 818 | } | 
| Dominik Brodowski | 90abdc3 | 2010-07-24 17:23:51 +0200 | [diff] [blame] | 819 | link->resource[1]->end = 8; | 
|  | 820 | link->resource[1]->flags |= IO_DATA_PATH_WIDTH_8; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 821 | if (local->dingo) { | 
|  | 822 | /* Take the Modem IO port from the CIS and scan for a free | 
|  | 823 | * Ethernet port */ | 
| Dominik Brodowski | 90abdc3 | 2010-07-24 17:23:51 +0200 | [diff] [blame] | 824 | link->resource[0]->end = 16; /* no Mako stuff anymore */ | 
| Dominik Brodowski | b54bf94 | 2008-08-02 14:28:43 +0200 | [diff] [blame] | 825 | if (!pcmcia_loop_config(link, xirc2ps_config_modem, NULL)) | 
|  | 826 | goto port_found; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 827 | } else { | 
| Dominik Brodowski | 90abdc3 | 2010-07-24 17:23:51 +0200 | [diff] [blame] | 828 | link->resource[0]->end = 18; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 829 | /* We do 2 passes here: The first one uses the regular mapping and | 
|  | 830 | * the second tries again, thereby considering that the 32 ports are | 
|  | 831 | * mirrored every 32 bytes. Actually we use a mirrored port for | 
|  | 832 | * the Mako if (on the first pass) the COR bit 5 is set. | 
|  | 833 | */ | 
| Dominik Brodowski | b54bf94 | 2008-08-02 14:28:43 +0200 | [diff] [blame] | 834 | for (pass=0; pass < 2; pass++) | 
|  | 835 | if (!pcmcia_loop_config(link, xirc2ps_config_check, &pass)) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 836 | goto port_found; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 837 | /* if special option: | 
|  | 838 | * try to configure as Ethernet only. | 
|  | 839 | * .... */ | 
|  | 840 | } | 
|  | 841 | printk(KNOT_XIRC "no ports available\n"); | 
|  | 842 | } else { | 
| Dominik Brodowski | 90abdc3 | 2010-07-24 17:23:51 +0200 | [diff] [blame] | 843 | link->resource[0]->end = 16; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 844 | for (ioaddr = 0x300; ioaddr < 0x400; ioaddr += 0x10) { | 
| Dominik Brodowski | 90abdc3 | 2010-07-24 17:23:51 +0200 | [diff] [blame] | 845 | link->resource[0]->start = ioaddr; | 
|  | 846 | if (!(err = pcmcia_request_io(link))) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 847 | goto port_found; | 
|  | 848 | } | 
| Dominik Brodowski | 90abdc3 | 2010-07-24 17:23:51 +0200 | [diff] [blame] | 849 | link->resource[0]->start = 0; /* let CS decide */ | 
|  | 850 | if ((err = pcmcia_request_io(link))) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 851 | goto config_error; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 852 | } | 
|  | 853 | port_found: | 
|  | 854 | if (err) | 
|  | 855 | goto config_error; | 
|  | 856 |  | 
|  | 857 | /**************** | 
|  | 858 | * Now allocate an interrupt line.	Note that this does not | 
|  | 859 | * actually assign a handler to the interrupt. | 
|  | 860 | */ | 
| Dominik Brodowski | eb14120 | 2010-03-07 12:21:16 +0100 | [diff] [blame] | 861 | if ((err=pcmcia_request_irq(link, xirc2ps_interrupt))) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 862 | goto config_error; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 863 |  | 
|  | 864 | /**************** | 
|  | 865 | * This actually configures the PCMCIA socket -- setting up | 
|  | 866 | * the I/O windows and the interrupt mapping. | 
|  | 867 | */ | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 868 | if ((err=pcmcia_request_configuration(link, &link->conf))) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 869 | goto config_error; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 870 |  | 
|  | 871 | if (local->dingo) { | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 872 | /* Reset the modem's BAR to the correct value | 
|  | 873 | * This is necessary because in the RequestConfiguration call, | 
|  | 874 | * the base address of the ethernet port (BasePort1) is written | 
|  | 875 | * to the BAR registers of the modem. | 
|  | 876 | */ | 
| Dominik Brodowski | 9a017a9 | 2010-07-24 15:58:54 +0200 | [diff] [blame] | 877 | err = pcmcia_write_config_byte(link, CISREG_IOBASE_0, (u8) | 
|  | 878 | link->resource[1]->start & 0xff); | 
| Dominik Brodowski | 1d5cc19 | 2010-07-24 12:23:21 +0200 | [diff] [blame] | 879 | if (err) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 880 | goto config_error; | 
| Dominik Brodowski | 1d5cc19 | 2010-07-24 12:23:21 +0200 | [diff] [blame] | 881 |  | 
|  | 882 | err = pcmcia_write_config_byte(link, CISREG_IOBASE_1, | 
| Dominik Brodowski | 9a017a9 | 2010-07-24 15:58:54 +0200 | [diff] [blame] | 883 | (link->resource[1]->start >> 8) & 0xff); | 
| Dominik Brodowski | 1d5cc19 | 2010-07-24 12:23:21 +0200 | [diff] [blame] | 884 | if (err) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 885 | goto config_error; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 886 |  | 
|  | 887 | /* There is no config entry for the Ethernet part which | 
|  | 888 | * is at 0x0800. So we allocate a window into the attribute | 
|  | 889 | * memory and write direct to the CIS registers | 
|  | 890 | */ | 
| Dominik Brodowski | cdb1380 | 2010-07-28 10:59:06 +0200 | [diff] [blame^] | 891 | link->resource[2]->flags = WIN_DATA_WIDTH_8 | WIN_MEMORY_TYPE_AM | | 
|  | 892 | WIN_ENABLE; | 
|  | 893 | link->resource[2]->start = link->resource[2]->end = 0; | 
|  | 894 | if ((err = pcmcia_request_window(link, link->resource[2], 0))) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 895 | goto config_error; | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 896 |  | 
| Dominik Brodowski | cdb1380 | 2010-07-28 10:59:06 +0200 | [diff] [blame^] | 897 | local->dingo_ccr = ioremap(link->resource[2]->start, 0x1000) + 0x0800; | 
|  | 898 | if ((err = pcmcia_map_mem_page(link, link->resource[2], 0))) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 899 | goto config_error; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 900 |  | 
|  | 901 | /* Setup the CCRs; there are no infos in the CIS about the Ethernet | 
|  | 902 | * part. | 
|  | 903 | */ | 
|  | 904 | writeb(0x47, local->dingo_ccr + CISREG_COR); | 
| Dominik Brodowski | 9a017a9 | 2010-07-24 15:58:54 +0200 | [diff] [blame] | 905 | ioaddr = link->resource[0]->start; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 906 | writeb(ioaddr & 0xff	  , local->dingo_ccr + CISREG_IOBASE_0); | 
|  | 907 | writeb((ioaddr >> 8)&0xff , local->dingo_ccr + CISREG_IOBASE_1); | 
|  | 908 |  | 
|  | 909 | #if 0 | 
|  | 910 | { | 
|  | 911 | u_char tmp; | 
|  | 912 | printk(KERN_INFO "ECOR:"); | 
|  | 913 | for (i=0; i < 7; i++) { | 
|  | 914 | tmp = readb(local->dingo_ccr + i*2); | 
|  | 915 | printk(" %02x", tmp); | 
|  | 916 | } | 
|  | 917 | printk("\n"); | 
|  | 918 | printk(KERN_INFO "DCOR:"); | 
|  | 919 | for (i=0; i < 4; i++) { | 
|  | 920 | tmp = readb(local->dingo_ccr + 0x20 + i*2); | 
|  | 921 | printk(" %02x", tmp); | 
|  | 922 | } | 
|  | 923 | printk("\n"); | 
|  | 924 | printk(KERN_INFO "SCOR:"); | 
|  | 925 | for (i=0; i < 10; i++) { | 
|  | 926 | tmp = readb(local->dingo_ccr + 0x40 + i*2); | 
|  | 927 | printk(" %02x", tmp); | 
|  | 928 | } | 
|  | 929 | printk("\n"); | 
|  | 930 | } | 
|  | 931 | #endif | 
|  | 932 |  | 
|  | 933 | writeb(0x01, local->dingo_ccr + 0x20); | 
|  | 934 | writeb(0x0c, local->dingo_ccr + 0x22); | 
|  | 935 | writeb(0x00, local->dingo_ccr + 0x24); | 
|  | 936 | writeb(0x00, local->dingo_ccr + 0x26); | 
|  | 937 | writeb(0x00, local->dingo_ccr + 0x28); | 
|  | 938 | } | 
|  | 939 |  | 
|  | 940 | /* The if_port symbol can be set when the module is loaded */ | 
|  | 941 | local->probe_port=0; | 
|  | 942 | if (!if_port) { | 
|  | 943 | local->probe_port = dev->if_port = 1; | 
|  | 944 | } else if ((if_port >= 1 && if_port <= 2) || | 
|  | 945 | (local->mohawk && if_port==4)) | 
|  | 946 | dev->if_port = if_port; | 
|  | 947 | else | 
|  | 948 | printk(KNOT_XIRC "invalid if_port requested\n"); | 
|  | 949 |  | 
|  | 950 | /* we can now register the device with the net subsystem */ | 
| Dominik Brodowski | eb14120 | 2010-03-07 12:21:16 +0100 | [diff] [blame] | 951 | dev->irq = link->irq; | 
| Dominik Brodowski | 9a017a9 | 2010-07-24 15:58:54 +0200 | [diff] [blame] | 952 | dev->base_addr = link->resource[0]->start; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 953 |  | 
|  | 954 | if (local->dingo) | 
|  | 955 | do_reset(dev, 1); /* a kludge to make the cem56 work */ | 
|  | 956 |  | 
| Dominik Brodowski | dd2e5a1 | 2009-11-03 10:27:34 +0100 | [diff] [blame] | 957 | SET_NETDEV_DEV(dev, &link->dev); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 958 |  | 
|  | 959 | if ((err=register_netdev(dev))) { | 
|  | 960 | printk(KNOT_XIRC "register_netdev() failed\n"); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 961 | goto config_error; | 
|  | 962 | } | 
|  | 963 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 964 | /* give some infos about the hardware */ | 
| Johannes Berg | e174961 | 2008-10-27 15:59:26 -0700 | [diff] [blame] | 965 | printk(KERN_INFO "%s: %s: port %#3lx, irq %d, hwaddr %pM\n", | 
| Joe Perches | 0795af5 | 2007-10-03 17:59:30 -0700 | [diff] [blame] | 966 | dev->name, local->manf_str,(u_long)dev->base_addr, (int)dev->irq, | 
| Johannes Berg | e174961 | 2008-10-27 15:59:26 -0700 | [diff] [blame] | 967 | dev->dev_addr); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 968 |  | 
| Dominik Brodowski | 15b99ac | 2006-03-31 17:26:06 +0200 | [diff] [blame] | 969 | return 0; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 970 |  | 
|  | 971 | config_error: | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 972 | xirc2ps_release(link); | 
| Dominik Brodowski | 15b99ac | 2006-03-31 17:26:06 +0200 | [diff] [blame] | 973 | return -ENODEV; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 974 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 975 | failure: | 
| Dominik Brodowski | 15b99ac | 2006-03-31 17:26:06 +0200 | [diff] [blame] | 976 | return -ENODEV; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 977 | } /* xirc2ps_config */ | 
|  | 978 |  | 
|  | 979 | /**************** | 
|  | 980 | * After a card is removed, xirc2ps_release() will unregister the net | 
|  | 981 | * device, and release the PCMCIA configuration.  If the device is | 
|  | 982 | * still open, this will be postponed until it is closed. | 
|  | 983 | */ | 
|  | 984 | static void | 
| Dominik Brodowski | fba395e | 2006-03-31 17:21:06 +0200 | [diff] [blame] | 985 | xirc2ps_release(struct pcmcia_device *link) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 986 | { | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 987 | dev_dbg(&link->dev, "release\n"); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 988 |  | 
| Dominik Brodowski | cdb1380 | 2010-07-28 10:59:06 +0200 | [diff] [blame^] | 989 | if (link->resource[2]->end) { | 
| Dominik Brodowski | 5f2a71f | 2006-01-15 09:32:39 +0100 | [diff] [blame] | 990 | struct net_device *dev = link->priv; | 
|  | 991 | local_info_t *local = netdev_priv(dev); | 
|  | 992 | if (local->dingo) | 
|  | 993 | iounmap(local->dingo_ccr - 0x0800); | 
|  | 994 | } | 
| Dominik Brodowski | fba395e | 2006-03-31 17:21:06 +0200 | [diff] [blame] | 995 | pcmcia_disable_device(link); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 996 | } /* xirc2ps_release */ | 
|  | 997 |  | 
|  | 998 | /*====================================================================*/ | 
|  | 999 |  | 
| Dominik Brodowski | 98e4c28 | 2005-11-14 21:21:18 +0100 | [diff] [blame] | 1000 |  | 
| Dominik Brodowski | fba395e | 2006-03-31 17:21:06 +0200 | [diff] [blame] | 1001 | static int xirc2ps_suspend(struct pcmcia_device *link) | 
| Dominik Brodowski | 98e4c28 | 2005-11-14 21:21:18 +0100 | [diff] [blame] | 1002 | { | 
| Dominik Brodowski | 98e4c28 | 2005-11-14 21:21:18 +0100 | [diff] [blame] | 1003 | struct net_device *dev = link->priv; | 
|  | 1004 |  | 
| Dominik Brodowski | e2d4096 | 2006-03-02 00:09:29 +0100 | [diff] [blame] | 1005 | if (link->open) { | 
|  | 1006 | netif_device_detach(dev); | 
|  | 1007 | do_powerdown(dev); | 
| Dominik Brodowski | 98e4c28 | 2005-11-14 21:21:18 +0100 | [diff] [blame] | 1008 | } | 
|  | 1009 |  | 
|  | 1010 | return 0; | 
|  | 1011 | } | 
|  | 1012 |  | 
| Dominik Brodowski | fba395e | 2006-03-31 17:21:06 +0200 | [diff] [blame] | 1013 | static int xirc2ps_resume(struct pcmcia_device *link) | 
| Dominik Brodowski | 98e4c28 | 2005-11-14 21:21:18 +0100 | [diff] [blame] | 1014 | { | 
| Dominik Brodowski | 98e4c28 | 2005-11-14 21:21:18 +0100 | [diff] [blame] | 1015 | struct net_device *dev = link->priv; | 
|  | 1016 |  | 
| Dominik Brodowski | e2d4096 | 2006-03-02 00:09:29 +0100 | [diff] [blame] | 1017 | if (link->open) { | 
| Dominik Brodowski | 8661bb5 | 2006-03-02 00:02:33 +0100 | [diff] [blame] | 1018 | do_reset(dev,1); | 
|  | 1019 | netif_device_attach(dev); | 
| Dominik Brodowski | 98e4c28 | 2005-11-14 21:21:18 +0100 | [diff] [blame] | 1020 | } | 
|  | 1021 |  | 
|  | 1022 | return 0; | 
|  | 1023 | } | 
|  | 1024 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1025 |  | 
|  | 1026 | /*====================================================================*/ | 
|  | 1027 |  | 
|  | 1028 | /**************** | 
|  | 1029 | * This is the Interrupt service route. | 
|  | 1030 | */ | 
|  | 1031 | static irqreturn_t | 
| David Howells | 7d12e78 | 2006-10-05 14:55:46 +0100 | [diff] [blame] | 1032 | xirc2ps_interrupt(int irq, void *dev_id) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1033 | { | 
|  | 1034 | struct net_device *dev = (struct net_device *)dev_id; | 
|  | 1035 | local_info_t *lp = netdev_priv(dev); | 
| Olof Johansson | 906da80 | 2008-02-04 22:27:35 -0800 | [diff] [blame] | 1036 | unsigned int ioaddr; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1037 | u_char saved_page; | 
|  | 1038 | unsigned bytes_rcvd; | 
|  | 1039 | unsigned int_status, eth_status, rx_status, tx_status; | 
|  | 1040 | unsigned rsr, pktlen; | 
|  | 1041 | ulong start_ticks = jiffies; /* fixme: jiffies rollover every 497 days | 
|  | 1042 | * is this something to worry about? | 
|  | 1043 | * -- on a laptop? | 
|  | 1044 | */ | 
|  | 1045 |  | 
|  | 1046 | if (!netif_device_present(dev)) | 
|  | 1047 | return IRQ_HANDLED; | 
|  | 1048 |  | 
|  | 1049 | ioaddr = dev->base_addr; | 
|  | 1050 | if (lp->mohawk) { /* must disable the interrupt */ | 
|  | 1051 | PutByte(XIRCREG_CR, 0); | 
|  | 1052 | } | 
|  | 1053 |  | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1054 | pr_debug("%s: interrupt %d at %#x.\n", dev->name, irq, ioaddr); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1055 |  | 
|  | 1056 | saved_page = GetByte(XIRCREG_PR); | 
|  | 1057 | /* Read the ISR to see whats the cause for the interrupt. | 
|  | 1058 | * This also clears the interrupt flags on CE2 cards | 
|  | 1059 | */ | 
|  | 1060 | int_status = GetByte(XIRCREG_ISR); | 
|  | 1061 | bytes_rcvd = 0; | 
|  | 1062 | loop_entry: | 
|  | 1063 | if (int_status == 0xff) { /* card may be ejected */ | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1064 | pr_debug("%s: interrupt %d for dead card\n", dev->name, irq); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1065 | goto leave; | 
|  | 1066 | } | 
|  | 1067 | eth_status = GetByte(XIRCREG_ESR); | 
|  | 1068 |  | 
|  | 1069 | SelectPage(0x40); | 
|  | 1070 | rx_status  = GetByte(XIRCREG40_RXST0); | 
|  | 1071 | PutByte(XIRCREG40_RXST0, (~rx_status & 0xff)); | 
|  | 1072 | tx_status = GetByte(XIRCREG40_TXST0); | 
|  | 1073 | tx_status |= GetByte(XIRCREG40_TXST1) << 8; | 
|  | 1074 | PutByte(XIRCREG40_TXST0, 0); | 
|  | 1075 | PutByte(XIRCREG40_TXST1, 0); | 
|  | 1076 |  | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1077 | pr_debug("%s: ISR=%#2.2x ESR=%#2.2x RSR=%#2.2x TSR=%#4.4x\n", | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1078 | dev->name, int_status, eth_status, rx_status, tx_status); | 
|  | 1079 |  | 
|  | 1080 | /***** receive section ******/ | 
|  | 1081 | SelectPage(0); | 
|  | 1082 | while (eth_status & FullPktRcvd) { | 
|  | 1083 | rsr = GetByte(XIRCREG0_RSR); | 
|  | 1084 | if (bytes_rcvd > maxrx_bytes && (rsr & PktRxOk)) { | 
|  | 1085 | /* too many bytes received during this int, drop the rest of the | 
|  | 1086 | * packets */ | 
| Stephen Hemminger | 6394d7c | 2009-03-20 19:36:07 +0000 | [diff] [blame] | 1087 | dev->stats.rx_dropped++; | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1088 | pr_debug("%s: RX drop, too much done\n", dev->name); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1089 | } else if (rsr & PktRxOk) { | 
|  | 1090 | struct sk_buff *skb; | 
|  | 1091 |  | 
|  | 1092 | pktlen = GetWord(XIRCREG0_RBC); | 
|  | 1093 | bytes_rcvd += pktlen; | 
|  | 1094 |  | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1095 | pr_debug("rsr=%#02x packet_length=%u\n", rsr, pktlen); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1096 |  | 
|  | 1097 | skb = dev_alloc_skb(pktlen+3); /* 1 extra so we can use insw */ | 
|  | 1098 | if (!skb) { | 
|  | 1099 | printk(KNOT_XIRC "low memory, packet dropped (size=%u)\n", | 
|  | 1100 | pktlen); | 
| Stephen Hemminger | 6394d7c | 2009-03-20 19:36:07 +0000 | [diff] [blame] | 1101 | dev->stats.rx_dropped++; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1102 | } else { /* okay get the packet */ | 
|  | 1103 | skb_reserve(skb, 2); | 
|  | 1104 | if (lp->silicon == 0 ) { /* work around a hardware bug */ | 
|  | 1105 | unsigned rhsa; /* receive start address */ | 
|  | 1106 |  | 
|  | 1107 | SelectPage(5); | 
|  | 1108 | rhsa = GetWord(XIRCREG5_RHSA0); | 
|  | 1109 | SelectPage(0); | 
|  | 1110 | rhsa += 3; /* skip control infos */ | 
|  | 1111 | if (rhsa >= 0x8000) | 
|  | 1112 | rhsa = 0; | 
|  | 1113 | if (rhsa + pktlen > 0x8000) { | 
|  | 1114 | unsigned i; | 
|  | 1115 | u_char *buf = skb_put(skb, pktlen); | 
|  | 1116 | for (i=0; i < pktlen ; i++, rhsa++) { | 
|  | 1117 | buf[i] = GetByte(XIRCREG_EDP); | 
|  | 1118 | if (rhsa == 0x8000) { | 
|  | 1119 | rhsa = 0; | 
|  | 1120 | i--; | 
|  | 1121 | } | 
|  | 1122 | } | 
|  | 1123 | } else { | 
|  | 1124 | insw(ioaddr+XIRCREG_EDP, | 
|  | 1125 | skb_put(skb, pktlen), (pktlen+1)>>1); | 
|  | 1126 | } | 
|  | 1127 | } | 
|  | 1128 | #if 0 | 
|  | 1129 | else if (lp->mohawk) { | 
|  | 1130 | /* To use this 32 bit access we should use | 
|  | 1131 | * a manual optimized loop | 
|  | 1132 | * Also the words are swapped, we can get more | 
|  | 1133 | * performance by using 32 bit access and swapping | 
|  | 1134 | * the words in a register. Will need this for cardbus | 
|  | 1135 | * | 
|  | 1136 | * Note: don't forget to change the ALLOC_SKB to .. +3 | 
|  | 1137 | */ | 
|  | 1138 | unsigned i; | 
|  | 1139 | u_long *p = skb_put(skb, pktlen); | 
|  | 1140 | register u_long a; | 
| Olof Johansson | 906da80 | 2008-02-04 22:27:35 -0800 | [diff] [blame] | 1141 | unsigned int edpreg = ioaddr+XIRCREG_EDP-2; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1142 | for (i=0; i < len ; i += 4, p++) { | 
|  | 1143 | a = inl(edpreg); | 
|  | 1144 | __asm__("rorl $16,%0\n\t" | 
|  | 1145 | :"=q" (a) | 
|  | 1146 | : "0" (a)); | 
|  | 1147 | *p = a; | 
|  | 1148 | } | 
|  | 1149 | } | 
|  | 1150 | #endif | 
|  | 1151 | else { | 
|  | 1152 | insw(ioaddr+XIRCREG_EDP, skb_put(skb, pktlen), | 
|  | 1153 | (pktlen+1)>>1); | 
|  | 1154 | } | 
|  | 1155 | skb->protocol = eth_type_trans(skb, dev); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1156 | netif_rx(skb); | 
| Stephen Hemminger | 6394d7c | 2009-03-20 19:36:07 +0000 | [diff] [blame] | 1157 | dev->stats.rx_packets++; | 
|  | 1158 | dev->stats.rx_bytes += pktlen; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1159 | if (!(rsr & PhyPkt)) | 
| Stephen Hemminger | 6394d7c | 2009-03-20 19:36:07 +0000 | [diff] [blame] | 1160 | dev->stats.multicast++; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1161 | } | 
|  | 1162 | } else { /* bad packet */ | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1163 | pr_debug("rsr=%#02x\n", rsr); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1164 | } | 
|  | 1165 | if (rsr & PktTooLong) { | 
| Stephen Hemminger | 6394d7c | 2009-03-20 19:36:07 +0000 | [diff] [blame] | 1166 | dev->stats.rx_frame_errors++; | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1167 | pr_debug("%s: Packet too long\n", dev->name); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1168 | } | 
|  | 1169 | if (rsr & CRCErr) { | 
| Stephen Hemminger | 6394d7c | 2009-03-20 19:36:07 +0000 | [diff] [blame] | 1170 | dev->stats.rx_crc_errors++; | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1171 | pr_debug("%s: CRC error\n", dev->name); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1172 | } | 
|  | 1173 | if (rsr & AlignErr) { | 
| Stephen Hemminger | 6394d7c | 2009-03-20 19:36:07 +0000 | [diff] [blame] | 1174 | dev->stats.rx_fifo_errors++; /* okay ? */ | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1175 | pr_debug("%s: Alignment error\n", dev->name); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1176 | } | 
|  | 1177 |  | 
|  | 1178 | /* clear the received/dropped/error packet */ | 
|  | 1179 | PutWord(XIRCREG0_DO, 0x8000); /* issue cmd: skip_rx_packet */ | 
|  | 1180 |  | 
|  | 1181 | /* get the new ethernet status */ | 
|  | 1182 | eth_status = GetByte(XIRCREG_ESR); | 
|  | 1183 | } | 
|  | 1184 | if (rx_status & 0x10) { /* Receive overrun */ | 
| Stephen Hemminger | 6394d7c | 2009-03-20 19:36:07 +0000 | [diff] [blame] | 1185 | dev->stats.rx_over_errors++; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1186 | PutByte(XIRCREG_CR, ClearRxOvrun); | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1187 | pr_debug("receive overrun cleared\n"); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1188 | } | 
|  | 1189 |  | 
|  | 1190 | /***** transmit section ******/ | 
|  | 1191 | if (int_status & PktTxed) { | 
|  | 1192 | unsigned n, nn; | 
|  | 1193 |  | 
|  | 1194 | n = lp->last_ptr_value; | 
|  | 1195 | nn = GetByte(XIRCREG0_PTR); | 
|  | 1196 | lp->last_ptr_value = nn; | 
|  | 1197 | if (nn < n) /* rollover */ | 
| Stephen Hemminger | 6394d7c | 2009-03-20 19:36:07 +0000 | [diff] [blame] | 1198 | dev->stats.tx_packets += 256 - n; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1199 | else if (n == nn) { /* happens sometimes - don't know why */ | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1200 | pr_debug("PTR not changed?\n"); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1201 | } else | 
| Stephen Hemminger | 6394d7c | 2009-03-20 19:36:07 +0000 | [diff] [blame] | 1202 | dev->stats.tx_packets += lp->last_ptr_value - n; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1203 | netif_wake_queue(dev); | 
|  | 1204 | } | 
|  | 1205 | if (tx_status & 0x0002) {	/* Execessive collissions */ | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1206 | pr_debug("tx restarted due to execssive collissions\n"); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1207 | PutByte(XIRCREG_CR, RestartTx);  /* restart transmitter process */ | 
|  | 1208 | } | 
|  | 1209 | if (tx_status & 0x0040) | 
| Stephen Hemminger | 6394d7c | 2009-03-20 19:36:07 +0000 | [diff] [blame] | 1210 | dev->stats.tx_aborted_errors++; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1211 |  | 
|  | 1212 | /* recalculate our work chunk so that we limit the duration of this | 
|  | 1213 | * ISR to about 1/10 of a second. | 
|  | 1214 | * Calculate only if we received a reasonable amount of bytes. | 
|  | 1215 | */ | 
|  | 1216 | if (bytes_rcvd > 1000) { | 
|  | 1217 | u_long duration = jiffies - start_ticks; | 
|  | 1218 |  | 
|  | 1219 | if (duration >= HZ/10) { /* if more than about 1/10 second */ | 
|  | 1220 | maxrx_bytes = (bytes_rcvd * (HZ/10)) / duration; | 
|  | 1221 | if (maxrx_bytes < 2000) | 
|  | 1222 | maxrx_bytes = 2000; | 
|  | 1223 | else if (maxrx_bytes > 22000) | 
|  | 1224 | maxrx_bytes = 22000; | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1225 | pr_debug("set maxrx=%u (rcvd=%u ticks=%lu)\n", | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1226 | maxrx_bytes, bytes_rcvd, duration); | 
|  | 1227 | } else if (!duration && maxrx_bytes < 22000) { | 
|  | 1228 | /* now much faster */ | 
|  | 1229 | maxrx_bytes += 2000; | 
|  | 1230 | if (maxrx_bytes > 22000) | 
|  | 1231 | maxrx_bytes = 22000; | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1232 | pr_debug("set maxrx=%u\n", maxrx_bytes); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1233 | } | 
|  | 1234 | } | 
|  | 1235 |  | 
|  | 1236 | leave: | 
|  | 1237 | if (lockup_hack) { | 
|  | 1238 | if (int_status != 0xff && (int_status = GetByte(XIRCREG_ISR)) != 0) | 
|  | 1239 | goto loop_entry; | 
|  | 1240 | } | 
|  | 1241 | SelectPage(saved_page); | 
|  | 1242 | PutByte(XIRCREG_CR, EnableIntr);  /* re-enable interrupts */ | 
|  | 1243 | /* Instead of dropping packets during a receive, we could | 
|  | 1244 | * force an interrupt with this command: | 
|  | 1245 | *	  PutByte(XIRCREG_CR, EnableIntr|ForceIntr); | 
|  | 1246 | */ | 
|  | 1247 | return IRQ_HANDLED; | 
|  | 1248 | } /* xirc2ps_interrupt */ | 
|  | 1249 |  | 
|  | 1250 | /*====================================================================*/ | 
|  | 1251 |  | 
|  | 1252 | static void | 
| David Howells | c402895 | 2006-11-22 14:57:56 +0000 | [diff] [blame] | 1253 | xirc2ps_tx_timeout_task(struct work_struct *work) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1254 | { | 
| David Howells | c402895 | 2006-11-22 14:57:56 +0000 | [diff] [blame] | 1255 | local_info_t *local = | 
|  | 1256 | container_of(work, local_info_t, tx_timeout_task); | 
|  | 1257 | struct net_device *dev = local->dev; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1258 | /* reset the card */ | 
|  | 1259 | do_reset(dev,1); | 
| Eric Dumazet | 1ae5dc3 | 2010-05-10 05:01:31 -0700 | [diff] [blame] | 1260 | dev->trans_start = jiffies; /* prevent tx timeout */ | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1261 | netif_wake_queue(dev); | 
|  | 1262 | } | 
|  | 1263 |  | 
| Joerg Ahrens | 9a469ab | 2006-08-20 21:51:57 +0100 | [diff] [blame] | 1264 | static void | 
| Arjan van de Ven | ed4cb13 | 2008-10-05 07:35:05 +0000 | [diff] [blame] | 1265 | xirc_tx_timeout(struct net_device *dev) | 
| Joerg Ahrens | 9a469ab | 2006-08-20 21:51:57 +0100 | [diff] [blame] | 1266 | { | 
|  | 1267 | local_info_t *lp = netdev_priv(dev); | 
| Stephen Hemminger | 6394d7c | 2009-03-20 19:36:07 +0000 | [diff] [blame] | 1268 | dev->stats.tx_errors++; | 
| Joerg Ahrens | 9a469ab | 2006-08-20 21:51:57 +0100 | [diff] [blame] | 1269 | printk(KERN_NOTICE "%s: transmit timed out\n", dev->name); | 
|  | 1270 | schedule_work(&lp->tx_timeout_task); | 
|  | 1271 | } | 
|  | 1272 |  | 
| Stephen Hemminger | dbf02fa | 2009-08-31 19:50:49 +0000 | [diff] [blame] | 1273 | static netdev_tx_t | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1274 | do_start_xmit(struct sk_buff *skb, struct net_device *dev) | 
|  | 1275 | { | 
|  | 1276 | local_info_t *lp = netdev_priv(dev); | 
| Olof Johansson | 906da80 | 2008-02-04 22:27:35 -0800 | [diff] [blame] | 1277 | unsigned int ioaddr = dev->base_addr; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1278 | int okay; | 
|  | 1279 | unsigned freespace; | 
| Eric Sesterhenn | da4f5cc | 2006-06-21 16:10:48 +0200 | [diff] [blame] | 1280 | unsigned pktlen = skb->len; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1281 |  | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1282 | pr_debug("do_start_xmit(skb=%p, dev=%p) len=%u\n", | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1283 | skb, dev, pktlen); | 
|  | 1284 |  | 
|  | 1285 |  | 
|  | 1286 | /* adjust the packet length to min. required | 
|  | 1287 | * and hope that the buffer is large enough | 
|  | 1288 | * to provide some random data. | 
|  | 1289 | * fixme: For Mohawk we can change this by sending | 
|  | 1290 | * a larger packetlen than we actually have; the chip will | 
|  | 1291 | * pad this in his buffer with random bytes | 
|  | 1292 | */ | 
|  | 1293 | if (pktlen < ETH_ZLEN) | 
|  | 1294 | { | 
| Herbert Xu | 5b057c6 | 2006-06-23 02:06:41 -0700 | [diff] [blame] | 1295 | if (skb_padto(skb, ETH_ZLEN)) | 
| Patrick McHardy | 6ed1065 | 2009-06-23 06:03:08 +0000 | [diff] [blame] | 1296 | return NETDEV_TX_OK; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1297 | pktlen = ETH_ZLEN; | 
|  | 1298 | } | 
|  | 1299 |  | 
|  | 1300 | netif_stop_queue(dev); | 
|  | 1301 | SelectPage(0); | 
|  | 1302 | PutWord(XIRCREG0_TRS, (u_short)pktlen+2); | 
|  | 1303 | freespace = GetWord(XIRCREG0_TSO); | 
|  | 1304 | okay = freespace & 0x8000; | 
|  | 1305 | freespace &= 0x7fff; | 
|  | 1306 | /* TRS doesn't work - (indeed it is eliminated with sil-rev 1) */ | 
|  | 1307 | okay = pktlen +2 < freespace; | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1308 | pr_debug("%s: avail. tx space=%u%s\n", | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1309 | dev->name, freespace, okay ? " (okay)":" (not enough)"); | 
|  | 1310 | if (!okay) { /* not enough space */ | 
| Patrick McHardy | 5b54814 | 2009-06-12 06:22:29 +0000 | [diff] [blame] | 1311 | return NETDEV_TX_BUSY;  /* upper layer may decide to requeue this packet */ | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1312 | } | 
|  | 1313 | /* send the packet */ | 
|  | 1314 | PutWord(XIRCREG_EDP, (u_short)pktlen); | 
|  | 1315 | outsw(ioaddr+XIRCREG_EDP, skb->data, pktlen>>1); | 
|  | 1316 | if (pktlen & 1) | 
|  | 1317 | PutByte(XIRCREG_EDP, skb->data[pktlen-1]); | 
|  | 1318 |  | 
|  | 1319 | if (lp->mohawk) | 
|  | 1320 | PutByte(XIRCREG_CR, TransmitPacket|EnableIntr); | 
|  | 1321 |  | 
|  | 1322 | dev_kfree_skb (skb); | 
| Stephen Hemminger | 6394d7c | 2009-03-20 19:36:07 +0000 | [diff] [blame] | 1323 | dev->stats.tx_bytes += pktlen; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1324 | netif_start_queue(dev); | 
| Patrick McHardy | 6ed1065 | 2009-06-23 06:03:08 +0000 | [diff] [blame] | 1325 | return NETDEV_TX_OK; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1326 | } | 
|  | 1327 |  | 
| Jiri Pirko | 91fea58 | 2010-02-19 08:48:47 +0000 | [diff] [blame] | 1328 | struct set_address_info { | 
|  | 1329 | int reg_nr; | 
|  | 1330 | int page_nr; | 
|  | 1331 | int mohawk; | 
|  | 1332 | unsigned int ioaddr; | 
|  | 1333 | }; | 
|  | 1334 |  | 
|  | 1335 | static void set_address(struct set_address_info *sa_info, char *addr) | 
|  | 1336 | { | 
|  | 1337 | unsigned int ioaddr = sa_info->ioaddr; | 
|  | 1338 | int i; | 
|  | 1339 |  | 
|  | 1340 | for (i = 0; i < 6; i++) { | 
|  | 1341 | if (sa_info->reg_nr > 15) { | 
|  | 1342 | sa_info->reg_nr = 8; | 
|  | 1343 | sa_info->page_nr++; | 
|  | 1344 | SelectPage(sa_info->page_nr); | 
|  | 1345 | } | 
|  | 1346 | if (sa_info->mohawk) | 
|  | 1347 | PutByte(sa_info->reg_nr++, addr[5 - i]); | 
|  | 1348 | else | 
|  | 1349 | PutByte(sa_info->reg_nr++, addr[i]); | 
|  | 1350 | } | 
|  | 1351 | } | 
|  | 1352 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1353 | /**************** | 
|  | 1354 | * Set all addresses: This first one is the individual address, | 
|  | 1355 | * the next 9 addresses are taken from the multicast list and | 
|  | 1356 | * the rest is filled with the individual address. | 
|  | 1357 | */ | 
| Jiri Pirko | 91fea58 | 2010-02-19 08:48:47 +0000 | [diff] [blame] | 1358 | static void set_addresses(struct net_device *dev) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1359 | { | 
| Jiri Pirko | 91fea58 | 2010-02-19 08:48:47 +0000 | [diff] [blame] | 1360 | unsigned int ioaddr = dev->base_addr; | 
|  | 1361 | local_info_t *lp = netdev_priv(dev); | 
| Jiri Pirko | 22bedad | 2010-04-01 21:22:57 +0000 | [diff] [blame] | 1362 | struct netdev_hw_addr *ha; | 
| Jiri Pirko | 91fea58 | 2010-02-19 08:48:47 +0000 | [diff] [blame] | 1363 | struct set_address_info sa_info; | 
|  | 1364 | int i; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1365 |  | 
| Jiri Pirko | 91fea58 | 2010-02-19 08:48:47 +0000 | [diff] [blame] | 1366 | /* | 
|  | 1367 | * Setup the info structure so that by first set_address call it will do | 
|  | 1368 | * SelectPage with the right page number. Hence these ones here. | 
|  | 1369 | */ | 
|  | 1370 | sa_info.reg_nr = 15 + 1; | 
|  | 1371 | sa_info.page_nr = 0x50 - 1; | 
|  | 1372 | sa_info.mohawk = lp->mohawk; | 
|  | 1373 | sa_info.ioaddr = ioaddr; | 
|  | 1374 |  | 
|  | 1375 | set_address(&sa_info, dev->dev_addr); | 
|  | 1376 | i = 0; | 
| Jiri Pirko | 22bedad | 2010-04-01 21:22:57 +0000 | [diff] [blame] | 1377 | netdev_for_each_mc_addr(ha, dev) { | 
| Jiri Pirko | 91fea58 | 2010-02-19 08:48:47 +0000 | [diff] [blame] | 1378 | if (i++ == 9) | 
|  | 1379 | break; | 
| Jiri Pirko | 22bedad | 2010-04-01 21:22:57 +0000 | [diff] [blame] | 1380 | set_address(&sa_info, ha->addr); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1381 | } | 
| Jiri Pirko | 91fea58 | 2010-02-19 08:48:47 +0000 | [diff] [blame] | 1382 | while (i++ < 9) | 
|  | 1383 | set_address(&sa_info, dev->dev_addr); | 
|  | 1384 | SelectPage(0); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1385 | } | 
|  | 1386 |  | 
|  | 1387 | /**************** | 
|  | 1388 | * Set or clear the multicast filter for this adaptor. | 
|  | 1389 | * We can filter up to 9 addresses, if more are requested we set | 
|  | 1390 | * multicast promiscuous mode. | 
|  | 1391 | */ | 
|  | 1392 |  | 
|  | 1393 | static void | 
|  | 1394 | set_multicast_list(struct net_device *dev) | 
|  | 1395 | { | 
| Olof Johansson | 906da80 | 2008-02-04 22:27:35 -0800 | [diff] [blame] | 1396 | unsigned int ioaddr = dev->base_addr; | 
| Komuro | 43fc63d | 2008-04-20 14:32:34 +0900 | [diff] [blame] | 1397 | unsigned value; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1398 |  | 
|  | 1399 | SelectPage(0x42); | 
| Komuro | 43fc63d | 2008-04-20 14:32:34 +0900 | [diff] [blame] | 1400 | value = GetByte(XIRCREG42_SWC1) & 0xC0; | 
|  | 1401 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1402 | if (dev->flags & IFF_PROMISC) { /* snoop */ | 
| Komuro | 43fc63d | 2008-04-20 14:32:34 +0900 | [diff] [blame] | 1403 | PutByte(XIRCREG42_SWC1, value | 0x06); /* set MPE and PME */ | 
| Jiri Pirko | 4cd24ea | 2010-02-08 04:30:35 +0000 | [diff] [blame] | 1404 | } else if (netdev_mc_count(dev) > 9 || (dev->flags & IFF_ALLMULTI)) { | 
| Komuro | 43fc63d | 2008-04-20 14:32:34 +0900 | [diff] [blame] | 1405 | PutByte(XIRCREG42_SWC1, value | 0x02); /* set MPE */ | 
| Jiri Pirko | 4cd24ea | 2010-02-08 04:30:35 +0000 | [diff] [blame] | 1406 | } else if (!netdev_mc_empty(dev)) { | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1407 | /* the chip can filter 9 addresses perfectly */ | 
| Komuro | 43fc63d | 2008-04-20 14:32:34 +0900 | [diff] [blame] | 1408 | PutByte(XIRCREG42_SWC1, value | 0x01); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1409 | SelectPage(0x40); | 
|  | 1410 | PutByte(XIRCREG40_CMD0, Offline); | 
|  | 1411 | set_addresses(dev); | 
|  | 1412 | SelectPage(0x40); | 
|  | 1413 | PutByte(XIRCREG40_CMD0, EnableRecv | Online); | 
|  | 1414 | } else { /* standard usage */ | 
| Komuro | 43fc63d | 2008-04-20 14:32:34 +0900 | [diff] [blame] | 1415 | PutByte(XIRCREG42_SWC1, value | 0x00); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1416 | } | 
|  | 1417 | SelectPage(0); | 
|  | 1418 | } | 
|  | 1419 |  | 
|  | 1420 | static int | 
|  | 1421 | do_config(struct net_device *dev, struct ifmap *map) | 
|  | 1422 | { | 
|  | 1423 | local_info_t *local = netdev_priv(dev); | 
|  | 1424 |  | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1425 | pr_debug("do_config(%p)\n", dev); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1426 | if (map->port != 255 && map->port != dev->if_port) { | 
|  | 1427 | if (map->port > 4) | 
|  | 1428 | return -EINVAL; | 
|  | 1429 | if (!map->port) { | 
|  | 1430 | local->probe_port = 1; | 
|  | 1431 | dev->if_port = 1; | 
|  | 1432 | } else { | 
|  | 1433 | local->probe_port = 0; | 
|  | 1434 | dev->if_port = map->port; | 
|  | 1435 | } | 
|  | 1436 | printk(KERN_INFO "%s: switching to %s port\n", | 
|  | 1437 | dev->name, if_names[dev->if_port]); | 
|  | 1438 | do_reset(dev,1);  /* not the fine way :-) */ | 
|  | 1439 | } | 
|  | 1440 | return 0; | 
|  | 1441 | } | 
|  | 1442 |  | 
|  | 1443 | /**************** | 
|  | 1444 | * Open the driver | 
|  | 1445 | */ | 
|  | 1446 | static int | 
|  | 1447 | do_open(struct net_device *dev) | 
|  | 1448 | { | 
|  | 1449 | local_info_t *lp = netdev_priv(dev); | 
| Dominik Brodowski | fba395e | 2006-03-31 17:21:06 +0200 | [diff] [blame] | 1450 | struct pcmcia_device *link = lp->p_dev; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1451 |  | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1452 | dev_dbg(&link->dev, "do_open(%p)\n", dev); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1453 |  | 
|  | 1454 | /* Check that the PCMCIA card is still here. */ | 
|  | 1455 | /* Physical device present signature. */ | 
| Dominik Brodowski | 9940ec3 | 2006-03-05 11:04:33 +0100 | [diff] [blame] | 1456 | if (!pcmcia_dev_present(link)) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1457 | return -ENODEV; | 
|  | 1458 |  | 
|  | 1459 | /* okay */ | 
|  | 1460 | link->open++; | 
|  | 1461 |  | 
|  | 1462 | netif_start_queue(dev); | 
|  | 1463 | do_reset(dev,1); | 
|  | 1464 |  | 
|  | 1465 | return 0; | 
|  | 1466 | } | 
|  | 1467 |  | 
|  | 1468 | static void netdev_get_drvinfo(struct net_device *dev, | 
|  | 1469 | struct ethtool_drvinfo *info) | 
|  | 1470 | { | 
|  | 1471 | strcpy(info->driver, "xirc2ps_cs"); | 
|  | 1472 | sprintf(info->bus_info, "PCMCIA 0x%lx", dev->base_addr); | 
|  | 1473 | } | 
|  | 1474 |  | 
| Jeff Garzik | 7282d49 | 2006-09-13 14:30:00 -0400 | [diff] [blame] | 1475 | static const struct ethtool_ops netdev_ethtool_ops = { | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1476 | .get_drvinfo		= netdev_get_drvinfo, | 
|  | 1477 | }; | 
|  | 1478 |  | 
|  | 1479 | static int | 
|  | 1480 | do_ioctl(struct net_device *dev, struct ifreq *rq, int cmd) | 
|  | 1481 | { | 
|  | 1482 | local_info_t *local = netdev_priv(dev); | 
| Olof Johansson | 906da80 | 2008-02-04 22:27:35 -0800 | [diff] [blame] | 1483 | unsigned int ioaddr = dev->base_addr; | 
| Ben Hutchings | 0fa0ee05 | 2009-09-03 10:41:17 +0000 | [diff] [blame] | 1484 | struct mii_ioctl_data *data = if_mii(rq); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1485 |  | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1486 | pr_debug("%s: ioctl(%-.6s, %#04x) %04x %04x %04x %04x\n", | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1487 | dev->name, rq->ifr_ifrn.ifrn_name, cmd, | 
| Ben Hutchings | 0fa0ee05 | 2009-09-03 10:41:17 +0000 | [diff] [blame] | 1488 | data->phy_id, data->reg_num, data->val_in, data->val_out); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1489 |  | 
|  | 1490 | if (!local->mohawk) | 
|  | 1491 | return -EOPNOTSUPP; | 
|  | 1492 |  | 
|  | 1493 | switch(cmd) { | 
|  | 1494 | case SIOCGMIIPHY:		/* Get the address of the PHY in use. */ | 
| Ben Hutchings | 0fa0ee05 | 2009-09-03 10:41:17 +0000 | [diff] [blame] | 1495 | data->phy_id = 0;	/* we have only this address */ | 
| Adrian Bunk | 93b1fae | 2006-01-10 00:13:33 +0100 | [diff] [blame] | 1496 | /* fall through */ | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1497 | case SIOCGMIIREG:		/* Read the specified MII register. */ | 
| Ben Hutchings | 0fa0ee05 | 2009-09-03 10:41:17 +0000 | [diff] [blame] | 1498 | data->val_out = mii_rd(ioaddr, data->phy_id & 0x1f, | 
|  | 1499 | data->reg_num & 0x1f); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1500 | break; | 
|  | 1501 | case SIOCSMIIREG:		/* Write the specified MII register */ | 
| Ben Hutchings | 0fa0ee05 | 2009-09-03 10:41:17 +0000 | [diff] [blame] | 1502 | mii_wr(ioaddr, data->phy_id & 0x1f, data->reg_num & 0x1f, data->val_in, | 
|  | 1503 | 16); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1504 | break; | 
|  | 1505 | default: | 
|  | 1506 | return -EOPNOTSUPP; | 
|  | 1507 | } | 
|  | 1508 | return 0; | 
|  | 1509 | } | 
|  | 1510 |  | 
|  | 1511 | static void | 
|  | 1512 | hardreset(struct net_device *dev) | 
|  | 1513 | { | 
|  | 1514 | local_info_t *local = netdev_priv(dev); | 
| Olof Johansson | 906da80 | 2008-02-04 22:27:35 -0800 | [diff] [blame] | 1515 | unsigned int ioaddr = dev->base_addr; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1516 |  | 
|  | 1517 | SelectPage(4); | 
|  | 1518 | udelay(1); | 
|  | 1519 | PutByte(XIRCREG4_GPR1, 0);	     /* clear bit 0: power down */ | 
|  | 1520 | msleep(40);				     /* wait 40 msec */ | 
|  | 1521 | if (local->mohawk) | 
|  | 1522 | PutByte(XIRCREG4_GPR1, 1);	 /* set bit 0: power up */ | 
|  | 1523 | else | 
|  | 1524 | PutByte(XIRCREG4_GPR1, 1 | 4);	 /* set bit 0: power up, bit 2: AIC */ | 
|  | 1525 | msleep(20);			     /* wait 20 msec */ | 
|  | 1526 | } | 
|  | 1527 |  | 
|  | 1528 | static void | 
|  | 1529 | do_reset(struct net_device *dev, int full) | 
|  | 1530 | { | 
|  | 1531 | local_info_t *local = netdev_priv(dev); | 
| Olof Johansson | 906da80 | 2008-02-04 22:27:35 -0800 | [diff] [blame] | 1532 | unsigned int ioaddr = dev->base_addr; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1533 | unsigned value; | 
|  | 1534 |  | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1535 | pr_debug("%s: do_reset(%p,%d)\n", dev? dev->name:"eth?", dev, full); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1536 |  | 
|  | 1537 | hardreset(dev); | 
|  | 1538 | PutByte(XIRCREG_CR, SoftReset); /* set */ | 
|  | 1539 | msleep(20);			     /* wait 20 msec */ | 
|  | 1540 | PutByte(XIRCREG_CR, 0);	     /* clear */ | 
|  | 1541 | msleep(40);			     /* wait 40 msec */ | 
|  | 1542 | if (local->mohawk) { | 
|  | 1543 | SelectPage(4); | 
|  | 1544 | /* set pin GP1 and GP2 to output  (0x0c) | 
|  | 1545 | * set GP1 to low to power up the ML6692 (0x00) | 
|  | 1546 | * set GP2 to high to power up the 10Mhz chip  (0x02) | 
|  | 1547 | */ | 
|  | 1548 | PutByte(XIRCREG4_GPR0, 0x0e); | 
|  | 1549 | } | 
|  | 1550 |  | 
|  | 1551 | /* give the circuits some time to power up */ | 
|  | 1552 | msleep(500);			/* about 500ms */ | 
|  | 1553 |  | 
|  | 1554 | local->last_ptr_value = 0; | 
|  | 1555 | local->silicon = local->mohawk ? (GetByte(XIRCREG4_BOV) & 0x70) >> 4 | 
|  | 1556 | : (GetByte(XIRCREG4_BOV) & 0x30) >> 4; | 
|  | 1557 |  | 
|  | 1558 | if (local->probe_port) { | 
|  | 1559 | if (!local->mohawk) { | 
|  | 1560 | SelectPage(4); | 
|  | 1561 | PutByte(XIRCREG4_GPR0, 4); | 
|  | 1562 | local->probe_port = 0; | 
|  | 1563 | } | 
|  | 1564 | } else if (dev->if_port == 2) { /* enable 10Base2 */ | 
|  | 1565 | SelectPage(0x42); | 
|  | 1566 | PutByte(XIRCREG42_SWC1, 0xC0); | 
|  | 1567 | } else { /* enable 10BaseT */ | 
|  | 1568 | SelectPage(0x42); | 
|  | 1569 | PutByte(XIRCREG42_SWC1, 0x80); | 
|  | 1570 | } | 
|  | 1571 | msleep(40);			     /* wait 40 msec to let it complete */ | 
|  | 1572 |  | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1573 | #if 0 | 
|  | 1574 | { | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1575 | SelectPage(0); | 
|  | 1576 | value = GetByte(XIRCREG_ESR);	 /* read the ESR */ | 
|  | 1577 | printk(KERN_DEBUG "%s: ESR is: %#02x\n", dev->name, value); | 
|  | 1578 | } | 
|  | 1579 | #endif | 
|  | 1580 |  | 
|  | 1581 | /* setup the ECR */ | 
|  | 1582 | SelectPage(1); | 
|  | 1583 | PutByte(XIRCREG1_IMR0, 0xff); /* allow all ints */ | 
|  | 1584 | PutByte(XIRCREG1_IMR1, 1	); /* and Set TxUnderrunDetect */ | 
|  | 1585 | value = GetByte(XIRCREG1_ECR); | 
|  | 1586 | #if 0 | 
|  | 1587 | if (local->mohawk) | 
|  | 1588 | value |= DisableLinkPulse; | 
|  | 1589 | PutByte(XIRCREG1_ECR, value); | 
|  | 1590 | #endif | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1591 | pr_debug("%s: ECR is: %#02x\n", dev->name, value); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1592 |  | 
|  | 1593 | SelectPage(0x42); | 
|  | 1594 | PutByte(XIRCREG42_SWC0, 0x20); /* disable source insertion */ | 
|  | 1595 |  | 
|  | 1596 | if (local->silicon != 1) { | 
|  | 1597 | /* set the local memory dividing line. | 
|  | 1598 | * The comments in the sample code say that this is only | 
|  | 1599 | * settable with the scipper version 2 which is revision 0. | 
|  | 1600 | * Always for CE3 cards | 
|  | 1601 | */ | 
|  | 1602 | SelectPage(2); | 
|  | 1603 | PutWord(XIRCREG2_RBS, 0x2000); | 
|  | 1604 | } | 
|  | 1605 |  | 
|  | 1606 | if (full) | 
|  | 1607 | set_addresses(dev); | 
|  | 1608 |  | 
|  | 1609 | /* Hardware workaround: | 
|  | 1610 | * The receive byte pointer after reset is off by 1 so we need | 
|  | 1611 | * to move the offset pointer back to 0. | 
|  | 1612 | */ | 
|  | 1613 | SelectPage(0); | 
|  | 1614 | PutWord(XIRCREG0_DO, 0x2000); /* change offset command, off=0 */ | 
|  | 1615 |  | 
|  | 1616 | /* setup MAC IMRs and clear status registers */ | 
|  | 1617 | SelectPage(0x40);		     /* Bit 7 ... bit 0 */ | 
|  | 1618 | PutByte(XIRCREG40_RMASK0, 0xff); /* ROK, RAB, rsv, RO, CRC, AE, PTL, MP */ | 
|  | 1619 | PutByte(XIRCREG40_TMASK0, 0xff); /* TOK, TAB, SQE, LL, TU, JAB, EXC, CRS */ | 
|  | 1620 | PutByte(XIRCREG40_TMASK1, 0xb0); /* rsv, rsv, PTD, EXT, rsv,rsv,rsv, rsv*/ | 
|  | 1621 | PutByte(XIRCREG40_RXST0,  0x00); /* ROK, RAB, REN, RO, CRC, AE, PTL, MP */ | 
|  | 1622 | PutByte(XIRCREG40_TXST0,  0x00); /* TOK, TAB, SQE, LL, TU, JAB, EXC, CRS */ | 
|  | 1623 | PutByte(XIRCREG40_TXST1,  0x00); /* TEN, rsv, PTD, EXT, retry_counter:4  */ | 
|  | 1624 |  | 
|  | 1625 | if (full && local->mohawk && init_mii(dev)) { | 
|  | 1626 | if (dev->if_port == 4 || local->dingo || local->new_mii) { | 
|  | 1627 | printk(KERN_INFO "%s: MII selected\n", dev->name); | 
|  | 1628 | SelectPage(2); | 
|  | 1629 | PutByte(XIRCREG2_MSR, GetByte(XIRCREG2_MSR) | 0x08); | 
|  | 1630 | msleep(20); | 
|  | 1631 | } else { | 
|  | 1632 | printk(KERN_INFO "%s: MII detected; using 10mbs\n", | 
|  | 1633 | dev->name); | 
|  | 1634 | SelectPage(0x42); | 
|  | 1635 | if (dev->if_port == 2) /* enable 10Base2 */ | 
|  | 1636 | PutByte(XIRCREG42_SWC1, 0xC0); | 
|  | 1637 | else  /* enable 10BaseT */ | 
|  | 1638 | PutByte(XIRCREG42_SWC1, 0x80); | 
|  | 1639 | msleep(40);			/* wait 40 msec to let it complete */ | 
|  | 1640 | } | 
|  | 1641 | if (full_duplex) | 
|  | 1642 | PutByte(XIRCREG1_ECR, GetByte(XIRCREG1_ECR | FullDuplex)); | 
|  | 1643 | } else {  /* No MII */ | 
|  | 1644 | SelectPage(0); | 
|  | 1645 | value = GetByte(XIRCREG_ESR);	 /* read the ESR */ | 
|  | 1646 | dev->if_port = (value & MediaSelect) ? 1 : 2; | 
|  | 1647 | } | 
|  | 1648 |  | 
|  | 1649 | /* configure the LEDs */ | 
|  | 1650 | SelectPage(2); | 
|  | 1651 | if (dev->if_port == 1 || dev->if_port == 4) /* TP: Link and Activity */ | 
|  | 1652 | PutByte(XIRCREG2_LED, 0x3b); | 
|  | 1653 | else			      /* Coax: Not-Collision and Activity */ | 
|  | 1654 | PutByte(XIRCREG2_LED, 0x3a); | 
|  | 1655 |  | 
|  | 1656 | if (local->dingo) | 
|  | 1657 | PutByte(0x0b, 0x04); /* 100 Mbit LED */ | 
|  | 1658 |  | 
|  | 1659 | /* enable receiver and put the mac online */ | 
|  | 1660 | if (full) { | 
| Komuro | 43fc63d | 2008-04-20 14:32:34 +0900 | [diff] [blame] | 1661 | set_multicast_list(dev); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1662 | SelectPage(0x40); | 
|  | 1663 | PutByte(XIRCREG40_CMD0, EnableRecv | Online); | 
|  | 1664 | } | 
|  | 1665 |  | 
|  | 1666 | /* setup Ethernet IMR and enable interrupts */ | 
|  | 1667 | SelectPage(1); | 
|  | 1668 | PutByte(XIRCREG1_IMR0, 0xff); | 
|  | 1669 | udelay(1); | 
|  | 1670 | SelectPage(0); | 
|  | 1671 | PutByte(XIRCREG_CR, EnableIntr); | 
|  | 1672 | if (local->modem && !local->dingo) { /* do some magic */ | 
|  | 1673 | if (!(GetByte(0x10) & 0x01)) | 
|  | 1674 | PutByte(0x10, 0x11); /* unmask master-int bit */ | 
|  | 1675 | } | 
|  | 1676 |  | 
|  | 1677 | if (full) | 
|  | 1678 | printk(KERN_INFO "%s: media %s, silicon revision %d\n", | 
|  | 1679 | dev->name, if_names[dev->if_port], local->silicon); | 
|  | 1680 | /* We should switch back to page 0 to avoid a bug in revision 0 | 
|  | 1681 | * where regs with offset below 8 can't be read after an access | 
|  | 1682 | * to the MAC registers */ | 
|  | 1683 | SelectPage(0); | 
|  | 1684 | } | 
|  | 1685 |  | 
|  | 1686 | /**************** | 
|  | 1687 | * Initialize the Media-Independent-Interface | 
|  | 1688 | * Returns: True if we have a good MII | 
|  | 1689 | */ | 
|  | 1690 | static int | 
|  | 1691 | init_mii(struct net_device *dev) | 
|  | 1692 | { | 
|  | 1693 | local_info_t *local = netdev_priv(dev); | 
| Olof Johansson | 906da80 | 2008-02-04 22:27:35 -0800 | [diff] [blame] | 1694 | unsigned int ioaddr = dev->base_addr; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1695 | unsigned control, status, linkpartner; | 
|  | 1696 | int i; | 
|  | 1697 |  | 
|  | 1698 | if (if_port == 4 || if_port == 1) { /* force 100BaseT or 10BaseT */ | 
|  | 1699 | dev->if_port = if_port; | 
|  | 1700 | local->probe_port = 0; | 
|  | 1701 | return 1; | 
|  | 1702 | } | 
|  | 1703 |  | 
|  | 1704 | status = mii_rd(ioaddr,  0, 1); | 
|  | 1705 | if ((status & 0xff00) != 0x7800) | 
|  | 1706 | return 0; /* No MII */ | 
|  | 1707 |  | 
|  | 1708 | local->new_mii = (mii_rd(ioaddr, 0, 2) != 0xffff); | 
|  | 1709 |  | 
|  | 1710 | if (local->probe_port) | 
|  | 1711 | control = 0x1000; /* auto neg */ | 
|  | 1712 | else if (dev->if_port == 4) | 
|  | 1713 | control = 0x2000; /* no auto neg, 100mbs mode */ | 
|  | 1714 | else | 
|  | 1715 | control = 0x0000; /* no auto neg, 10mbs mode */ | 
|  | 1716 | mii_wr(ioaddr,  0, 0, control, 16); | 
|  | 1717 | udelay(100); | 
|  | 1718 | control = mii_rd(ioaddr, 0, 0); | 
|  | 1719 |  | 
|  | 1720 | if (control & 0x0400) { | 
|  | 1721 | printk(KERN_NOTICE "%s can't take PHY out of isolation mode\n", | 
|  | 1722 | dev->name); | 
|  | 1723 | local->probe_port = 0; | 
|  | 1724 | return 0; | 
|  | 1725 | } | 
|  | 1726 |  | 
|  | 1727 | if (local->probe_port) { | 
|  | 1728 | /* according to the DP83840A specs the auto negotiation process | 
|  | 1729 | * may take up to 3.5 sec, so we use this also for our ML6692 | 
|  | 1730 | * Fixme: Better to use a timer here! | 
|  | 1731 | */ | 
|  | 1732 | for (i=0; i < 35; i++) { | 
|  | 1733 | msleep(100);	 /* wait 100 msec */ | 
|  | 1734 | status = mii_rd(ioaddr,  0, 1); | 
|  | 1735 | if ((status & 0x0020) && (status & 0x0004)) | 
|  | 1736 | break; | 
|  | 1737 | } | 
|  | 1738 |  | 
|  | 1739 | if (!(status & 0x0020)) { | 
|  | 1740 | printk(KERN_INFO "%s: autonegotiation failed;" | 
|  | 1741 | " using 10mbs\n", dev->name); | 
|  | 1742 | if (!local->new_mii) { | 
|  | 1743 | control = 0x0000; | 
|  | 1744 | mii_wr(ioaddr,  0, 0, control, 16); | 
|  | 1745 | udelay(100); | 
|  | 1746 | SelectPage(0); | 
|  | 1747 | dev->if_port = (GetByte(XIRCREG_ESR) & MediaSelect) ? 1 : 2; | 
|  | 1748 | } | 
|  | 1749 | } else { | 
|  | 1750 | linkpartner = mii_rd(ioaddr, 0, 5); | 
|  | 1751 | printk(KERN_INFO "%s: MII link partner: %04x\n", | 
|  | 1752 | dev->name, linkpartner); | 
|  | 1753 | if (linkpartner & 0x0080) { | 
|  | 1754 | dev->if_port = 4; | 
|  | 1755 | } else | 
|  | 1756 | dev->if_port = 1; | 
|  | 1757 | } | 
|  | 1758 | } | 
|  | 1759 |  | 
|  | 1760 | return 1; | 
|  | 1761 | } | 
|  | 1762 |  | 
|  | 1763 | static void | 
|  | 1764 | do_powerdown(struct net_device *dev) | 
|  | 1765 | { | 
|  | 1766 |  | 
| Olof Johansson | 906da80 | 2008-02-04 22:27:35 -0800 | [diff] [blame] | 1767 | unsigned int ioaddr = dev->base_addr; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1768 |  | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1769 | pr_debug("do_powerdown(%p)\n", dev); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1770 |  | 
|  | 1771 | SelectPage(4); | 
|  | 1772 | PutByte(XIRCREG4_GPR1, 0);	     /* clear bit 0: power down */ | 
|  | 1773 | SelectPage(0); | 
|  | 1774 | } | 
|  | 1775 |  | 
|  | 1776 | static int | 
|  | 1777 | do_stop(struct net_device *dev) | 
|  | 1778 | { | 
| Olof Johansson | 906da80 | 2008-02-04 22:27:35 -0800 | [diff] [blame] | 1779 | unsigned int ioaddr = dev->base_addr; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1780 | local_info_t *lp = netdev_priv(dev); | 
| Dominik Brodowski | fba395e | 2006-03-31 17:21:06 +0200 | [diff] [blame] | 1781 | struct pcmcia_device *link = lp->p_dev; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1782 |  | 
| Dominik Brodowski | dd0fab5 | 2009-10-24 15:51:05 +0200 | [diff] [blame] | 1783 | dev_dbg(&link->dev, "do_stop(%p)\n", dev); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1784 |  | 
|  | 1785 | if (!link) | 
|  | 1786 | return -ENODEV; | 
|  | 1787 |  | 
|  | 1788 | netif_stop_queue(dev); | 
|  | 1789 |  | 
|  | 1790 | SelectPage(0); | 
|  | 1791 | PutByte(XIRCREG_CR, 0);  /* disable interrupts */ | 
|  | 1792 | SelectPage(0x01); | 
|  | 1793 | PutByte(XIRCREG1_IMR0, 0x00); /* forbid all ints */ | 
|  | 1794 | SelectPage(4); | 
|  | 1795 | PutByte(XIRCREG4_GPR1, 0);	/* clear bit 0: power down */ | 
|  | 1796 | SelectPage(0); | 
|  | 1797 |  | 
|  | 1798 | link->open--; | 
|  | 1799 | return 0; | 
|  | 1800 | } | 
|  | 1801 |  | 
| Dominik Brodowski | ff07bb1 | 2005-06-27 16:28:28 -0700 | [diff] [blame] | 1802 | static struct pcmcia_device_id xirc2ps_ids[] = { | 
|  | 1803 | PCMCIA_PFC_DEVICE_MANF_CARD(0, 0x0089, 0x110a), | 
|  | 1804 | PCMCIA_PFC_DEVICE_MANF_CARD(0, 0x0138, 0x110a), | 
|  | 1805 | PCMCIA_PFC_DEVICE_PROD_ID13(0, "Xircom", "CEM28", 0x2e3ee845, 0x0ea978ea), | 
|  | 1806 | PCMCIA_PFC_DEVICE_PROD_ID13(0, "Xircom", "CEM33", 0x2e3ee845, 0x80609023), | 
|  | 1807 | PCMCIA_PFC_DEVICE_PROD_ID13(0, "Xircom", "CEM56", 0x2e3ee845, 0xa650c32a), | 
|  | 1808 | PCMCIA_PFC_DEVICE_PROD_ID13(0, "Xircom", "REM10", 0x2e3ee845, 0x76df1d29), | 
|  | 1809 | PCMCIA_PFC_DEVICE_PROD_ID13(0, "Xircom", "XEM5600", 0x2e3ee845, 0xf1403719), | 
| Komuro | d277ad0 | 2005-07-28 01:07:24 -0700 | [diff] [blame] | 1810 | PCMCIA_PFC_DEVICE_PROD_ID12(0, "Xircom", "CreditCard Ethernet+Modem II", 0x2e3ee845, 0xeca401bf), | 
| Dominik Brodowski | ff07bb1 | 2005-06-27 16:28:28 -0700 | [diff] [blame] | 1811 | PCMCIA_DEVICE_MANF_CARD(0x01bf, 0x010a), | 
|  | 1812 | PCMCIA_DEVICE_PROD_ID13("Toshiba Information Systems", "TPCENET", 0x1b3b94fe, 0xf381c1a2), | 
|  | 1813 | PCMCIA_DEVICE_PROD_ID13("Xircom", "CE3-10/100", 0x2e3ee845, 0x0ec0ac37), | 
|  | 1814 | PCMCIA_DEVICE_PROD_ID13("Xircom", "PS-CE2-10", 0x2e3ee845, 0x947d9073), | 
|  | 1815 | PCMCIA_DEVICE_PROD_ID13("Xircom", "R2E-100BTX", 0x2e3ee845, 0x2464a6e3), | 
|  | 1816 | PCMCIA_DEVICE_PROD_ID13("Xircom", "RE-10", 0x2e3ee845, 0x3e08d609), | 
|  | 1817 | PCMCIA_DEVICE_PROD_ID13("Xircom", "XE2000", 0x2e3ee845, 0xf7188e46), | 
|  | 1818 | PCMCIA_DEVICE_PROD_ID12("Compaq", "Ethernet LAN Card", 0x54f7c49c, 0x9fd2f0a2), | 
|  | 1819 | PCMCIA_DEVICE_PROD_ID12("Compaq", "Netelligent 10/100 PC Card", 0x54f7c49c, 0xefe96769), | 
|  | 1820 | PCMCIA_DEVICE_PROD_ID12("Intel", "EtherExpress(TM) PRO/100 PC Card Mobile Adapter16", 0x816cc815, 0x174397db), | 
|  | 1821 | PCMCIA_DEVICE_PROD_ID12("Toshiba", "10/100 Ethernet PC Card", 0x44a09d9c, 0xb44deecf), | 
|  | 1822 | /* also matches CFE-10 cards! */ | 
|  | 1823 | /* PCMCIA_DEVICE_MANF_CARD(0x0105, 0x010a), */ | 
|  | 1824 | PCMCIA_DEVICE_NULL, | 
|  | 1825 | }; | 
|  | 1826 | MODULE_DEVICE_TABLE(pcmcia, xirc2ps_ids); | 
|  | 1827 |  | 
|  | 1828 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1829 | static struct pcmcia_driver xirc2ps_cs_driver = { | 
|  | 1830 | .owner		= THIS_MODULE, | 
|  | 1831 | .drv		= { | 
|  | 1832 | .name	= "xirc2ps_cs", | 
|  | 1833 | }, | 
| Dominik Brodowski | 15b99ac | 2006-03-31 17:26:06 +0200 | [diff] [blame] | 1834 | .probe		= xirc2ps_probe, | 
| Dominik Brodowski | cc3b486 | 2005-11-14 21:23:14 +0100 | [diff] [blame] | 1835 | .remove		= xirc2ps_detach, | 
| Dominik Brodowski | ff07bb1 | 2005-06-27 16:28:28 -0700 | [diff] [blame] | 1836 | .id_table       = xirc2ps_ids, | 
| Dominik Brodowski | 98e4c28 | 2005-11-14 21:21:18 +0100 | [diff] [blame] | 1837 | .suspend	= xirc2ps_suspend, | 
|  | 1838 | .resume		= xirc2ps_resume, | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1839 | }; | 
|  | 1840 |  | 
|  | 1841 | static int __init | 
|  | 1842 | init_xirc2ps_cs(void) | 
|  | 1843 | { | 
|  | 1844 | return pcmcia_register_driver(&xirc2ps_cs_driver); | 
|  | 1845 | } | 
|  | 1846 |  | 
|  | 1847 | static void __exit | 
|  | 1848 | exit_xirc2ps_cs(void) | 
|  | 1849 | { | 
|  | 1850 | pcmcia_unregister_driver(&xirc2ps_cs_driver); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1851 | } | 
|  | 1852 |  | 
|  | 1853 | module_init(init_xirc2ps_cs); | 
|  | 1854 | module_exit(exit_xirc2ps_cs); | 
|  | 1855 |  | 
|  | 1856 | #ifndef MODULE | 
|  | 1857 | static int __init setup_xirc2ps_cs(char *str) | 
|  | 1858 | { | 
|  | 1859 | /* if_port, full_duplex, do_sound, lockup_hack | 
|  | 1860 | */ | 
|  | 1861 | int ints[10] = { -1 }; | 
|  | 1862 |  | 
|  | 1863 | str = get_options(str, 9, ints); | 
|  | 1864 |  | 
|  | 1865 | #define MAYBE_SET(X,Y) if (ints[0] >= Y && ints[Y] != -1) { X = ints[Y]; } | 
|  | 1866 | MAYBE_SET(if_port, 3); | 
|  | 1867 | MAYBE_SET(full_duplex, 4); | 
|  | 1868 | MAYBE_SET(do_sound, 5); | 
|  | 1869 | MAYBE_SET(lockup_hack, 6); | 
|  | 1870 | #undef  MAYBE_SET | 
|  | 1871 |  | 
| OGAWA Hirofumi | 9b41046 | 2006-03-31 02:30:33 -0800 | [diff] [blame] | 1872 | return 1; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1873 | } | 
|  | 1874 |  | 
|  | 1875 | __setup("xirc2ps_cs=", setup_xirc2ps_cs); | 
|  | 1876 | #endif |