xref: /onnv-gate/usr/src/grub/grub-0.97/netboot/tulip.c (revision 8044:b3af80bbf173)
1*8044SWilliam.Kucharski@Sun.COM /* -*- Mode:C; c-basic-offset:4; -*- */
2*8044SWilliam.Kucharski@Sun.COM 
3*8044SWilliam.Kucharski@Sun.COM /*
4*8044SWilliam.Kucharski@Sun.COM   Tulip and clone Etherboot Driver
5*8044SWilliam.Kucharski@Sun.COM 
6*8044SWilliam.Kucharski@Sun.COM   By Marty Connor (mdc@thinguin.org)
7*8044SWilliam.Kucharski@Sun.COM   Copyright (C) 2001 Entity Cyber, Inc.
8*8044SWilliam.Kucharski@Sun.COM 
9*8044SWilliam.Kucharski@Sun.COM   This software may be used and distributed according to the terms
10*8044SWilliam.Kucharski@Sun.COM   of the GNU Public License, incorporated herein by reference.
11*8044SWilliam.Kucharski@Sun.COM 
12*8044SWilliam.Kucharski@Sun.COM   As of April 2001 this driver should support most tulip cards that
13*8044SWilliam.Kucharski@Sun.COM   the Linux tulip driver supports because Donald Becker's Linux media
14*8044SWilliam.Kucharski@Sun.COM   detection code is now included.
15*8044SWilliam.Kucharski@Sun.COM 
16*8044SWilliam.Kucharski@Sun.COM   Based on Ken Yap's Tulip Etherboot Driver and Donald Becker's
17*8044SWilliam.Kucharski@Sun.COM   Linux Tulip Driver. Supports N-Way speed auto-configuration on
18*8044SWilliam.Kucharski@Sun.COM   MX98715, MX98715A and MX98725. Support inexpensive PCI 10/100 cards
19*8044SWilliam.Kucharski@Sun.COM   based on the Macronix MX987x5 chip, such as the SOHOware Fast
20*8044SWilliam.Kucharski@Sun.COM   model SFA110A, and the LinkSYS model LNE100TX. The NetGear
21*8044SWilliam.Kucharski@Sun.COM   model FA310X, based on the LC82C168 chip is supported.
22*8044SWilliam.Kucharski@Sun.COM   The TRENDnet TE100-PCIA NIC which uses a genuine Intel 21143-PD
23*8044SWilliam.Kucharski@Sun.COM   chipset is supported. Also, Davicom DM9102's.
24*8044SWilliam.Kucharski@Sun.COM 
25*8044SWilliam.Kucharski@Sun.COM   Documentation and source code used:
26*8044SWilliam.Kucharski@Sun.COM   Source for Etherboot driver at
27*8044SWilliam.Kucharski@Sun.COM   http://etherboot.sourceforge.net/
28*8044SWilliam.Kucharski@Sun.COM   MX98715A Data Sheet and MX98715A Application Note
29*8044SWilliam.Kucharski@Sun.COM   on http://www.macronix.com/  (PDF format files)
30*8044SWilliam.Kucharski@Sun.COM   Source for Linux tulip driver at
31*8044SWilliam.Kucharski@Sun.COM   http://cesdis.gsfc.nasa.gov/linux/drivers/tulip.html
32*8044SWilliam.Kucharski@Sun.COM 
33*8044SWilliam.Kucharski@Sun.COM   Adapted by Ken Yap from
34*8044SWilliam.Kucharski@Sun.COM   FreeBSD netboot DEC 21143 driver
35*8044SWilliam.Kucharski@Sun.COM   Author: David Sharp
36*8044SWilliam.Kucharski@Sun.COM   date: Nov/98
37*8044SWilliam.Kucharski@Sun.COM 
38*8044SWilliam.Kucharski@Sun.COM   Some code fragments were taken from verious places, Ken Yap's
39*8044SWilliam.Kucharski@Sun.COM   etherboot, FreeBSD's if_de.c, and various Linux related files.
40*8044SWilliam.Kucharski@Sun.COM   DEC's manuals for the 21143 and SROM format were very helpful.
41*8044SWilliam.Kucharski@Sun.COM   The Linux de driver development page has a number of links to
42*8044SWilliam.Kucharski@Sun.COM   useful related information.  Have a look at:
43*8044SWilliam.Kucharski@Sun.COM   ftp://cesdis.gsfc.nasa.gov/pub/linux/drivers/tulip-devel.html
44*8044SWilliam.Kucharski@Sun.COM */
45*8044SWilliam.Kucharski@Sun.COM 
46*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
47*8044SWilliam.Kucharski@Sun.COM /* Revision History                                                  */
48*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
49*8044SWilliam.Kucharski@Sun.COM 
50*8044SWilliam.Kucharski@Sun.COM /*
51*8044SWilliam.Kucharski@Sun.COM   07 Sep 2003  timlegge	Multicast Support Added
52*8044SWilliam.Kucharski@Sun.COM   11 Apr 2001  mdc     [patch to etherboot 4.7.24]
53*8044SWilliam.Kucharski@Sun.COM      Major rewrite to include Linux tulip driver media detection
54*8044SWilliam.Kucharski@Sun.COM      code.  This driver should support a lot more cards now.
55*8044SWilliam.Kucharski@Sun.COM   16 Jul 2000  mdc     0.75b11
56*8044SWilliam.Kucharski@Sun.COM      Added support for ADMtek 0985 Centaur-P, a "Comet" tulip clone
57*8044SWilliam.Kucharski@Sun.COM      which is used on the LinkSYS LNE100TX v4.x cards.  We already
58*8044SWilliam.Kucharski@Sun.COM      support LNE100TX v2.0 cards, which use a different controller.
59*8044SWilliam.Kucharski@Sun.COM   04 Jul 2000   jam     ?
60*8044SWilliam.Kucharski@Sun.COM      Added test of status after receiving a packet from the card.
61*8044SWilliam.Kucharski@Sun.COM      Also uncommented the tulip_disable routine.  Stray packets
62*8044SWilliam.Kucharski@Sun.COM      seemed to be causing problems.
63*8044SWilliam.Kucharski@Sun.COM   27 Apr 2000   njl     ?
64*8044SWilliam.Kucharski@Sun.COM   29 Feb 2000   mdc     0.75b7
65*8044SWilliam.Kucharski@Sun.COM      Increased reset delay to 3 seconds because Macronix cards seem to
66*8044SWilliam.Kucharski@Sun.COM      need more reset time before card comes back to a usable state.
67*8044SWilliam.Kucharski@Sun.COM   26 Feb 2000   mdc     0.75b6
68*8044SWilliam.Kucharski@Sun.COM      Added a 1 second delay after initializing the transmitter because
69*8044SWilliam.Kucharski@Sun.COM      some cards seem to need the time or they drop the first packet
70*8044SWilliam.Kucharski@Sun.COM      transmitted.
71*8044SWilliam.Kucharski@Sun.COM   23 Feb 2000   mdc     0.75b5
72*8044SWilliam.Kucharski@Sun.COM      removed udelay code and used currticks() for more reliable delay
73*8044SWilliam.Kucharski@Sun.COM      code in reset pause and sanity timeouts.  Added function prototypes
74*8044SWilliam.Kucharski@Sun.COM      and TX debugging code.
75*8044SWilliam.Kucharski@Sun.COM   21 Feb 2000   mdc     patch to Etherboot 4.4.3
76*8044SWilliam.Kucharski@Sun.COM      Incorporated patches from Bob Edwards and Paul Mackerras of
77*8044SWilliam.Kucharski@Sun.COM      Linuxcare's OZLabs to deal with inefficiencies in tulip_transmit
78*8044SWilliam.Kucharski@Sun.COM      and udelay.  We now wait for packet transmission to complete
79*8044SWilliam.Kucharski@Sun.COM      (or sanity timeout).
80*8044SWilliam.Kucharski@Sun.COM   04 Feb 2000   Robert.Edwards@anu.edu.au patch to Etherboot 4.4.2
81*8044SWilliam.Kucharski@Sun.COM      patch to tulip.c that implements the automatic selection of the MII
82*8044SWilliam.Kucharski@Sun.COM      interface on cards using the Intel/DEC 21143 reference design, in
83*8044SWilliam.Kucharski@Sun.COM      particular, the TRENDnet TE100-PCIA NIC which uses a genuine Intel
84*8044SWilliam.Kucharski@Sun.COM      21143-PD chipset.
85*8044SWilliam.Kucharski@Sun.COM   11 Jan 2000   mdc     0.75b4
86*8044SWilliam.Kucharski@Sun.COM      Added support for NetGear FA310TX card based on the LC82C168
87*8044SWilliam.Kucharski@Sun.COM      chip.  This should also support Lite-On LC82C168 boards.
88*8044SWilliam.Kucharski@Sun.COM      Added simple MII support. Re-arranged code to better modularize
89*8044SWilliam.Kucharski@Sun.COM      initializations.
90*8044SWilliam.Kucharski@Sun.COM   04 Dec 1999   mdc     0.75b3
91*8044SWilliam.Kucharski@Sun.COM      Added preliminary support for LNE100TX PCI cards.  Should work for
92*8044SWilliam.Kucharski@Sun.COM      PNIC2 cards. No MII support, but single interface (RJ45) tulip
93*8044SWilliam.Kucharski@Sun.COM      cards seem to not care.
94*8044SWilliam.Kucharski@Sun.COM   03 Dec 1999   mdc     0.75b2
95*8044SWilliam.Kucharski@Sun.COM      Renamed from mx987x5 to tulip, merged in original tulip init code
96*8044SWilliam.Kucharski@Sun.COM      from tulip.c to support other tulip compatible cards.
97*8044SWilliam.Kucharski@Sun.COM   02 Dec 1999   mdc     0.75b1
98*8044SWilliam.Kucharski@Sun.COM      Released Beta MX987x5 Driver for code review and testing to netboot
99*8044SWilliam.Kucharski@Sun.COM      and thinguin mailing lists.
100*8044SWilliam.Kucharski@Sun.COM */
101*8044SWilliam.Kucharski@Sun.COM 
102*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
103*8044SWilliam.Kucharski@Sun.COM /* Declarations                                                      */
104*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
105*8044SWilliam.Kucharski@Sun.COM 
106*8044SWilliam.Kucharski@Sun.COM #include "etherboot.h"
107*8044SWilliam.Kucharski@Sun.COM #include "nic.h"
108*8044SWilliam.Kucharski@Sun.COM #include "pci.h"
109*8044SWilliam.Kucharski@Sun.COM 
110*8044SWilliam.Kucharski@Sun.COM /* User settable parameters */
111*8044SWilliam.Kucharski@Sun.COM 
112*8044SWilliam.Kucharski@Sun.COM #undef	TULIP_DEBUG
113*8044SWilliam.Kucharski@Sun.COM #undef	TULIP_DEBUG_WHERE
114*8044SWilliam.Kucharski@Sun.COM #ifdef	TULIP_DEBUG
115*8044SWilliam.Kucharski@Sun.COM static int tulip_debug = 2;             /* 1 normal messages, 0 quiet .. 7 verbose. */
116*8044SWilliam.Kucharski@Sun.COM #endif
117*8044SWilliam.Kucharski@Sun.COM 
118*8044SWilliam.Kucharski@Sun.COM #define TX_TIME_OUT       2*TICKS_PER_SEC
119*8044SWilliam.Kucharski@Sun.COM 
120*8044SWilliam.Kucharski@Sun.COM typedef uint8_t    u8;
121*8044SWilliam.Kucharski@Sun.COM typedef  int8_t    s8;
122*8044SWilliam.Kucharski@Sun.COM typedef uint16_t   u16;
123*8044SWilliam.Kucharski@Sun.COM typedef  int16_t   s16;
124*8044SWilliam.Kucharski@Sun.COM typedef uint32_t   u32;
125*8044SWilliam.Kucharski@Sun.COM typedef  int32_t   s32;
126*8044SWilliam.Kucharski@Sun.COM 
127*8044SWilliam.Kucharski@Sun.COM /* helpful macros if on a big_endian machine for changing byte order.
128*8044SWilliam.Kucharski@Sun.COM    not strictly needed on Intel */
129*8044SWilliam.Kucharski@Sun.COM #define get_unaligned(ptr) (*(ptr))
130*8044SWilliam.Kucharski@Sun.COM #define put_unaligned(val, ptr) ((void)( *(ptr) = (val) ))
131*8044SWilliam.Kucharski@Sun.COM #define get_u16(ptr) (*(u16 *)(ptr))
132*8044SWilliam.Kucharski@Sun.COM #define virt_to_le32desc(addr)  virt_to_bus(addr)
133*8044SWilliam.Kucharski@Sun.COM 
134*8044SWilliam.Kucharski@Sun.COM #define TULIP_IOTYPE  PCI_USES_MASTER | PCI_USES_IO | PCI_ADDR0
135*8044SWilliam.Kucharski@Sun.COM #define TULIP_SIZE 0x80
136*8044SWilliam.Kucharski@Sun.COM 
137*8044SWilliam.Kucharski@Sun.COM /* This is a mysterious value that can be written to CSR11 in the 21040 (only)
138*8044SWilliam.Kucharski@Sun.COM    to support a pre-NWay full-duplex signaling mechanism using short frames.
139*8044SWilliam.Kucharski@Sun.COM    No one knows what it should be, but if left at its default value some
140*8044SWilliam.Kucharski@Sun.COM    10base2(!) packets trigger a full-duplex-request interrupt. */
141*8044SWilliam.Kucharski@Sun.COM #define FULL_DUPLEX_MAGIC       0x6969
142*8044SWilliam.Kucharski@Sun.COM 
143*8044SWilliam.Kucharski@Sun.COM static const int csr0 = 0x01A00000 | 0x8000;
144*8044SWilliam.Kucharski@Sun.COM 
145*8044SWilliam.Kucharski@Sun.COM /*  The possible media types that can be set in options[] are: */
146*8044SWilliam.Kucharski@Sun.COM #define MEDIA_MASK 31
147*8044SWilliam.Kucharski@Sun.COM static const char * const medianame[32] = {
148*8044SWilliam.Kucharski@Sun.COM     "10baseT", "10base2", "AUI", "100baseTx",
149*8044SWilliam.Kucharski@Sun.COM     "10baseT-FDX", "100baseTx-FDX", "100baseT4", "100baseFx",
150*8044SWilliam.Kucharski@Sun.COM     "100baseFx-FDX", "MII 10baseT", "MII 10baseT-FDX", "MII",
151*8044SWilliam.Kucharski@Sun.COM     "10baseT(forced)", "MII 100baseTx", "MII 100baseTx-FDX", "MII 100baseT4",
152*8044SWilliam.Kucharski@Sun.COM     "MII 100baseFx-HDX", "MII 100baseFx-FDX", "Home-PNA 1Mbps", "Invalid-19",
153*8044SWilliam.Kucharski@Sun.COM };
154*8044SWilliam.Kucharski@Sun.COM 
155*8044SWilliam.Kucharski@Sun.COM /* This much match tulip_tbl[]!  Note 21142 == 21143. */
156*8044SWilliam.Kucharski@Sun.COM enum tulip_chips {
157*8044SWilliam.Kucharski@Sun.COM     DC21040=0, DC21041=1, DC21140=2, DC21142=3, DC21143=3,
158*8044SWilliam.Kucharski@Sun.COM     LC82C168, MX98713, MX98715, MX98725, AX88141, AX88140, PNIC2, COMET,
159*8044SWilliam.Kucharski@Sun.COM     COMPEX9881, I21145, XIRCOM
160*8044SWilliam.Kucharski@Sun.COM };
161*8044SWilliam.Kucharski@Sun.COM 
162*8044SWilliam.Kucharski@Sun.COM enum pci_id_flags_bits {
163*8044SWilliam.Kucharski@Sun.COM     /* Set PCI command register bits before calling probe1(). */
164*8044SWilliam.Kucharski@Sun.COM     PCI_USES_IO=1, PCI_USES_MEM=2, PCI_USES_MASTER=4,
165*8044SWilliam.Kucharski@Sun.COM     /* Read and map the single following PCI BAR. */
166*8044SWilliam.Kucharski@Sun.COM     PCI_ADDR0=0<<4, PCI_ADDR1=1<<4, PCI_ADDR2=2<<4, PCI_ADDR3=3<<4,
167*8044SWilliam.Kucharski@Sun.COM     PCI_ADDR_64BITS=0x100, PCI_NO_ACPI_WAKE=0x200, PCI_NO_MIN_LATENCY=0x400,
168*8044SWilliam.Kucharski@Sun.COM     PCI_UNUSED_IRQ=0x800,
169*8044SWilliam.Kucharski@Sun.COM };
170*8044SWilliam.Kucharski@Sun.COM 
171*8044SWilliam.Kucharski@Sun.COM struct pci_id_info {
172*8044SWilliam.Kucharski@Sun.COM     char *name;
173*8044SWilliam.Kucharski@Sun.COM     struct match_info {
174*8044SWilliam.Kucharski@Sun.COM         u32 pci, pci_mask, subsystem, subsystem_mask;
175*8044SWilliam.Kucharski@Sun.COM         u32 revision, revision_mask;                            /* Only 8 bits. */
176*8044SWilliam.Kucharski@Sun.COM     } id;
177*8044SWilliam.Kucharski@Sun.COM     enum pci_id_flags_bits pci_flags;
178*8044SWilliam.Kucharski@Sun.COM     int io_size;                                /* Needed for I/O region check or ioremap(). */
179*8044SWilliam.Kucharski@Sun.COM     int drv_flags;                              /* Driver use, intended as capability flags. */
180*8044SWilliam.Kucharski@Sun.COM };
181*8044SWilliam.Kucharski@Sun.COM 
182*8044SWilliam.Kucharski@Sun.COM static struct pci_id_info pci_id_tbl[] = {
183*8044SWilliam.Kucharski@Sun.COM     { "Digital DC21040 Tulip", { 0x00021011, 0xffffffff, 0, 0, 0, 0 },
184*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 0x80, DC21040 },
185*8044SWilliam.Kucharski@Sun.COM     { "Digital DC21041 Tulip", { 0x00141011, 0xffffffff, 0, 0, 0, 0 },
186*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 0x80, DC21041 },
187*8044SWilliam.Kucharski@Sun.COM     { "Digital DS21140A Tulip", { 0x00091011, 0xffffffff, 0,0, 0x20,0xf0 },
188*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 0x80, DC21140 },
189*8044SWilliam.Kucharski@Sun.COM     { "Digital DS21140 Tulip", { 0x00091011, 0xffffffff, 0, 0, 0, 0 },
190*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 0x80, DC21140 },
191*8044SWilliam.Kucharski@Sun.COM     { "Digital DS21143 Tulip", { 0x00191011, 0xffffffff, 0,0, 65,0xff },
192*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, TULIP_SIZE, DC21142 },
193*8044SWilliam.Kucharski@Sun.COM     { "Digital DS21142 Tulip", { 0x00191011, 0xffffffff, 0, 0, 0, 0 },
194*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, TULIP_SIZE, DC21142 },
195*8044SWilliam.Kucharski@Sun.COM     { "Kingston KNE110tx (PNIC)", { 0x000211AD, 0xffffffff, 0xf0022646, 0xffffffff, 0, 0 },
196*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 256, LC82C168 },
197*8044SWilliam.Kucharski@Sun.COM     { "Lite-On 82c168 PNIC", { 0x000211AD, 0xffffffff, 0, 0, 0, 0 },
198*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 256, LC82C168 },
199*8044SWilliam.Kucharski@Sun.COM     { "Macronix 98713 PMAC", { 0x051210d9, 0xffffffff, 0, 0, 0, 0 },
200*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 256, MX98713 },
201*8044SWilliam.Kucharski@Sun.COM     { "Macronix 98715 PMAC", { 0x053110d9, 0xffffffff, 0, 0, 0, 0 },
202*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 256, MX98715 },
203*8044SWilliam.Kucharski@Sun.COM     { "Macronix 98725 PMAC", { 0x053110d9, 0xffffffff, 0, 0, 0, 0 },
204*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 256, MX98725 },
205*8044SWilliam.Kucharski@Sun.COM     { "ASIX AX88141", { 0x1400125B, 0xffffffff, 0,0, 0x10, 0xf0 },
206*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 128, AX88141 },
207*8044SWilliam.Kucharski@Sun.COM     { "ASIX AX88140", { 0x1400125B, 0xffffffff, 0, 0, 0, 0 },
208*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 128, AX88140 },
209*8044SWilliam.Kucharski@Sun.COM     { "Lite-On LC82C115 PNIC-II", { 0xc11511AD, 0xffffffff, 0, 0, 0, 0 },
210*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 256, PNIC2 },
211*8044SWilliam.Kucharski@Sun.COM     { "ADMtek AN981 Comet", { 0x09811317, 0xffffffff, 0, 0, 0, 0 },
212*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 256, COMET },
213*8044SWilliam.Kucharski@Sun.COM     { "ADMTek AN983 Comet", { 0x12161113, 0xffffffff, 0, 0, 0, 0 },
214*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 256, COMET },
215*8044SWilliam.Kucharski@Sun.COM     { "ADMtek Centaur-P", { 0x09851317, 0xffffffff, 0, 0, 0, 0 },
216*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 256, COMET },
217*8044SWilliam.Kucharski@Sun.COM     { "ADMtek Centaur-C", { 0x19851317, 0xffffffff, 0, 0, 0, 0 },
218*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 256, COMET },
219*8044SWilliam.Kucharski@Sun.COM     { "Compex RL100-TX", { 0x988111F6, 0xffffffff, 0, 0, 0, 0 },
220*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 128, COMPEX9881 },
221*8044SWilliam.Kucharski@Sun.COM     { "Intel 21145 Tulip", { 0x00398086, 0xffffffff, 0, 0, 0, 0 },
222*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 128, I21145 },
223*8044SWilliam.Kucharski@Sun.COM     { "Xircom Tulip clone", { 0x0003115d, 0xffffffff, 0, 0, 0, 0 },
224*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 128, XIRCOM },
225*8044SWilliam.Kucharski@Sun.COM     { "Davicom DM9102", { 0x91021282, 0xffffffff, 0, 0, 0, 0 },
226*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 0x80, DC21140 },
227*8044SWilliam.Kucharski@Sun.COM     { "Davicom DM9100", { 0x91001282, 0xffffffff, 0, 0, 0, 0 },
228*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 0x80, DC21140 },
229*8044SWilliam.Kucharski@Sun.COM     { "Macronix mxic-98715 (EN1217)", { 0x12171113, 0xffffffff, 0, 0, 0, 0 },
230*8044SWilliam.Kucharski@Sun.COM       TULIP_IOTYPE, 256, MX98715 },
231*8044SWilliam.Kucharski@Sun.COM     { 0, { 0, 0, 0, 0, 0, 0 }, 0, 0, 0 },
232*8044SWilliam.Kucharski@Sun.COM };
233*8044SWilliam.Kucharski@Sun.COM 
234*8044SWilliam.Kucharski@Sun.COM enum tbl_flag {
235*8044SWilliam.Kucharski@Sun.COM     HAS_MII=1, HAS_MEDIA_TABLE=2, CSR12_IN_SROM=4, ALWAYS_CHECK_MII=8,
236*8044SWilliam.Kucharski@Sun.COM     HAS_PWRDWN=0x10, MC_HASH_ONLY=0x20, /* Hash-only multicast filter. */
237*8044SWilliam.Kucharski@Sun.COM     HAS_PNICNWAY=0x80, HAS_NWAY=0x40,   /* Uses internal NWay xcvr. */
238*8044SWilliam.Kucharski@Sun.COM     HAS_INTR_MITIGATION=0x100, IS_ASIX=0x200, HAS_8023X=0x400,
239*8044SWilliam.Kucharski@Sun.COM };
240*8044SWilliam.Kucharski@Sun.COM 
241*8044SWilliam.Kucharski@Sun.COM /* Note: this table must match  enum tulip_chips  above. */
242*8044SWilliam.Kucharski@Sun.COM static struct tulip_chip_table {
243*8044SWilliam.Kucharski@Sun.COM     char *chip_name;
244*8044SWilliam.Kucharski@Sun.COM     int flags;
245*8044SWilliam.Kucharski@Sun.COM } tulip_tbl[] = {
246*8044SWilliam.Kucharski@Sun.COM     { "Digital DC21040 Tulip", 0},
247*8044SWilliam.Kucharski@Sun.COM     { "Digital DC21041 Tulip", HAS_MEDIA_TABLE | HAS_NWAY },
248*8044SWilliam.Kucharski@Sun.COM     { "Digital DS21140 Tulip", HAS_MII | HAS_MEDIA_TABLE | CSR12_IN_SROM },
249*8044SWilliam.Kucharski@Sun.COM     { "Digital DS21143 Tulip", HAS_MII | HAS_MEDIA_TABLE | ALWAYS_CHECK_MII
250*8044SWilliam.Kucharski@Sun.COM       | HAS_PWRDWN | HAS_NWAY   | HAS_INTR_MITIGATION },
251*8044SWilliam.Kucharski@Sun.COM     { "Lite-On 82c168 PNIC", HAS_MII | HAS_PNICNWAY },
252*8044SWilliam.Kucharski@Sun.COM     { "Macronix 98713 PMAC", HAS_MII | HAS_MEDIA_TABLE | CSR12_IN_SROM },
253*8044SWilliam.Kucharski@Sun.COM     { "Macronix 98715 PMAC", HAS_MEDIA_TABLE },
254*8044SWilliam.Kucharski@Sun.COM     { "Macronix 98725 PMAC", HAS_MEDIA_TABLE },
255*8044SWilliam.Kucharski@Sun.COM     { "ASIX AX88140", HAS_MII | HAS_MEDIA_TABLE | CSR12_IN_SROM
256*8044SWilliam.Kucharski@Sun.COM       | MC_HASH_ONLY | IS_ASIX },
257*8044SWilliam.Kucharski@Sun.COM     { "ASIX AX88141", HAS_MII | HAS_MEDIA_TABLE | CSR12_IN_SROM | MC_HASH_ONLY
258*8044SWilliam.Kucharski@Sun.COM       | IS_ASIX },
259*8044SWilliam.Kucharski@Sun.COM     { "Lite-On PNIC-II", HAS_MII | HAS_NWAY | HAS_8023X },
260*8044SWilliam.Kucharski@Sun.COM     { "ADMtek Comet", MC_HASH_ONLY },
261*8044SWilliam.Kucharski@Sun.COM     { "Compex 9881 PMAC",       HAS_MII | HAS_MEDIA_TABLE | CSR12_IN_SROM },
262*8044SWilliam.Kucharski@Sun.COM     { "Intel DS21145 Tulip", HAS_MII | HAS_MEDIA_TABLE | ALWAYS_CHECK_MII
263*8044SWilliam.Kucharski@Sun.COM       | HAS_PWRDWN | HAS_NWAY },
264*8044SWilliam.Kucharski@Sun.COM     { "Xircom tulip work-alike", HAS_MII | HAS_MEDIA_TABLE | ALWAYS_CHECK_MII
265*8044SWilliam.Kucharski@Sun.COM       | HAS_PWRDWN | HAS_NWAY },
266*8044SWilliam.Kucharski@Sun.COM     { 0, 0 },
267*8044SWilliam.Kucharski@Sun.COM };
268*8044SWilliam.Kucharski@Sun.COM 
269*8044SWilliam.Kucharski@Sun.COM /* A full-duplex map for media types. */
270*8044SWilliam.Kucharski@Sun.COM enum MediaIs {
271*8044SWilliam.Kucharski@Sun.COM     MediaIsFD = 1, MediaAlwaysFD=2, MediaIsMII=4, MediaIsFx=8,
272*8044SWilliam.Kucharski@Sun.COM     MediaIs100=16};
273*8044SWilliam.Kucharski@Sun.COM 
274*8044SWilliam.Kucharski@Sun.COM static const char media_cap[32] =
275*8044SWilliam.Kucharski@Sun.COM {0,0,0,16,  3,19,16,24,  27,4,7,5, 0,20,23,20, 20,31,0,0, };
276*8044SWilliam.Kucharski@Sun.COM static u8 t21040_csr13[] = {2,0x0C,8,4,  4,0,0,0, 0,0,0,0, 4,0,0,0};
277*8044SWilliam.Kucharski@Sun.COM 
278*8044SWilliam.Kucharski@Sun.COM /* 21041 transceiver register settings: 10-T, 10-2, AUI, 10-T, 10T-FD */
279*8044SWilliam.Kucharski@Sun.COM static u16 t21041_csr13[] = { 0xEF01, 0xEF09, 0xEF09, 0xEF01, 0xEF09, };
280*8044SWilliam.Kucharski@Sun.COM static u16 t21041_csr14[] = { 0xFFFF, 0xF7FD, 0xF7FD, 0x7F3F, 0x7F3D, };
281*8044SWilliam.Kucharski@Sun.COM static u16 t21041_csr15[] = { 0x0008, 0x0006, 0x000E, 0x0008, 0x0008, };
282*8044SWilliam.Kucharski@Sun.COM 
283*8044SWilliam.Kucharski@Sun.COM /* not used
284*8044SWilliam.Kucharski@Sun.COM static u16 t21142_csr13[] = { 0x0001, 0x0009, 0x0009, 0x0000, 0x0001, };
285*8044SWilliam.Kucharski@Sun.COM */
286*8044SWilliam.Kucharski@Sun.COM static u16 t21142_csr14[] = { 0xFFFF, 0x0705, 0x0705, 0x0000, 0x7F3D, };
287*8044SWilliam.Kucharski@Sun.COM /* not used
288*8044SWilliam.Kucharski@Sun.COM static u16 t21142_csr15[] = { 0x0008, 0x0006, 0x000E, 0x0008, 0x0008, };
289*8044SWilliam.Kucharski@Sun.COM */
290*8044SWilliam.Kucharski@Sun.COM 
291*8044SWilliam.Kucharski@Sun.COM /* Offsets to the Command and Status Registers, "CSRs".  All accesses
292*8044SWilliam.Kucharski@Sun.COM    must be longword instructions and quadword aligned. */
293*8044SWilliam.Kucharski@Sun.COM enum tulip_offsets {
294*8044SWilliam.Kucharski@Sun.COM     CSR0=0,     CSR1=0x08,  CSR2=0x10,  CSR3=0x18,  CSR4=0x20,  CSR5=0x28,
295*8044SWilliam.Kucharski@Sun.COM     CSR6=0x30,  CSR7=0x38,  CSR8=0x40,  CSR9=0x48, CSR10=0x50, CSR11=0x58,
296*8044SWilliam.Kucharski@Sun.COM     CSR12=0x60, CSR13=0x68, CSR14=0x70, CSR15=0x78, CSR16=0x80, CSR20=0xA0
297*8044SWilliam.Kucharski@Sun.COM };
298*8044SWilliam.Kucharski@Sun.COM 
299*8044SWilliam.Kucharski@Sun.COM /* The bits in the CSR5 status registers, mostly interrupt sources. */
300*8044SWilliam.Kucharski@Sun.COM enum status_bits {
301*8044SWilliam.Kucharski@Sun.COM     TimerInt=0x800, TPLnkFail=0x1000, TPLnkPass=0x10,
302*8044SWilliam.Kucharski@Sun.COM     NormalIntr=0x10000, AbnormalIntr=0x8000,
303*8044SWilliam.Kucharski@Sun.COM     RxJabber=0x200, RxDied=0x100, RxNoBuf=0x80, RxIntr=0x40,
304*8044SWilliam.Kucharski@Sun.COM     TxFIFOUnderflow=0x20, TxJabber=0x08, TxNoBuf=0x04, TxDied=0x02, TxIntr=0x01,
305*8044SWilliam.Kucharski@Sun.COM };
306*8044SWilliam.Kucharski@Sun.COM 
307*8044SWilliam.Kucharski@Sun.COM /* The configuration bits in CSR6. */
308*8044SWilliam.Kucharski@Sun.COM enum csr6_mode_bits {
309*8044SWilliam.Kucharski@Sun.COM 	TxOn=0x2000, RxOn=0x0002, FullDuplex=0x0200,
310*8044SWilliam.Kucharski@Sun.COM 	AcceptBroadcast=0x0100, AcceptAllMulticast=0x0080,
311*8044SWilliam.Kucharski@Sun.COM 	AcceptAllPhys=0x0040, AcceptRunt=0x0008,
312*8044SWilliam.Kucharski@Sun.COM };
313*8044SWilliam.Kucharski@Sun.COM 
314*8044SWilliam.Kucharski@Sun.COM 
315*8044SWilliam.Kucharski@Sun.COM enum desc_status_bits {
316*8044SWilliam.Kucharski@Sun.COM     DescOwnded=0x80000000, RxDescFatalErr=0x8000, RxWholePkt=0x0300,
317*8044SWilliam.Kucharski@Sun.COM };
318*8044SWilliam.Kucharski@Sun.COM 
319*8044SWilliam.Kucharski@Sun.COM struct medialeaf {
320*8044SWilliam.Kucharski@Sun.COM     u8 type;
321*8044SWilliam.Kucharski@Sun.COM     u8 media;
322*8044SWilliam.Kucharski@Sun.COM     unsigned char *leafdata;
323*8044SWilliam.Kucharski@Sun.COM };
324*8044SWilliam.Kucharski@Sun.COM 
325*8044SWilliam.Kucharski@Sun.COM struct mediatable {
326*8044SWilliam.Kucharski@Sun.COM     u16 defaultmedia;
327*8044SWilliam.Kucharski@Sun.COM     u8 leafcount, csr12dir;                             /* General purpose pin directions. */
328*8044SWilliam.Kucharski@Sun.COM     unsigned has_mii:1, has_nonmii:1, has_reset:6;
329*8044SWilliam.Kucharski@Sun.COM     u32 csr15dir, csr15val;                             /* 21143 NWay setting. */
330*8044SWilliam.Kucharski@Sun.COM     struct medialeaf mleaf[0];
331*8044SWilliam.Kucharski@Sun.COM };
332*8044SWilliam.Kucharski@Sun.COM 
333*8044SWilliam.Kucharski@Sun.COM struct mediainfo {
334*8044SWilliam.Kucharski@Sun.COM     struct mediainfo *next;
335*8044SWilliam.Kucharski@Sun.COM     int info_type;
336*8044SWilliam.Kucharski@Sun.COM     int index;
337*8044SWilliam.Kucharski@Sun.COM     unsigned char *info;
338*8044SWilliam.Kucharski@Sun.COM };
339*8044SWilliam.Kucharski@Sun.COM 
340*8044SWilliam.Kucharski@Sun.COM /* EEPROM Address width definitions */
341*8044SWilliam.Kucharski@Sun.COM #define EEPROM_ADDRLEN 6
342*8044SWilliam.Kucharski@Sun.COM #define EEPROM_SIZE    128              /* 2 << EEPROM_ADDRLEN */
343*8044SWilliam.Kucharski@Sun.COM 
344*8044SWilliam.Kucharski@Sun.COM /* The EEPROM commands include the alway-set leading bit. */
345*8044SWilliam.Kucharski@Sun.COM #define EE_WRITE_CMD    (5 << addr_len)
346*8044SWilliam.Kucharski@Sun.COM #define EE_READ_CMD     (6 << addr_len)
347*8044SWilliam.Kucharski@Sun.COM #define EE_ERASE_CMD    (7 << addr_len)
348*8044SWilliam.Kucharski@Sun.COM 
349*8044SWilliam.Kucharski@Sun.COM /* EEPROM_Ctrl bits. */
350*8044SWilliam.Kucharski@Sun.COM #define EE_SHIFT_CLK    0x02    /* EEPROM shift clock. */
351*8044SWilliam.Kucharski@Sun.COM #define EE_CS           0x01    /* EEPROM chip select. */
352*8044SWilliam.Kucharski@Sun.COM #define EE_DATA_WRITE   0x04    /* EEPROM chip data in. */
353*8044SWilliam.Kucharski@Sun.COM #define EE_WRITE_0      0x01
354*8044SWilliam.Kucharski@Sun.COM #define EE_WRITE_1      0x05
355*8044SWilliam.Kucharski@Sun.COM #define EE_DATA_READ    0x08    /* EEPROM chip data out. */
356*8044SWilliam.Kucharski@Sun.COM #define EE_ENB          (0x4800 | EE_CS)
357*8044SWilliam.Kucharski@Sun.COM 
358*8044SWilliam.Kucharski@Sun.COM /* Delay between EEPROM clock transitions.  Even at 33Mhz current PCI
359*8044SWilliam.Kucharski@Sun.COM    implementations don't overrun the EEPROM clock.  We add a bus
360*8044SWilliam.Kucharski@Sun.COM    turn-around to insure that this remains true.  */
361*8044SWilliam.Kucharski@Sun.COM #define eeprom_delay()  inl(ee_addr)
362*8044SWilliam.Kucharski@Sun.COM 
363*8044SWilliam.Kucharski@Sun.COM /* Size of transmit and receive buffers */
364*8044SWilliam.Kucharski@Sun.COM #define BUFLEN 1536
365*8044SWilliam.Kucharski@Sun.COM 
366*8044SWilliam.Kucharski@Sun.COM /* Ring-wrap flag in length field, use for last ring entry.
367*8044SWilliam.Kucharski@Sun.COM    0x01000000 means chain on buffer2 address,
368*8044SWilliam.Kucharski@Sun.COM    0x02000000 means use the ring start address in CSR2/3.
369*8044SWilliam.Kucharski@Sun.COM    Note: Some work-alike chips do not function correctly in chained mode.
370*8044SWilliam.Kucharski@Sun.COM    The ASIX chip works only in chained mode.
371*8044SWilliam.Kucharski@Sun.COM    Thus we indicate ring mode, but always write the 'next' field for
372*8044SWilliam.Kucharski@Sun.COM    chained mode as well. */
373*8044SWilliam.Kucharski@Sun.COM #define DESC_RING_WRAP 0x02000000
374*8044SWilliam.Kucharski@Sun.COM 
375*8044SWilliam.Kucharski@Sun.COM /* transmit and receive descriptor format */
376*8044SWilliam.Kucharski@Sun.COM struct tulip_rx_desc {
377*8044SWilliam.Kucharski@Sun.COM     volatile u32 status;
378*8044SWilliam.Kucharski@Sun.COM     u32 length;
379*8044SWilliam.Kucharski@Sun.COM     u32 buffer1, buffer2;
380*8044SWilliam.Kucharski@Sun.COM };
381*8044SWilliam.Kucharski@Sun.COM 
382*8044SWilliam.Kucharski@Sun.COM struct tulip_tx_desc {
383*8044SWilliam.Kucharski@Sun.COM     volatile u32 status;
384*8044SWilliam.Kucharski@Sun.COM     u32 length;
385*8044SWilliam.Kucharski@Sun.COM     u32 buffer1, buffer2;
386*8044SWilliam.Kucharski@Sun.COM };
387*8044SWilliam.Kucharski@Sun.COM 
388*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
389*8044SWilliam.Kucharski@Sun.COM /* Global Storage                                                    */
390*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
391*8044SWilliam.Kucharski@Sun.COM 
392*8044SWilliam.Kucharski@Sun.COM static u32 ioaddr;
393*8044SWilliam.Kucharski@Sun.COM 
394*8044SWilliam.Kucharski@Sun.COM /* Note: transmit and receive buffers must be longword aligned and
395*8044SWilliam.Kucharski@Sun.COM    longword divisable */
396*8044SWilliam.Kucharski@Sun.COM 
397*8044SWilliam.Kucharski@Sun.COM #define TX_RING_SIZE	2
398*8044SWilliam.Kucharski@Sun.COM static struct tulip_tx_desc tx_ring[TX_RING_SIZE] __attribute__ ((aligned(4)));
399*8044SWilliam.Kucharski@Sun.COM static unsigned char txb[BUFLEN] __attribute__ ((aligned(4)));
400*8044SWilliam.Kucharski@Sun.COM 
401*8044SWilliam.Kucharski@Sun.COM #define RX_RING_SIZE	4
402*8044SWilliam.Kucharski@Sun.COM static struct tulip_rx_desc rx_ring[RX_RING_SIZE] __attribute__ ((aligned(4)));
403*8044SWilliam.Kucharski@Sun.COM static unsigned char rxb[RX_RING_SIZE * BUFLEN] __attribute__ ((aligned(4)));
404*8044SWilliam.Kucharski@Sun.COM 
405*8044SWilliam.Kucharski@Sun.COM static struct tulip_private {
406*8044SWilliam.Kucharski@Sun.COM     int cur_rx;
407*8044SWilliam.Kucharski@Sun.COM     int chip_id;                        /* index into tulip_tbl[]  */
408*8044SWilliam.Kucharski@Sun.COM     int pci_id_idx;                     /* index into pci_id_tbl[] */
409*8044SWilliam.Kucharski@Sun.COM     int revision;
410*8044SWilliam.Kucharski@Sun.COM     int flags;
411*8044SWilliam.Kucharski@Sun.COM     unsigned short vendor_id;           /* PCI card vendor code */
412*8044SWilliam.Kucharski@Sun.COM     unsigned short dev_id;              /* PCI card device code */
413*8044SWilliam.Kucharski@Sun.COM     unsigned char ehdr[ETH_HLEN];       /* buffer for ethernet header */
414*8044SWilliam.Kucharski@Sun.COM     const char *nic_name;
415*8044SWilliam.Kucharski@Sun.COM     unsigned int csr0, csr6;            /* Current CSR0, CSR6 settings. */
416*8044SWilliam.Kucharski@Sun.COM     unsigned int if_port;
417*8044SWilliam.Kucharski@Sun.COM     unsigned int full_duplex;         /* Full-duplex operation requested. */
418*8044SWilliam.Kucharski@Sun.COM     unsigned int full_duplex_lock;
419*8044SWilliam.Kucharski@Sun.COM     unsigned int medialock;           /* Do not sense media type. */
420*8044SWilliam.Kucharski@Sun.COM     unsigned int mediasense;          /* Media sensing in progress. */
421*8044SWilliam.Kucharski@Sun.COM     unsigned int nway, nwayset;     /* 21143 internal NWay. */
422*8044SWilliam.Kucharski@Sun.COM     unsigned int default_port;
423*8044SWilliam.Kucharski@Sun.COM     unsigned char eeprom[EEPROM_SIZE];  /* Serial EEPROM contents. */
424*8044SWilliam.Kucharski@Sun.COM     u8 media_table_storage[(sizeof(struct mediatable) + 32*sizeof(struct medialeaf))];
425*8044SWilliam.Kucharski@Sun.COM     u16 sym_advertise, mii_advertise;   /* NWay to-advertise. */
426*8044SWilliam.Kucharski@Sun.COM     struct mediatable *mtable;
427*8044SWilliam.Kucharski@Sun.COM     u16 lpar;                           /* 21143 Link partner ability. */
428*8044SWilliam.Kucharski@Sun.COM     u16 advertising[4];                 /* MII advertise, from SROM table. */
429*8044SWilliam.Kucharski@Sun.COM     signed char phys[4], mii_cnt;       /* MII device addresses. */
430*8044SWilliam.Kucharski@Sun.COM     int cur_index;                      /* Current media index. */
431*8044SWilliam.Kucharski@Sun.COM     int saved_if_port;
432*8044SWilliam.Kucharski@Sun.COM } tpx;
433*8044SWilliam.Kucharski@Sun.COM 
434*8044SWilliam.Kucharski@Sun.COM static struct tulip_private *tp;
435*8044SWilliam.Kucharski@Sun.COM 
436*8044SWilliam.Kucharski@Sun.COM /* Known cards that have old-style EEPROMs.
437*8044SWilliam.Kucharski@Sun.COM    Writing this table is described at
438*8044SWilliam.Kucharski@Sun.COM    http://cesdis.gsfc.nasa.gov/linux/drivers/tulip-drivers/tulip-media.html */
439*8044SWilliam.Kucharski@Sun.COM static struct fixups {
440*8044SWilliam.Kucharski@Sun.COM     char *name;
441*8044SWilliam.Kucharski@Sun.COM     unsigned char addr0, addr1, addr2;
442*8044SWilliam.Kucharski@Sun.COM     u16 newtable[32];                           /* Max length below. */
443*8044SWilliam.Kucharski@Sun.COM } eeprom_fixups[] = {
444*8044SWilliam.Kucharski@Sun.COM     {"Asante", 0, 0, 0x94, {0x1e00, 0x0000, 0x0800, 0x0100, 0x018c,
445*8044SWilliam.Kucharski@Sun.COM                             0x0000, 0x0000, 0xe078, 0x0001, 0x0050, 0x0018 }},
446*8044SWilliam.Kucharski@Sun.COM     {"SMC9332DST", 0, 0, 0xC0, { 0x1e00, 0x0000, 0x0800, 0x041f,
447*8044SWilliam.Kucharski@Sun.COM                                  0x0000, 0x009E, /* 10baseT */
448*8044SWilliam.Kucharski@Sun.COM                                  0x0004, 0x009E, /* 10baseT-FD */
449*8044SWilliam.Kucharski@Sun.COM                                  0x0903, 0x006D, /* 100baseTx */
450*8044SWilliam.Kucharski@Sun.COM                                  0x0905, 0x006D, /* 100baseTx-FD */ }},
451*8044SWilliam.Kucharski@Sun.COM     {"Cogent EM100", 0, 0, 0x92, { 0x1e00, 0x0000, 0x0800, 0x063f,
452*8044SWilliam.Kucharski@Sun.COM                                    0x0107, 0x8021, /* 100baseFx */
453*8044SWilliam.Kucharski@Sun.COM                                    0x0108, 0x8021, /* 100baseFx-FD */
454*8044SWilliam.Kucharski@Sun.COM                                    0x0100, 0x009E, /* 10baseT */
455*8044SWilliam.Kucharski@Sun.COM                                    0x0104, 0x009E, /* 10baseT-FD */
456*8044SWilliam.Kucharski@Sun.COM                                    0x0103, 0x006D, /* 100baseTx */
457*8044SWilliam.Kucharski@Sun.COM                                    0x0105, 0x006D, /* 100baseTx-FD */ }},
458*8044SWilliam.Kucharski@Sun.COM     {"Maxtech NX-110", 0, 0, 0xE8, { 0x1e00, 0x0000, 0x0800, 0x0513,
459*8044SWilliam.Kucharski@Sun.COM                                      0x1001, 0x009E, /* 10base2, CSR12 0x10*/
460*8044SWilliam.Kucharski@Sun.COM                                      0x0000, 0x009E, /* 10baseT */
461*8044SWilliam.Kucharski@Sun.COM                                      0x0004, 0x009E, /* 10baseT-FD */
462*8044SWilliam.Kucharski@Sun.COM                                      0x0303, 0x006D, /* 100baseTx, CSR12 0x03 */
463*8044SWilliam.Kucharski@Sun.COM                                      0x0305, 0x006D, /* 100baseTx-FD CSR12 0x03 */}},
464*8044SWilliam.Kucharski@Sun.COM     {"Accton EN1207", 0, 0, 0xE8, { 0x1e00, 0x0000, 0x0800, 0x051F,
465*8044SWilliam.Kucharski@Sun.COM                                     0x1B01, 0x0000, /* 10base2,   CSR12 0x1B */
466*8044SWilliam.Kucharski@Sun.COM                                     0x0B00, 0x009E, /* 10baseT,   CSR12 0x0B */
467*8044SWilliam.Kucharski@Sun.COM                                     0x0B04, 0x009E, /* 10baseT-FD,CSR12 0x0B */
468*8044SWilliam.Kucharski@Sun.COM                                     0x1B03, 0x006D, /* 100baseTx, CSR12 0x1B */
469*8044SWilliam.Kucharski@Sun.COM                                     0x1B05, 0x006D, /* 100baseTx-FD CSR12 0x1B */
470*8044SWilliam.Kucharski@Sun.COM     }},
471*8044SWilliam.Kucharski@Sun.COM     {0, 0, 0, 0, {}}};
472*8044SWilliam.Kucharski@Sun.COM 
473*8044SWilliam.Kucharski@Sun.COM static const char * block_name[] = {"21140 non-MII", "21140 MII PHY",
474*8044SWilliam.Kucharski@Sun.COM                                     "21142 Serial PHY", "21142 MII PHY", "21143 SYM PHY", "21143 reset method"};
475*8044SWilliam.Kucharski@Sun.COM 
476*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
477*8044SWilliam.Kucharski@Sun.COM /* Function Prototypes                                               */
478*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
479*8044SWilliam.Kucharski@Sun.COM static int mdio_read(struct nic *nic, int phy_id, int location);
480*8044SWilliam.Kucharski@Sun.COM static void mdio_write(struct nic *nic, int phy_id, int location, int value);
481*8044SWilliam.Kucharski@Sun.COM static int read_eeprom(unsigned long ioaddr, int location, int addr_len);
482*8044SWilliam.Kucharski@Sun.COM static void parse_eeprom(struct nic *nic);
483*8044SWilliam.Kucharski@Sun.COM static int tulip_probe(struct dev *dev, struct pci_device *pci);
484*8044SWilliam.Kucharski@Sun.COM static void tulip_init_ring(struct nic *nic);
485*8044SWilliam.Kucharski@Sun.COM static void tulip_reset(struct nic *nic);
486*8044SWilliam.Kucharski@Sun.COM static void tulip_transmit(struct nic *nic, const char *d, unsigned int t,
487*8044SWilliam.Kucharski@Sun.COM                            unsigned int s, const char *p);
488*8044SWilliam.Kucharski@Sun.COM static int tulip_poll(struct nic *nic, int retrieve);
489*8044SWilliam.Kucharski@Sun.COM static void tulip_disable(struct dev *dev);
490*8044SWilliam.Kucharski@Sun.COM static void nway_start(struct nic *nic);
491*8044SWilliam.Kucharski@Sun.COM static void pnic_do_nway(struct nic *nic);
492*8044SWilliam.Kucharski@Sun.COM static void select_media(struct nic *nic, int startup);
493*8044SWilliam.Kucharski@Sun.COM static void init_media(struct nic *nic);
494*8044SWilliam.Kucharski@Sun.COM static void start_link(struct nic *nic);
495*8044SWilliam.Kucharski@Sun.COM static int tulip_check_duplex(struct nic *nic);
496*8044SWilliam.Kucharski@Sun.COM 
497*8044SWilliam.Kucharski@Sun.COM static void tulip_wait(unsigned int nticks);
498*8044SWilliam.Kucharski@Sun.COM 
499*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG_WHERE
500*8044SWilliam.Kucharski@Sun.COM static void whereami(const char *str);
501*8044SWilliam.Kucharski@Sun.COM #endif
502*8044SWilliam.Kucharski@Sun.COM 
503*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
504*8044SWilliam.Kucharski@Sun.COM static void tulip_more(void);
505*8044SWilliam.Kucharski@Sun.COM #endif
506*8044SWilliam.Kucharski@Sun.COM 
507*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
508*8044SWilliam.Kucharski@Sun.COM /* Utility Routines                                                  */
509*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
510*8044SWilliam.Kucharski@Sun.COM 
511*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG_WHERE
whereami(const char * str)512*8044SWilliam.Kucharski@Sun.COM static void whereami (const char *str)
513*8044SWilliam.Kucharski@Sun.COM {
514*8044SWilliam.Kucharski@Sun.COM     printf("%s: %s\n", tp->nic_name, str);
515*8044SWilliam.Kucharski@Sun.COM     /* sleep(2); */
516*8044SWilliam.Kucharski@Sun.COM }
517*8044SWilliam.Kucharski@Sun.COM #endif
518*8044SWilliam.Kucharski@Sun.COM 
519*8044SWilliam.Kucharski@Sun.COM #ifdef  TULIP_DEBUG
tulip_more(void)520*8044SWilliam.Kucharski@Sun.COM static void tulip_more(void)
521*8044SWilliam.Kucharski@Sun.COM {
522*8044SWilliam.Kucharski@Sun.COM     printf("\n\n-- more --");
523*8044SWilliam.Kucharski@Sun.COM     while (!iskey())
524*8044SWilliam.Kucharski@Sun.COM         /* wait */;
525*8044SWilliam.Kucharski@Sun.COM     getchar();
526*8044SWilliam.Kucharski@Sun.COM     printf("\n\n");
527*8044SWilliam.Kucharski@Sun.COM }
528*8044SWilliam.Kucharski@Sun.COM #endif /* TULIP_DEBUG */
529*8044SWilliam.Kucharski@Sun.COM 
tulip_wait(unsigned int nticks)530*8044SWilliam.Kucharski@Sun.COM static void tulip_wait(unsigned int nticks)
531*8044SWilliam.Kucharski@Sun.COM {
532*8044SWilliam.Kucharski@Sun.COM     unsigned int to = currticks() + nticks;
533*8044SWilliam.Kucharski@Sun.COM     while (currticks() < to)
534*8044SWilliam.Kucharski@Sun.COM         /* wait */ ;
535*8044SWilliam.Kucharski@Sun.COM }
536*8044SWilliam.Kucharski@Sun.COM 
537*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
538*8044SWilliam.Kucharski@Sun.COM /* Media Descriptor Code                                             */
539*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
540*8044SWilliam.Kucharski@Sun.COM 
541*8044SWilliam.Kucharski@Sun.COM /* MII transceiver control section.
542*8044SWilliam.Kucharski@Sun.COM    Read and write the MII registers using software-generated serial
543*8044SWilliam.Kucharski@Sun.COM    MDIO protocol.  See the MII specifications or DP83840A data sheet
544*8044SWilliam.Kucharski@Sun.COM    for details. */
545*8044SWilliam.Kucharski@Sun.COM 
546*8044SWilliam.Kucharski@Sun.COM /* The maximum data clock rate is 2.5 Mhz.  The minimum timing is usually
547*8044SWilliam.Kucharski@Sun.COM    met by back-to-back PCI I/O cycles, but we insert a delay to avoid
548*8044SWilliam.Kucharski@Sun.COM    "overclocking" issues or future 66Mhz PCI. */
549*8044SWilliam.Kucharski@Sun.COM #define mdio_delay() inl(mdio_addr)
550*8044SWilliam.Kucharski@Sun.COM 
551*8044SWilliam.Kucharski@Sun.COM /* Read and write the MII registers using software-generated serial
552*8044SWilliam.Kucharski@Sun.COM    MDIO protocol.  It is just different enough from the EEPROM protocol
553*8044SWilliam.Kucharski@Sun.COM    to not share code.  The maxium data clock rate is 2.5 Mhz. */
554*8044SWilliam.Kucharski@Sun.COM #define MDIO_SHIFT_CLK  0x10000
555*8044SWilliam.Kucharski@Sun.COM #define MDIO_DATA_WRITE0 0x00000
556*8044SWilliam.Kucharski@Sun.COM #define MDIO_DATA_WRITE1 0x20000
557*8044SWilliam.Kucharski@Sun.COM #define MDIO_ENB                0x00000         /* Ignore the 0x02000 databook setting. */
558*8044SWilliam.Kucharski@Sun.COM #define MDIO_ENB_IN             0x40000
559*8044SWilliam.Kucharski@Sun.COM #define MDIO_DATA_READ  0x80000
560*8044SWilliam.Kucharski@Sun.COM 
561*8044SWilliam.Kucharski@Sun.COM /* MII transceiver control section.
562*8044SWilliam.Kucharski@Sun.COM    Read and write the MII registers using software-generated serial
563*8044SWilliam.Kucharski@Sun.COM    MDIO protocol.  See the MII specifications or DP83840A data sheet
564*8044SWilliam.Kucharski@Sun.COM    for details. */
565*8044SWilliam.Kucharski@Sun.COM 
mdio_read(struct nic * nic __unused,int phy_id,int location)566*8044SWilliam.Kucharski@Sun.COM int mdio_read(struct nic *nic __unused, int phy_id, int location)
567*8044SWilliam.Kucharski@Sun.COM {
568*8044SWilliam.Kucharski@Sun.COM     int i;
569*8044SWilliam.Kucharski@Sun.COM     int read_cmd = (0xf6 << 10) | (phy_id << 5) | location;
570*8044SWilliam.Kucharski@Sun.COM     int retval = 0;
571*8044SWilliam.Kucharski@Sun.COM     long mdio_addr = ioaddr + CSR9;
572*8044SWilliam.Kucharski@Sun.COM 
573*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG_WHERE
574*8044SWilliam.Kucharski@Sun.COM     whereami("mdio_read\n");
575*8044SWilliam.Kucharski@Sun.COM #endif
576*8044SWilliam.Kucharski@Sun.COM 
577*8044SWilliam.Kucharski@Sun.COM     if (tp->chip_id == LC82C168) {
578*8044SWilliam.Kucharski@Sun.COM 	int i = 1000;
579*8044SWilliam.Kucharski@Sun.COM 	outl(0x60020000 + (phy_id<<23) + (location<<18), ioaddr + 0xA0);
580*8044SWilliam.Kucharski@Sun.COM 	inl(ioaddr + 0xA0);
581*8044SWilliam.Kucharski@Sun.COM 	inl(ioaddr + 0xA0);
582*8044SWilliam.Kucharski@Sun.COM 	while (--i > 0)
583*8044SWilliam.Kucharski@Sun.COM 	    if ( ! ((retval = inl(ioaddr + 0xA0)) & 0x80000000))
584*8044SWilliam.Kucharski@Sun.COM 		return retval & 0xffff;
585*8044SWilliam.Kucharski@Sun.COM 	return 0xffff;
586*8044SWilliam.Kucharski@Sun.COM     }
587*8044SWilliam.Kucharski@Sun.COM 
588*8044SWilliam.Kucharski@Sun.COM     if (tp->chip_id == COMET) {
589*8044SWilliam.Kucharski@Sun.COM 	if (phy_id == 1) {
590*8044SWilliam.Kucharski@Sun.COM 	    if (location < 7)
591*8044SWilliam.Kucharski@Sun.COM 		return inl(ioaddr + 0xB4 + (location<<2));
592*8044SWilliam.Kucharski@Sun.COM 	    else if (location == 17)
593*8044SWilliam.Kucharski@Sun.COM 		return inl(ioaddr + 0xD0);
594*8044SWilliam.Kucharski@Sun.COM 	    else if (location >= 29 && location <= 31)
595*8044SWilliam.Kucharski@Sun.COM 		return inl(ioaddr + 0xD4 + ((location-29)<<2));
596*8044SWilliam.Kucharski@Sun.COM 	}
597*8044SWilliam.Kucharski@Sun.COM 	return 0xffff;
598*8044SWilliam.Kucharski@Sun.COM     }
599*8044SWilliam.Kucharski@Sun.COM 
600*8044SWilliam.Kucharski@Sun.COM     /* Establish sync by sending at least 32 logic ones. */
601*8044SWilliam.Kucharski@Sun.COM     for (i = 32; i >= 0; i--) {
602*8044SWilliam.Kucharski@Sun.COM 	outl(MDIO_ENB | MDIO_DATA_WRITE1, mdio_addr);
603*8044SWilliam.Kucharski@Sun.COM 	mdio_delay();
604*8044SWilliam.Kucharski@Sun.COM 	outl(MDIO_ENB | MDIO_DATA_WRITE1 | MDIO_SHIFT_CLK, mdio_addr);
605*8044SWilliam.Kucharski@Sun.COM 	mdio_delay();
606*8044SWilliam.Kucharski@Sun.COM     }
607*8044SWilliam.Kucharski@Sun.COM     /* Shift the read command bits out. */
608*8044SWilliam.Kucharski@Sun.COM     for (i = 15; i >= 0; i--) {
609*8044SWilliam.Kucharski@Sun.COM 	int dataval = (read_cmd & (1 << i)) ? MDIO_DATA_WRITE1 : 0;
610*8044SWilliam.Kucharski@Sun.COM 
611*8044SWilliam.Kucharski@Sun.COM 	outl(MDIO_ENB | dataval, mdio_addr);
612*8044SWilliam.Kucharski@Sun.COM 	mdio_delay();
613*8044SWilliam.Kucharski@Sun.COM 	outl(MDIO_ENB | dataval | MDIO_SHIFT_CLK, mdio_addr);
614*8044SWilliam.Kucharski@Sun.COM 	mdio_delay();
615*8044SWilliam.Kucharski@Sun.COM     }
616*8044SWilliam.Kucharski@Sun.COM     /* Read the two transition, 16 data, and wire-idle bits. */
617*8044SWilliam.Kucharski@Sun.COM     for (i = 19; i > 0; i--) {
618*8044SWilliam.Kucharski@Sun.COM 	outl(MDIO_ENB_IN, mdio_addr);
619*8044SWilliam.Kucharski@Sun.COM 	mdio_delay();
620*8044SWilliam.Kucharski@Sun.COM 	retval = (retval << 1) | ((inl(mdio_addr) & MDIO_DATA_READ) ? 1 : 0);
621*8044SWilliam.Kucharski@Sun.COM 	outl(MDIO_ENB_IN | MDIO_SHIFT_CLK, mdio_addr);
622*8044SWilliam.Kucharski@Sun.COM 	mdio_delay();
623*8044SWilliam.Kucharski@Sun.COM     }
624*8044SWilliam.Kucharski@Sun.COM     return (retval>>1) & 0xffff;
625*8044SWilliam.Kucharski@Sun.COM }
626*8044SWilliam.Kucharski@Sun.COM 
mdio_write(struct nic * nic __unused,int phy_id,int location,int value)627*8044SWilliam.Kucharski@Sun.COM void mdio_write(struct nic *nic __unused, int phy_id, int location, int value)
628*8044SWilliam.Kucharski@Sun.COM {
629*8044SWilliam.Kucharski@Sun.COM     int i;
630*8044SWilliam.Kucharski@Sun.COM     int cmd = (0x5002 << 16) | (phy_id << 23) | (location<<18) | value;
631*8044SWilliam.Kucharski@Sun.COM     long mdio_addr = ioaddr + CSR9;
632*8044SWilliam.Kucharski@Sun.COM 
633*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG_WHERE
634*8044SWilliam.Kucharski@Sun.COM     whereami("mdio_write\n");
635*8044SWilliam.Kucharski@Sun.COM #endif
636*8044SWilliam.Kucharski@Sun.COM 
637*8044SWilliam.Kucharski@Sun.COM     if (tp->chip_id == LC82C168) {
638*8044SWilliam.Kucharski@Sun.COM 	int i = 1000;
639*8044SWilliam.Kucharski@Sun.COM 	outl(cmd, ioaddr + 0xA0);
640*8044SWilliam.Kucharski@Sun.COM 	do
641*8044SWilliam.Kucharski@Sun.COM 	    if ( ! (inl(ioaddr + 0xA0) & 0x80000000))
642*8044SWilliam.Kucharski@Sun.COM 		break;
643*8044SWilliam.Kucharski@Sun.COM 	while (--i > 0);
644*8044SWilliam.Kucharski@Sun.COM 	return;
645*8044SWilliam.Kucharski@Sun.COM     }
646*8044SWilliam.Kucharski@Sun.COM 
647*8044SWilliam.Kucharski@Sun.COM     if (tp->chip_id == COMET) {
648*8044SWilliam.Kucharski@Sun.COM 	if (phy_id != 1)
649*8044SWilliam.Kucharski@Sun.COM 	    return;
650*8044SWilliam.Kucharski@Sun.COM 	if (location < 7)
651*8044SWilliam.Kucharski@Sun.COM 	    outl(value, ioaddr + 0xB4 + (location<<2));
652*8044SWilliam.Kucharski@Sun.COM 	else if (location == 17)
653*8044SWilliam.Kucharski@Sun.COM 	    outl(value, ioaddr + 0xD0);
654*8044SWilliam.Kucharski@Sun.COM 	else if (location >= 29 && location <= 31)
655*8044SWilliam.Kucharski@Sun.COM 	    outl(value, ioaddr + 0xD4 + ((location-29)<<2));
656*8044SWilliam.Kucharski@Sun.COM 	return;
657*8044SWilliam.Kucharski@Sun.COM     }
658*8044SWilliam.Kucharski@Sun.COM 
659*8044SWilliam.Kucharski@Sun.COM     /* Establish sync by sending 32 logic ones. */
660*8044SWilliam.Kucharski@Sun.COM     for (i = 32; i >= 0; i--) {
661*8044SWilliam.Kucharski@Sun.COM 	outl(MDIO_ENB | MDIO_DATA_WRITE1, mdio_addr);
662*8044SWilliam.Kucharski@Sun.COM 	mdio_delay();
663*8044SWilliam.Kucharski@Sun.COM 	outl(MDIO_ENB | MDIO_DATA_WRITE1 | MDIO_SHIFT_CLK, mdio_addr);
664*8044SWilliam.Kucharski@Sun.COM 	mdio_delay();
665*8044SWilliam.Kucharski@Sun.COM     }
666*8044SWilliam.Kucharski@Sun.COM     /* Shift the command bits out. */
667*8044SWilliam.Kucharski@Sun.COM     for (i = 31; i >= 0; i--) {
668*8044SWilliam.Kucharski@Sun.COM 	int dataval = (cmd & (1 << i)) ? MDIO_DATA_WRITE1 : 0;
669*8044SWilliam.Kucharski@Sun.COM 	outl(MDIO_ENB | dataval, mdio_addr);
670*8044SWilliam.Kucharski@Sun.COM 	mdio_delay();
671*8044SWilliam.Kucharski@Sun.COM 	outl(MDIO_ENB | dataval | MDIO_SHIFT_CLK, mdio_addr);
672*8044SWilliam.Kucharski@Sun.COM 	mdio_delay();
673*8044SWilliam.Kucharski@Sun.COM     }
674*8044SWilliam.Kucharski@Sun.COM     /* Clear out extra bits. */
675*8044SWilliam.Kucharski@Sun.COM     for (i = 2; i > 0; i--) {
676*8044SWilliam.Kucharski@Sun.COM 	outl(MDIO_ENB_IN, mdio_addr);
677*8044SWilliam.Kucharski@Sun.COM 	mdio_delay();
678*8044SWilliam.Kucharski@Sun.COM 	outl(MDIO_ENB_IN | MDIO_SHIFT_CLK, mdio_addr);
679*8044SWilliam.Kucharski@Sun.COM 	mdio_delay();
680*8044SWilliam.Kucharski@Sun.COM     }
681*8044SWilliam.Kucharski@Sun.COM }
682*8044SWilliam.Kucharski@Sun.COM 
683*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
684*8044SWilliam.Kucharski@Sun.COM /* EEPROM Reading Code                                               */
685*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
686*8044SWilliam.Kucharski@Sun.COM /* EEPROM routines adapted from the Linux Tulip Code */
687*8044SWilliam.Kucharski@Sun.COM /* Reading a serial EEPROM is a "bit" grungy, but we work our way
688*8044SWilliam.Kucharski@Sun.COM    through:->.
689*8044SWilliam.Kucharski@Sun.COM */
read_eeprom(unsigned long ioaddr,int location,int addr_len)690*8044SWilliam.Kucharski@Sun.COM static int read_eeprom(unsigned long ioaddr, int location, int addr_len)
691*8044SWilliam.Kucharski@Sun.COM {
692*8044SWilliam.Kucharski@Sun.COM     int i;
693*8044SWilliam.Kucharski@Sun.COM     unsigned short retval = 0;
694*8044SWilliam.Kucharski@Sun.COM     long ee_addr = ioaddr + CSR9;
695*8044SWilliam.Kucharski@Sun.COM     int read_cmd = location | EE_READ_CMD;
696*8044SWilliam.Kucharski@Sun.COM 
697*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG_WHERE
698*8044SWilliam.Kucharski@Sun.COM     whereami("read_eeprom\n");
699*8044SWilliam.Kucharski@Sun.COM #endif
700*8044SWilliam.Kucharski@Sun.COM 
701*8044SWilliam.Kucharski@Sun.COM     outl(EE_ENB & ~EE_CS, ee_addr);
702*8044SWilliam.Kucharski@Sun.COM     outl(EE_ENB, ee_addr);
703*8044SWilliam.Kucharski@Sun.COM 
704*8044SWilliam.Kucharski@Sun.COM     /* Shift the read command bits out. */
705*8044SWilliam.Kucharski@Sun.COM     for (i = 4 + addr_len; i >= 0; i--) {
706*8044SWilliam.Kucharski@Sun.COM         short dataval = (read_cmd & (1 << i)) ? EE_DATA_WRITE : 0;
707*8044SWilliam.Kucharski@Sun.COM         outl(EE_ENB | dataval, ee_addr);
708*8044SWilliam.Kucharski@Sun.COM         eeprom_delay();
709*8044SWilliam.Kucharski@Sun.COM         outl(EE_ENB | dataval | EE_SHIFT_CLK, ee_addr);
710*8044SWilliam.Kucharski@Sun.COM         eeprom_delay();
711*8044SWilliam.Kucharski@Sun.COM     }
712*8044SWilliam.Kucharski@Sun.COM     outl(EE_ENB, ee_addr);
713*8044SWilliam.Kucharski@Sun.COM 
714*8044SWilliam.Kucharski@Sun.COM     for (i = 16; i > 0; i--) {
715*8044SWilliam.Kucharski@Sun.COM         outl(EE_ENB | EE_SHIFT_CLK, ee_addr);
716*8044SWilliam.Kucharski@Sun.COM         eeprom_delay();
717*8044SWilliam.Kucharski@Sun.COM         retval = (retval << 1) | ((inl(ee_addr) & EE_DATA_READ) ? 1 : 0);
718*8044SWilliam.Kucharski@Sun.COM         outl(EE_ENB, ee_addr);
719*8044SWilliam.Kucharski@Sun.COM         eeprom_delay();
720*8044SWilliam.Kucharski@Sun.COM     }
721*8044SWilliam.Kucharski@Sun.COM 
722*8044SWilliam.Kucharski@Sun.COM     /* Terminate the EEPROM access. */
723*8044SWilliam.Kucharski@Sun.COM     outl(EE_ENB & ~EE_CS, ee_addr);
724*8044SWilliam.Kucharski@Sun.COM     return retval;
725*8044SWilliam.Kucharski@Sun.COM }
726*8044SWilliam.Kucharski@Sun.COM 
727*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
728*8044SWilliam.Kucharski@Sun.COM /* EEPROM Parsing Code                                               */
729*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
parse_eeprom(struct nic * nic)730*8044SWilliam.Kucharski@Sun.COM static void parse_eeprom(struct nic *nic)
731*8044SWilliam.Kucharski@Sun.COM {
732*8044SWilliam.Kucharski@Sun.COM     unsigned char *p, *ee_data = tp->eeprom;
733*8044SWilliam.Kucharski@Sun.COM     int new_advertise = 0;
734*8044SWilliam.Kucharski@Sun.COM     int i;
735*8044SWilliam.Kucharski@Sun.COM 
736*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG_WHERE
737*8044SWilliam.Kucharski@Sun.COM     whereami("parse_eeprom\n");
738*8044SWilliam.Kucharski@Sun.COM #endif
739*8044SWilliam.Kucharski@Sun.COM 
740*8044SWilliam.Kucharski@Sun.COM     tp->mtable = 0;
741*8044SWilliam.Kucharski@Sun.COM     /* Detect an old-style (SA only) EEPROM layout:
742*8044SWilliam.Kucharski@Sun.COM        memcmp(ee_data, ee_data+16, 8). */
743*8044SWilliam.Kucharski@Sun.COM     for (i = 0; i < 8; i ++)
744*8044SWilliam.Kucharski@Sun.COM         if (ee_data[i] != ee_data[16+i])
745*8044SWilliam.Kucharski@Sun.COM             break;
746*8044SWilliam.Kucharski@Sun.COM     if (i >= 8) {
747*8044SWilliam.Kucharski@Sun.COM         /* Do a fix-up based on the vendor half of the station address. */
748*8044SWilliam.Kucharski@Sun.COM         for (i = 0; eeprom_fixups[i].name; i++) {
749*8044SWilliam.Kucharski@Sun.COM             if (nic->node_addr[0] == eeprom_fixups[i].addr0
750*8044SWilliam.Kucharski@Sun.COM                 &&  nic->node_addr[1] == eeprom_fixups[i].addr1
751*8044SWilliam.Kucharski@Sun.COM                 &&  nic->node_addr[2] == eeprom_fixups[i].addr2) {
752*8044SWilliam.Kucharski@Sun.COM                 if (nic->node_addr[2] == 0xE8  &&  ee_data[0x1a] == 0x55)
753*8044SWilliam.Kucharski@Sun.COM                     i++;                /* An Accton EN1207, not an outlaw Maxtech. */
754*8044SWilliam.Kucharski@Sun.COM                 memcpy(ee_data + 26, eeprom_fixups[i].newtable,
755*8044SWilliam.Kucharski@Sun.COM                        sizeof(eeprom_fixups[i].newtable));
756*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
757*8044SWilliam.Kucharski@Sun.COM                 printf("%s: Old format EEPROM on '%s' board.\n%s: Using substitute media control info.\n",
758*8044SWilliam.Kucharski@Sun.COM                        tp->nic_name, eeprom_fixups[i].name, tp->nic_name);
759*8044SWilliam.Kucharski@Sun.COM #endif
760*8044SWilliam.Kucharski@Sun.COM                 break;
761*8044SWilliam.Kucharski@Sun.COM             }
762*8044SWilliam.Kucharski@Sun.COM         }
763*8044SWilliam.Kucharski@Sun.COM         if (eeprom_fixups[i].name == NULL) { /* No fixup found. */
764*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
765*8044SWilliam.Kucharski@Sun.COM             printf("%s: Old style EEPROM with no media selection information.\n",
766*8044SWilliam.Kucharski@Sun.COM                    tp->nic_name);
767*8044SWilliam.Kucharski@Sun.COM #endif
768*8044SWilliam.Kucharski@Sun.COM             return;
769*8044SWilliam.Kucharski@Sun.COM         }
770*8044SWilliam.Kucharski@Sun.COM     }
771*8044SWilliam.Kucharski@Sun.COM 
772*8044SWilliam.Kucharski@Sun.COM     if (ee_data[19] > 1) {
773*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
774*8044SWilliam.Kucharski@Sun.COM         printf("%s:  Multiport cards (%d ports) may not work correctly.\n",
775*8044SWilliam.Kucharski@Sun.COM                tp->nic_name, ee_data[19]);
776*8044SWilliam.Kucharski@Sun.COM #endif
777*8044SWilliam.Kucharski@Sun.COM     }
778*8044SWilliam.Kucharski@Sun.COM 
779*8044SWilliam.Kucharski@Sun.COM     p = (void *)ee_data + ee_data[27];
780*8044SWilliam.Kucharski@Sun.COM 
781*8044SWilliam.Kucharski@Sun.COM     if (ee_data[27] == 0) {             /* No valid media table. */
782*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
783*8044SWilliam.Kucharski@Sun.COM         if (tulip_debug > 1) {
784*8044SWilliam.Kucharski@Sun.COM             printf("%s:  No Valid Media Table. ee_data[27] = %hhX\n",
785*8044SWilliam.Kucharski@Sun.COM                    tp->nic_name, ee_data[27]);
786*8044SWilliam.Kucharski@Sun.COM         }
787*8044SWilliam.Kucharski@Sun.COM #endif
788*8044SWilliam.Kucharski@Sun.COM     } else if (tp->chip_id == DC21041) {
789*8044SWilliam.Kucharski@Sun.COM         int media = get_u16(p);
790*8044SWilliam.Kucharski@Sun.COM         int count = p[2];
791*8044SWilliam.Kucharski@Sun.COM         p += 3;
792*8044SWilliam.Kucharski@Sun.COM 
793*8044SWilliam.Kucharski@Sun.COM         printf("%s: 21041 Media table, default media %hX (%s).\n",
794*8044SWilliam.Kucharski@Sun.COM                tp->nic_name, media,
795*8044SWilliam.Kucharski@Sun.COM                media & 0x0800 ? "Autosense" : medianame[media & 15]);
796*8044SWilliam.Kucharski@Sun.COM         for (i = 0; i < count; i++) {
797*8044SWilliam.Kucharski@Sun.COM             unsigned char media_block = *p++;
798*8044SWilliam.Kucharski@Sun.COM             int media_code = media_block & MEDIA_MASK;
799*8044SWilliam.Kucharski@Sun.COM             if (media_block & 0x40)
800*8044SWilliam.Kucharski@Sun.COM                 p += 6;
801*8044SWilliam.Kucharski@Sun.COM             switch(media_code) {
802*8044SWilliam.Kucharski@Sun.COM             case 0: new_advertise |= 0x0020; break;
803*8044SWilliam.Kucharski@Sun.COM             case 4: new_advertise |= 0x0040; break;
804*8044SWilliam.Kucharski@Sun.COM             }
805*8044SWilliam.Kucharski@Sun.COM             printf("%s:  21041 media #%d, %s.\n",
806*8044SWilliam.Kucharski@Sun.COM                    tp->nic_name, media_code, medianame[media_code]);
807*8044SWilliam.Kucharski@Sun.COM         }
808*8044SWilliam.Kucharski@Sun.COM     } else {
809*8044SWilliam.Kucharski@Sun.COM         unsigned char csr12dir = 0;
810*8044SWilliam.Kucharski@Sun.COM         int count;
811*8044SWilliam.Kucharski@Sun.COM         struct mediatable *mtable;
812*8044SWilliam.Kucharski@Sun.COM         u16 media = get_u16(p);
813*8044SWilliam.Kucharski@Sun.COM 
814*8044SWilliam.Kucharski@Sun.COM         p += 2;
815*8044SWilliam.Kucharski@Sun.COM         if (tp->flags & CSR12_IN_SROM)
816*8044SWilliam.Kucharski@Sun.COM             csr12dir = *p++;
817*8044SWilliam.Kucharski@Sun.COM         count = *p++;
818*8044SWilliam.Kucharski@Sun.COM 
819*8044SWilliam.Kucharski@Sun.COM         tp->mtable = mtable = (struct mediatable *)&tp->media_table_storage[0];
820*8044SWilliam.Kucharski@Sun.COM 
821*8044SWilliam.Kucharski@Sun.COM         mtable->defaultmedia = media;
822*8044SWilliam.Kucharski@Sun.COM         mtable->leafcount = count;
823*8044SWilliam.Kucharski@Sun.COM         mtable->csr12dir = csr12dir;
824*8044SWilliam.Kucharski@Sun.COM         mtable->has_nonmii = mtable->has_mii = mtable->has_reset = 0;
825*8044SWilliam.Kucharski@Sun.COM         mtable->csr15dir = mtable->csr15val = 0;
826*8044SWilliam.Kucharski@Sun.COM 
827*8044SWilliam.Kucharski@Sun.COM         printf("%s:  EEPROM default media type %s.\n", tp->nic_name,
828*8044SWilliam.Kucharski@Sun.COM                media & 0x0800 ? "Autosense" : medianame[media & MEDIA_MASK]);
829*8044SWilliam.Kucharski@Sun.COM 
830*8044SWilliam.Kucharski@Sun.COM         for (i = 0; i < count; i++) {
831*8044SWilliam.Kucharski@Sun.COM             struct medialeaf *leaf = &mtable->mleaf[i];
832*8044SWilliam.Kucharski@Sun.COM 
833*8044SWilliam.Kucharski@Sun.COM             if ((p[0] & 0x80) == 0) { /* 21140 Compact block. */
834*8044SWilliam.Kucharski@Sun.COM                 leaf->type = 0;
835*8044SWilliam.Kucharski@Sun.COM                 leaf->media = p[0] & 0x3f;
836*8044SWilliam.Kucharski@Sun.COM                 leaf->leafdata = p;
837*8044SWilliam.Kucharski@Sun.COM                 if ((p[2] & 0x61) == 0x01)      /* Bogus, but Znyx boards do it. */
838*8044SWilliam.Kucharski@Sun.COM                     mtable->has_mii = 1;
839*8044SWilliam.Kucharski@Sun.COM                 p += 4;
840*8044SWilliam.Kucharski@Sun.COM             } else {
841*8044SWilliam.Kucharski@Sun.COM                 switch(leaf->type = p[1]) {
842*8044SWilliam.Kucharski@Sun.COM                 case 5:
843*8044SWilliam.Kucharski@Sun.COM                     mtable->has_reset = i;
844*8044SWilliam.Kucharski@Sun.COM                     leaf->media = p[2] & 0x0f;
845*8044SWilliam.Kucharski@Sun.COM                     break;
846*8044SWilliam.Kucharski@Sun.COM                 case 1: case 3:
847*8044SWilliam.Kucharski@Sun.COM                     mtable->has_mii = 1;
848*8044SWilliam.Kucharski@Sun.COM                     leaf->media = 11;
849*8044SWilliam.Kucharski@Sun.COM                     break;
850*8044SWilliam.Kucharski@Sun.COM                 case 2:
851*8044SWilliam.Kucharski@Sun.COM                     if ((p[2] & 0x3f) == 0) {
852*8044SWilliam.Kucharski@Sun.COM                         u32 base15 = (p[2] & 0x40) ? get_u16(p + 7) : 0x0008;
853*8044SWilliam.Kucharski@Sun.COM                         u16 *p1 = (u16 *)(p + (p[2] & 0x40 ? 9 : 3));
854*8044SWilliam.Kucharski@Sun.COM                         mtable->csr15dir = (get_unaligned(p1 + 0)<<16) + base15;
855*8044SWilliam.Kucharski@Sun.COM                         mtable->csr15val = (get_unaligned(p1 + 1)<<16) + base15;
856*8044SWilliam.Kucharski@Sun.COM                     }
857*8044SWilliam.Kucharski@Sun.COM                     /* Fall through. */
858*8044SWilliam.Kucharski@Sun.COM                 case 0: case 4:
859*8044SWilliam.Kucharski@Sun.COM                     mtable->has_nonmii = 1;
860*8044SWilliam.Kucharski@Sun.COM                     leaf->media = p[2] & MEDIA_MASK;
861*8044SWilliam.Kucharski@Sun.COM                     switch (leaf->media) {
862*8044SWilliam.Kucharski@Sun.COM                     case 0: new_advertise |= 0x0020; break;
863*8044SWilliam.Kucharski@Sun.COM                     case 4: new_advertise |= 0x0040; break;
864*8044SWilliam.Kucharski@Sun.COM                     case 3: new_advertise |= 0x0080; break;
865*8044SWilliam.Kucharski@Sun.COM                     case 5: new_advertise |= 0x0100; break;
866*8044SWilliam.Kucharski@Sun.COM                     case 6: new_advertise |= 0x0200; break;
867*8044SWilliam.Kucharski@Sun.COM                     }
868*8044SWilliam.Kucharski@Sun.COM                     break;
869*8044SWilliam.Kucharski@Sun.COM                 default:
870*8044SWilliam.Kucharski@Sun.COM                     leaf->media = 19;
871*8044SWilliam.Kucharski@Sun.COM                 }
872*8044SWilliam.Kucharski@Sun.COM                 leaf->leafdata = p + 2;
873*8044SWilliam.Kucharski@Sun.COM                 p += (p[0] & 0x3f) + 1;
874*8044SWilliam.Kucharski@Sun.COM             }
875*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
876*8044SWilliam.Kucharski@Sun.COM             if (tulip_debug > 1  &&  leaf->media == 11) {
877*8044SWilliam.Kucharski@Sun.COM                 unsigned char *bp = leaf->leafdata;
878*8044SWilliam.Kucharski@Sun.COM                 printf("%s:  MII interface PHY %d, setup/reset sequences %d/%d long, capabilities %hhX %hhX.\n",
879*8044SWilliam.Kucharski@Sun.COM                        tp->nic_name, bp[0], bp[1], bp[2 + bp[1]*2],
880*8044SWilliam.Kucharski@Sun.COM                        bp[5 + bp[2 + bp[1]*2]*2], bp[4 + bp[2 + bp[1]*2]*2]);
881*8044SWilliam.Kucharski@Sun.COM             }
882*8044SWilliam.Kucharski@Sun.COM #endif
883*8044SWilliam.Kucharski@Sun.COM             printf("%s:  Index #%d - Media %s (#%d) described "
884*8044SWilliam.Kucharski@Sun.COM                    "by a %s (%d) block.\n",
885*8044SWilliam.Kucharski@Sun.COM                    tp->nic_name, i, medianame[leaf->media], leaf->media,
886*8044SWilliam.Kucharski@Sun.COM                    leaf->type < 6 ? block_name[leaf->type] : "UNKNOWN",
887*8044SWilliam.Kucharski@Sun.COM                    leaf->type);
888*8044SWilliam.Kucharski@Sun.COM         }
889*8044SWilliam.Kucharski@Sun.COM         if (new_advertise)
890*8044SWilliam.Kucharski@Sun.COM             tp->sym_advertise = new_advertise;
891*8044SWilliam.Kucharski@Sun.COM     }
892*8044SWilliam.Kucharski@Sun.COM }
893*8044SWilliam.Kucharski@Sun.COM 
894*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
895*8044SWilliam.Kucharski@Sun.COM /* tulip_init_ring - setup the tx and rx descriptors                */
896*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
tulip_init_ring(struct nic * nic __unused)897*8044SWilliam.Kucharski@Sun.COM static void tulip_init_ring(struct nic *nic __unused)
898*8044SWilliam.Kucharski@Sun.COM {
899*8044SWilliam.Kucharski@Sun.COM     int i;
900*8044SWilliam.Kucharski@Sun.COM 
901*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG_WHERE
902*8044SWilliam.Kucharski@Sun.COM     whereami("tulip_init_ring\n");
903*8044SWilliam.Kucharski@Sun.COM #endif
904*8044SWilliam.Kucharski@Sun.COM 
905*8044SWilliam.Kucharski@Sun.COM     tp->cur_rx = 0;
906*8044SWilliam.Kucharski@Sun.COM 
907*8044SWilliam.Kucharski@Sun.COM     for (i = 0; i < RX_RING_SIZE; i++) {
908*8044SWilliam.Kucharski@Sun.COM 	rx_ring[i].status  = cpu_to_le32(0x80000000);
909*8044SWilliam.Kucharski@Sun.COM 	rx_ring[i].length  = cpu_to_le32(BUFLEN);
910*8044SWilliam.Kucharski@Sun.COM 	rx_ring[i].buffer1 = virt_to_le32desc(&rxb[i * BUFLEN]);
911*8044SWilliam.Kucharski@Sun.COM 	rx_ring[i].buffer2 = virt_to_le32desc(&rx_ring[i+1]);
912*8044SWilliam.Kucharski@Sun.COM     }
913*8044SWilliam.Kucharski@Sun.COM     /* Mark the last entry as wrapping the ring. */
914*8044SWilliam.Kucharski@Sun.COM     rx_ring[i-1].length    = cpu_to_le32(DESC_RING_WRAP | BUFLEN);
915*8044SWilliam.Kucharski@Sun.COM     rx_ring[i-1].buffer2   = virt_to_le32desc(&rx_ring[0]);
916*8044SWilliam.Kucharski@Sun.COM 
917*8044SWilliam.Kucharski@Sun.COM     /* We only use 1 transmit buffer, but we use 2 descriptors so
918*8044SWilliam.Kucharski@Sun.COM        transmit engines have somewhere to point to if they feel the need */
919*8044SWilliam.Kucharski@Sun.COM 
920*8044SWilliam.Kucharski@Sun.COM     tx_ring[0].status  = 0x00000000;
921*8044SWilliam.Kucharski@Sun.COM     tx_ring[0].buffer1 = virt_to_le32desc(&txb[0]);
922*8044SWilliam.Kucharski@Sun.COM     tx_ring[0].buffer2 = virt_to_le32desc(&tx_ring[1]);
923*8044SWilliam.Kucharski@Sun.COM 
924*8044SWilliam.Kucharski@Sun.COM     /* this descriptor should never get used, since it will never be owned
925*8044SWilliam.Kucharski@Sun.COM        by the machine (status will always == 0) */
926*8044SWilliam.Kucharski@Sun.COM     tx_ring[1].status  = 0x00000000;
927*8044SWilliam.Kucharski@Sun.COM     tx_ring[1].buffer1 = virt_to_le32desc(&txb[0]);
928*8044SWilliam.Kucharski@Sun.COM     tx_ring[1].buffer2 = virt_to_le32desc(&tx_ring[0]);
929*8044SWilliam.Kucharski@Sun.COM 
930*8044SWilliam.Kucharski@Sun.COM     /* Mark the last entry as wrapping the ring, though this should never happen */
931*8044SWilliam.Kucharski@Sun.COM     tx_ring[1].length  = cpu_to_le32(DESC_RING_WRAP | BUFLEN);
932*8044SWilliam.Kucharski@Sun.COM }
933*8044SWilliam.Kucharski@Sun.COM 
set_rx_mode(struct nic * nic __unused)934*8044SWilliam.Kucharski@Sun.COM static void set_rx_mode(struct nic *nic __unused) {
935*8044SWilliam.Kucharski@Sun.COM 	int csr6 = inl(ioaddr + CSR6) & ~0x00D5;
936*8044SWilliam.Kucharski@Sun.COM 
937*8044SWilliam.Kucharski@Sun.COM 	tp->csr6 &= ~0x00D5;
938*8044SWilliam.Kucharski@Sun.COM 
939*8044SWilliam.Kucharski@Sun.COM 	/* !IFF_PROMISC */
940*8044SWilliam.Kucharski@Sun.COM 	tp->csr6 |= AcceptAllMulticast;
941*8044SWilliam.Kucharski@Sun.COM 	csr6 |= AcceptAllMulticast;
942*8044SWilliam.Kucharski@Sun.COM 
943*8044SWilliam.Kucharski@Sun.COM 	outl(csr6, ioaddr + CSR6);
944*8044SWilliam.Kucharski@Sun.COM 
945*8044SWilliam.Kucharski@Sun.COM 
946*8044SWilliam.Kucharski@Sun.COM 
947*8044SWilliam.Kucharski@Sun.COM }
948*8044SWilliam.Kucharski@Sun.COM 
949*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
950*8044SWilliam.Kucharski@Sun.COM /* eth_reset - Reset adapter                                         */
951*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
tulip_reset(struct nic * nic)952*8044SWilliam.Kucharski@Sun.COM static void tulip_reset(struct nic *nic)
953*8044SWilliam.Kucharski@Sun.COM {
954*8044SWilliam.Kucharski@Sun.COM     int i;
955*8044SWilliam.Kucharski@Sun.COM     unsigned long to;
956*8044SWilliam.Kucharski@Sun.COM 
957*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG_WHERE
958*8044SWilliam.Kucharski@Sun.COM     whereami("tulip_reset\n");
959*8044SWilliam.Kucharski@Sun.COM #endif
960*8044SWilliam.Kucharski@Sun.COM 
961*8044SWilliam.Kucharski@Sun.COM     /* Stop Tx and RX */
962*8044SWilliam.Kucharski@Sun.COM     outl(inl(ioaddr + CSR6) & ~0x00002002, ioaddr + CSR6);
963*8044SWilliam.Kucharski@Sun.COM 
964*8044SWilliam.Kucharski@Sun.COM     /* On some chip revs we must set the MII/SYM port before the reset!? */
965*8044SWilliam.Kucharski@Sun.COM     if (tp->mii_cnt  ||  (tp->mtable  &&  tp->mtable->has_mii)) {
966*8044SWilliam.Kucharski@Sun.COM 	outl(0x814C0000, ioaddr + CSR6);
967*8044SWilliam.Kucharski@Sun.COM     }
968*8044SWilliam.Kucharski@Sun.COM 
969*8044SWilliam.Kucharski@Sun.COM     /* Reset the chip, holding bit 0 set at least 50 PCI cycles. */
970*8044SWilliam.Kucharski@Sun.COM     outl(0x00000001, ioaddr + CSR0);
971*8044SWilliam.Kucharski@Sun.COM     tulip_wait(1);
972*8044SWilliam.Kucharski@Sun.COM 
973*8044SWilliam.Kucharski@Sun.COM     /* turn off reset and set cache align=16lword, burst=unlimit */
974*8044SWilliam.Kucharski@Sun.COM     outl(tp->csr0, ioaddr + CSR0);
975*8044SWilliam.Kucharski@Sun.COM 
976*8044SWilliam.Kucharski@Sun.COM     /*  Wait the specified 50 PCI cycles after a reset */
977*8044SWilliam.Kucharski@Sun.COM     tulip_wait(1);
978*8044SWilliam.Kucharski@Sun.COM 
979*8044SWilliam.Kucharski@Sun.COM     /* set up transmit and receive descriptors */
980*8044SWilliam.Kucharski@Sun.COM     tulip_init_ring(nic);
981*8044SWilliam.Kucharski@Sun.COM 
982*8044SWilliam.Kucharski@Sun.COM     if (tp->chip_id == PNIC2) {
983*8044SWilliam.Kucharski@Sun.COM         u32 addr_high = (nic->node_addr[1]<<8) + (nic->node_addr[0]<<0);
984*8044SWilliam.Kucharski@Sun.COM         /* This address setting does not appear to impact chip operation?? */
985*8044SWilliam.Kucharski@Sun.COM         outl((nic->node_addr[5]<<8) + nic->node_addr[4] +
986*8044SWilliam.Kucharski@Sun.COM              (nic->node_addr[3]<<24) + (nic->node_addr[2]<<16),
987*8044SWilliam.Kucharski@Sun.COM              ioaddr + 0xB0);
988*8044SWilliam.Kucharski@Sun.COM         outl(addr_high + (addr_high<<16), ioaddr + 0xB8);
989*8044SWilliam.Kucharski@Sun.COM     }
990*8044SWilliam.Kucharski@Sun.COM 
991*8044SWilliam.Kucharski@Sun.COM     /* MC_HASH_ONLY boards don't support setup packets */
992*8044SWilliam.Kucharski@Sun.COM     if (tp->flags & MC_HASH_ONLY) {
993*8044SWilliam.Kucharski@Sun.COM         u32 addr_low = cpu_to_le32(get_unaligned((u32 *)nic->node_addr));
994*8044SWilliam.Kucharski@Sun.COM         u32 addr_high = cpu_to_le32(get_unaligned((u16 *)(nic->node_addr+4)));
995*8044SWilliam.Kucharski@Sun.COM 
996*8044SWilliam.Kucharski@Sun.COM 	/* clear multicast hash filters and setup MAC address filters */
997*8044SWilliam.Kucharski@Sun.COM 	if (tp->flags & IS_ASIX) {
998*8044SWilliam.Kucharski@Sun.COM             outl(0, ioaddr + CSR13);
999*8044SWilliam.Kucharski@Sun.COM             outl(addr_low,  ioaddr + CSR14);
1000*8044SWilliam.Kucharski@Sun.COM             outl(1, ioaddr + CSR13);
1001*8044SWilliam.Kucharski@Sun.COM             outl(addr_high, ioaddr + CSR14);
1002*8044SWilliam.Kucharski@Sun.COM 	    outl(2, ioaddr + CSR13);
1003*8044SWilliam.Kucharski@Sun.COM 	    outl(0, ioaddr + CSR14);
1004*8044SWilliam.Kucharski@Sun.COM 	    outl(3, ioaddr + CSR13);
1005*8044SWilliam.Kucharski@Sun.COM 	    outl(0, ioaddr + CSR14);
1006*8044SWilliam.Kucharski@Sun.COM 	} else if (tp->chip_id == COMET) {
1007*8044SWilliam.Kucharski@Sun.COM             outl(addr_low,  ioaddr + 0xA4);
1008*8044SWilliam.Kucharski@Sun.COM             outl(addr_high, ioaddr + 0xA8);
1009*8044SWilliam.Kucharski@Sun.COM             outl(0, ioaddr + 0xAC);
1010*8044SWilliam.Kucharski@Sun.COM             outl(0, ioaddr + 0xB0);
1011*8044SWilliam.Kucharski@Sun.COM 	}
1012*8044SWilliam.Kucharski@Sun.COM     } else {
1013*8044SWilliam.Kucharski@Sun.COM 	/* for other boards we send a setup packet to initialize
1014*8044SWilliam.Kucharski@Sun.COM 	   the filters */
1015*8044SWilliam.Kucharski@Sun.COM 	u32 tx_flags = 0x08000000 | 192;
1016*8044SWilliam.Kucharski@Sun.COM 
1017*8044SWilliam.Kucharski@Sun.COM 	/* construct perfect filter frame with mac address as first match
1018*8044SWilliam.Kucharski@Sun.COM 	   and broadcast address for all others */
1019*8044SWilliam.Kucharski@Sun.COM 	for (i=0; i<192; i++)
1020*8044SWilliam.Kucharski@Sun.COM 	    txb[i] = 0xFF;
1021*8044SWilliam.Kucharski@Sun.COM 	txb[0] = nic->node_addr[0];
1022*8044SWilliam.Kucharski@Sun.COM 	txb[1] = nic->node_addr[1];
1023*8044SWilliam.Kucharski@Sun.COM 	txb[4] = nic->node_addr[2];
1024*8044SWilliam.Kucharski@Sun.COM 	txb[5] = nic->node_addr[3];
1025*8044SWilliam.Kucharski@Sun.COM 	txb[8] = nic->node_addr[4];
1026*8044SWilliam.Kucharski@Sun.COM 	txb[9] = nic->node_addr[5];
1027*8044SWilliam.Kucharski@Sun.COM 
1028*8044SWilliam.Kucharski@Sun.COM 	tx_ring[0].length  = cpu_to_le32(tx_flags);
1029*8044SWilliam.Kucharski@Sun.COM 	tx_ring[0].buffer1 = virt_to_le32desc(&txb[0]);
1030*8044SWilliam.Kucharski@Sun.COM 	tx_ring[0].status  = cpu_to_le32(0x80000000);
1031*8044SWilliam.Kucharski@Sun.COM     }
1032*8044SWilliam.Kucharski@Sun.COM 
1033*8044SWilliam.Kucharski@Sun.COM     /* Point to rx and tx descriptors */
1034*8044SWilliam.Kucharski@Sun.COM     outl(virt_to_le32desc(&rx_ring[0]), ioaddr + CSR3);
1035*8044SWilliam.Kucharski@Sun.COM     outl(virt_to_le32desc(&tx_ring[0]), ioaddr + CSR4);
1036*8044SWilliam.Kucharski@Sun.COM 
1037*8044SWilliam.Kucharski@Sun.COM     init_media(nic);
1038*8044SWilliam.Kucharski@Sun.COM 
1039*8044SWilliam.Kucharski@Sun.COM     /* set the chip's operating mode (but don't turn on xmit and recv yet) */
1040*8044SWilliam.Kucharski@Sun.COM     outl((tp->csr6 & ~0x00002002), ioaddr + CSR6);
1041*8044SWilliam.Kucharski@Sun.COM 
1042*8044SWilliam.Kucharski@Sun.COM     /* send setup packet for cards that support it */
1043*8044SWilliam.Kucharski@Sun.COM     if (!(tp->flags & MC_HASH_ONLY)) {
1044*8044SWilliam.Kucharski@Sun.COM 	/* enable transmit  wait for completion */
1045*8044SWilliam.Kucharski@Sun.COM 	outl(tp->csr6 | 0x00002000, ioaddr + CSR6);
1046*8044SWilliam.Kucharski@Sun.COM 	/* immediate transmit demand */
1047*8044SWilliam.Kucharski@Sun.COM 	outl(0, ioaddr + CSR1);
1048*8044SWilliam.Kucharski@Sun.COM 
1049*8044SWilliam.Kucharski@Sun.COM 	to = currticks() + TX_TIME_OUT;
1050*8044SWilliam.Kucharski@Sun.COM 	while ((tx_ring[0].status & 0x80000000) && (currticks() < to))
1051*8044SWilliam.Kucharski@Sun.COM 	    /* wait */ ;
1052*8044SWilliam.Kucharski@Sun.COM 
1053*8044SWilliam.Kucharski@Sun.COM 	if (currticks() >= to) {
1054*8044SWilliam.Kucharski@Sun.COM 	    printf ("%s: TX Setup Timeout.\n", tp->nic_name);
1055*8044SWilliam.Kucharski@Sun.COM 	}
1056*8044SWilliam.Kucharski@Sun.COM     }
1057*8044SWilliam.Kucharski@Sun.COM 
1058*8044SWilliam.Kucharski@Sun.COM     if (tp->chip_id == LC82C168)
1059*8044SWilliam.Kucharski@Sun.COM 	tulip_check_duplex(nic);
1060*8044SWilliam.Kucharski@Sun.COM 
1061*8044SWilliam.Kucharski@Sun.COM     set_rx_mode(nic);
1062*8044SWilliam.Kucharski@Sun.COM 
1063*8044SWilliam.Kucharski@Sun.COM     /* enable transmit and receive */
1064*8044SWilliam.Kucharski@Sun.COM     outl(tp->csr6 | 0x00002002, ioaddr + CSR6);
1065*8044SWilliam.Kucharski@Sun.COM }
1066*8044SWilliam.Kucharski@Sun.COM 
1067*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
1068*8044SWilliam.Kucharski@Sun.COM /* eth_transmit - Transmit a frame                                   */
1069*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
tulip_transmit(struct nic * nic,const char * d,unsigned int t,unsigned int s,const char * p)1070*8044SWilliam.Kucharski@Sun.COM static void tulip_transmit(struct nic *nic, const char *d, unsigned int t,
1071*8044SWilliam.Kucharski@Sun.COM                            unsigned int s, const char *p)
1072*8044SWilliam.Kucharski@Sun.COM {
1073*8044SWilliam.Kucharski@Sun.COM     u16 nstype;
1074*8044SWilliam.Kucharski@Sun.COM     u32 to;
1075*8044SWilliam.Kucharski@Sun.COM     u32 csr6 = inl(ioaddr + CSR6);
1076*8044SWilliam.Kucharski@Sun.COM 
1077*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG_WHERE
1078*8044SWilliam.Kucharski@Sun.COM     whereami("tulip_transmit\n");
1079*8044SWilliam.Kucharski@Sun.COM #endif
1080*8044SWilliam.Kucharski@Sun.COM 
1081*8044SWilliam.Kucharski@Sun.COM     /* Disable Tx */
1082*8044SWilliam.Kucharski@Sun.COM     outl(csr6 & ~0x00002000, ioaddr + CSR6);
1083*8044SWilliam.Kucharski@Sun.COM 
1084*8044SWilliam.Kucharski@Sun.COM     memcpy(txb, d, ETH_ALEN);
1085*8044SWilliam.Kucharski@Sun.COM     memcpy(txb + ETH_ALEN, nic->node_addr, ETH_ALEN);
1086*8044SWilliam.Kucharski@Sun.COM     nstype = htons((u16) t);
1087*8044SWilliam.Kucharski@Sun.COM     memcpy(txb + 2 * ETH_ALEN, (u8 *)&nstype, 2);
1088*8044SWilliam.Kucharski@Sun.COM     memcpy(txb + ETH_HLEN, p, s);
1089*8044SWilliam.Kucharski@Sun.COM 
1090*8044SWilliam.Kucharski@Sun.COM     s += ETH_HLEN;
1091*8044SWilliam.Kucharski@Sun.COM     s &= 0x0FFF;
1092*8044SWilliam.Kucharski@Sun.COM 
1093*8044SWilliam.Kucharski@Sun.COM     /* pad to minimum packet size */
1094*8044SWilliam.Kucharski@Sun.COM     while (s < ETH_ZLEN)
1095*8044SWilliam.Kucharski@Sun.COM         txb[s++] = '\0';
1096*8044SWilliam.Kucharski@Sun.COM 
1097*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
1098*8044SWilliam.Kucharski@Sun.COM     if (tulip_debug > 1)
1099*8044SWilliam.Kucharski@Sun.COM 	printf("%s: sending %d bytes ethtype %hX\n", tp->nic_name, s, t);
1100*8044SWilliam.Kucharski@Sun.COM #endif
1101*8044SWilliam.Kucharski@Sun.COM 
1102*8044SWilliam.Kucharski@Sun.COM     /* setup the transmit descriptor */
1103*8044SWilliam.Kucharski@Sun.COM     /* 0x60000000 = no interrupt on completion */
1104*8044SWilliam.Kucharski@Sun.COM     tx_ring[0].length = cpu_to_le32(0x60000000 | s);
1105*8044SWilliam.Kucharski@Sun.COM     tx_ring[0].status = cpu_to_le32(0x80000000);
1106*8044SWilliam.Kucharski@Sun.COM 
1107*8044SWilliam.Kucharski@Sun.COM     /* Point to transmit descriptor */
1108*8044SWilliam.Kucharski@Sun.COM     outl(virt_to_le32desc(&tx_ring[0]), ioaddr + CSR4);
1109*8044SWilliam.Kucharski@Sun.COM 
1110*8044SWilliam.Kucharski@Sun.COM     /* Enable Tx */
1111*8044SWilliam.Kucharski@Sun.COM     outl(csr6 | 0x00002000, ioaddr + CSR6);
1112*8044SWilliam.Kucharski@Sun.COM     /* immediate transmit demand */
1113*8044SWilliam.Kucharski@Sun.COM     outl(0, ioaddr + CSR1);
1114*8044SWilliam.Kucharski@Sun.COM 
1115*8044SWilliam.Kucharski@Sun.COM     to = currticks() + TX_TIME_OUT;
1116*8044SWilliam.Kucharski@Sun.COM     while ((tx_ring[0].status & 0x80000000) && (currticks() < to))
1117*8044SWilliam.Kucharski@Sun.COM         /* wait */ ;
1118*8044SWilliam.Kucharski@Sun.COM 
1119*8044SWilliam.Kucharski@Sun.COM     if (currticks() >= to) {
1120*8044SWilliam.Kucharski@Sun.COM         printf ("TX Timeout!\n");
1121*8044SWilliam.Kucharski@Sun.COM     }
1122*8044SWilliam.Kucharski@Sun.COM 
1123*8044SWilliam.Kucharski@Sun.COM     /* Disable Tx */
1124*8044SWilliam.Kucharski@Sun.COM     outl(csr6 & ~0x00002000, ioaddr + CSR6);
1125*8044SWilliam.Kucharski@Sun.COM }
1126*8044SWilliam.Kucharski@Sun.COM 
1127*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
1128*8044SWilliam.Kucharski@Sun.COM /* eth_poll - Wait for a frame                                       */
1129*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
tulip_poll(struct nic * nic,int retrieve)1130*8044SWilliam.Kucharski@Sun.COM static int tulip_poll(struct nic *nic, int retrieve)
1131*8044SWilliam.Kucharski@Sun.COM {
1132*8044SWilliam.Kucharski@Sun.COM 
1133*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG_WHERE
1134*8044SWilliam.Kucharski@Sun.COM     whereami("tulip_poll\n");
1135*8044SWilliam.Kucharski@Sun.COM #endif
1136*8044SWilliam.Kucharski@Sun.COM 
1137*8044SWilliam.Kucharski@Sun.COM     /* no packet waiting. packet still owned by NIC */
1138*8044SWilliam.Kucharski@Sun.COM     if (rx_ring[tp->cur_rx].status & 0x80000000)
1139*8044SWilliam.Kucharski@Sun.COM         return 0;
1140*8044SWilliam.Kucharski@Sun.COM 
1141*8044SWilliam.Kucharski@Sun.COM     if ( ! retrieve ) return 1;
1142*8044SWilliam.Kucharski@Sun.COM 
1143*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG_WHERE
1144*8044SWilliam.Kucharski@Sun.COM     whereami("tulip_poll got one\n");
1145*8044SWilliam.Kucharski@Sun.COM #endif
1146*8044SWilliam.Kucharski@Sun.COM 
1147*8044SWilliam.Kucharski@Sun.COM     nic->packetlen = (rx_ring[tp->cur_rx].status & 0x3FFF0000) >> 16;
1148*8044SWilliam.Kucharski@Sun.COM 
1149*8044SWilliam.Kucharski@Sun.COM     /* if we get a corrupted packet. throw it away and move on */
1150*8044SWilliam.Kucharski@Sun.COM     if (rx_ring[tp->cur_rx].status & 0x00008000) {
1151*8044SWilliam.Kucharski@Sun.COM 	/* return the descriptor and buffer to receive ring */
1152*8044SWilliam.Kucharski@Sun.COM         rx_ring[tp->cur_rx].status = 0x80000000;
1153*8044SWilliam.Kucharski@Sun.COM 	tp->cur_rx = (++tp->cur_rx) % RX_RING_SIZE;
1154*8044SWilliam.Kucharski@Sun.COM         return 0;
1155*8044SWilliam.Kucharski@Sun.COM     }
1156*8044SWilliam.Kucharski@Sun.COM 
1157*8044SWilliam.Kucharski@Sun.COM     /* copy packet to working buffer */
1158*8044SWilliam.Kucharski@Sun.COM     memcpy(nic->packet, rxb + tp->cur_rx * BUFLEN, nic->packetlen);
1159*8044SWilliam.Kucharski@Sun.COM 
1160*8044SWilliam.Kucharski@Sun.COM     /* return the descriptor and buffer to receive ring */
1161*8044SWilliam.Kucharski@Sun.COM     rx_ring[tp->cur_rx].status = 0x80000000;
1162*8044SWilliam.Kucharski@Sun.COM     tp->cur_rx = (++tp->cur_rx) % RX_RING_SIZE;
1163*8044SWilliam.Kucharski@Sun.COM 
1164*8044SWilliam.Kucharski@Sun.COM     return 1;
1165*8044SWilliam.Kucharski@Sun.COM }
1166*8044SWilliam.Kucharski@Sun.COM 
1167*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
1168*8044SWilliam.Kucharski@Sun.COM /* eth_disable - Disable the interface                               */
1169*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
tulip_disable(struct dev * dev)1170*8044SWilliam.Kucharski@Sun.COM static void tulip_disable(struct dev *dev)
1171*8044SWilliam.Kucharski@Sun.COM {
1172*8044SWilliam.Kucharski@Sun.COM     struct nic *nic = (struct nic *)dev;
1173*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG_WHERE
1174*8044SWilliam.Kucharski@Sun.COM     whereami("tulip_disable\n");
1175*8044SWilliam.Kucharski@Sun.COM #endif
1176*8044SWilliam.Kucharski@Sun.COM 
1177*8044SWilliam.Kucharski@Sun.COM     /* merge reset and disable */
1178*8044SWilliam.Kucharski@Sun.COM     tulip_reset(nic);
1179*8044SWilliam.Kucharski@Sun.COM 
1180*8044SWilliam.Kucharski@Sun.COM     /* disable interrupts */
1181*8044SWilliam.Kucharski@Sun.COM     outl(0x00000000, ioaddr + CSR7);
1182*8044SWilliam.Kucharski@Sun.COM 
1183*8044SWilliam.Kucharski@Sun.COM     /* Stop the chip's Tx and Rx processes. */
1184*8044SWilliam.Kucharski@Sun.COM     outl(inl(ioaddr + CSR6) & ~0x00002002, ioaddr + CSR6);
1185*8044SWilliam.Kucharski@Sun.COM 
1186*8044SWilliam.Kucharski@Sun.COM     /* Clear the missed-packet counter. */
1187*8044SWilliam.Kucharski@Sun.COM     (volatile unsigned long)inl(ioaddr + CSR8);
1188*8044SWilliam.Kucharski@Sun.COM }
1189*8044SWilliam.Kucharski@Sun.COM 
1190*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
1191*8044SWilliam.Kucharski@Sun.COM /*IRQ - Enable, Disable, or Force interrupts                         */
1192*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
tulip_irq(struct nic * nic __unused,irq_action_t action __unused)1193*8044SWilliam.Kucharski@Sun.COM static void tulip_irq(struct nic *nic __unused, irq_action_t action __unused)
1194*8044SWilliam.Kucharski@Sun.COM {
1195*8044SWilliam.Kucharski@Sun.COM   switch ( action ) {
1196*8044SWilliam.Kucharski@Sun.COM   case DISABLE :
1197*8044SWilliam.Kucharski@Sun.COM     break;
1198*8044SWilliam.Kucharski@Sun.COM   case ENABLE :
1199*8044SWilliam.Kucharski@Sun.COM     break;
1200*8044SWilliam.Kucharski@Sun.COM   case FORCE :
1201*8044SWilliam.Kucharski@Sun.COM     break;
1202*8044SWilliam.Kucharski@Sun.COM   }
1203*8044SWilliam.Kucharski@Sun.COM }
1204*8044SWilliam.Kucharski@Sun.COM 
1205*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
1206*8044SWilliam.Kucharski@Sun.COM /* eth_probe - Look for an adapter                                   */
1207*8044SWilliam.Kucharski@Sun.COM /*********************************************************************/
tulip_probe(struct dev * dev,struct pci_device * pci)1208*8044SWilliam.Kucharski@Sun.COM static int tulip_probe(struct dev *dev, struct pci_device *pci)
1209*8044SWilliam.Kucharski@Sun.COM {
1210*8044SWilliam.Kucharski@Sun.COM     struct nic *nic = (struct nic *)dev;
1211*8044SWilliam.Kucharski@Sun.COM     u32 i;
1212*8044SWilliam.Kucharski@Sun.COM     u8  chip_rev;
1213*8044SWilliam.Kucharski@Sun.COM     u8 ee_data[EEPROM_SIZE];
1214*8044SWilliam.Kucharski@Sun.COM     unsigned short sum;
1215*8044SWilliam.Kucharski@Sun.COM     int chip_idx;
1216*8044SWilliam.Kucharski@Sun.COM     static unsigned char last_phys_addr[ETH_ALEN] = {0x00, 'L', 'i', 'n', 'u', 'x'};
1217*8044SWilliam.Kucharski@Sun.COM 
1218*8044SWilliam.Kucharski@Sun.COM     if (pci->ioaddr == 0)
1219*8044SWilliam.Kucharski@Sun.COM         return 0;
1220*8044SWilliam.Kucharski@Sun.COM 
1221*8044SWilliam.Kucharski@Sun.COM     ioaddr         = pci->ioaddr;
1222*8044SWilliam.Kucharski@Sun.COM     nic->ioaddr    = pci->ioaddr & ~3;
1223*8044SWilliam.Kucharski@Sun.COM     nic->irqno     = 0;
1224*8044SWilliam.Kucharski@Sun.COM 
1225*8044SWilliam.Kucharski@Sun.COM     /* point to private storage */
1226*8044SWilliam.Kucharski@Sun.COM     tp = &tpx;
1227*8044SWilliam.Kucharski@Sun.COM 
1228*8044SWilliam.Kucharski@Sun.COM     tp->vendor_id  = pci->vendor;
1229*8044SWilliam.Kucharski@Sun.COM     tp->dev_id     = pci->dev_id;
1230*8044SWilliam.Kucharski@Sun.COM     tp->nic_name   = pci->name;
1231*8044SWilliam.Kucharski@Sun.COM 
1232*8044SWilliam.Kucharski@Sun.COM     tp->if_port = 0;
1233*8044SWilliam.Kucharski@Sun.COM     tp->default_port = 0;
1234*8044SWilliam.Kucharski@Sun.COM 
1235*8044SWilliam.Kucharski@Sun.COM     adjust_pci_device(pci);
1236*8044SWilliam.Kucharski@Sun.COM 
1237*8044SWilliam.Kucharski@Sun.COM     /* disable interrupts */
1238*8044SWilliam.Kucharski@Sun.COM     outl(0x00000000, ioaddr + CSR7);
1239*8044SWilliam.Kucharski@Sun.COM 
1240*8044SWilliam.Kucharski@Sun.COM     /* Stop the chip's Tx and Rx processes. */
1241*8044SWilliam.Kucharski@Sun.COM     outl(inl(ioaddr + CSR6) & ~0x00002002, ioaddr + CSR6);
1242*8044SWilliam.Kucharski@Sun.COM 
1243*8044SWilliam.Kucharski@Sun.COM     /* Clear the missed-packet counter. */
1244*8044SWilliam.Kucharski@Sun.COM     (volatile unsigned long)inl(ioaddr + CSR8);
1245*8044SWilliam.Kucharski@Sun.COM 
1246*8044SWilliam.Kucharski@Sun.COM     printf("\n");                /* so we start on a fresh line */
1247*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG_WHERE
1248*8044SWilliam.Kucharski@Sun.COM     whereami("tulip_probe\n");
1249*8044SWilliam.Kucharski@Sun.COM #endif
1250*8044SWilliam.Kucharski@Sun.COM 
1251*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
1252*8044SWilliam.Kucharski@Sun.COM     if (tulip_debug > 1)
1253*8044SWilliam.Kucharski@Sun.COM 	printf ("%s: Looking for Tulip Chip: Vendor=%hX  Device=%hX\n", tp->nic_name,
1254*8044SWilliam.Kucharski@Sun.COM 		tp->vendor_id, tp->dev_id);
1255*8044SWilliam.Kucharski@Sun.COM #endif
1256*8044SWilliam.Kucharski@Sun.COM 
1257*8044SWilliam.Kucharski@Sun.COM     /* Figure out which chip we're dealing with */
1258*8044SWilliam.Kucharski@Sun.COM     i = 0;
1259*8044SWilliam.Kucharski@Sun.COM     chip_idx = -1;
1260*8044SWilliam.Kucharski@Sun.COM 
1261*8044SWilliam.Kucharski@Sun.COM     while (pci_id_tbl[i].name) {
1262*8044SWilliam.Kucharski@Sun.COM         if ( (((u32) tp->dev_id << 16) | tp->vendor_id) ==
1263*8044SWilliam.Kucharski@Sun.COM              (pci_id_tbl[i].id.pci & pci_id_tbl[i].id.pci_mask) ) {
1264*8044SWilliam.Kucharski@Sun.COM             chip_idx = pci_id_tbl[i].drv_flags;
1265*8044SWilliam.Kucharski@Sun.COM             break;
1266*8044SWilliam.Kucharski@Sun.COM         }
1267*8044SWilliam.Kucharski@Sun.COM         i++;
1268*8044SWilliam.Kucharski@Sun.COM     }
1269*8044SWilliam.Kucharski@Sun.COM 
1270*8044SWilliam.Kucharski@Sun.COM     if (chip_idx == -1) {
1271*8044SWilliam.Kucharski@Sun.COM         printf ("%s: Unknown Tulip Chip: Vendor=%hX  Device=%hX\n", tp->nic_name,
1272*8044SWilliam.Kucharski@Sun.COM                 tp->vendor_id, tp->dev_id);
1273*8044SWilliam.Kucharski@Sun.COM         return 0;
1274*8044SWilliam.Kucharski@Sun.COM     }
1275*8044SWilliam.Kucharski@Sun.COM 
1276*8044SWilliam.Kucharski@Sun.COM     tp->pci_id_idx = i;
1277*8044SWilliam.Kucharski@Sun.COM     tp->flags = tulip_tbl[chip_idx].flags;
1278*8044SWilliam.Kucharski@Sun.COM 
1279*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
1280*8044SWilliam.Kucharski@Sun.COM     if (tulip_debug > 1) {
1281*8044SWilliam.Kucharski@Sun.COM 	printf ("%s: tp->pci_id_idx == %d,  name == %s\n", tp->nic_name,
1282*8044SWilliam.Kucharski@Sun.COM 		tp->pci_id_idx, pci_id_tbl[tp->pci_id_idx].name);
1283*8044SWilliam.Kucharski@Sun.COM 	printf ("%s: chip_idx == %d, name == %s\n", tp->nic_name, chip_idx,
1284*8044SWilliam.Kucharski@Sun.COM 		tulip_tbl[chip_idx].chip_name);
1285*8044SWilliam.Kucharski@Sun.COM     }
1286*8044SWilliam.Kucharski@Sun.COM #endif
1287*8044SWilliam.Kucharski@Sun.COM 
1288*8044SWilliam.Kucharski@Sun.COM     /* Bring the 21041/21143 out of sleep mode.
1289*8044SWilliam.Kucharski@Sun.COM        Caution: Snooze mode does not work with some boards! */
1290*8044SWilliam.Kucharski@Sun.COM     if (tp->flags & HAS_PWRDWN)
1291*8044SWilliam.Kucharski@Sun.COM         pcibios_write_config_dword(pci->bus, pci->devfn, 0x40, 0x00000000);
1292*8044SWilliam.Kucharski@Sun.COM 
1293*8044SWilliam.Kucharski@Sun.COM     if (inl(ioaddr + CSR5) == 0xFFFFFFFF) {
1294*8044SWilliam.Kucharski@Sun.COM         printf("%s: The Tulip chip at %X is not functioning.\n",
1295*8044SWilliam.Kucharski@Sun.COM                tp->nic_name, ioaddr);
1296*8044SWilliam.Kucharski@Sun.COM         return 0;
1297*8044SWilliam.Kucharski@Sun.COM     }
1298*8044SWilliam.Kucharski@Sun.COM 
1299*8044SWilliam.Kucharski@Sun.COM     pcibios_read_config_byte(pci->bus, pci->devfn, PCI_REVISION, &chip_rev);
1300*8044SWilliam.Kucharski@Sun.COM 
1301*8044SWilliam.Kucharski@Sun.COM     printf("%s: [chip: %s] rev %d at %hX\n", tp->nic_name,
1302*8044SWilliam.Kucharski@Sun.COM            tulip_tbl[chip_idx].chip_name, chip_rev, ioaddr);
1303*8044SWilliam.Kucharski@Sun.COM     printf("%s: Vendor=%hX  Device=%hX", tp->nic_name, tp->vendor_id, tp->dev_id);
1304*8044SWilliam.Kucharski@Sun.COM 
1305*8044SWilliam.Kucharski@Sun.COM     if (chip_idx == DC21041  &&  inl(ioaddr + CSR9) & 0x8000) {
1306*8044SWilliam.Kucharski@Sun.COM         printf(" 21040 compatible mode.");
1307*8044SWilliam.Kucharski@Sun.COM         chip_idx = DC21040;
1308*8044SWilliam.Kucharski@Sun.COM     }
1309*8044SWilliam.Kucharski@Sun.COM 
1310*8044SWilliam.Kucharski@Sun.COM     printf("\n");
1311*8044SWilliam.Kucharski@Sun.COM 
1312*8044SWilliam.Kucharski@Sun.COM     /* The SROM/EEPROM interface varies dramatically. */
1313*8044SWilliam.Kucharski@Sun.COM     sum = 0;
1314*8044SWilliam.Kucharski@Sun.COM     if (chip_idx == DC21040) {
1315*8044SWilliam.Kucharski@Sun.COM         outl(0, ioaddr + CSR9);         /* Reset the pointer with a dummy write. */
1316*8044SWilliam.Kucharski@Sun.COM         for (i = 0; i < ETH_ALEN; i++) {
1317*8044SWilliam.Kucharski@Sun.COM             int value, boguscnt = 100000;
1318*8044SWilliam.Kucharski@Sun.COM             do
1319*8044SWilliam.Kucharski@Sun.COM                 value = inl(ioaddr + CSR9);
1320*8044SWilliam.Kucharski@Sun.COM             while (value < 0  && --boguscnt > 0);
1321*8044SWilliam.Kucharski@Sun.COM             nic->node_addr[i] = value;
1322*8044SWilliam.Kucharski@Sun.COM             sum += value & 0xff;
1323*8044SWilliam.Kucharski@Sun.COM         }
1324*8044SWilliam.Kucharski@Sun.COM     } else if (chip_idx == LC82C168) {
1325*8044SWilliam.Kucharski@Sun.COM         for (i = 0; i < 3; i++) {
1326*8044SWilliam.Kucharski@Sun.COM             int value, boguscnt = 100000;
1327*8044SWilliam.Kucharski@Sun.COM             outl(0x600 | i, ioaddr + 0x98);
1328*8044SWilliam.Kucharski@Sun.COM             do
1329*8044SWilliam.Kucharski@Sun.COM                 value = inl(ioaddr + CSR9);
1330*8044SWilliam.Kucharski@Sun.COM             while (value < 0  && --boguscnt > 0);
1331*8044SWilliam.Kucharski@Sun.COM             put_unaligned(le16_to_cpu(value), ((u16*)nic->node_addr) + i);
1332*8044SWilliam.Kucharski@Sun.COM             sum += value & 0xffff;
1333*8044SWilliam.Kucharski@Sun.COM         }
1334*8044SWilliam.Kucharski@Sun.COM     } else if (chip_idx == COMET) {
1335*8044SWilliam.Kucharski@Sun.COM         /* No need to read the EEPROM. */
1336*8044SWilliam.Kucharski@Sun.COM         put_unaligned(inl(ioaddr + 0xA4), (u32 *)nic->node_addr);
1337*8044SWilliam.Kucharski@Sun.COM         put_unaligned(inl(ioaddr + 0xA8), (u16 *)(nic->node_addr + 4));
1338*8044SWilliam.Kucharski@Sun.COM         for (i = 0; i < ETH_ALEN; i ++)
1339*8044SWilliam.Kucharski@Sun.COM             sum += nic->node_addr[i];
1340*8044SWilliam.Kucharski@Sun.COM     } else {
1341*8044SWilliam.Kucharski@Sun.COM         /* A serial EEPROM interface, we read now and sort it out later. */
1342*8044SWilliam.Kucharski@Sun.COM         int sa_offset = 0;
1343*8044SWilliam.Kucharski@Sun.COM         int ee_addr_size = read_eeprom(ioaddr, 0xff, 8) & 0x40000 ? 8 : 6;
1344*8044SWilliam.Kucharski@Sun.COM 
1345*8044SWilliam.Kucharski@Sun.COM         for (i = 0; i < sizeof(ee_data)/2; i++)
1346*8044SWilliam.Kucharski@Sun.COM             ((u16 *)ee_data)[i] =
1347*8044SWilliam.Kucharski@Sun.COM                 le16_to_cpu(read_eeprom(ioaddr, i, ee_addr_size));
1348*8044SWilliam.Kucharski@Sun.COM 
1349*8044SWilliam.Kucharski@Sun.COM         /* DEC now has a specification (see Notes) but early board makers
1350*8044SWilliam.Kucharski@Sun.COM            just put the address in the first EEPROM locations. */
1351*8044SWilliam.Kucharski@Sun.COM         /* This does  memcmp(eedata, eedata+16, 8) */
1352*8044SWilliam.Kucharski@Sun.COM         for (i = 0; i < 8; i ++)
1353*8044SWilliam.Kucharski@Sun.COM             if (ee_data[i] != ee_data[16+i])
1354*8044SWilliam.Kucharski@Sun.COM                 sa_offset = 20;
1355*8044SWilliam.Kucharski@Sun.COM         if (ee_data[0] == 0xff  &&  ee_data[1] == 0xff &&  ee_data[2] == 0) {
1356*8044SWilliam.Kucharski@Sun.COM             sa_offset = 2;              /* Grrr, damn Matrox boards. */
1357*8044SWilliam.Kucharski@Sun.COM         }
1358*8044SWilliam.Kucharski@Sun.COM         for (i = 0; i < ETH_ALEN; i ++) {
1359*8044SWilliam.Kucharski@Sun.COM             nic->node_addr[i] = ee_data[i + sa_offset];
1360*8044SWilliam.Kucharski@Sun.COM             sum += ee_data[i + sa_offset];
1361*8044SWilliam.Kucharski@Sun.COM         }
1362*8044SWilliam.Kucharski@Sun.COM     }
1363*8044SWilliam.Kucharski@Sun.COM     /* Lite-On boards have the address byte-swapped. */
1364*8044SWilliam.Kucharski@Sun.COM     if ((nic->node_addr[0] == 0xA0  ||  nic->node_addr[0] == 0xC0)
1365*8044SWilliam.Kucharski@Sun.COM         &&  nic->node_addr[1] == 0x00)
1366*8044SWilliam.Kucharski@Sun.COM         for (i = 0; i < ETH_ALEN; i+=2) {
1367*8044SWilliam.Kucharski@Sun.COM             char tmp = nic->node_addr[i];
1368*8044SWilliam.Kucharski@Sun.COM             nic->node_addr[i] = nic->node_addr[i+1];
1369*8044SWilliam.Kucharski@Sun.COM             nic->node_addr[i+1] = tmp;
1370*8044SWilliam.Kucharski@Sun.COM         }
1371*8044SWilliam.Kucharski@Sun.COM 
1372*8044SWilliam.Kucharski@Sun.COM     if (sum == 0  || sum == ETH_ALEN*0xff) {
1373*8044SWilliam.Kucharski@Sun.COM         printf("%s: EEPROM not present!\n", tp->nic_name);
1374*8044SWilliam.Kucharski@Sun.COM         for (i = 0; i < ETH_ALEN-1; i++)
1375*8044SWilliam.Kucharski@Sun.COM             nic->node_addr[i] = last_phys_addr[i];
1376*8044SWilliam.Kucharski@Sun.COM         nic->node_addr[i] = last_phys_addr[i] + 1;
1377*8044SWilliam.Kucharski@Sun.COM     }
1378*8044SWilliam.Kucharski@Sun.COM 
1379*8044SWilliam.Kucharski@Sun.COM     for (i = 0; i < ETH_ALEN; i++)
1380*8044SWilliam.Kucharski@Sun.COM         last_phys_addr[i] = nic->node_addr[i];
1381*8044SWilliam.Kucharski@Sun.COM 
1382*8044SWilliam.Kucharski@Sun.COM     printf("%s: %! at ioaddr %hX\n", tp->nic_name, nic->node_addr, ioaddr);
1383*8044SWilliam.Kucharski@Sun.COM 
1384*8044SWilliam.Kucharski@Sun.COM     tp->chip_id = chip_idx;
1385*8044SWilliam.Kucharski@Sun.COM     tp->revision = chip_rev;
1386*8044SWilliam.Kucharski@Sun.COM     tp->csr0 = csr0;
1387*8044SWilliam.Kucharski@Sun.COM 
1388*8044SWilliam.Kucharski@Sun.COM     /* BugFixes: The 21143-TD hangs with PCI Write-and-Invalidate cycles.
1389*8044SWilliam.Kucharski@Sun.COM        And the ASIX must have a burst limit or horrible things happen. */
1390*8044SWilliam.Kucharski@Sun.COM     if (chip_idx == DC21143  &&  chip_rev == 65)
1391*8044SWilliam.Kucharski@Sun.COM         tp->csr0 &= ~0x01000000;
1392*8044SWilliam.Kucharski@Sun.COM     else if (tp->flags & IS_ASIX)
1393*8044SWilliam.Kucharski@Sun.COM         tp->csr0 |= 0x2000;
1394*8044SWilliam.Kucharski@Sun.COM 
1395*8044SWilliam.Kucharski@Sun.COM     if (media_cap[tp->default_port] & MediaIsMII) {
1396*8044SWilliam.Kucharski@Sun.COM         u16 media2advert[] = { 0x20, 0x40, 0x03e0, 0x60, 0x80, 0x100, 0x200 };
1397*8044SWilliam.Kucharski@Sun.COM         tp->mii_advertise = media2advert[tp->default_port - 9];
1398*8044SWilliam.Kucharski@Sun.COM         tp->mii_advertise |= (tp->flags & HAS_8023X); /* Matching bits! */
1399*8044SWilliam.Kucharski@Sun.COM     }
1400*8044SWilliam.Kucharski@Sun.COM 
1401*8044SWilliam.Kucharski@Sun.COM     /* This is logically part of the probe routine, but too complex
1402*8044SWilliam.Kucharski@Sun.COM        to write inline. */
1403*8044SWilliam.Kucharski@Sun.COM     if (tp->flags & HAS_MEDIA_TABLE) {
1404*8044SWilliam.Kucharski@Sun.COM         memcpy(tp->eeprom, ee_data, sizeof(tp->eeprom));
1405*8044SWilliam.Kucharski@Sun.COM         parse_eeprom(nic);
1406*8044SWilliam.Kucharski@Sun.COM     }
1407*8044SWilliam.Kucharski@Sun.COM 
1408*8044SWilliam.Kucharski@Sun.COM     start_link(nic);
1409*8044SWilliam.Kucharski@Sun.COM 
1410*8044SWilliam.Kucharski@Sun.COM     /* reset the device and make ready for tx and rx of packets */
1411*8044SWilliam.Kucharski@Sun.COM     tulip_reset(nic);
1412*8044SWilliam.Kucharski@Sun.COM 
1413*8044SWilliam.Kucharski@Sun.COM     dev->disable  = tulip_disable;
1414*8044SWilliam.Kucharski@Sun.COM     nic->poll     = tulip_poll;
1415*8044SWilliam.Kucharski@Sun.COM     nic->transmit = tulip_transmit;
1416*8044SWilliam.Kucharski@Sun.COM     nic->irq      = tulip_irq;
1417*8044SWilliam.Kucharski@Sun.COM 
1418*8044SWilliam.Kucharski@Sun.COM     /* give the board a chance to reset before returning */
1419*8044SWilliam.Kucharski@Sun.COM     tulip_wait(4*TICKS_PER_SEC);
1420*8044SWilliam.Kucharski@Sun.COM 
1421*8044SWilliam.Kucharski@Sun.COM     return 1;
1422*8044SWilliam.Kucharski@Sun.COM }
1423*8044SWilliam.Kucharski@Sun.COM 
start_link(struct nic * nic)1424*8044SWilliam.Kucharski@Sun.COM static void start_link(struct nic *nic)
1425*8044SWilliam.Kucharski@Sun.COM {
1426*8044SWilliam.Kucharski@Sun.COM     int i;
1427*8044SWilliam.Kucharski@Sun.COM 
1428*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG_WHERE
1429*8044SWilliam.Kucharski@Sun.COM     whereami("start_link\n");
1430*8044SWilliam.Kucharski@Sun.COM #endif
1431*8044SWilliam.Kucharski@Sun.COM 
1432*8044SWilliam.Kucharski@Sun.COM     if ((tp->flags & ALWAYS_CHECK_MII) ||
1433*8044SWilliam.Kucharski@Sun.COM         (tp->mtable  &&  tp->mtable->has_mii) ||
1434*8044SWilliam.Kucharski@Sun.COM         ( ! tp->mtable  &&  (tp->flags & HAS_MII))) {
1435*8044SWilliam.Kucharski@Sun.COM         unsigned int phy, phy_idx;
1436*8044SWilliam.Kucharski@Sun.COM         if (tp->mtable  &&  tp->mtable->has_mii) {
1437*8044SWilliam.Kucharski@Sun.COM             for (i = 0; i < tp->mtable->leafcount; i++)
1438*8044SWilliam.Kucharski@Sun.COM                 if (tp->mtable->mleaf[i].media == 11) {
1439*8044SWilliam.Kucharski@Sun.COM                     tp->cur_index = i;
1440*8044SWilliam.Kucharski@Sun.COM                     tp->saved_if_port = tp->if_port;
1441*8044SWilliam.Kucharski@Sun.COM                     select_media(nic, 2);
1442*8044SWilliam.Kucharski@Sun.COM                     tp->if_port = tp->saved_if_port;
1443*8044SWilliam.Kucharski@Sun.COM                     break;
1444*8044SWilliam.Kucharski@Sun.COM                 }
1445*8044SWilliam.Kucharski@Sun.COM         }
1446*8044SWilliam.Kucharski@Sun.COM 
1447*8044SWilliam.Kucharski@Sun.COM         /* Find the connected MII xcvrs. */
1448*8044SWilliam.Kucharski@Sun.COM         for (phy = 0, phy_idx = 0; phy < 32 && phy_idx < sizeof(tp->phys);
1449*8044SWilliam.Kucharski@Sun.COM              phy++) {
1450*8044SWilliam.Kucharski@Sun.COM             int mii_status = mdio_read(nic, phy, 1);
1451*8044SWilliam.Kucharski@Sun.COM             if ((mii_status & 0x8301) == 0x8001 ||
1452*8044SWilliam.Kucharski@Sun.COM                 ((mii_status & 0x8000) == 0  && (mii_status & 0x7800) != 0)) {
1453*8044SWilliam.Kucharski@Sun.COM                 int mii_reg0 = mdio_read(nic, phy, 0);
1454*8044SWilliam.Kucharski@Sun.COM                 int mii_advert = mdio_read(nic, phy, 4);
1455*8044SWilliam.Kucharski@Sun.COM                 int to_advert;
1456*8044SWilliam.Kucharski@Sun.COM 
1457*8044SWilliam.Kucharski@Sun.COM                 if (tp->mii_advertise)
1458*8044SWilliam.Kucharski@Sun.COM                     to_advert = tp->mii_advertise;
1459*8044SWilliam.Kucharski@Sun.COM                 else if (tp->advertising[phy_idx])
1460*8044SWilliam.Kucharski@Sun.COM                     to_advert = tp->advertising[phy_idx];
1461*8044SWilliam.Kucharski@Sun.COM                 else                    /* Leave unchanged. */
1462*8044SWilliam.Kucharski@Sun.COM                     tp->mii_advertise = to_advert = mii_advert;
1463*8044SWilliam.Kucharski@Sun.COM 
1464*8044SWilliam.Kucharski@Sun.COM                 tp->phys[phy_idx++] = phy;
1465*8044SWilliam.Kucharski@Sun.COM                 printf("%s:  MII transceiver %d config %hX status %hX advertising %hX.\n",
1466*8044SWilliam.Kucharski@Sun.COM                        tp->nic_name, phy, mii_reg0, mii_status, mii_advert);
1467*8044SWilliam.Kucharski@Sun.COM                                 /* Fixup for DLink with miswired PHY. */
1468*8044SWilliam.Kucharski@Sun.COM                 if (mii_advert != to_advert) {
1469*8044SWilliam.Kucharski@Sun.COM                     printf("%s:  Advertising %hX on PHY %d previously advertising %hX.\n",
1470*8044SWilliam.Kucharski@Sun.COM                            tp->nic_name, to_advert, phy, mii_advert);
1471*8044SWilliam.Kucharski@Sun.COM                     mdio_write(nic, phy, 4, to_advert);
1472*8044SWilliam.Kucharski@Sun.COM                 }
1473*8044SWilliam.Kucharski@Sun.COM                                 /* Enable autonegotiation: some boards default to off. */
1474*8044SWilliam.Kucharski@Sun.COM                 mdio_write(nic, phy, 0, mii_reg0 |
1475*8044SWilliam.Kucharski@Sun.COM                            (tp->full_duplex ? 0x1100 : 0x1000) |
1476*8044SWilliam.Kucharski@Sun.COM                            (media_cap[tp->default_port]&MediaIs100 ? 0x2000:0));
1477*8044SWilliam.Kucharski@Sun.COM             }
1478*8044SWilliam.Kucharski@Sun.COM         }
1479*8044SWilliam.Kucharski@Sun.COM         tp->mii_cnt = phy_idx;
1480*8044SWilliam.Kucharski@Sun.COM         if (tp->mtable  &&  tp->mtable->has_mii  &&  phy_idx == 0) {
1481*8044SWilliam.Kucharski@Sun.COM             printf("%s: ***WARNING***: No MII transceiver found!\n",
1482*8044SWilliam.Kucharski@Sun.COM                    tp->nic_name);
1483*8044SWilliam.Kucharski@Sun.COM             tp->phys[0] = 1;
1484*8044SWilliam.Kucharski@Sun.COM         }
1485*8044SWilliam.Kucharski@Sun.COM     }
1486*8044SWilliam.Kucharski@Sun.COM 
1487*8044SWilliam.Kucharski@Sun.COM     /* Reset the xcvr interface and turn on heartbeat. */
1488*8044SWilliam.Kucharski@Sun.COM     switch (tp->chip_id) {
1489*8044SWilliam.Kucharski@Sun.COM     case DC21040:
1490*8044SWilliam.Kucharski@Sun.COM         outl(0x00000000, ioaddr + CSR13);
1491*8044SWilliam.Kucharski@Sun.COM         outl(0x00000004, ioaddr + CSR13);
1492*8044SWilliam.Kucharski@Sun.COM         break;
1493*8044SWilliam.Kucharski@Sun.COM     case DC21041:
1494*8044SWilliam.Kucharski@Sun.COM         /* This is nway_start(). */
1495*8044SWilliam.Kucharski@Sun.COM         if (tp->sym_advertise == 0)
1496*8044SWilliam.Kucharski@Sun.COM             tp->sym_advertise = 0x0061;
1497*8044SWilliam.Kucharski@Sun.COM         outl(0x00000000, ioaddr + CSR13);
1498*8044SWilliam.Kucharski@Sun.COM         outl(0xFFFFFFFF, ioaddr + CSR14);
1499*8044SWilliam.Kucharski@Sun.COM         outl(0x00000008, ioaddr + CSR15); /* Listen on AUI also. */
1500*8044SWilliam.Kucharski@Sun.COM         outl(inl(ioaddr + CSR6) | 0x0200, ioaddr + CSR6);
1501*8044SWilliam.Kucharski@Sun.COM         outl(0x0000EF01, ioaddr + CSR13);
1502*8044SWilliam.Kucharski@Sun.COM         break;
1503*8044SWilliam.Kucharski@Sun.COM     case DC21140: default:
1504*8044SWilliam.Kucharski@Sun.COM         if (tp->mtable)
1505*8044SWilliam.Kucharski@Sun.COM             outl(tp->mtable->csr12dir | 0x100, ioaddr + CSR12);
1506*8044SWilliam.Kucharski@Sun.COM         break;
1507*8044SWilliam.Kucharski@Sun.COM     case DC21142:
1508*8044SWilliam.Kucharski@Sun.COM     case PNIC2:
1509*8044SWilliam.Kucharski@Sun.COM         if (tp->mii_cnt  ||  media_cap[tp->if_port] & MediaIsMII) {
1510*8044SWilliam.Kucharski@Sun.COM             outl(0x82020000, ioaddr + CSR6);
1511*8044SWilliam.Kucharski@Sun.COM             outl(0x0000, ioaddr + CSR13);
1512*8044SWilliam.Kucharski@Sun.COM             outl(0x0000, ioaddr + CSR14);
1513*8044SWilliam.Kucharski@Sun.COM             outl(0x820E0000, ioaddr + CSR6);
1514*8044SWilliam.Kucharski@Sun.COM         } else
1515*8044SWilliam.Kucharski@Sun.COM             nway_start(nic);
1516*8044SWilliam.Kucharski@Sun.COM         break;
1517*8044SWilliam.Kucharski@Sun.COM     case LC82C168:
1518*8044SWilliam.Kucharski@Sun.COM         if ( ! tp->mii_cnt) {
1519*8044SWilliam.Kucharski@Sun.COM             tp->nway = 1;
1520*8044SWilliam.Kucharski@Sun.COM             tp->nwayset = 0;
1521*8044SWilliam.Kucharski@Sun.COM             outl(0x00420000, ioaddr + CSR6);
1522*8044SWilliam.Kucharski@Sun.COM             outl(0x30, ioaddr + CSR12);
1523*8044SWilliam.Kucharski@Sun.COM             outl(0x0001F078, ioaddr + 0xB8);
1524*8044SWilliam.Kucharski@Sun.COM             outl(0x0201F078, ioaddr + 0xB8); /* Turn on autonegotiation. */
1525*8044SWilliam.Kucharski@Sun.COM         }
1526*8044SWilliam.Kucharski@Sun.COM         break;
1527*8044SWilliam.Kucharski@Sun.COM     case MX98713: case COMPEX9881:
1528*8044SWilliam.Kucharski@Sun.COM         outl(0x00000000, ioaddr + CSR6);
1529*8044SWilliam.Kucharski@Sun.COM         outl(0x000711C0, ioaddr + CSR14); /* Turn on NWay. */
1530*8044SWilliam.Kucharski@Sun.COM         outl(0x00000001, ioaddr + CSR13);
1531*8044SWilliam.Kucharski@Sun.COM         break;
1532*8044SWilliam.Kucharski@Sun.COM     case MX98715: case MX98725:
1533*8044SWilliam.Kucharski@Sun.COM         outl(0x01a80000, ioaddr + CSR6);
1534*8044SWilliam.Kucharski@Sun.COM         outl(0xFFFFFFFF, ioaddr + CSR14);
1535*8044SWilliam.Kucharski@Sun.COM         outl(0x00001000, ioaddr + CSR12);
1536*8044SWilliam.Kucharski@Sun.COM         break;
1537*8044SWilliam.Kucharski@Sun.COM     case COMET:
1538*8044SWilliam.Kucharski@Sun.COM         /* No initialization necessary. */
1539*8044SWilliam.Kucharski@Sun.COM         break;
1540*8044SWilliam.Kucharski@Sun.COM     }
1541*8044SWilliam.Kucharski@Sun.COM }
1542*8044SWilliam.Kucharski@Sun.COM 
nway_start(struct nic * nic __unused)1543*8044SWilliam.Kucharski@Sun.COM static void nway_start(struct nic *nic __unused)
1544*8044SWilliam.Kucharski@Sun.COM {
1545*8044SWilliam.Kucharski@Sun.COM     int csr14 = ((tp->sym_advertise & 0x0780) << 9)  |
1546*8044SWilliam.Kucharski@Sun.COM         ((tp->sym_advertise&0x0020)<<1) | 0xffbf;
1547*8044SWilliam.Kucharski@Sun.COM 
1548*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG_WHERE
1549*8044SWilliam.Kucharski@Sun.COM     whereami("nway_start\n");
1550*8044SWilliam.Kucharski@Sun.COM #endif
1551*8044SWilliam.Kucharski@Sun.COM 
1552*8044SWilliam.Kucharski@Sun.COM     tp->if_port = 0;
1553*8044SWilliam.Kucharski@Sun.COM     tp->nway = tp->mediasense = 1;
1554*8044SWilliam.Kucharski@Sun.COM     tp->nwayset = tp->lpar = 0;
1555*8044SWilliam.Kucharski@Sun.COM     if (tp->chip_id == PNIC2) {
1556*8044SWilliam.Kucharski@Sun.COM         tp->csr6 = 0x01000000 | (tp->sym_advertise & 0x0040 ? 0x0200 : 0);
1557*8044SWilliam.Kucharski@Sun.COM         return;
1558*8044SWilliam.Kucharski@Sun.COM     }
1559*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
1560*8044SWilliam.Kucharski@Sun.COM     if (tulip_debug > 1)
1561*8044SWilliam.Kucharski@Sun.COM         printf("%s: Restarting internal NWay autonegotiation, %X.\n",
1562*8044SWilliam.Kucharski@Sun.COM                tp->nic_name, csr14);
1563*8044SWilliam.Kucharski@Sun.COM #endif
1564*8044SWilliam.Kucharski@Sun.COM     outl(0x0001, ioaddr + CSR13);
1565*8044SWilliam.Kucharski@Sun.COM     outl(csr14, ioaddr + CSR14);
1566*8044SWilliam.Kucharski@Sun.COM     tp->csr6 = 0x82420000 | (tp->sym_advertise & 0x0040 ? 0x0200 : 0);
1567*8044SWilliam.Kucharski@Sun.COM     outl(tp->csr6, ioaddr + CSR6);
1568*8044SWilliam.Kucharski@Sun.COM     if (tp->mtable  &&  tp->mtable->csr15dir) {
1569*8044SWilliam.Kucharski@Sun.COM         outl(tp->mtable->csr15dir, ioaddr + CSR15);
1570*8044SWilliam.Kucharski@Sun.COM         outl(tp->mtable->csr15val, ioaddr + CSR15);
1571*8044SWilliam.Kucharski@Sun.COM     } else if (tp->chip_id != PNIC2)
1572*8044SWilliam.Kucharski@Sun.COM         outw(0x0008, ioaddr + CSR15);
1573*8044SWilliam.Kucharski@Sun.COM     if (tp->chip_id == DC21041)                 /* Trigger NWAY. */
1574*8044SWilliam.Kucharski@Sun.COM         outl(0xEF01, ioaddr + CSR12);
1575*8044SWilliam.Kucharski@Sun.COM     else
1576*8044SWilliam.Kucharski@Sun.COM         outl(0x1301, ioaddr + CSR12);
1577*8044SWilliam.Kucharski@Sun.COM }
1578*8044SWilliam.Kucharski@Sun.COM 
init_media(struct nic * nic)1579*8044SWilliam.Kucharski@Sun.COM static void init_media(struct nic *nic)
1580*8044SWilliam.Kucharski@Sun.COM {
1581*8044SWilliam.Kucharski@Sun.COM     int i;
1582*8044SWilliam.Kucharski@Sun.COM 
1583*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG_WHERE
1584*8044SWilliam.Kucharski@Sun.COM     whereami("init_media\n");
1585*8044SWilliam.Kucharski@Sun.COM #endif
1586*8044SWilliam.Kucharski@Sun.COM 
1587*8044SWilliam.Kucharski@Sun.COM     tp->saved_if_port = tp->if_port;
1588*8044SWilliam.Kucharski@Sun.COM     if (tp->if_port == 0)
1589*8044SWilliam.Kucharski@Sun.COM         tp->if_port = tp->default_port;
1590*8044SWilliam.Kucharski@Sun.COM 
1591*8044SWilliam.Kucharski@Sun.COM     /* Allow selecting a default media. */
1592*8044SWilliam.Kucharski@Sun.COM     i = 0;
1593*8044SWilliam.Kucharski@Sun.COM     if (tp->mtable == NULL)
1594*8044SWilliam.Kucharski@Sun.COM         goto media_picked;
1595*8044SWilliam.Kucharski@Sun.COM     if (tp->if_port) {
1596*8044SWilliam.Kucharski@Sun.COM         int looking_for = media_cap[tp->if_port] & MediaIsMII ? 11 :
1597*8044SWilliam.Kucharski@Sun.COM             (tp->if_port == 12 ? 0 : tp->if_port);
1598*8044SWilliam.Kucharski@Sun.COM         for (i = 0; i < tp->mtable->leafcount; i++)
1599*8044SWilliam.Kucharski@Sun.COM             if (tp->mtable->mleaf[i].media == looking_for) {
1600*8044SWilliam.Kucharski@Sun.COM                 printf("%s: Using user-specified media %s.\n",
1601*8044SWilliam.Kucharski@Sun.COM                        tp->nic_name, medianame[tp->if_port]);
1602*8044SWilliam.Kucharski@Sun.COM                 goto media_picked;
1603*8044SWilliam.Kucharski@Sun.COM             }
1604*8044SWilliam.Kucharski@Sun.COM     }
1605*8044SWilliam.Kucharski@Sun.COM     if ((tp->mtable->defaultmedia & 0x0800) == 0) {
1606*8044SWilliam.Kucharski@Sun.COM         int looking_for = tp->mtable->defaultmedia & 15;
1607*8044SWilliam.Kucharski@Sun.COM         for (i = 0; i < tp->mtable->leafcount; i++)
1608*8044SWilliam.Kucharski@Sun.COM             if (tp->mtable->mleaf[i].media == looking_for) {
1609*8044SWilliam.Kucharski@Sun.COM                 printf("%s: Using EEPROM-set media %s.\n",
1610*8044SWilliam.Kucharski@Sun.COM                        tp->nic_name, medianame[looking_for]);
1611*8044SWilliam.Kucharski@Sun.COM                 goto media_picked;
1612*8044SWilliam.Kucharski@Sun.COM             }
1613*8044SWilliam.Kucharski@Sun.COM     }
1614*8044SWilliam.Kucharski@Sun.COM     /* Start sensing first non-full-duplex media. */
1615*8044SWilliam.Kucharski@Sun.COM     for (i = tp->mtable->leafcount - 1;
1616*8044SWilliam.Kucharski@Sun.COM          (media_cap[tp->mtable->mleaf[i].media] & MediaAlwaysFD) && i > 0; i--)
1617*8044SWilliam.Kucharski@Sun.COM         ;
1618*8044SWilliam.Kucharski@Sun.COM  media_picked:
1619*8044SWilliam.Kucharski@Sun.COM 
1620*8044SWilliam.Kucharski@Sun.COM     tp->csr6 = 0;
1621*8044SWilliam.Kucharski@Sun.COM     tp->cur_index = i;
1622*8044SWilliam.Kucharski@Sun.COM     tp->nwayset = 0;
1623*8044SWilliam.Kucharski@Sun.COM 
1624*8044SWilliam.Kucharski@Sun.COM     if (tp->if_port) {
1625*8044SWilliam.Kucharski@Sun.COM         if (tp->chip_id == DC21143  &&  media_cap[tp->if_port] & MediaIsMII) {
1626*8044SWilliam.Kucharski@Sun.COM             /* We must reset the media CSRs when we force-select MII mode. */
1627*8044SWilliam.Kucharski@Sun.COM             outl(0x0000, ioaddr + CSR13);
1628*8044SWilliam.Kucharski@Sun.COM             outl(0x0000, ioaddr + CSR14);
1629*8044SWilliam.Kucharski@Sun.COM             outl(0x0008, ioaddr + CSR15);
1630*8044SWilliam.Kucharski@Sun.COM         }
1631*8044SWilliam.Kucharski@Sun.COM         select_media(nic, 1);
1632*8044SWilliam.Kucharski@Sun.COM         return;
1633*8044SWilliam.Kucharski@Sun.COM     }
1634*8044SWilliam.Kucharski@Sun.COM     switch(tp->chip_id) {
1635*8044SWilliam.Kucharski@Sun.COM     case DC21041:
1636*8044SWilliam.Kucharski@Sun.COM         /* tp->nway = 1;*/
1637*8044SWilliam.Kucharski@Sun.COM         nway_start(nic);
1638*8044SWilliam.Kucharski@Sun.COM         break;
1639*8044SWilliam.Kucharski@Sun.COM     case DC21142:
1640*8044SWilliam.Kucharski@Sun.COM         if (tp->mii_cnt) {
1641*8044SWilliam.Kucharski@Sun.COM             select_media(nic, 1);
1642*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
1643*8044SWilliam.Kucharski@Sun.COM             if (tulip_debug > 1)
1644*8044SWilliam.Kucharski@Sun.COM                 printf("%s: Using MII transceiver %d, status %hX.\n",
1645*8044SWilliam.Kucharski@Sun.COM                        tp->nic_name, tp->phys[0], mdio_read(nic, tp->phys[0], 1));
1646*8044SWilliam.Kucharski@Sun.COM #endif
1647*8044SWilliam.Kucharski@Sun.COM             outl(0x82020000, ioaddr + CSR6);
1648*8044SWilliam.Kucharski@Sun.COM             tp->csr6 = 0x820E0000;
1649*8044SWilliam.Kucharski@Sun.COM             tp->if_port = 11;
1650*8044SWilliam.Kucharski@Sun.COM             outl(0x0000, ioaddr + CSR13);
1651*8044SWilliam.Kucharski@Sun.COM             outl(0x0000, ioaddr + CSR14);
1652*8044SWilliam.Kucharski@Sun.COM         } else
1653*8044SWilliam.Kucharski@Sun.COM             nway_start(nic);
1654*8044SWilliam.Kucharski@Sun.COM         break;
1655*8044SWilliam.Kucharski@Sun.COM     case PNIC2:
1656*8044SWilliam.Kucharski@Sun.COM         nway_start(nic);
1657*8044SWilliam.Kucharski@Sun.COM         break;
1658*8044SWilliam.Kucharski@Sun.COM     case LC82C168:
1659*8044SWilliam.Kucharski@Sun.COM         if (tp->mii_cnt) {
1660*8044SWilliam.Kucharski@Sun.COM             tp->if_port = 11;
1661*8044SWilliam.Kucharski@Sun.COM             tp->csr6 = 0x814C0000 | (tp->full_duplex ? 0x0200 : 0);
1662*8044SWilliam.Kucharski@Sun.COM             outl(0x0001, ioaddr + CSR15);
1663*8044SWilliam.Kucharski@Sun.COM         } else if (inl(ioaddr + CSR5) & TPLnkPass)
1664*8044SWilliam.Kucharski@Sun.COM             pnic_do_nway(nic);
1665*8044SWilliam.Kucharski@Sun.COM         else {
1666*8044SWilliam.Kucharski@Sun.COM             /* Start with 10mbps to do autonegotiation. */
1667*8044SWilliam.Kucharski@Sun.COM             outl(0x32, ioaddr + CSR12);
1668*8044SWilliam.Kucharski@Sun.COM             tp->csr6 = 0x00420000;
1669*8044SWilliam.Kucharski@Sun.COM             outl(0x0001B078, ioaddr + 0xB8);
1670*8044SWilliam.Kucharski@Sun.COM             outl(0x0201B078, ioaddr + 0xB8);
1671*8044SWilliam.Kucharski@Sun.COM         }
1672*8044SWilliam.Kucharski@Sun.COM         break;
1673*8044SWilliam.Kucharski@Sun.COM     case MX98713: case COMPEX9881:
1674*8044SWilliam.Kucharski@Sun.COM         tp->if_port = 0;
1675*8044SWilliam.Kucharski@Sun.COM         tp->csr6 = 0x01880000 | (tp->full_duplex ? 0x0200 : 0);
1676*8044SWilliam.Kucharski@Sun.COM         outl(0x0f370000 | inw(ioaddr + 0x80), ioaddr + 0x80);
1677*8044SWilliam.Kucharski@Sun.COM         break;
1678*8044SWilliam.Kucharski@Sun.COM     case MX98715: case MX98725:
1679*8044SWilliam.Kucharski@Sun.COM         /* Provided by BOLO, Macronix - 12/10/1998. */
1680*8044SWilliam.Kucharski@Sun.COM         tp->if_port = 0;
1681*8044SWilliam.Kucharski@Sun.COM         tp->csr6 = 0x01a80200;
1682*8044SWilliam.Kucharski@Sun.COM         outl(0x0f370000 | inw(ioaddr + 0x80), ioaddr + 0x80);
1683*8044SWilliam.Kucharski@Sun.COM         outl(0x11000 | inw(ioaddr + 0xa0), ioaddr + 0xa0);
1684*8044SWilliam.Kucharski@Sun.COM         break;
1685*8044SWilliam.Kucharski@Sun.COM     case COMET:
1686*8044SWilliam.Kucharski@Sun.COM         tp->if_port = 0;
1687*8044SWilliam.Kucharski@Sun.COM 	tp->csr6 = 0x00040000;
1688*8044SWilliam.Kucharski@Sun.COM         break;
1689*8044SWilliam.Kucharski@Sun.COM     case AX88140: case AX88141:
1690*8044SWilliam.Kucharski@Sun.COM         tp->csr6 = tp->mii_cnt ? 0x00040100 : 0x00000100;
1691*8044SWilliam.Kucharski@Sun.COM         break;
1692*8044SWilliam.Kucharski@Sun.COM     default:
1693*8044SWilliam.Kucharski@Sun.COM         select_media(nic, 1);
1694*8044SWilliam.Kucharski@Sun.COM     }
1695*8044SWilliam.Kucharski@Sun.COM }
1696*8044SWilliam.Kucharski@Sun.COM 
pnic_do_nway(struct nic * nic __unused)1697*8044SWilliam.Kucharski@Sun.COM static void pnic_do_nway(struct nic *nic __unused)
1698*8044SWilliam.Kucharski@Sun.COM {
1699*8044SWilliam.Kucharski@Sun.COM     u32 phy_reg = inl(ioaddr + 0xB8);
1700*8044SWilliam.Kucharski@Sun.COM     u32 new_csr6 = tp->csr6 & ~0x40C40200;
1701*8044SWilliam.Kucharski@Sun.COM 
1702*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG_WHERE
1703*8044SWilliam.Kucharski@Sun.COM     whereami("pnic_do_nway\n");
1704*8044SWilliam.Kucharski@Sun.COM #endif
1705*8044SWilliam.Kucharski@Sun.COM 
1706*8044SWilliam.Kucharski@Sun.COM     if (phy_reg & 0x78000000) { /* Ignore baseT4 */
1707*8044SWilliam.Kucharski@Sun.COM         if (phy_reg & 0x20000000)               tp->if_port = 5;
1708*8044SWilliam.Kucharski@Sun.COM         else if (phy_reg & 0x40000000)  tp->if_port = 3;
1709*8044SWilliam.Kucharski@Sun.COM         else if (phy_reg & 0x10000000)  tp->if_port = 4;
1710*8044SWilliam.Kucharski@Sun.COM         else if (phy_reg & 0x08000000)  tp->if_port = 0;
1711*8044SWilliam.Kucharski@Sun.COM         tp->nwayset = 1;
1712*8044SWilliam.Kucharski@Sun.COM         new_csr6 = (tp->if_port & 1) ? 0x01860000 : 0x00420000;
1713*8044SWilliam.Kucharski@Sun.COM         outl(0x32 | (tp->if_port & 1), ioaddr + CSR12);
1714*8044SWilliam.Kucharski@Sun.COM         if (tp->if_port & 1)
1715*8044SWilliam.Kucharski@Sun.COM             outl(0x1F868, ioaddr + 0xB8);
1716*8044SWilliam.Kucharski@Sun.COM         if (phy_reg & 0x30000000) {
1717*8044SWilliam.Kucharski@Sun.COM             tp->full_duplex = 1;
1718*8044SWilliam.Kucharski@Sun.COM             new_csr6 |= 0x00000200;
1719*8044SWilliam.Kucharski@Sun.COM         }
1720*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
1721*8044SWilliam.Kucharski@Sun.COM         if (tulip_debug > 1)
1722*8044SWilliam.Kucharski@Sun.COM             printf("%s: PNIC autonegotiated status %X, %s.\n",
1723*8044SWilliam.Kucharski@Sun.COM                    tp->nic_name, phy_reg, medianame[tp->if_port]);
1724*8044SWilliam.Kucharski@Sun.COM #endif
1725*8044SWilliam.Kucharski@Sun.COM         if (tp->csr6 != new_csr6) {
1726*8044SWilliam.Kucharski@Sun.COM             tp->csr6 = new_csr6;
1727*8044SWilliam.Kucharski@Sun.COM             outl(tp->csr6 | 0x0002, ioaddr + CSR6);     /* Restart Tx */
1728*8044SWilliam.Kucharski@Sun.COM             outl(tp->csr6 | 0x2002, ioaddr + CSR6);
1729*8044SWilliam.Kucharski@Sun.COM         }
1730*8044SWilliam.Kucharski@Sun.COM     }
1731*8044SWilliam.Kucharski@Sun.COM }
1732*8044SWilliam.Kucharski@Sun.COM 
1733*8044SWilliam.Kucharski@Sun.COM /* Set up the transceiver control registers for the selected media type. */
select_media(struct nic * nic,int startup)1734*8044SWilliam.Kucharski@Sun.COM static void select_media(struct nic *nic, int startup)
1735*8044SWilliam.Kucharski@Sun.COM {
1736*8044SWilliam.Kucharski@Sun.COM     struct mediatable *mtable = tp->mtable;
1737*8044SWilliam.Kucharski@Sun.COM     u32 new_csr6;
1738*8044SWilliam.Kucharski@Sun.COM     int i;
1739*8044SWilliam.Kucharski@Sun.COM 
1740*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG_WHERE
1741*8044SWilliam.Kucharski@Sun.COM     whereami("select_media\n");
1742*8044SWilliam.Kucharski@Sun.COM #endif
1743*8044SWilliam.Kucharski@Sun.COM 
1744*8044SWilliam.Kucharski@Sun.COM     if (mtable) {
1745*8044SWilliam.Kucharski@Sun.COM         struct medialeaf *mleaf = &mtable->mleaf[tp->cur_index];
1746*8044SWilliam.Kucharski@Sun.COM         unsigned char *p = mleaf->leafdata;
1747*8044SWilliam.Kucharski@Sun.COM         switch (mleaf->type) {
1748*8044SWilliam.Kucharski@Sun.COM         case 0:                                 /* 21140 non-MII xcvr. */
1749*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
1750*8044SWilliam.Kucharski@Sun.COM             if (tulip_debug > 1)
1751*8044SWilliam.Kucharski@Sun.COM                 printf("%s: Using a 21140 non-MII transceiver"
1752*8044SWilliam.Kucharski@Sun.COM                        " with control setting %hhX.\n",
1753*8044SWilliam.Kucharski@Sun.COM                        tp->nic_name, p[1]);
1754*8044SWilliam.Kucharski@Sun.COM #endif
1755*8044SWilliam.Kucharski@Sun.COM             tp->if_port = p[0];
1756*8044SWilliam.Kucharski@Sun.COM             if (startup)
1757*8044SWilliam.Kucharski@Sun.COM                 outl(mtable->csr12dir | 0x100, ioaddr + CSR12);
1758*8044SWilliam.Kucharski@Sun.COM             outl(p[1], ioaddr + CSR12);
1759*8044SWilliam.Kucharski@Sun.COM             new_csr6 = 0x02000000 | ((p[2] & 0x71) << 18);
1760*8044SWilliam.Kucharski@Sun.COM             break;
1761*8044SWilliam.Kucharski@Sun.COM         case 2: case 4: {
1762*8044SWilliam.Kucharski@Sun.COM             u16 setup[5];
1763*8044SWilliam.Kucharski@Sun.COM             u32 csr13val, csr14val, csr15dir, csr15val;
1764*8044SWilliam.Kucharski@Sun.COM             for (i = 0; i < 5; i++)
1765*8044SWilliam.Kucharski@Sun.COM                 setup[i] = get_u16(&p[i*2 + 1]);
1766*8044SWilliam.Kucharski@Sun.COM 
1767*8044SWilliam.Kucharski@Sun.COM             tp->if_port = p[0] & 15;
1768*8044SWilliam.Kucharski@Sun.COM             if (media_cap[tp->if_port] & MediaAlwaysFD)
1769*8044SWilliam.Kucharski@Sun.COM                 tp->full_duplex = 1;
1770*8044SWilliam.Kucharski@Sun.COM 
1771*8044SWilliam.Kucharski@Sun.COM             if (startup && mtable->has_reset) {
1772*8044SWilliam.Kucharski@Sun.COM                 struct medialeaf *rleaf = &mtable->mleaf[mtable->has_reset];
1773*8044SWilliam.Kucharski@Sun.COM                 unsigned char *rst = rleaf->leafdata;
1774*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
1775*8044SWilliam.Kucharski@Sun.COM                 if (tulip_debug > 1)
1776*8044SWilliam.Kucharski@Sun.COM                     printf("%s: Resetting the transceiver.\n",
1777*8044SWilliam.Kucharski@Sun.COM                            tp->nic_name);
1778*8044SWilliam.Kucharski@Sun.COM #endif
1779*8044SWilliam.Kucharski@Sun.COM                 for (i = 0; i < rst[0]; i++)
1780*8044SWilliam.Kucharski@Sun.COM                     outl(get_u16(rst + 1 + (i<<1)) << 16, ioaddr + CSR15);
1781*8044SWilliam.Kucharski@Sun.COM             }
1782*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
1783*8044SWilliam.Kucharski@Sun.COM             if (tulip_debug > 1)
1784*8044SWilliam.Kucharski@Sun.COM                 printf("%s: 21143 non-MII %s transceiver control "
1785*8044SWilliam.Kucharski@Sun.COM                        "%hX/%hX.\n",
1786*8044SWilliam.Kucharski@Sun.COM                        tp->nic_name, medianame[tp->if_port], setup[0], setup[1]);
1787*8044SWilliam.Kucharski@Sun.COM #endif
1788*8044SWilliam.Kucharski@Sun.COM             if (p[0] & 0x40) {  /* SIA (CSR13-15) setup values are provided. */
1789*8044SWilliam.Kucharski@Sun.COM                 csr13val = setup[0];
1790*8044SWilliam.Kucharski@Sun.COM                 csr14val = setup[1];
1791*8044SWilliam.Kucharski@Sun.COM                 csr15dir = (setup[3]<<16) | setup[2];
1792*8044SWilliam.Kucharski@Sun.COM                 csr15val = (setup[4]<<16) | setup[2];
1793*8044SWilliam.Kucharski@Sun.COM                 outl(0, ioaddr + CSR13);
1794*8044SWilliam.Kucharski@Sun.COM                 outl(csr14val, ioaddr + CSR14);
1795*8044SWilliam.Kucharski@Sun.COM                 outl(csr15dir, ioaddr + CSR15); /* Direction */
1796*8044SWilliam.Kucharski@Sun.COM                 outl(csr15val, ioaddr + CSR15); /* Data */
1797*8044SWilliam.Kucharski@Sun.COM                 outl(csr13val, ioaddr + CSR13);
1798*8044SWilliam.Kucharski@Sun.COM             } else {
1799*8044SWilliam.Kucharski@Sun.COM                 csr13val = 1;
1800*8044SWilliam.Kucharski@Sun.COM                 csr14val = 0x0003FF7F;
1801*8044SWilliam.Kucharski@Sun.COM                 csr15dir = (setup[0]<<16) | 0x0008;
1802*8044SWilliam.Kucharski@Sun.COM                 csr15val = (setup[1]<<16) | 0x0008;
1803*8044SWilliam.Kucharski@Sun.COM                 if (tp->if_port <= 4)
1804*8044SWilliam.Kucharski@Sun.COM                     csr14val = t21142_csr14[tp->if_port];
1805*8044SWilliam.Kucharski@Sun.COM                 if (startup) {
1806*8044SWilliam.Kucharski@Sun.COM                     outl(0, ioaddr + CSR13);
1807*8044SWilliam.Kucharski@Sun.COM                     outl(csr14val, ioaddr + CSR14);
1808*8044SWilliam.Kucharski@Sun.COM                 }
1809*8044SWilliam.Kucharski@Sun.COM                 outl(csr15dir, ioaddr + CSR15); /* Direction */
1810*8044SWilliam.Kucharski@Sun.COM                 outl(csr15val, ioaddr + CSR15); /* Data */
1811*8044SWilliam.Kucharski@Sun.COM                 if (startup) outl(csr13val, ioaddr + CSR13);
1812*8044SWilliam.Kucharski@Sun.COM             }
1813*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
1814*8044SWilliam.Kucharski@Sun.COM             if (tulip_debug > 1)
1815*8044SWilliam.Kucharski@Sun.COM                 printf("%s:  Setting CSR15 to %X/%X.\n",
1816*8044SWilliam.Kucharski@Sun.COM                        tp->nic_name, csr15dir, csr15val);
1817*8044SWilliam.Kucharski@Sun.COM #endif
1818*8044SWilliam.Kucharski@Sun.COM             if (mleaf->type == 4)
1819*8044SWilliam.Kucharski@Sun.COM                 new_csr6 = 0x82020000 | ((setup[2] & 0x71) << 18);
1820*8044SWilliam.Kucharski@Sun.COM             else
1821*8044SWilliam.Kucharski@Sun.COM                 new_csr6 = 0x82420000;
1822*8044SWilliam.Kucharski@Sun.COM             break;
1823*8044SWilliam.Kucharski@Sun.COM         }
1824*8044SWilliam.Kucharski@Sun.COM         case 1: case 3: {
1825*8044SWilliam.Kucharski@Sun.COM             int phy_num = p[0];
1826*8044SWilliam.Kucharski@Sun.COM             int init_length = p[1];
1827*8044SWilliam.Kucharski@Sun.COM             u16 *misc_info;
1828*8044SWilliam.Kucharski@Sun.COM 
1829*8044SWilliam.Kucharski@Sun.COM             tp->if_port = 11;
1830*8044SWilliam.Kucharski@Sun.COM             new_csr6 = 0x020E0000;
1831*8044SWilliam.Kucharski@Sun.COM             if (mleaf->type == 3) {     /* 21142 */
1832*8044SWilliam.Kucharski@Sun.COM                 u16 *init_sequence = (u16*)(p+2);
1833*8044SWilliam.Kucharski@Sun.COM                 u16 *reset_sequence = &((u16*)(p+3))[init_length];
1834*8044SWilliam.Kucharski@Sun.COM                 int reset_length = p[2 + init_length*2];
1835*8044SWilliam.Kucharski@Sun.COM                 misc_info = reset_sequence + reset_length;
1836*8044SWilliam.Kucharski@Sun.COM                 if (startup)
1837*8044SWilliam.Kucharski@Sun.COM                     for (i = 0; i < reset_length; i++)
1838*8044SWilliam.Kucharski@Sun.COM                         outl(get_u16(&reset_sequence[i]) << 16, ioaddr + CSR15);
1839*8044SWilliam.Kucharski@Sun.COM                 for (i = 0; i < init_length; i++)
1840*8044SWilliam.Kucharski@Sun.COM                     outl(get_u16(&init_sequence[i]) << 16, ioaddr + CSR15);
1841*8044SWilliam.Kucharski@Sun.COM             } else {
1842*8044SWilliam.Kucharski@Sun.COM                 u8 *init_sequence = p + 2;
1843*8044SWilliam.Kucharski@Sun.COM                 u8 *reset_sequence = p + 3 + init_length;
1844*8044SWilliam.Kucharski@Sun.COM                 int reset_length = p[2 + init_length];
1845*8044SWilliam.Kucharski@Sun.COM                 misc_info = (u16*)(reset_sequence + reset_length);
1846*8044SWilliam.Kucharski@Sun.COM                 if (startup) {
1847*8044SWilliam.Kucharski@Sun.COM                     outl(mtable->csr12dir | 0x100, ioaddr + CSR12);
1848*8044SWilliam.Kucharski@Sun.COM                     for (i = 0; i < reset_length; i++)
1849*8044SWilliam.Kucharski@Sun.COM                         outl(reset_sequence[i], ioaddr + CSR12);
1850*8044SWilliam.Kucharski@Sun.COM                 }
1851*8044SWilliam.Kucharski@Sun.COM                 for (i = 0; i < init_length; i++)
1852*8044SWilliam.Kucharski@Sun.COM                     outl(init_sequence[i], ioaddr + CSR12);
1853*8044SWilliam.Kucharski@Sun.COM             }
1854*8044SWilliam.Kucharski@Sun.COM             tp->advertising[phy_num] = get_u16(&misc_info[1]) | 1;
1855*8044SWilliam.Kucharski@Sun.COM             if (startup < 2) {
1856*8044SWilliam.Kucharski@Sun.COM                 if (tp->mii_advertise == 0)
1857*8044SWilliam.Kucharski@Sun.COM                     tp->mii_advertise = tp->advertising[phy_num];
1858*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
1859*8044SWilliam.Kucharski@Sun.COM                 if (tulip_debug > 1)
1860*8044SWilliam.Kucharski@Sun.COM                     printf("%s:  Advertising %hX on MII %d.\n",
1861*8044SWilliam.Kucharski@Sun.COM                            tp->nic_name, tp->mii_advertise, tp->phys[phy_num]);
1862*8044SWilliam.Kucharski@Sun.COM #endif
1863*8044SWilliam.Kucharski@Sun.COM                 mdio_write(nic, tp->phys[phy_num], 4, tp->mii_advertise);
1864*8044SWilliam.Kucharski@Sun.COM             }
1865*8044SWilliam.Kucharski@Sun.COM             break;
1866*8044SWilliam.Kucharski@Sun.COM         }
1867*8044SWilliam.Kucharski@Sun.COM         default:
1868*8044SWilliam.Kucharski@Sun.COM             printf("%s:  Invalid media table selection %d.\n",
1869*8044SWilliam.Kucharski@Sun.COM                    tp->nic_name, mleaf->type);
1870*8044SWilliam.Kucharski@Sun.COM             new_csr6 = 0x020E0000;
1871*8044SWilliam.Kucharski@Sun.COM         }
1872*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
1873*8044SWilliam.Kucharski@Sun.COM         if (tulip_debug > 1)
1874*8044SWilliam.Kucharski@Sun.COM             printf("%s: Using media type %s, CSR12 is %hhX.\n",
1875*8044SWilliam.Kucharski@Sun.COM                    tp->nic_name, medianame[tp->if_port],
1876*8044SWilliam.Kucharski@Sun.COM                    inl(ioaddr + CSR12) & 0xff);
1877*8044SWilliam.Kucharski@Sun.COM #endif
1878*8044SWilliam.Kucharski@Sun.COM     } else if (tp->chip_id == DC21041) {
1879*8044SWilliam.Kucharski@Sun.COM         int port = tp->if_port <= 4 ? tp->if_port : 0;
1880*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
1881*8044SWilliam.Kucharski@Sun.COM         if (tulip_debug > 1)
1882*8044SWilliam.Kucharski@Sun.COM             printf("%s: 21041 using media %s, CSR12 is %hX.\n",
1883*8044SWilliam.Kucharski@Sun.COM                    tp->nic_name, medianame[port == 3 ? 12: port],
1884*8044SWilliam.Kucharski@Sun.COM                    inl(ioaddr + CSR12));
1885*8044SWilliam.Kucharski@Sun.COM #endif
1886*8044SWilliam.Kucharski@Sun.COM         outl(0x00000000, ioaddr + CSR13); /* Reset the serial interface */
1887*8044SWilliam.Kucharski@Sun.COM         outl(t21041_csr14[port], ioaddr + CSR14);
1888*8044SWilliam.Kucharski@Sun.COM         outl(t21041_csr15[port], ioaddr + CSR15);
1889*8044SWilliam.Kucharski@Sun.COM         outl(t21041_csr13[port], ioaddr + CSR13);
1890*8044SWilliam.Kucharski@Sun.COM         new_csr6 = 0x80020000;
1891*8044SWilliam.Kucharski@Sun.COM     } else if (tp->chip_id == LC82C168) {
1892*8044SWilliam.Kucharski@Sun.COM         if (startup && ! tp->medialock)
1893*8044SWilliam.Kucharski@Sun.COM             tp->if_port = tp->mii_cnt ? 11 : 0;
1894*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
1895*8044SWilliam.Kucharski@Sun.COM         if (tulip_debug > 1)
1896*8044SWilliam.Kucharski@Sun.COM 	    printf("%s: PNIC PHY status is %hX, media %s.\n",
1897*8044SWilliam.Kucharski@Sun.COM                    tp->nic_name, inl(ioaddr + 0xB8), medianame[tp->if_port]);
1898*8044SWilliam.Kucharski@Sun.COM #endif
1899*8044SWilliam.Kucharski@Sun.COM         if (tp->mii_cnt) {
1900*8044SWilliam.Kucharski@Sun.COM             new_csr6 = 0x810C0000;
1901*8044SWilliam.Kucharski@Sun.COM             outl(0x0001, ioaddr + CSR15);
1902*8044SWilliam.Kucharski@Sun.COM             outl(0x0201B07A, ioaddr + 0xB8);
1903*8044SWilliam.Kucharski@Sun.COM         } else if (startup) {
1904*8044SWilliam.Kucharski@Sun.COM             /* Start with 10mbps to do autonegotiation. */
1905*8044SWilliam.Kucharski@Sun.COM             outl(0x32, ioaddr + CSR12);
1906*8044SWilliam.Kucharski@Sun.COM             new_csr6 = 0x00420000;
1907*8044SWilliam.Kucharski@Sun.COM             outl(0x0001B078, ioaddr + 0xB8);
1908*8044SWilliam.Kucharski@Sun.COM             outl(0x0201B078, ioaddr + 0xB8);
1909*8044SWilliam.Kucharski@Sun.COM         } else if (tp->if_port == 3  ||  tp->if_port == 5) {
1910*8044SWilliam.Kucharski@Sun.COM             outl(0x33, ioaddr + CSR12);
1911*8044SWilliam.Kucharski@Sun.COM             new_csr6 = 0x01860000;
1912*8044SWilliam.Kucharski@Sun.COM             /* Trigger autonegotiation. */
1913*8044SWilliam.Kucharski@Sun.COM             outl(startup ? 0x0201F868 : 0x0001F868, ioaddr + 0xB8);
1914*8044SWilliam.Kucharski@Sun.COM         } else {
1915*8044SWilliam.Kucharski@Sun.COM             outl(0x32, ioaddr + CSR12);
1916*8044SWilliam.Kucharski@Sun.COM             new_csr6 = 0x00420000;
1917*8044SWilliam.Kucharski@Sun.COM             outl(0x1F078, ioaddr + 0xB8);
1918*8044SWilliam.Kucharski@Sun.COM         }
1919*8044SWilliam.Kucharski@Sun.COM     } else if (tp->chip_id == DC21040) {                                        /* 21040 */
1920*8044SWilliam.Kucharski@Sun.COM         /* Turn on the xcvr interface. */
1921*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
1922*8044SWilliam.Kucharski@Sun.COM         int csr12 = inl(ioaddr + CSR12);
1923*8044SWilliam.Kucharski@Sun.COM         if (tulip_debug > 1)
1924*8044SWilliam.Kucharski@Sun.COM             printf("%s: 21040 media type is %s, CSR12 is %hhX.\n",
1925*8044SWilliam.Kucharski@Sun.COM                    tp->nic_name, medianame[tp->if_port], csr12);
1926*8044SWilliam.Kucharski@Sun.COM #endif
1927*8044SWilliam.Kucharski@Sun.COM         if (media_cap[tp->if_port] & MediaAlwaysFD)
1928*8044SWilliam.Kucharski@Sun.COM             tp->full_duplex = 1;
1929*8044SWilliam.Kucharski@Sun.COM         new_csr6 = 0x20000;
1930*8044SWilliam.Kucharski@Sun.COM         /* Set the full duplux match frame. */
1931*8044SWilliam.Kucharski@Sun.COM         outl(FULL_DUPLEX_MAGIC, ioaddr + CSR11);
1932*8044SWilliam.Kucharski@Sun.COM         outl(0x00000000, ioaddr + CSR13); /* Reset the serial interface */
1933*8044SWilliam.Kucharski@Sun.COM         if (t21040_csr13[tp->if_port] & 8) {
1934*8044SWilliam.Kucharski@Sun.COM             outl(0x0705, ioaddr + CSR14);
1935*8044SWilliam.Kucharski@Sun.COM             outl(0x0006, ioaddr + CSR15);
1936*8044SWilliam.Kucharski@Sun.COM         } else {
1937*8044SWilliam.Kucharski@Sun.COM             outl(0xffff, ioaddr + CSR14);
1938*8044SWilliam.Kucharski@Sun.COM             outl(0x0000, ioaddr + CSR15);
1939*8044SWilliam.Kucharski@Sun.COM         }
1940*8044SWilliam.Kucharski@Sun.COM         outl(0x8f01 | t21040_csr13[tp->if_port], ioaddr + CSR13);
1941*8044SWilliam.Kucharski@Sun.COM     } else {                                    /* Unknown chip type with no media table. */
1942*8044SWilliam.Kucharski@Sun.COM         if (tp->default_port == 0)
1943*8044SWilliam.Kucharski@Sun.COM             tp->if_port = tp->mii_cnt ? 11 : 3;
1944*8044SWilliam.Kucharski@Sun.COM         if (media_cap[tp->if_port] & MediaIsMII) {
1945*8044SWilliam.Kucharski@Sun.COM             new_csr6 = 0x020E0000;
1946*8044SWilliam.Kucharski@Sun.COM         } else if (media_cap[tp->if_port] & MediaIsFx) {
1947*8044SWilliam.Kucharski@Sun.COM             new_csr6 = 0x028600000;
1948*8044SWilliam.Kucharski@Sun.COM         } else
1949*8044SWilliam.Kucharski@Sun.COM             new_csr6 = 0x038600000;
1950*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
1951*8044SWilliam.Kucharski@Sun.COM         if (tulip_debug > 1)
1952*8044SWilliam.Kucharski@Sun.COM             printf("%s: No media description table, assuming "
1953*8044SWilliam.Kucharski@Sun.COM                    "%s transceiver, CSR12 %hhX.\n",
1954*8044SWilliam.Kucharski@Sun.COM                    tp->nic_name, medianame[tp->if_port],
1955*8044SWilliam.Kucharski@Sun.COM                    inl(ioaddr + CSR12));
1956*8044SWilliam.Kucharski@Sun.COM #endif
1957*8044SWilliam.Kucharski@Sun.COM     }
1958*8044SWilliam.Kucharski@Sun.COM 
1959*8044SWilliam.Kucharski@Sun.COM     tp->csr6 = new_csr6 | (tp->csr6 & 0xfdff) | (tp->full_duplex ? 0x0200 : 0);
1960*8044SWilliam.Kucharski@Sun.COM     return;
1961*8044SWilliam.Kucharski@Sun.COM }
1962*8044SWilliam.Kucharski@Sun.COM 
1963*8044SWilliam.Kucharski@Sun.COM /*
1964*8044SWilliam.Kucharski@Sun.COM   Check the MII negotiated duplex and change the CSR6 setting if
1965*8044SWilliam.Kucharski@Sun.COM   required.
1966*8044SWilliam.Kucharski@Sun.COM   Return 0 if everything is OK.
1967*8044SWilliam.Kucharski@Sun.COM   Return < 0 if the transceiver is missing or has no link beat.
1968*8044SWilliam.Kucharski@Sun.COM */
tulip_check_duplex(struct nic * nic)1969*8044SWilliam.Kucharski@Sun.COM static int tulip_check_duplex(struct nic *nic)
1970*8044SWilliam.Kucharski@Sun.COM {
1971*8044SWilliam.Kucharski@Sun.COM         unsigned int bmsr, lpa, negotiated, new_csr6;
1972*8044SWilliam.Kucharski@Sun.COM 
1973*8044SWilliam.Kucharski@Sun.COM         bmsr = mdio_read(nic, tp->phys[0], 1);
1974*8044SWilliam.Kucharski@Sun.COM         lpa = mdio_read(nic, tp->phys[0], 5);
1975*8044SWilliam.Kucharski@Sun.COM 
1976*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
1977*8044SWilliam.Kucharski@Sun.COM         if (tulip_debug > 1)
1978*8044SWilliam.Kucharski@Sun.COM                 printf("%s: MII status %#x, Link partner report "
1979*8044SWilliam.Kucharski@Sun.COM                            "%#x.\n", tp->nic_name, bmsr, lpa);
1980*8044SWilliam.Kucharski@Sun.COM #endif
1981*8044SWilliam.Kucharski@Sun.COM 
1982*8044SWilliam.Kucharski@Sun.COM         if (bmsr == 0xffff)
1983*8044SWilliam.Kucharski@Sun.COM                 return -2;
1984*8044SWilliam.Kucharski@Sun.COM         if ((bmsr & 4) == 0) {
1985*8044SWilliam.Kucharski@Sun.COM                 int new_bmsr = mdio_read(nic, tp->phys[0], 1);
1986*8044SWilliam.Kucharski@Sun.COM                 if ((new_bmsr & 4) == 0) {
1987*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
1988*8044SWilliam.Kucharski@Sun.COM                         if (tulip_debug  > 1)
1989*8044SWilliam.Kucharski@Sun.COM                                 printf("%s: No link beat on the MII interface,"
1990*8044SWilliam.Kucharski@Sun.COM                                            " status %#x.\n", tp->nic_name,
1991*8044SWilliam.Kucharski@Sun.COM                                            new_bmsr);
1992*8044SWilliam.Kucharski@Sun.COM #endif
1993*8044SWilliam.Kucharski@Sun.COM                         return -1;
1994*8044SWilliam.Kucharski@Sun.COM                 }
1995*8044SWilliam.Kucharski@Sun.COM         }
1996*8044SWilliam.Kucharski@Sun.COM         tp->full_duplex = lpa & 0x140;
1997*8044SWilliam.Kucharski@Sun.COM 
1998*8044SWilliam.Kucharski@Sun.COM         new_csr6 = tp->csr6;
1999*8044SWilliam.Kucharski@Sun.COM         negotiated = lpa & tp->advertising[0];
2000*8044SWilliam.Kucharski@Sun.COM 
2001*8044SWilliam.Kucharski@Sun.COM         if(negotiated & 0x380) new_csr6 &= ~0x400000;
2002*8044SWilliam.Kucharski@Sun.COM         else                   new_csr6 |= 0x400000;
2003*8044SWilliam.Kucharski@Sun.COM         if (tp->full_duplex)   new_csr6 |= 0x200;
2004*8044SWilliam.Kucharski@Sun.COM         else                   new_csr6 &= ~0x200;
2005*8044SWilliam.Kucharski@Sun.COM 
2006*8044SWilliam.Kucharski@Sun.COM         if (new_csr6 != tp->csr6) {
2007*8044SWilliam.Kucharski@Sun.COM                 tp->csr6 = new_csr6;
2008*8044SWilliam.Kucharski@Sun.COM 
2009*8044SWilliam.Kucharski@Sun.COM #ifdef TULIP_DEBUG
2010*8044SWilliam.Kucharski@Sun.COM                 if (tulip_debug > 0)
2011*8044SWilliam.Kucharski@Sun.COM                         printf("%s: Setting %s-duplex based on MII"
2012*8044SWilliam.Kucharski@Sun.COM                                    "#%d link partner capability of %#x.\n",
2013*8044SWilliam.Kucharski@Sun.COM                                    tp->nic_name,
2014*8044SWilliam.Kucharski@Sun.COM                                    tp->full_duplex ? "full" : "half",
2015*8044SWilliam.Kucharski@Sun.COM                                    tp->phys[0], lpa);
2016*8044SWilliam.Kucharski@Sun.COM #endif
2017*8044SWilliam.Kucharski@Sun.COM                 return 1;
2018*8044SWilliam.Kucharski@Sun.COM         }
2019*8044SWilliam.Kucharski@Sun.COM 
2020*8044SWilliam.Kucharski@Sun.COM         return 0;
2021*8044SWilliam.Kucharski@Sun.COM }
2022*8044SWilliam.Kucharski@Sun.COM 
2023*8044SWilliam.Kucharski@Sun.COM static struct pci_id tulip_nics[] = {
2024*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x1011, 0x0002, "dc21040",     "Digital Tulip"),
2025*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x1011, 0x0009, "ds21140",     "Digital Tulip Fast"),
2026*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x1011, 0x0014, "dc21041",     "Digital Tulip+"),
2027*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x1011, 0x0019, "ds21142",     "Digital Tulip 21142"),
2028*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x10b7, 0x9300, "3csoho100b-tx","3ComSOHO100B-TX"),
2029*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x10b9, 0x5261, "ali1563",     "ALi 1563 integrated ethernet"),
2030*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x10d9, 0x0512, "mx98713",     "Macronix MX987x3"),
2031*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x10d9, 0x0531, "mx98715",     "Macronix MX987x5"),
2032*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x1113, 0x1217, "mxic-98715",  "Macronix MX987x5"),
2033*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x11ad, 0xc115, "lc82c115",    "LinkSys LNE100TX"),
2034*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x11ad, 0x0002, "82c168",      "Netgear FA310TX"),
2035*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x1282, 0x9100, "dm9100",      "Davicom 9100"),
2036*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x1282, 0x9102, "dm9102",      "Davicom 9102"),
2037*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x1282, 0x9009, "dm9009",      "Davicom 9009"),
2038*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x1282, 0x9132, "dm9132",      "Davicom 9132"),
2039*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x1317, 0x0985, "centaur-p",   "ADMtek Centaur-P"),
2040*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x1317, 0x0981, "an981",       "ADMtek AN981 Comet"),		/* ADMTek Centaur-P (stmicro) */
2041*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x1113, 0x1216, "an983",       "ADMTek AN983 Comet"),
2042*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x1317, 0x9511, "an983b",      "ADMTek Comet 983b"),
2043*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x1317, 0x1985, "centaur-c",   "ADMTek Centaur-C"),
2044*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x8086, 0x0039, "intel21145",  "Intel Tulip"),
2045*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x125b, 0x1400, "ax88140",     "ASIX AX88140"),
2046*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x11f6, 0x9881, "rl100tx",     "Compex RL100-TX"),
2047*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x115d, 0x0003, "xircomtulip", "Xircom Tulip"),
2048*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x104a, 0x0981, "tulip-0981",  "Tulip 0x104a 0x0981"),
2049*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x104a, 0x2774, "tulip-2774",  "Tulip 0x104a 0x2774"),
2050*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x1113, 0x9511, "tulip-9511",  "Tulip 0x1113 0x9511"),
2051*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x1186, 0x1561, "tulip-1561",  "Tulip 0x1186 0x1561"),
2052*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x1259, 0xa120, "tulip-a120",  "Tulip 0x1259 0xa120"),
2053*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x13d1, 0xab02, "tulip-ab02",  "Tulip 0x13d1 0xab02"),
2054*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x13d1, 0xab03, "tulip-ab03",  "Tulip 0x13d1 0xab03"),
2055*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x13d1, 0xab08, "tulip-ab08",  "Tulip 0x13d1 0xab08"),
2056*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x14f1, 0x1803, "lanfinity",   "Conexant LANfinity"),
2057*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x1626, 0x8410, "tulip-8410",  "Tulip 0x1626 0x8410"),
2058*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x1737, 0xab08, "tulip-1737-ab08","Tulip 0x1737 0xab08"),
2059*8044SWilliam.Kucharski@Sun.COM PCI_ROM(0x1737, 0xab09, "tulip-ab09",  "Tulip 0x1737 0xab09"),
2060*8044SWilliam.Kucharski@Sun.COM };
2061*8044SWilliam.Kucharski@Sun.COM 
2062*8044SWilliam.Kucharski@Sun.COM struct pci_driver tulip_driver = {
2063*8044SWilliam.Kucharski@Sun.COM 	.type     = NIC_DRIVER,
2064*8044SWilliam.Kucharski@Sun.COM 	.name     = "Tulip",
2065*8044SWilliam.Kucharski@Sun.COM 	.probe    = tulip_probe,
2066*8044SWilliam.Kucharski@Sun.COM 	.ids      = tulip_nics,
2067*8044SWilliam.Kucharski@Sun.COM 	.id_count = sizeof(tulip_nics)/sizeof(tulip_nics[0]),
2068*8044SWilliam.Kucharski@Sun.COM 	.class    = 0,
2069*8044SWilliam.Kucharski@Sun.COM };
2070