1 | /* |
---|
2 | * 3c515.c -- 3COM 3C515 Fast Etherlink ISA 10/100BASE-TX driver for etherboot |
---|
3 | * Copyright (C) 2002 Timothy Legge <tlegge@rogers.com> |
---|
4 | * |
---|
5 | * This program is free software; you can redistribute it and/or modify |
---|
6 | * it under the terms of the GNU General Public License as published by |
---|
7 | * the Free Software Foundation; either version 2 of the License, or |
---|
8 | * (at your option) any later version. |
---|
9 | * |
---|
10 | * This program is distributed in the hope that it will be useful, |
---|
11 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
---|
12 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
---|
13 | * GNU General Public License for more details. |
---|
14 | * |
---|
15 | * You should have received a copy of the GNU General Public License |
---|
16 | * along with this program; if not, write to the Free Software |
---|
17 | * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. |
---|
18 | * |
---|
19 | * Portions of this code: |
---|
20 | * Copyright (C) 1997-2002 Donald Becker 3c515.c: A 3Com ISA EtherLink XL "Corkscrew" ethernet driver for linux. |
---|
21 | * Copyright (C) 2001 P.J.H.Fox (fox@roestock.demon.co.uk) ISAPNP Tools |
---|
22 | * Copyright (c) 2002 Jaroslav Kysela <perex@suse.cz> ISA Plug & Play support Linux Kernel |
---|
23 | * Copyright (C) 2000 Shusuke Nisiyama <shu@athena.qe.eng.hokudai.ac.jp> etherboot-5.0.5 3c595.c |
---|
24 | * Coptright (C) 1995 Martin Renters etherboot-5.0.5 3c509.c |
---|
25 | * Copyright (C) 1999 LightSys Technology Services, Inc. etherboot-5.0.5 3c90x.c |
---|
26 | * Portions Copyright (C) 1999 Steve Smith etherboot-5.0.5 3c90x.c |
---|
27 | * |
---|
28 | * The probe and reset functions and defines are direct copies from the |
---|
29 | * Becker code modified where necessary to make it work for etherboot |
---|
30 | * |
---|
31 | * The poll and transmit functions either contain code from or were written by referencing |
---|
32 | * the above referenced etherboot drivers. This driver would not have been |
---|
33 | * possible without this prior work |
---|
34 | * |
---|
35 | * REVISION HISTORY: |
---|
36 | * ================ |
---|
37 | * v0.10 4-17-2002 TJL Initial implementation. |
---|
38 | * v0.11 4-17-2002 TJL Cleanup of the code |
---|
39 | * v0.12 4-26-2002 TJL Added ISA Plug and Play for Non-PNP Bioses |
---|
40 | * v0.13 6-10-2002 TJL Fixed ISA_PNP MAC Address problem |
---|
41 | * v0.14 9-23-2003 TJL Replaced delay with currticks |
---|
42 | * |
---|
43 | * Indent Options: indent -kr -i8 |
---|
44 | * *********************************************************/ |
---|
45 | |
---|
46 | FILE_LICENCE ( GPL2_OR_LATER ); |
---|
47 | |
---|
48 | /* to get some global routines like printf */ |
---|
49 | #include "etherboot.h" |
---|
50 | /* to get the interface to the body of the program */ |
---|
51 | #include "nic.h" |
---|
52 | #include <gpxe/isapnp.h> |
---|
53 | #include <gpxe/isa.h> /* for ISA_ROM */ |
---|
54 | #include <gpxe/ethernet.h> |
---|
55 | |
---|
56 | static void t3c515_wait(unsigned int nticks) |
---|
57 | { |
---|
58 | unsigned int to = currticks() + nticks; |
---|
59 | while (currticks() < to) |
---|
60 | /* wait */ ; |
---|
61 | } |
---|
62 | |
---|
63 | /* TJL definations */ |
---|
64 | #define HZ 100 |
---|
65 | static int if_port; |
---|
66 | static struct corkscrew_private *vp; |
---|
67 | /* Brought directly from 3c515.c by Becker */ |
---|
68 | #define CORKSCREW 1 |
---|
69 | |
---|
70 | /* Maximum events (Rx packets, etc.) to handle at each interrupt. |
---|
71 | static int max_interrupt_work = 20; |
---|
72 | */ |
---|
73 | |
---|
74 | /* Enable the automatic media selection code -- usually set. */ |
---|
75 | #define AUTOMEDIA 1 |
---|
76 | |
---|
77 | /* Allow the use of fragment bus master transfers instead of only |
---|
78 | programmed-I/O for Vortex cards. Full-bus-master transfers are always |
---|
79 | enabled by default on Boomerang cards. If VORTEX_BUS_MASTER is defined, |
---|
80 | the feature may be turned on using 'options'. */ |
---|
81 | #define VORTEX_BUS_MASTER |
---|
82 | |
---|
83 | /* A few values that may be tweaked. */ |
---|
84 | /* Keep the ring sizes a power of two for efficiency. */ |
---|
85 | #define TX_RING_SIZE 16 |
---|
86 | #define RX_RING_SIZE 16 |
---|
87 | #define PKT_BUF_SZ 1536 /* Size of each temporary Rx buffer. */ |
---|
88 | |
---|
89 | /* "Knobs" for adjusting internal parameters. */ |
---|
90 | /* Put out somewhat more debugging messages. (0 - no msg, 1 minimal msgs). */ |
---|
91 | #define DRIVER_DEBUG 1 |
---|
92 | /* Some values here only for performance evaluation and path-coverage |
---|
93 | debugging. |
---|
94 | static int rx_nocopy, rx_copy, queued_packet; |
---|
95 | */ |
---|
96 | |
---|
97 | #define CORKSCREW_ID 10 |
---|
98 | |
---|
99 | #define EL3WINDOW(win_num) \ |
---|
100 | outw(SelectWindow + (win_num), nic->ioaddr + EL3_CMD) |
---|
101 | #define EL3_CMD 0x0e |
---|
102 | #define EL3_STATUS 0x0e |
---|
103 | #define RX_BYTES_MASK (unsigned short) (0x07ff) |
---|
104 | |
---|
105 | enum corkscrew_cmd { |
---|
106 | TotalReset = 0 << 11, SelectWindow = 1 << 11, StartCoax = 2 << 11, |
---|
107 | RxDisable = 3 << 11, RxEnable = 4 << 11, RxReset = 5 << 11, |
---|
108 | UpStall = 6 << 11, UpUnstall = (6 << 11) + 1, |
---|
109 | DownStall = (6 << 11) + 2, DownUnstall = (6 << 11) + 3, |
---|
110 | RxDiscard = 8 << 11, TxEnable = 9 << 11, TxDisable = |
---|
111 | 10 << 11, TxReset = 11 << 11, |
---|
112 | FakeIntr = 12 << 11, AckIntr = 13 << 11, SetIntrEnb = 14 << 11, |
---|
113 | SetStatusEnb = 15 << 11, SetRxFilter = 16 << 11, SetRxThreshold = |
---|
114 | 17 << 11, |
---|
115 | SetTxThreshold = 18 << 11, SetTxStart = 19 << 11, |
---|
116 | StartDMAUp = 20 << 11, StartDMADown = (20 << 11) + 1, StatsEnable = |
---|
117 | 21 << 11, |
---|
118 | StatsDisable = 22 << 11, StopCoax = 23 << 11, |
---|
119 | }; |
---|
120 | |
---|
121 | /* The SetRxFilter command accepts the following classes: */ |
---|
122 | enum RxFilter { |
---|
123 | RxStation = 1, RxMulticast = 2, RxBroadcast = 4, RxProm = 8 |
---|
124 | }; |
---|
125 | |
---|
126 | /* Bits in the general status register. */ |
---|
127 | enum corkscrew_status { |
---|
128 | IntLatch = 0x0001, AdapterFailure = 0x0002, TxComplete = 0x0004, |
---|
129 | TxAvailable = 0x0008, RxComplete = 0x0010, RxEarly = 0x0020, |
---|
130 | IntReq = 0x0040, StatsFull = 0x0080, |
---|
131 | DMADone = 1 << 8, DownComplete = 1 << 9, UpComplete = 1 << 10, |
---|
132 | DMAInProgress = 1 << 11, /* DMA controller is still busy. */ |
---|
133 | CmdInProgress = 1 << 12, /* EL3_CMD is still busy. */ |
---|
134 | }; |
---|
135 | |
---|
136 | /* Register window 1 offsets, the window used in normal operation. |
---|
137 | On the Corkscrew this window is always mapped at offsets 0x10-0x1f. */ |
---|
138 | enum Window1 { |
---|
139 | TX_FIFO = 0x10, RX_FIFO = 0x10, RxErrors = 0x14, |
---|
140 | RxStatus = 0x18, Timer = 0x1A, TxStatus = 0x1B, |
---|
141 | TxFree = 0x1C, /* Remaining free bytes in Tx buffer. */ |
---|
142 | }; |
---|
143 | enum Window0 { |
---|
144 | Wn0IRQ = 0x08, |
---|
145 | #if defined(CORKSCREW) |
---|
146 | Wn0EepromCmd = 0x200A, /* Corkscrew EEPROM command register. */ |
---|
147 | Wn0EepromData = 0x200C, /* Corkscrew EEPROM results register. */ |
---|
148 | #else |
---|
149 | Wn0EepromCmd = 10, /* Window 0: EEPROM command register. */ |
---|
150 | Wn0EepromData = 12, /* Window 0: EEPROM results register. */ |
---|
151 | #endif |
---|
152 | }; |
---|
153 | enum Win0_EEPROM_bits { |
---|
154 | EEPROM_Read = 0x80, EEPROM_WRITE = 0x40, EEPROM_ERASE = 0xC0, |
---|
155 | EEPROM_EWENB = 0x30, /* Enable erasing/writing for 10 msec. */ |
---|
156 | EEPROM_EWDIS = 0x00, /* Disable EWENB before 10 msec timeout. */ |
---|
157 | }; |
---|
158 | |
---|
159 | enum Window3 { /* Window 3: MAC/config bits. */ |
---|
160 | Wn3_Config = 0, Wn3_MAC_Ctrl = 6, Wn3_Options = 8, |
---|
161 | }; |
---|
162 | union wn3_config { |
---|
163 | int i; |
---|
164 | struct w3_config_fields { |
---|
165 | unsigned int ram_size:3, ram_width:1, ram_speed:2, |
---|
166 | rom_size:2; |
---|
167 | int pad8:8; |
---|
168 | unsigned int ram_split:2, pad18:2, xcvr:3, pad21:1, |
---|
169 | autoselect:1; |
---|
170 | int pad24:7; |
---|
171 | } u; |
---|
172 | }; |
---|
173 | |
---|
174 | enum Window4 { |
---|
175 | Wn4_NetDiag = 6, Wn4_Media = 10, /* Window 4: Xcvr/media bits. */ |
---|
176 | }; |
---|
177 | enum Win4_Media_bits { |
---|
178 | Media_SQE = 0x0008, /* Enable SQE error counting for AUI. */ |
---|
179 | Media_10TP = 0x00C0, /* Enable link beat and jabber for 10baseT. */ |
---|
180 | Media_Lnk = 0x0080, /* Enable just link beat for 100TX/100FX. */ |
---|
181 | Media_LnkBeat = 0x0800, |
---|
182 | }; |
---|
183 | enum Window7 { /* Window 7: Bus Master control. */ |
---|
184 | Wn7_MasterAddr = 0, Wn7_MasterLen = 6, Wn7_MasterStatus = 12, |
---|
185 | }; |
---|
186 | |
---|
187 | /* Boomerang-style bus master control registers. Note ISA aliases! */ |
---|
188 | enum MasterCtrl { |
---|
189 | PktStatus = 0x400, DownListPtr = 0x404, FragAddr = 0x408, FragLen = |
---|
190 | 0x40c, |
---|
191 | TxFreeThreshold = 0x40f, UpPktStatus = 0x410, UpListPtr = 0x418, |
---|
192 | }; |
---|
193 | |
---|
194 | /* The Rx and Tx descriptor lists. |
---|
195 | Caution Alpha hackers: these types are 32 bits! Note also the 8 byte |
---|
196 | alignment contraint on tx_ring[] and rx_ring[]. */ |
---|
197 | struct boom_rx_desc { |
---|
198 | u32 next; |
---|
199 | s32 status; |
---|
200 | u32 addr; |
---|
201 | s32 length; |
---|
202 | }; |
---|
203 | |
---|
204 | /* Values for the Rx status entry. */ |
---|
205 | enum rx_desc_status { |
---|
206 | RxDComplete = 0x00008000, RxDError = 0x4000, |
---|
207 | /* See boomerang_rx() for actual error bits */ |
---|
208 | }; |
---|
209 | |
---|
210 | struct boom_tx_desc { |
---|
211 | u32 next; |
---|
212 | s32 status; |
---|
213 | u32 addr; |
---|
214 | s32 length; |
---|
215 | }; |
---|
216 | |
---|
217 | struct corkscrew_private { |
---|
218 | const char *product_name; |
---|
219 | struct net_device *next_module; |
---|
220 | /* The Rx and Tx rings are here to keep them quad-word-aligned. */ |
---|
221 | struct boom_rx_desc rx_ring[RX_RING_SIZE]; |
---|
222 | struct boom_tx_desc tx_ring[TX_RING_SIZE]; |
---|
223 | /* The addresses of transmit- and receive-in-place skbuffs. */ |
---|
224 | struct sk_buff *rx_skbuff[RX_RING_SIZE]; |
---|
225 | struct sk_buff *tx_skbuff[TX_RING_SIZE]; |
---|
226 | unsigned int cur_rx, cur_tx; /* The next free ring entry */ |
---|
227 | unsigned int dirty_rx, dirty_tx; /* The ring entries to be free()ed. */ |
---|
228 | struct sk_buff *tx_skb; /* Packet being eaten by bus master ctrl. */ |
---|
229 | int capabilities; /* Adapter capabilities word. */ |
---|
230 | int options; /* User-settable misc. driver options. */ |
---|
231 | int last_rx_packets; /* For media autoselection. */ |
---|
232 | unsigned int available_media:8, /* From Wn3_Options */ |
---|
233 | media_override:3, /* Passed-in media type. */ |
---|
234 | default_media:3, /* Read from the EEPROM. */ |
---|
235 | full_duplex:1, autoselect:1, bus_master:1, /* Vortex can only do a fragment bus-m. */ |
---|
236 | full_bus_master_tx:1, full_bus_master_rx:1, /* Boomerang */ |
---|
237 | tx_full:1; |
---|
238 | }; |
---|
239 | |
---|
240 | /* The action to take with a media selection timer tick. |
---|
241 | Note that we deviate from the 3Com order by checking 10base2 before AUI. |
---|
242 | */ |
---|
243 | enum xcvr_types { |
---|
244 | XCVR_10baseT = |
---|
245 | 0, XCVR_AUI, XCVR_10baseTOnly, XCVR_10base2, XCVR_100baseTx, |
---|
246 | XCVR_100baseFx, XCVR_MII = 6, XCVR_Default = 8, |
---|
247 | }; |
---|
248 | |
---|
249 | static struct media_table { |
---|
250 | char *name; |
---|
251 | unsigned int media_bits:16, /* Bits to set in Wn4_Media register. */ |
---|
252 | mask:8, /* The transceiver-present bit in Wn3_Config. */ |
---|
253 | next:8; /* The media type to try next. */ |
---|
254 | short wait; /* Time before we check media status. */ |
---|
255 | } media_tbl[] = { |
---|
256 | { |
---|
257 | "10baseT", Media_10TP, 0x08, XCVR_10base2, (14 * HZ) / 10} |
---|
258 | , { |
---|
259 | "10Mbs AUI", Media_SQE, 0x20, XCVR_Default, (1 * HZ) / 10} |
---|
260 | , { |
---|
261 | "undefined", 0, 0x80, XCVR_10baseT, 10000} |
---|
262 | , { |
---|
263 | "10base2", 0, 0x10, XCVR_AUI, (1 * HZ) / 10} |
---|
264 | , { |
---|
265 | "100baseTX", Media_Lnk, 0x02, XCVR_100baseFx, |
---|
266 | (14 * HZ) / 10} |
---|
267 | , { |
---|
268 | "100baseFX", Media_Lnk, 0x04, XCVR_MII, (14 * HZ) / 10} |
---|
269 | , { |
---|
270 | "MII", 0, 0x40, XCVR_10baseT, 3 * HZ} |
---|
271 | , { |
---|
272 | "undefined", 0, 0x01, XCVR_10baseT, 10000} |
---|
273 | , { |
---|
274 | "Default", 0, 0xFF, XCVR_10baseT, 10000} |
---|
275 | ,}; |
---|
276 | |
---|
277 | /* TILEG Modified to remove reference to dev */ |
---|
278 | static int corkscrew_found_device(int ioaddr, int irq, int product_index, |
---|
279 | int options, struct nic *nic); |
---|
280 | static int corkscrew_probe1(int ioaddr, int irq, int product_index, |
---|
281 | struct nic *nic); |
---|
282 | |
---|
283 | /* This driver uses 'options' to pass the media type, full-duplex flag, etc. */ |
---|
284 | /* Note: this is the only limit on the number of cards supported!! */ |
---|
285 | static int options = -1; |
---|
286 | |
---|
287 | /* End Brought directly from 3c515.c by Becker */ |
---|
288 | |
---|
289 | /************************************************************************** |
---|
290 | RESET - Reset adapter |
---|
291 | ***************************************************************************/ |
---|
292 | static void t515_reset(struct nic *nic) |
---|
293 | { |
---|
294 | union wn3_config config; |
---|
295 | int i; |
---|
296 | |
---|
297 | /* Before initializing select the active media port. */ |
---|
298 | EL3WINDOW(3); |
---|
299 | if (vp->full_duplex) |
---|
300 | outb(0x20, nic->ioaddr + Wn3_MAC_Ctrl); /* Set the full-duplex bit. */ |
---|
301 | config.i = inl(nic->ioaddr + Wn3_Config); |
---|
302 | |
---|
303 | if (vp->media_override != 7) { |
---|
304 | DBG ( "Media override to transceiver %d (%s).\n", |
---|
305 | vp->media_override, |
---|
306 | media_tbl[vp->media_override].name); |
---|
307 | if_port = vp->media_override; |
---|
308 | } else if (vp->autoselect) { |
---|
309 | /* Find first available media type, starting with 100baseTx. */ |
---|
310 | if_port = 4; |
---|
311 | while (!(vp->available_media & media_tbl[if_port].mask)) |
---|
312 | if_port = media_tbl[if_port].next; |
---|
313 | |
---|
314 | DBG ( "Initial media type %s.\n", |
---|
315 | media_tbl[if_port].name); |
---|
316 | } else |
---|
317 | if_port = vp->default_media; |
---|
318 | |
---|
319 | config.u.xcvr = if_port; |
---|
320 | outl(config.i, nic->ioaddr + Wn3_Config); |
---|
321 | |
---|
322 | DBG ( "corkscrew_open() InternalConfig 0x%hX.\n", |
---|
323 | config.i); |
---|
324 | |
---|
325 | outw(TxReset, nic->ioaddr + EL3_CMD); |
---|
326 | for (i = 20; i >= 0; i--) |
---|
327 | if (!(inw(nic->ioaddr + EL3_STATUS) & CmdInProgress)) |
---|
328 | break; |
---|
329 | |
---|
330 | outw(RxReset, nic->ioaddr + EL3_CMD); |
---|
331 | /* Wait a few ticks for the RxReset command to complete. */ |
---|
332 | for (i = 20; i >= 0; i--) |
---|
333 | if (!(inw(nic->ioaddr + EL3_STATUS) & CmdInProgress)) |
---|
334 | break; |
---|
335 | |
---|
336 | outw(SetStatusEnb | 0x00, nic->ioaddr + EL3_CMD); |
---|
337 | |
---|
338 | #ifdef debug_3c515 |
---|
339 | EL3WINDOW(4); |
---|
340 | DBG ( "FIXME: fix print for irq, not 9" ); |
---|
341 | DBG ( "corkscrew_open() irq %d media status 0x%hX.\n", |
---|
342 | 9, inw(nic->ioaddr + Wn4_Media) ); |
---|
343 | #endif |
---|
344 | |
---|
345 | /* Set the station address and mask in window 2 each time opened. */ |
---|
346 | EL3WINDOW(2); |
---|
347 | for (i = 0; i < 6; i++) |
---|
348 | outb(nic->node_addr[i], nic->ioaddr + i); |
---|
349 | for (; i < 12; i += 2) |
---|
350 | outw(0, nic->ioaddr + i); |
---|
351 | |
---|
352 | if (if_port == 3) |
---|
353 | /* Start the thinnet transceiver. We should really wait 50ms... */ |
---|
354 | outw(StartCoax, nic->ioaddr + EL3_CMD); |
---|
355 | EL3WINDOW(4); |
---|
356 | outw((inw(nic->ioaddr + Wn4_Media) & ~(Media_10TP | Media_SQE)) | |
---|
357 | media_tbl[if_port].media_bits, nic->ioaddr + Wn4_Media); |
---|
358 | |
---|
359 | /* Switch to the stats window, and clear all stats by reading. */ |
---|
360 | /* outw(StatsDisable, nic->ioaddr + EL3_CMD);*/ |
---|
361 | EL3WINDOW(6); |
---|
362 | for (i = 0; i < 10; i++) |
---|
363 | inb(nic->ioaddr + i); |
---|
364 | inw(nic->ioaddr + 10); |
---|
365 | inw(nic->ioaddr + 12); |
---|
366 | /* New: On the Vortex we must also clear the BadSSD counter. */ |
---|
367 | EL3WINDOW(4); |
---|
368 | inb(nic->ioaddr + 12); |
---|
369 | /* ..and on the Boomerang we enable the extra statistics bits. */ |
---|
370 | outw(0x0040, nic->ioaddr + Wn4_NetDiag); |
---|
371 | |
---|
372 | /* Switch to register set 7 for normal use. */ |
---|
373 | EL3WINDOW(7); |
---|
374 | |
---|
375 | /* Temporarily left in place. If these FIXMEs are printed |
---|
376 | it meand that special logic for that card may need to be added |
---|
377 | see Becker's 3c515.c driver */ |
---|
378 | if (vp->full_bus_master_rx) { /* Boomerang bus master. */ |
---|
379 | printf("FIXME: Is this if necessary"); |
---|
380 | vp->cur_rx = vp->dirty_rx = 0; |
---|
381 | DBG ( " Filling in the Rx ring.\n" ); |
---|
382 | for (i = 0; i < RX_RING_SIZE; i++) { |
---|
383 | printf("FIXME: Is this if necessary"); |
---|
384 | } |
---|
385 | } |
---|
386 | if (vp->full_bus_master_tx) { /* Boomerang bus master Tx. */ |
---|
387 | vp->cur_tx = vp->dirty_tx = 0; |
---|
388 | outb(PKT_BUF_SZ >> 8, nic->ioaddr + TxFreeThreshold); /* Room for a packet. */ |
---|
389 | /* Clear the Tx ring. */ |
---|
390 | for (i = 0; i < TX_RING_SIZE; i++) |
---|
391 | vp->tx_skbuff[i] = 0; |
---|
392 | outl(0, nic->ioaddr + DownListPtr); |
---|
393 | } |
---|
394 | /* Set receiver mode: presumably accept b-case and phys addr only. */ |
---|
395 | outw(SetRxFilter | RxStation | RxMulticast | RxBroadcast | RxProm, |
---|
396 | nic->ioaddr + EL3_CMD); |
---|
397 | |
---|
398 | outw(RxEnable, nic->ioaddr + EL3_CMD); /* Enable the receiver. */ |
---|
399 | outw(TxEnable, nic->ioaddr + EL3_CMD); /* Enable transmitter. */ |
---|
400 | /* Allow status bits to be seen. */ |
---|
401 | outw(SetStatusEnb | AdapterFailure | IntReq | StatsFull | |
---|
402 | (vp->full_bus_master_tx ? DownComplete : TxAvailable) | |
---|
403 | (vp->full_bus_master_rx ? UpComplete : RxComplete) | |
---|
404 | (vp->bus_master ? DMADone : 0), nic->ioaddr + EL3_CMD); |
---|
405 | /* Ack all pending events, and set active indicator mask. */ |
---|
406 | outw(AckIntr | IntLatch | TxAvailable | RxEarly | IntReq, |
---|
407 | nic->ioaddr + EL3_CMD); |
---|
408 | outw(SetIntrEnb | IntLatch | TxAvailable | RxComplete | StatsFull |
---|
409 | | (vp->bus_master ? DMADone : 0) | UpComplete | DownComplete, |
---|
410 | nic->ioaddr + EL3_CMD); |
---|
411 | |
---|
412 | } |
---|
413 | |
---|
414 | /************************************************************************** |
---|
415 | POLL - Wait for a frame |
---|
416 | ***************************************************************************/ |
---|
417 | static int t515_poll(struct nic *nic, int retrieve) |
---|
418 | { |
---|
419 | short status, cst; |
---|
420 | register short rx_fifo; |
---|
421 | |
---|
422 | cst = inw(nic->ioaddr + EL3_STATUS); |
---|
423 | |
---|
424 | if ((cst & RxComplete) == 0) { |
---|
425 | /* Ack all pending events, and set active indicator mask. */ |
---|
426 | outw(AckIntr | IntLatch | TxAvailable | RxEarly | IntReq, |
---|
427 | nic->ioaddr + EL3_CMD); |
---|
428 | outw(SetIntrEnb | IntLatch | TxAvailable | RxComplete | |
---|
429 | StatsFull | (vp-> |
---|
430 | bus_master ? DMADone : 0) | UpComplete | |
---|
431 | DownComplete, nic->ioaddr + EL3_CMD); |
---|
432 | return 0; |
---|
433 | } |
---|
434 | status = inw(nic->ioaddr + RxStatus); |
---|
435 | |
---|
436 | if (status & RxDError) { |
---|
437 | printf("RxDError\n"); |
---|
438 | outw(RxDiscard, nic->ioaddr + EL3_CMD); |
---|
439 | return 0; |
---|
440 | } |
---|
441 | |
---|
442 | rx_fifo = status & RX_BYTES_MASK; |
---|
443 | if (rx_fifo == 0) |
---|
444 | return 0; |
---|
445 | |
---|
446 | if ( ! retrieve ) return 1; |
---|
447 | |
---|
448 | DBG ( "[l=%d", rx_fifo ); |
---|
449 | insw(nic->ioaddr + RX_FIFO, nic->packet, rx_fifo / 2); |
---|
450 | if (rx_fifo & 1) |
---|
451 | nic->packet[rx_fifo - 1] = inb(nic->ioaddr + RX_FIFO); |
---|
452 | nic->packetlen = rx_fifo; |
---|
453 | |
---|
454 | while (1) { |
---|
455 | status = inw(nic->ioaddr + RxStatus); |
---|
456 | DBG ( "0x%hX*", status ); |
---|
457 | rx_fifo = status & RX_BYTES_MASK; |
---|
458 | |
---|
459 | if (rx_fifo > 0) { |
---|
460 | insw(nic->ioaddr + RX_FIFO, nic->packet + nic->packetlen, |
---|
461 | rx_fifo / 2); |
---|
462 | if (rx_fifo & 1) |
---|
463 | nic->packet[nic->packetlen + rx_fifo - 1] = |
---|
464 | inb(nic->ioaddr + RX_FIFO); |
---|
465 | nic->packetlen += rx_fifo; |
---|
466 | DBG ( "+%d", rx_fifo ); |
---|
467 | } |
---|
468 | if ((status & RxComplete) == 0) { |
---|
469 | DBG ( "=%d", nic->packetlen ); |
---|
470 | break; |
---|
471 | } |
---|
472 | udelay(1000); |
---|
473 | } |
---|
474 | |
---|
475 | /* acknowledge reception of packet */ |
---|
476 | outw(RxDiscard, nic->ioaddr + EL3_CMD); |
---|
477 | while (inw(nic->ioaddr + EL3_STATUS) & CmdInProgress); |
---|
478 | #ifdef debug_3c515 |
---|
479 | { |
---|
480 | unsigned short type = 0; |
---|
481 | type = (nic->packet[12] << 8) | nic->packet[13]; |
---|
482 | if (nic->packet[0] + nic->packet[1] + nic->packet[2] + |
---|
483 | nic->packet[3] + nic->packet[4] + nic->packet[5] == |
---|
484 | 0xFF * ETH_ALEN) |
---|
485 | DBG ( ",t=0x%hX,b]", type ); |
---|
486 | else |
---|
487 | DBG ( ",t=0x%hX]", type ); |
---|
488 | } |
---|
489 | #endif |
---|
490 | |
---|
491 | return 1; |
---|
492 | } |
---|
493 | |
---|
494 | /************************************************************************* |
---|
495 | 3Com 515 - specific routines |
---|
496 | **************************************************************************/ |
---|
497 | static char padmap[] = { |
---|
498 | 0, 3, 2, 1 |
---|
499 | }; |
---|
500 | /************************************************************************** |
---|
501 | TRANSMIT - Transmit a frame |
---|
502 | ***************************************************************************/ |
---|
503 | static void t515_transmit(struct nic *nic, const char *d, /* Destination */ |
---|
504 | unsigned int t, /* Type */ |
---|
505 | unsigned int s, /* size */ |
---|
506 | const char *p) |
---|
507 | { /* Packet */ |
---|
508 | register int len; |
---|
509 | int pad; |
---|
510 | int status; |
---|
511 | |
---|
512 | DBG ( "{l=%d,t=0x%hX}", s + ETH_HLEN, t ); |
---|
513 | |
---|
514 | /* swap bytes of type */ |
---|
515 | t = htons(t); |
---|
516 | |
---|
517 | len = s + ETH_HLEN; /* actual length of packet */ |
---|
518 | pad = padmap[len & 3]; |
---|
519 | |
---|
520 | /* |
---|
521 | * The 3c515 automatically pads short packets to minimum ethernet length, |
---|
522 | * but we drop packets that are too large. Perhaps we should truncate |
---|
523 | * them instead? |
---|
524 | Copied from 3c595. Is this true for the 3c515? |
---|
525 | */ |
---|
526 | if (len + pad > ETH_FRAME_LEN) { |
---|
527 | return; |
---|
528 | } |
---|
529 | /* drop acknowledgements */ |
---|
530 | while ((status = inb(nic->ioaddr + TxStatus)) & TxComplete) { |
---|
531 | /*if(status & (TXS_UNDERRUN|0x88|TXS_STATUS_OVERFLOW)) { */ |
---|
532 | outw(TxReset, nic->ioaddr + EL3_CMD); |
---|
533 | outw(TxEnable, nic->ioaddr + EL3_CMD); |
---|
534 | /* } */ |
---|
535 | |
---|
536 | outb(0x0, nic->ioaddr + TxStatus); |
---|
537 | } |
---|
538 | |
---|
539 | while (inw(nic->ioaddr + TxFree) < len + pad + 4) { |
---|
540 | /* no room in FIFO */ |
---|
541 | } |
---|
542 | |
---|
543 | outw(len, nic->ioaddr + TX_FIFO); |
---|
544 | outw(0x0, nic->ioaddr + TX_FIFO); /* Second dword meaningless */ |
---|
545 | |
---|
546 | /* write packet */ |
---|
547 | outsw(nic->ioaddr + TX_FIFO, d, ETH_ALEN / 2); |
---|
548 | outsw(nic->ioaddr + TX_FIFO, nic->node_addr, ETH_ALEN / 2); |
---|
549 | outw(t, nic->ioaddr + TX_FIFO); |
---|
550 | outsw(nic->ioaddr + TX_FIFO, p, s / 2); |
---|
551 | |
---|
552 | if (s & 1) |
---|
553 | outb(*(p + s - 1), nic->ioaddr + TX_FIFO); |
---|
554 | |
---|
555 | while (pad--) |
---|
556 | outb(0, nic->ioaddr + TX_FIFO); /* Padding */ |
---|
557 | |
---|
558 | /* wait for Tx complete */ |
---|
559 | while ((inw(nic->ioaddr + EL3_STATUS) & CmdInProgress) != 0); |
---|
560 | } |
---|
561 | |
---|
562 | /************************************************************************** |
---|
563 | DISABLE - Turn off ethernet interface |
---|
564 | ***************************************************************************/ |
---|
565 | static void t515_disable ( struct nic *nic, |
---|
566 | struct isapnp_device *isapnp ) { |
---|
567 | |
---|
568 | t515_reset(nic); |
---|
569 | |
---|
570 | /* This is a hack. Since ltsp worked on my |
---|
571 | system without any disable functionality I |
---|
572 | have no way to determine if this works */ |
---|
573 | |
---|
574 | /* Disable the receiver and transmitter. */ |
---|
575 | outw(RxDisable, nic->ioaddr + EL3_CMD); |
---|
576 | outw(TxDisable, nic->ioaddr + EL3_CMD); |
---|
577 | |
---|
578 | if (if_port == XCVR_10base2) |
---|
579 | /* Turn off thinnet power. Green! */ |
---|
580 | outw(StopCoax, nic->ioaddr + EL3_CMD); |
---|
581 | |
---|
582 | |
---|
583 | outw(SetIntrEnb | 0x0000, nic->ioaddr + EL3_CMD); |
---|
584 | |
---|
585 | deactivate_isapnp_device ( isapnp ); |
---|
586 | return; |
---|
587 | } |
---|
588 | |
---|
589 | static void t515_irq(struct nic *nic __unused, irq_action_t action __unused) |
---|
590 | { |
---|
591 | switch ( action ) { |
---|
592 | case DISABLE : |
---|
593 | break; |
---|
594 | case ENABLE : |
---|
595 | break; |
---|
596 | case FORCE : |
---|
597 | break; |
---|
598 | } |
---|
599 | } |
---|
600 | |
---|
601 | static struct nic_operations t515_operations = { |
---|
602 | .connect = dummy_connect, |
---|
603 | .poll = t515_poll, |
---|
604 | .transmit = t515_transmit, |
---|
605 | .irq = t515_irq, |
---|
606 | |
---|
607 | }; |
---|
608 | |
---|
609 | /************************************************************************** |
---|
610 | PROBE - Look for an adapter, this routine's visible to the outside |
---|
611 | You should omit the last argument struct pci_device * for a non-PCI NIC |
---|
612 | ***************************************************************************/ |
---|
613 | static int t515_probe ( struct nic *nic, struct isapnp_device *isapnp ) { |
---|
614 | |
---|
615 | /* Direct copy from Beckers 3c515.c removing any ISAPNP sections */ |
---|
616 | |
---|
617 | nic->ioaddr = isapnp->ioaddr; |
---|
618 | nic->irqno = isapnp->irqno; |
---|
619 | activate_isapnp_device ( isapnp ); |
---|
620 | |
---|
621 | /* Check the resource configuration for a matching ioaddr. */ |
---|
622 | if ((unsigned)(inw(nic->ioaddr + 0x2002) & 0x1f0) |
---|
623 | != (nic->ioaddr & 0x1f0)) { |
---|
624 | DBG ( "3c515 ioaddr mismatch\n" ); |
---|
625 | return 0; |
---|
626 | } |
---|
627 | |
---|
628 | /* Verify by reading the device ID from the EEPROM. */ |
---|
629 | { |
---|
630 | int timer; |
---|
631 | outw(EEPROM_Read + 7, nic->ioaddr + Wn0EepromCmd); |
---|
632 | /* Pause for at least 162 us. for the read to take place. */ |
---|
633 | for (timer = 4; timer >= 0; timer--) { |
---|
634 | t3c515_wait(1); |
---|
635 | if ((inw(nic->ioaddr + Wn0EepromCmd) & 0x0200) == 0) |
---|
636 | break; |
---|
637 | } |
---|
638 | if (inw(nic->ioaddr + Wn0EepromData) != 0x6d50) { |
---|
639 | DBG ( "3c515 read incorrect vendor ID from EEPROM" ); |
---|
640 | return 0; |
---|
641 | } |
---|
642 | |
---|
643 | } |
---|
644 | DBG ( "3c515 Resource configuration register 0x%X, DCR 0x%hX.\n", |
---|
645 | inl(nic->ioaddr + 0x2002), inw(nic->ioaddr + 0x2000) ); |
---|
646 | corkscrew_found_device(nic->ioaddr, nic->irqno, CORKSCREW_ID, |
---|
647 | options, nic); |
---|
648 | |
---|
649 | t515_reset(nic); |
---|
650 | nic->nic_op = &t515_operations; |
---|
651 | return 1; |
---|
652 | } |
---|
653 | |
---|
654 | static int |
---|
655 | corkscrew_found_device(int ioaddr, int irq, |
---|
656 | int product_index, int options, struct nic *nic) |
---|
657 | { |
---|
658 | /* Direct copy from Becker 3c515.c with unecessary parts removed */ |
---|
659 | vp->product_name = "3c515"; |
---|
660 | vp->options = options; |
---|
661 | if (options >= 0) { |
---|
662 | vp->media_override = |
---|
663 | ((options & 7) == 2) ? 0 : options & 7; |
---|
664 | vp->full_duplex = (options & 8) ? 1 : 0; |
---|
665 | vp->bus_master = (options & 16) ? 1 : 0; |
---|
666 | } else { |
---|
667 | vp->media_override = 7; |
---|
668 | vp->full_duplex = 0; |
---|
669 | vp->bus_master = 0; |
---|
670 | } |
---|
671 | |
---|
672 | corkscrew_probe1(ioaddr, irq, product_index, nic); |
---|
673 | return 0; |
---|
674 | } |
---|
675 | |
---|
676 | static int |
---|
677 | corkscrew_probe1(int ioaddr, int irq, int product_index __unused, |
---|
678 | struct nic *nic) |
---|
679 | { |
---|
680 | unsigned int eeprom[0x40], checksum = 0; /* EEPROM contents */ |
---|
681 | int i; |
---|
682 | |
---|
683 | printf("3Com %s at 0x%hX, ", vp->product_name, ioaddr); |
---|
684 | |
---|
685 | /* Read the station address from the EEPROM. */ |
---|
686 | EL3WINDOW(0); |
---|
687 | for (i = 0; i < 0x18; i++) { |
---|
688 | short *phys_addr = (short *) nic->node_addr; |
---|
689 | int timer; |
---|
690 | outw(EEPROM_Read + i, ioaddr + Wn0EepromCmd); |
---|
691 | /* Pause for at least 162 us. for the read to take place. */ |
---|
692 | for (timer = 4; timer >= 0; timer--) { |
---|
693 | t3c515_wait(1); |
---|
694 | if ((inw(ioaddr + Wn0EepromCmd) & 0x0200) == 0) |
---|
695 | break; |
---|
696 | } |
---|
697 | eeprom[i] = inw(ioaddr + Wn0EepromData); |
---|
698 | DBG ( "Value %d: %hX ", i, eeprom[i] ); |
---|
699 | checksum ^= eeprom[i]; |
---|
700 | if (i < 3) |
---|
701 | phys_addr[i] = htons(eeprom[i]); |
---|
702 | } |
---|
703 | checksum = (checksum ^ (checksum >> 8)) & 0xff; |
---|
704 | if (checksum != 0x00) |
---|
705 | printf(" ***INVALID CHECKSUM 0x%hX*** ", checksum); |
---|
706 | |
---|
707 | DBG ( "%s", eth_ntoa ( nic->node_addr ) ); |
---|
708 | |
---|
709 | if (eeprom[16] == 0x11c7) { /* Corkscrew */ |
---|
710 | |
---|
711 | } |
---|
712 | printf(", IRQ %d\n", irq); |
---|
713 | /* Tell them about an invalid IRQ. */ |
---|
714 | if ( (irq <= 0 || irq > 15) ) { |
---|
715 | DBG (" *** Warning: this IRQ is unlikely to work! ***\n" ); |
---|
716 | } |
---|
717 | |
---|
718 | { |
---|
719 | char *ram_split[] = { "5:3", "3:1", "1:1", "3:5" }; |
---|
720 | union wn3_config config; |
---|
721 | EL3WINDOW(3); |
---|
722 | vp->available_media = inw(ioaddr + Wn3_Options); |
---|
723 | config.i = inl(ioaddr + Wn3_Config); |
---|
724 | DBG ( " Internal config register is %4.4x, " |
---|
725 | "transceivers 0x%hX.\n", |
---|
726 | config.i, inw(ioaddr + Wn3_Options) ); |
---|
727 | printf |
---|
728 | (" %dK %s-wide RAM %s Rx:Tx split, %s%s interface.\n", |
---|
729 | 8 << config.u.ram_size, |
---|
730 | config.u.ram_width ? "word" : "byte", |
---|
731 | ram_split[config.u.ram_split], |
---|
732 | config.u.autoselect ? "autoselect/" : "", |
---|
733 | media_tbl[config.u.xcvr].name); |
---|
734 | if_port = config.u.xcvr; |
---|
735 | vp->default_media = config.u.xcvr; |
---|
736 | vp->autoselect = config.u.autoselect; |
---|
737 | } |
---|
738 | if (vp->media_override != 7) { |
---|
739 | printf(" Media override to transceiver type %d (%s).\n", |
---|
740 | vp->media_override, |
---|
741 | media_tbl[vp->media_override].name); |
---|
742 | if_port = vp->media_override; |
---|
743 | } |
---|
744 | |
---|
745 | vp->capabilities = eeprom[16]; |
---|
746 | vp->full_bus_master_tx = (vp->capabilities & 0x20) ? 1 : 0; |
---|
747 | /* Rx is broken at 10mbps, so we always disable it. */ |
---|
748 | /* vp->full_bus_master_rx = 0; */ |
---|
749 | vp->full_bus_master_rx = (vp->capabilities & 0x20) ? 1 : 0; |
---|
750 | |
---|
751 | return 0; |
---|
752 | } |
---|
753 | |
---|
754 | static struct isapnp_device_id t515_adapters[] = { |
---|
755 | { "3c515 (ISAPnP)", ISAPNP_VENDOR('T','C','M'), 0x5051 }, |
---|
756 | }; |
---|
757 | |
---|
758 | ISAPNP_DRIVER ( t515_driver, t515_adapters ); |
---|
759 | |
---|
760 | DRIVER ( "3c515", nic_driver, isapnp_driver, t515_driver, |
---|
761 | t515_probe, t515_disable ); |
---|
762 | |
---|
763 | ISA_ROM ( "3c515", "3c515 Fast EtherLink ISAPnP" ); |
---|