123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097 |
- /* -*- Mode:C; c-basic-offset:4; -*- */
-
- /*
- Tulip and clone Etherboot Driver
-
- By Marty Connor (mdc@etherboot.org)
- Copyright (C) 2001 Entity Cyber, Inc.
-
- This software may be used and distributed according to the terms
- of the GNU Public License, incorporated herein by reference.
-
- As of April 2001 this driver should support most tulip cards that
- the Linux tulip driver supports because Donald Becker's Linux media
- detection code is now included.
-
- Based on Ken Yap's Tulip Etherboot Driver and Donald Becker's
- Linux Tulip Driver. Supports N-Way speed auto-configuration on
- MX98715, MX98715A and MX98725. Support inexpensive PCI 10/100 cards
- based on the Macronix MX987x5 chip, such as the SOHOware Fast
- model SFA110A, and the LinkSYS model LNE100TX. The NetGear
- model FA310X, based on the LC82C168 chip is supported.
- The TRENDnet TE100-PCIA NIC which uses a genuine Intel 21143-PD
- chipset is supported. Also, Davicom DM9102's.
-
- Documentation and source code used:
- Source for Etherboot driver at
- http://etherboot.sourceforge.net/
- MX98715A Data Sheet and MX98715A Application Note
- on http://www.macronix.com/ (PDF format files)
- Source for Linux tulip driver at
- http://cesdis.gsfc.nasa.gov/linux/drivers/tulip.html
-
- Adapted by Ken Yap from
- FreeBSD netboot DEC 21143 driver
- Author: David Sharp
- date: Nov/98
-
- Some code fragments were taken from verious places, Ken Yap's
- etherboot, FreeBSD's if_de.c, and various Linux related files.
- DEC's manuals for the 21143 and SROM format were very helpful.
- The Linux de driver development page has a number of links to
- useful related information. Have a look at:
- ftp://cesdis.gsfc.nasa.gov/pub/linux/drivers/tulip-devel.html
- */
-
- /*********************************************************************/
- /* Revision History */
- /*********************************************************************/
-
- /*
- 08 Feb 2005 Ramesh Chander chhabaramesh at yahoo.co.in added table entries
- for SGThomson STE10/100A
- 07 Sep 2003 timlegge Multicast Support Added
- 11 Apr 2001 mdc [patch to etherboot 4.7.24]
- Major rewrite to include Linux tulip driver media detection
- code. This driver should support a lot more cards now.
- 16 Jul 2000 mdc 0.75b11
- Added support for ADMtek 0985 Centaur-P, a "Comet" tulip clone
- which is used on the LinkSYS LNE100TX v4.x cards. We already
- support LNE100TX v2.0 cards, which use a different controller.
- 04 Jul 2000 jam ?
- Added test of status after receiving a packet from the card.
- Also uncommented the tulip_disable routine. Stray packets
- seemed to be causing problems.
- 27 Apr 2000 njl ?
- 29 Feb 2000 mdc 0.75b7
- Increased reset delay to 3 seconds because Macronix cards seem to
- need more reset time before card comes back to a usable state.
- 26 Feb 2000 mdc 0.75b6
- Added a 1 second delay after initializing the transmitter because
- some cards seem to need the time or they drop the first packet
- transmitted.
- 23 Feb 2000 mdc 0.75b5
- removed udelay code and used currticks() for more reliable delay
- code in reset pause and sanity timeouts. Added function prototypes
- and TX debugging code.
- 21 Feb 2000 mdc patch to Etherboot 4.4.3
- Incorporated patches from Bob Edwards and Paul Mackerras of
- Linuxcare's OZLabs to deal with inefficiencies in tulip_transmit
- and udelay. We now wait for packet transmission to complete
- (or sanity timeout).
- 04 Feb 2000 Robert.Edwards@anu.edu.au patch to Etherboot 4.4.2
- patch to tulip.c that implements the automatic selection of the MII
- interface on cards using the Intel/DEC 21143 reference design, in
- particular, the TRENDnet TE100-PCIA NIC which uses a genuine Intel
- 21143-PD chipset.
- 11 Jan 2000 mdc 0.75b4
- Added support for NetGear FA310TX card based on the LC82C168
- chip. This should also support Lite-On LC82C168 boards.
- Added simple MII support. Re-arranged code to better modularize
- initializations.
- 04 Dec 1999 mdc 0.75b3
- Added preliminary support for LNE100TX PCI cards. Should work for
- PNIC2 cards. No MII support, but single interface (RJ45) tulip
- cards seem to not care.
- 03 Dec 1999 mdc 0.75b2
- Renamed from mx987x5 to tulip, merged in original tulip init code
- from tulip.c to support other tulip compatible cards.
- 02 Dec 1999 mdc 0.75b1
- Released Beta MX987x5 Driver for code review and testing to netboot
- and thinguin mailing lists.
- */
-
-
- /*********************************************************************/
- /* Declarations */
- /*********************************************************************/
-
- #include "etherboot.h"
- #include "nic.h"
-
- #include <gpxe/ethernet.h>
- #include <gpxe/pci.h>
-
- /* User settable parameters */
-
- #undef TULIP_DEBUG
- #undef TULIP_DEBUG_WHERE
- #ifdef TULIP_DEBUG
- static int tulip_debug = 2; /* 1 normal messages, 0 quiet .. 7 verbose. */
- #endif
-
- #define TX_TIME_OUT 2*TICKS_PER_SEC
-
- /* helpful macros if on a big_endian machine for changing byte order.
- not strictly needed on Intel */
- #define get_unaligned(ptr) (*(ptr))
- #define put_unaligned(val, ptr) ((void)( *(ptr) = (val) ))
- #define get_u16(ptr) (*(u16 *)(ptr))
- #define virt_to_le32desc(addr) virt_to_bus(addr)
-
- #define TULIP_IOTYPE PCI_USES_MASTER | PCI_USES_IO | PCI_ADDR0
- #define TULIP_SIZE 0x80
-
- /* This is a mysterious value that can be written to CSR11 in the 21040 (only)
- to support a pre-NWay full-duplex signaling mechanism using short frames.
- No one knows what it should be, but if left at its default value some
- 10base2(!) packets trigger a full-duplex-request interrupt. */
- #define FULL_DUPLEX_MAGIC 0x6969
-
- static const int csr0 = 0x01A00000 | 0x8000;
-
- /* The possible media types that can be set in options[] are: */
- #define MEDIA_MASK 31
- static const char * const medianame[32] = {
- "10baseT", "10base2", "AUI", "100baseTx",
- "10baseT-FDX", "100baseTx-FDX", "100baseT4", "100baseFx",
- "100baseFx-FDX", "MII 10baseT", "MII 10baseT-FDX", "MII",
- "10baseT(forced)", "MII 100baseTx", "MII 100baseTx-FDX", "MII 100baseT4",
- "MII 100baseFx-HDX", "MII 100baseFx-FDX", "Home-PNA 1Mbps", "Invalid-19",
- };
-
- /* This much match tulip_tbl[]! Note 21142 == 21143. */
- enum tulip_chips {
- DC21040=0, DC21041=1, DC21140=2, DC21142=3, DC21143=3,
- LC82C168, MX98713, MX98715, MX98725, AX88141, AX88140, PNIC2, COMET,
- COMPEX9881, I21145, XIRCOM, SGThomson, /*Ramesh Chander*/
- };
-
- enum pci_id_flags_bits {
- /* Set PCI command register bits before calling probe1(). */
- PCI_USES_IO=1, PCI_USES_MEM=2, PCI_USES_MASTER=4,
- /* Read and map the single following PCI BAR. */
- PCI_ADDR0=0<<4, PCI_ADDR1=1<<4, PCI_ADDR2=2<<4, PCI_ADDR3=3<<4,
- PCI_ADDR_64BITS=0x100, PCI_NO_ACPI_WAKE=0x200, PCI_NO_MIN_LATENCY=0x400,
- PCI_UNUSED_IRQ=0x800,
- };
-
- struct pci_id_info {
- char *name;
- struct match_info {
- u32 pci, pci_mask, subsystem, subsystem_mask;
- u32 revision, revision_mask; /* Only 8 bits. */
- } id;
- enum pci_id_flags_bits pci_flags;
- int io_size; /* Needed for I/O region check or ioremap(). */
- int drv_flags; /* Driver use, intended as capability flags. */
- };
-
- static const struct pci_id_info pci_id_tbl[] = {
- { "Digital DC21040 Tulip", { 0x00021011, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 0x80, DC21040 },
- { "Digital DC21041 Tulip", { 0x00141011, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 0x80, DC21041 },
- { "Digital DS21140A Tulip", { 0x00091011, 0xffffffff, 0,0, 0x20,0xf0 },
- TULIP_IOTYPE, 0x80, DC21140 },
- { "Digital DS21140 Tulip", { 0x00091011, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 0x80, DC21140 },
- { "Digital DS21143 Tulip", { 0x00191011, 0xffffffff, 0,0, 65,0xff },
- TULIP_IOTYPE, TULIP_SIZE, DC21142 },
- { "Digital DS21142 Tulip", { 0x00191011, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, TULIP_SIZE, DC21142 },
- { "Kingston KNE110tx (PNIC)", { 0x000211AD, 0xffffffff, 0xf0022646, 0xffffffff, 0, 0 },
- TULIP_IOTYPE, 256, LC82C168 },
- { "Lite-On 82c168 PNIC", { 0x000211AD, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 256, LC82C168 },
- { "Macronix 98713 PMAC", { 0x051210d9, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 256, MX98713 },
- { "Macronix 98715 PMAC", { 0x053110d9, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 256, MX98715 },
- { "Macronix 98725 PMAC", { 0x053110d9, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 256, MX98725 },
- { "ASIX AX88141", { 0x1400125B, 0xffffffff, 0,0, 0x10, 0xf0 },
- TULIP_IOTYPE, 128, AX88141 },
- { "ASIX AX88140", { 0x1400125B, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 128, AX88140 },
- { "Lite-On LC82C115 PNIC-II", { 0xc11511AD, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 256, PNIC2 },
- { "ADMtek AN981 Comet", { 0x09811317, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 256, COMET },
- { "ADMTek AN983 Comet", { 0x12161113, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 256, COMET },
- { "ADMTek Comet AN983b", { 0x95111317, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 256, COMET },
- { "ADMtek Centaur-P", { 0x09851317, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 256, COMET },
- { "ADMtek Centaur-C", { 0x19851317, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 256, COMET },
- { "Compex RL100-TX", { 0x988111F6, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 128, COMPEX9881 },
- { "Intel 21145 Tulip", { 0x00398086, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 128, I21145 },
- { "Xircom Tulip clone", { 0x0003115d, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 128, XIRCOM },
- { "Davicom DM9102", { 0x91021282, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 0x80, DC21140 },
- { "Davicom DM9100", { 0x91001282, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 0x80, DC21140 },
- { "Macronix mxic-98715 (EN1217)", { 0x12171113, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 256, MX98715 },
- { "3Com 3cSOHO100B-TX (ADMtek Centuar)", { 0x930010b7, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, TULIP_SIZE, COMET },
- { "SG Thomson STE10/100A", { 0x2774104a, 0xffffffff, 0, 0, 0, 0 },
- TULIP_IOTYPE, 256, COMET }, /*Ramesh Chander*/
- { 0, { 0, 0, 0, 0, 0, 0 }, 0, 0, 0 },
- };
-
- enum tbl_flag {
- HAS_MII=1, HAS_MEDIA_TABLE=2, CSR12_IN_SROM=4, ALWAYS_CHECK_MII=8,
- HAS_PWRDWN=0x10, MC_HASH_ONLY=0x20, /* Hash-only multicast filter. */
- HAS_PNICNWAY=0x80, HAS_NWAY=0x40, /* Uses internal NWay xcvr. */
- HAS_INTR_MITIGATION=0x100, IS_ASIX=0x200, HAS_8023X=0x400,
- };
-
- /* Note: this table must match enum tulip_chips above. */
- static struct tulip_chip_table {
- char *chip_name;
- int flags;
- } tulip_tbl[] = {
- { "Digital DC21040 Tulip", 0},
- { "Digital DC21041 Tulip", HAS_MEDIA_TABLE | HAS_NWAY },
- { "Digital DS21140 Tulip", HAS_MII | HAS_MEDIA_TABLE | CSR12_IN_SROM },
- { "Digital DS21143 Tulip", HAS_MII | HAS_MEDIA_TABLE | ALWAYS_CHECK_MII
- | HAS_PWRDWN | HAS_NWAY | HAS_INTR_MITIGATION },
- { "Lite-On 82c168 PNIC", HAS_MII | HAS_PNICNWAY },
- { "Macronix 98713 PMAC", HAS_MII | HAS_MEDIA_TABLE | CSR12_IN_SROM },
- { "Macronix 98715 PMAC", HAS_MEDIA_TABLE },
- { "Macronix 98725 PMAC", HAS_MEDIA_TABLE },
- { "ASIX AX88140", HAS_MII | HAS_MEDIA_TABLE | CSR12_IN_SROM
- | MC_HASH_ONLY | IS_ASIX },
- { "ASIX AX88141", HAS_MII | HAS_MEDIA_TABLE | CSR12_IN_SROM | MC_HASH_ONLY
- | IS_ASIX },
- { "Lite-On PNIC-II", HAS_MII | HAS_NWAY | HAS_8023X },
- { "ADMtek Comet", HAS_MII | MC_HASH_ONLY },
- { "Compex 9881 PMAC", HAS_MII | HAS_MEDIA_TABLE | CSR12_IN_SROM },
- { "Intel DS21145 Tulip", HAS_MII | HAS_MEDIA_TABLE | ALWAYS_CHECK_MII
- | HAS_PWRDWN | HAS_NWAY },
- { "Xircom tulip work-alike", HAS_MII | HAS_MEDIA_TABLE | ALWAYS_CHECK_MII
- | HAS_PWRDWN | HAS_NWAY },
- { "SGThomson STE10/100A", HAS_MII | MC_HASH_ONLY }, /*Ramesh Chander*/
- { 0, 0 },
- };
-
- /* A full-duplex map for media types. */
- enum MediaIs {
- MediaIsFD = 1, MediaAlwaysFD=2, MediaIsMII=4, MediaIsFx=8,
- MediaIs100=16};
-
- static const char media_cap[32] =
- {0,0,0,16, 3,19,16,24, 27,4,7,5, 0,20,23,20, 20,31,0,0, };
- static u8 t21040_csr13[] = {2,0x0C,8,4, 4,0,0,0, 0,0,0,0, 4,0,0,0};
-
- /* 21041 transceiver register settings: 10-T, 10-2, AUI, 10-T, 10T-FD */
- static u16 t21041_csr13[] = { 0xEF01, 0xEF09, 0xEF09, 0xEF01, 0xEF09, };
- static u16 t21041_csr14[] = { 0xFFFF, 0xF7FD, 0xF7FD, 0x7F3F, 0x7F3D, };
- static u16 t21041_csr15[] = { 0x0008, 0x0006, 0x000E, 0x0008, 0x0008, };
-
- /* not used
- static u16 t21142_csr13[] = { 0x0001, 0x0009, 0x0009, 0x0000, 0x0001, };
- */
- static u16 t21142_csr14[] = { 0xFFFF, 0x0705, 0x0705, 0x0000, 0x7F3D, };
- /* not used
- static u16 t21142_csr15[] = { 0x0008, 0x0006, 0x000E, 0x0008, 0x0008, };
- */
-
- /* Offsets to the Command and Status Registers, "CSRs". All accesses
- must be longword instructions and quadword aligned. */
- enum tulip_offsets {
- CSR0=0, CSR1=0x08, CSR2=0x10, CSR3=0x18, CSR4=0x20, CSR5=0x28,
- CSR6=0x30, CSR7=0x38, CSR8=0x40, CSR9=0x48, CSR10=0x50, CSR11=0x58,
- CSR12=0x60, CSR13=0x68, CSR14=0x70, CSR15=0x78, CSR16=0x80, CSR20=0xA0
- };
-
- /* The bits in the CSR5 status registers, mostly interrupt sources. */
- enum status_bits {
- TimerInt=0x800, TPLnkFail=0x1000, TPLnkPass=0x10,
- NormalIntr=0x10000, AbnormalIntr=0x8000,
- RxJabber=0x200, RxDied=0x100, RxNoBuf=0x80, RxIntr=0x40,
- TxFIFOUnderflow=0x20, TxJabber=0x08, TxNoBuf=0x04, TxDied=0x02, TxIntr=0x01,
- };
-
- /* The configuration bits in CSR6. */
- enum csr6_mode_bits {
- TxOn=0x2000, RxOn=0x0002, FullDuplex=0x0200,
- AcceptBroadcast=0x0100, AcceptAllMulticast=0x0080,
- AcceptAllPhys=0x0040, AcceptRunt=0x0008,
- };
-
-
- enum desc_status_bits {
- DescOwnded=0x80000000, RxDescFatalErr=0x8000, RxWholePkt=0x0300,
- };
-
- struct medialeaf {
- u8 type;
- u8 media;
- unsigned char *leafdata;
- };
-
- struct mediatable {
- u16 defaultmedia;
- u8 leafcount, csr12dir; /* General purpose pin directions. */
- unsigned has_mii:1, has_nonmii:1, has_reset:6;
- u32 csr15dir, csr15val; /* 21143 NWay setting. */
- struct medialeaf mleaf[0];
- };
-
- struct mediainfo {
- struct mediainfo *next;
- int info_type;
- int index;
- unsigned char *info;
- };
-
- /* EEPROM Address width definitions */
- #define EEPROM_ADDRLEN 6
- #define EEPROM_SIZE 128 /* 2 << EEPROM_ADDRLEN */
-
- /* The EEPROM commands include the alway-set leading bit. */
- #define EE_WRITE_CMD (5 << addr_len)
- #define EE_READ_CMD (6 << addr_len)
- #define EE_ERASE_CMD (7 << addr_len)
-
- /* EEPROM_Ctrl bits. */
- #define EE_SHIFT_CLK 0x02 /* EEPROM shift clock. */
- #define EE_CS 0x01 /* EEPROM chip select. */
- #define EE_DATA_WRITE 0x04 /* EEPROM chip data in. */
- #define EE_WRITE_0 0x01
- #define EE_WRITE_1 0x05
- #define EE_DATA_READ 0x08 /* EEPROM chip data out. */
- #define EE_ENB (0x4800 | EE_CS)
-
- /* Delay between EEPROM clock transitions. Even at 33Mhz current PCI
- implementations don't overrun the EEPROM clock. We add a bus
- turn-around to insure that this remains true. */
- #define eeprom_delay() inl(ee_addr)
-
- /* Size of transmit and receive buffers */
- #define BUFLEN 1536
-
- /* Ring-wrap flag in length field, use for last ring entry.
- 0x01000000 means chain on buffer2 address,
- 0x02000000 means use the ring start address in CSR2/3.
- Note: Some work-alike chips do not function correctly in chained mode.
- The ASIX chip works only in chained mode.
- Thus we indicate ring mode, but always write the 'next' field for
- chained mode as well. */
- #define DESC_RING_WRAP 0x02000000
-
- /* transmit and receive descriptor format */
- struct tulip_rx_desc {
- volatile u32 status;
- u32 length;
- u32 buffer1, buffer2;
- };
-
- struct tulip_tx_desc {
- volatile u32 status;
- u32 length;
- u32 buffer1, buffer2;
- };
-
- /*********************************************************************/
- /* Global Storage */
- /*********************************************************************/
-
- static u32 ioaddr;
-
- struct tulip_private {
- int cur_rx;
- int chip_id; /* index into tulip_tbl[] */
- int pci_id_idx; /* index into pci_id_tbl[] */
- int revision;
- int flags;
- unsigned short vendor_id; /* PCI card vendor code */
- unsigned short dev_id; /* PCI card device code */
- unsigned char ehdr[ETH_HLEN]; /* buffer for ethernet header */
- const char *nic_name;
- unsigned int csr0, csr6; /* Current CSR0, CSR6 settings. */
- unsigned int if_port;
- unsigned int full_duplex; /* Full-duplex operation requested. */
- unsigned int full_duplex_lock;
- unsigned int medialock; /* Do not sense media type. */
- unsigned int mediasense; /* Media sensing in progress. */
- unsigned int nway, nwayset; /* 21143 internal NWay. */
- unsigned int default_port;
- unsigned char eeprom[EEPROM_SIZE]; /* Serial EEPROM contents. */
- u8 media_table_storage[(sizeof(struct mediatable) + 32*sizeof(struct medialeaf))];
- u16 sym_advertise, mii_advertise; /* NWay to-advertise. */
- struct mediatable *mtable;
- u16 lpar; /* 21143 Link partner ability. */
- u16 advertising[4]; /* MII advertise, from SROM table. */
- signed char phys[4], mii_cnt; /* MII device addresses. */
- int cur_index; /* Current media index. */
- int saved_if_port;
- };
-
- /* Note: transmit and receive buffers must be longword aligned and
- longword divisable */
-
- #define TX_RING_SIZE 2
- #define RX_RING_SIZE 4
- struct {
- struct tulip_tx_desc tx_ring[TX_RING_SIZE];
- unsigned char txb[BUFLEN];
- struct tulip_rx_desc rx_ring[RX_RING_SIZE];
- unsigned char rxb[RX_RING_SIZE * BUFLEN];
- struct tulip_private tpx;
- } tulip_bss __shared __attribute__ ((aligned(4)));
- #define tx_ring tulip_bss.tx_ring
- #define txb tulip_bss.txb
- #define rx_ring tulip_bss.rx_ring
- #define rxb tulip_bss.rxb
-
- static struct tulip_private *tp;
-
- /* Known cards that have old-style EEPROMs.
- Writing this table is described at
- http://cesdis.gsfc.nasa.gov/linux/drivers/tulip-drivers/tulip-media.html */
- static struct fixups {
- char *name;
- unsigned char addr0, addr1, addr2;
- u16 newtable[32]; /* Max length below. */
- } eeprom_fixups[] = {
- {"Asante", 0, 0, 0x94, {0x1e00, 0x0000, 0x0800, 0x0100, 0x018c,
- 0x0000, 0x0000, 0xe078, 0x0001, 0x0050, 0x0018 }},
- {"SMC9332DST", 0, 0, 0xC0, { 0x1e00, 0x0000, 0x0800, 0x041f,
- 0x0000, 0x009E, /* 10baseT */
- 0x0004, 0x009E, /* 10baseT-FD */
- 0x0903, 0x006D, /* 100baseTx */
- 0x0905, 0x006D, /* 100baseTx-FD */ }},
- {"Cogent EM100", 0, 0, 0x92, { 0x1e00, 0x0000, 0x0800, 0x063f,
- 0x0107, 0x8021, /* 100baseFx */
- 0x0108, 0x8021, /* 100baseFx-FD */
- 0x0100, 0x009E, /* 10baseT */
- 0x0104, 0x009E, /* 10baseT-FD */
- 0x0103, 0x006D, /* 100baseTx */
- 0x0105, 0x006D, /* 100baseTx-FD */ }},
- {"Maxtech NX-110", 0, 0, 0xE8, { 0x1e00, 0x0000, 0x0800, 0x0513,
- 0x1001, 0x009E, /* 10base2, CSR12 0x10*/
- 0x0000, 0x009E, /* 10baseT */
- 0x0004, 0x009E, /* 10baseT-FD */
- 0x0303, 0x006D, /* 100baseTx, CSR12 0x03 */
- 0x0305, 0x006D, /* 100baseTx-FD CSR12 0x03 */}},
- {"Accton EN1207", 0, 0, 0xE8, { 0x1e00, 0x0000, 0x0800, 0x051F,
- 0x1B01, 0x0000, /* 10base2, CSR12 0x1B */
- 0x0B00, 0x009E, /* 10baseT, CSR12 0x0B */
- 0x0B04, 0x009E, /* 10baseT-FD,CSR12 0x0B */
- 0x1B03, 0x006D, /* 100baseTx, CSR12 0x1B */
- 0x1B05, 0x006D, /* 100baseTx-FD CSR12 0x1B */
- }},
- {0, 0, 0, 0, {}}};
-
- static const char * block_name[] = {"21140 non-MII", "21140 MII PHY",
- "21142 Serial PHY", "21142 MII PHY", "21143 SYM PHY", "21143 reset method"};
-
-
- /*********************************************************************/
- /* Function Prototypes */
- /*********************************************************************/
- static int mdio_read(struct nic *nic, int phy_id, int location);
- static void mdio_write(struct nic *nic, int phy_id, int location, int value);
- static int read_eeprom(unsigned long ioaddr, int location, int addr_len);
- static void parse_eeprom(struct nic *nic);
- static int tulip_probe(struct nic *nic,struct pci_device *pci);
- static void tulip_init_ring(struct nic *nic);
- static void tulip_reset(struct nic *nic);
- static void tulip_transmit(struct nic *nic, const char *d, unsigned int t,
- unsigned int s, const char *p);
- static int tulip_poll(struct nic *nic, int retrieve);
- static void tulip_disable(struct nic *nic);
- static void nway_start(struct nic *nic);
- static void pnic_do_nway(struct nic *nic);
- static void select_media(struct nic *nic, int startup);
- static void init_media(struct nic *nic);
- static void start_link(struct nic *nic);
- static int tulip_check_duplex(struct nic *nic);
-
- static void tulip_wait(unsigned int nticks);
-
- #ifdef TULIP_DEBUG_WHERE
- static void whereami(const char *str);
- #endif
-
- #ifdef TULIP_DEBUG
- static void tulip_more(void);
- #endif
-
-
- /*********************************************************************/
- /* Utility Routines */
- /*********************************************************************/
-
- #ifdef TULIP_DEBUG_WHERE
- static void whereami (const char *str, struct pci_device *pci)
- {
- printf("%s: %s\n", tp->nic_name, str);
- /* sleep(2); */
- }
- #endif
-
- #ifdef TULIP_DEBUG
- static void tulip_more(void)
- {
- printf("\n\n-- more --");
- while (!iskey())
- /* wait */;
- getchar();
- printf("\n\n");
- }
- #endif /* TULIP_DEBUG */
-
- static void tulip_wait(unsigned int nticks)
- {
- unsigned int to = currticks() + nticks;
- while (currticks() < to)
- /* wait */ ;
- }
-
-
- /*********************************************************************/
- /* Media Descriptor Code */
- /*********************************************************************/
-
- /* MII transceiver control section.
- Read and write the MII registers using software-generated serial
- MDIO protocol. See the MII specifications or DP83840A data sheet
- for details. */
-
- /* The maximum data clock rate is 2.5 Mhz. The minimum timing is usually
- met by back-to-back PCI I/O cycles, but we insert a delay to avoid
- "overclocking" issues or future 66Mhz PCI. */
- #define mdio_delay() inl(mdio_addr)
-
- /* Read and write the MII registers using software-generated serial
- MDIO protocol. It is just different enough from the EEPROM protocol
- to not share code. The maxium data clock rate is 2.5 Mhz. */
- #define MDIO_SHIFT_CLK 0x10000
- #define MDIO_DATA_WRITE0 0x00000
- #define MDIO_DATA_WRITE1 0x20000
- #define MDIO_ENB 0x00000 /* Ignore the 0x02000 databook setting. */
- #define MDIO_ENB_IN 0x40000
- #define MDIO_DATA_READ 0x80000
-
- /* MII transceiver control section.
- Read and write the MII registers using software-generated serial
- MDIO protocol. See the MII specifications or DP83840A data sheet
- for details. */
-
- int mdio_read(struct nic *nic __unused, int phy_id, int location)
- {
- int i;
- int read_cmd = (0xf6 << 10) | (phy_id << 5) | location;
- int retval = 0;
- long mdio_addr = ioaddr + CSR9;
-
- #ifdef TULIP_DEBUG_WHERE
- whereami("mdio_read\n");
- #endif
-
- if (tp->chip_id == LC82C168) {
- int i = 1000;
- outl(0x60020000 + (phy_id<<23) + (location<<18), ioaddr + 0xA0);
- inl(ioaddr + 0xA0);
- inl(ioaddr + 0xA0);
- while (--i > 0)
- if ( ! ((retval = inl(ioaddr + 0xA0)) & 0x80000000))
- return retval & 0xffff;
- return 0xffff;
- }
-
- if (tp->chip_id == COMET) {
- if (phy_id == 1) {
- if (location < 7)
- return inl(ioaddr + 0xB4 + (location<<2));
- else if (location == 17)
- return inl(ioaddr + 0xD0);
- else if (location >= 29 && location <= 31)
- return inl(ioaddr + 0xD4 + ((location-29)<<2));
- }
- return 0xffff;
- }
-
- /* Establish sync by sending at least 32 logic ones. */
- for (i = 32; i >= 0; i--) {
- outl(MDIO_ENB | MDIO_DATA_WRITE1, mdio_addr);
- mdio_delay();
- outl(MDIO_ENB | MDIO_DATA_WRITE1 | MDIO_SHIFT_CLK, mdio_addr);
- mdio_delay();
- }
- /* Shift the read command bits out. */
- for (i = 15; i >= 0; i--) {
- int dataval = (read_cmd & (1 << i)) ? MDIO_DATA_WRITE1 : 0;
-
- outl(MDIO_ENB | dataval, mdio_addr);
- mdio_delay();
- outl(MDIO_ENB | dataval | MDIO_SHIFT_CLK, mdio_addr);
- mdio_delay();
- }
- /* Read the two transition, 16 data, and wire-idle bits. */
- for (i = 19; i > 0; i--) {
- outl(MDIO_ENB_IN, mdio_addr);
- mdio_delay();
- retval = (retval << 1) | ((inl(mdio_addr) & MDIO_DATA_READ) ? 1 : 0);
- outl(MDIO_ENB_IN | MDIO_SHIFT_CLK, mdio_addr);
- mdio_delay();
- }
- return (retval>>1) & 0xffff;
- }
-
- void mdio_write(struct nic *nic __unused, int phy_id, int location, int value)
- {
- int i;
- int cmd = (0x5002 << 16) | (phy_id << 23) | (location<<18) | value;
- long mdio_addr = ioaddr + CSR9;
-
- #ifdef TULIP_DEBUG_WHERE
- whereami("mdio_write\n");
- #endif
-
- if (tp->chip_id == LC82C168) {
- int i = 1000;
- outl(cmd, ioaddr + 0xA0);
- do
- if ( ! (inl(ioaddr + 0xA0) & 0x80000000))
- break;
- while (--i > 0);
- return;
- }
-
- if (tp->chip_id == COMET) {
- if (phy_id != 1)
- return;
- if (location < 7)
- outl(value, ioaddr + 0xB4 + (location<<2));
- else if (location == 17)
- outl(value, ioaddr + 0xD0);
- else if (location >= 29 && location <= 31)
- outl(value, ioaddr + 0xD4 + ((location-29)<<2));
- return;
- }
-
- /* Establish sync by sending 32 logic ones. */
- for (i = 32; i >= 0; i--) {
- outl(MDIO_ENB | MDIO_DATA_WRITE1, mdio_addr);
- mdio_delay();
- outl(MDIO_ENB | MDIO_DATA_WRITE1 | MDIO_SHIFT_CLK, mdio_addr);
- mdio_delay();
- }
- /* Shift the command bits out. */
- for (i = 31; i >= 0; i--) {
- int dataval = (cmd & (1 << i)) ? MDIO_DATA_WRITE1 : 0;
- outl(MDIO_ENB | dataval, mdio_addr);
- mdio_delay();
- outl(MDIO_ENB | dataval | MDIO_SHIFT_CLK, mdio_addr);
- mdio_delay();
- }
- /* Clear out extra bits. */
- for (i = 2; i > 0; i--) {
- outl(MDIO_ENB_IN, mdio_addr);
- mdio_delay();
- outl(MDIO_ENB_IN | MDIO_SHIFT_CLK, mdio_addr);
- mdio_delay();
- }
- }
-
-
- /*********************************************************************/
- /* EEPROM Reading Code */
- /*********************************************************************/
- /* EEPROM routines adapted from the Linux Tulip Code */
- /* Reading a serial EEPROM is a "bit" grungy, but we work our way
- through:->.
- */
- static int read_eeprom(unsigned long ioaddr, int location, int addr_len)
- {
- int i;
- unsigned short retval = 0;
- long ee_addr = ioaddr + CSR9;
- int read_cmd = location | EE_READ_CMD;
-
- #ifdef TULIP_DEBUG_WHERE
- whereami("read_eeprom\n");
- #endif
-
- outl(EE_ENB & ~EE_CS, ee_addr);
- outl(EE_ENB, ee_addr);
-
- /* Shift the read command bits out. */
- for (i = 4 + addr_len; i >= 0; i--) {
- short dataval = (read_cmd & (1 << i)) ? EE_DATA_WRITE : 0;
- outl(EE_ENB | dataval, ee_addr);
- eeprom_delay();
- outl(EE_ENB | dataval | EE_SHIFT_CLK, ee_addr);
- eeprom_delay();
- }
- outl(EE_ENB, ee_addr);
-
- for (i = 16; i > 0; i--) {
- outl(EE_ENB | EE_SHIFT_CLK, ee_addr);
- eeprom_delay();
- retval = (retval << 1) | ((inl(ee_addr) & EE_DATA_READ) ? 1 : 0);
- outl(EE_ENB, ee_addr);
- eeprom_delay();
- }
-
- /* Terminate the EEPROM access. */
- outl(EE_ENB & ~EE_CS, ee_addr);
- return retval;
- }
-
-
- /*********************************************************************/
- /* EEPROM Parsing Code */
- /*********************************************************************/
- static void parse_eeprom(struct nic *nic)
- {
- unsigned char *p, *ee_data = tp->eeprom;
- int new_advertise = 0;
- int i;
-
- #ifdef TULIP_DEBUG_WHERE
- whereami("parse_eeprom\n");
- #endif
-
- tp->mtable = 0;
- /* Detect an old-style (SA only) EEPROM layout:
- memcmp(ee_data, ee_data+16, 8). */
- for (i = 0; i < 8; i ++)
- if (ee_data[i] != ee_data[16+i])
- break;
- if (i >= 8) {
- /* Do a fix-up based on the vendor half of the station address. */
- for (i = 0; eeprom_fixups[i].name; i++) {
- if (nic->node_addr[0] == eeprom_fixups[i].addr0
- && nic->node_addr[1] == eeprom_fixups[i].addr1
- && nic->node_addr[2] == eeprom_fixups[i].addr2) {
- if (nic->node_addr[2] == 0xE8 && ee_data[0x1a] == 0x55)
- i++; /* An Accton EN1207, not an outlaw Maxtech. */
- memcpy(ee_data + 26, eeprom_fixups[i].newtable,
- sizeof(eeprom_fixups[i].newtable));
- #ifdef TULIP_DEBUG
- printf("%s: Old format EEPROM on '%s' board.\n%s: Using substitute media control info.\n",
- tp->nic_name, eeprom_fixups[i].name, tp->nic_name);
- #endif
- break;
- }
- }
- if (eeprom_fixups[i].name == NULL) { /* No fixup found. */
- #ifdef TULIP_DEBUG
- printf("%s: Old style EEPROM with no media selection information.\n",
- tp->nic_name);
- #endif
- return;
- }
- }
-
- if (ee_data[19] > 1) {
- #ifdef TULIP_DEBUG
- printf("%s: Multiport cards (%d ports) may not work correctly.\n",
- tp->nic_name, ee_data[19]);
- #endif
- }
-
- p = (void *)ee_data + ee_data[27];
-
- if (ee_data[27] == 0) { /* No valid media table. */
- #ifdef TULIP_DEBUG
- if (tulip_debug > 1) {
- printf("%s: No Valid Media Table. ee_data[27] = %hhX\n",
- tp->nic_name, ee_data[27]);
- }
- #endif
- } else if (tp->chip_id == DC21041) {
- int media = get_u16(p);
- int count = p[2];
- p += 3;
-
- printf("%s: 21041 Media table, default media %hX (%s).\n",
- tp->nic_name, media,
- media & 0x0800 ? "Autosense" : medianame[media & 15]);
- for (i = 0; i < count; i++) {
- unsigned char media_block = *p++;
- int media_code = media_block & MEDIA_MASK;
- if (media_block & 0x40)
- p += 6;
- switch(media_code) {
- case 0: new_advertise |= 0x0020; break;
- case 4: new_advertise |= 0x0040; break;
- }
- printf("%s: 21041 media #%d, %s.\n",
- tp->nic_name, media_code, medianame[media_code]);
- }
- } else {
- unsigned char csr12dir = 0;
- int count;
- struct mediatable *mtable;
- u16 media = get_u16(p);
-
- p += 2;
- if (tp->flags & CSR12_IN_SROM)
- csr12dir = *p++;
- count = *p++;
-
- tp->mtable = mtable = (struct mediatable *)&tp->media_table_storage[0];
-
- mtable->defaultmedia = media;
- mtable->leafcount = count;
- mtable->csr12dir = csr12dir;
- mtable->has_nonmii = mtable->has_mii = mtable->has_reset = 0;
- mtable->csr15dir = mtable->csr15val = 0;
-
- printf("%s: EEPROM default media type %s.\n", tp->nic_name,
- media & 0x0800 ? "Autosense" : medianame[media & MEDIA_MASK]);
-
- for (i = 0; i < count; i++) {
- struct medialeaf *leaf = &mtable->mleaf[i];
-
- if ((p[0] & 0x80) == 0) { /* 21140 Compact block. */
- leaf->type = 0;
- leaf->media = p[0] & 0x3f;
- leaf->leafdata = p;
- if ((p[2] & 0x61) == 0x01) /* Bogus, but Znyx boards do it. */
- mtable->has_mii = 1;
- p += 4;
- } else {
- switch(leaf->type = p[1]) {
- case 5:
- mtable->has_reset = i;
- leaf->media = p[2] & 0x0f;
- break;
- case 1: case 3:
- mtable->has_mii = 1;
- leaf->media = 11;
- break;
- case 2:
- if ((p[2] & 0x3f) == 0) {
- u32 base15 = (p[2] & 0x40) ? get_u16(p + 7) : 0x0008;
- u16 *p1 = (u16 *)(p + (p[2] & 0x40 ? 9 : 3));
- mtable->csr15dir = (get_unaligned(p1 + 0)<<16) + base15;
- mtable->csr15val = (get_unaligned(p1 + 1)<<16) + base15;
- }
- /* Fall through. */
- case 0: case 4:
- mtable->has_nonmii = 1;
- leaf->media = p[2] & MEDIA_MASK;
- switch (leaf->media) {
- case 0: new_advertise |= 0x0020; break;
- case 4: new_advertise |= 0x0040; break;
- case 3: new_advertise |= 0x0080; break;
- case 5: new_advertise |= 0x0100; break;
- case 6: new_advertise |= 0x0200; break;
- }
- break;
- default:
- leaf->media = 19;
- }
- leaf->leafdata = p + 2;
- p += (p[0] & 0x3f) + 1;
- }
- #ifdef TULIP_DEBUG
- if (tulip_debug > 1 && leaf->media == 11) {
- unsigned char *bp = leaf->leafdata;
- printf("%s: MII interface PHY %d, setup/reset sequences %d/%d long, capabilities %hhX %hhX.\n",
- tp->nic_name, bp[0], bp[1], bp[2 + bp[1]*2],
- bp[5 + bp[2 + bp[1]*2]*2], bp[4 + bp[2 + bp[1]*2]*2]);
- }
- #endif
- printf("%s: Index #%d - Media %s (#%d) described "
- "by a %s (%d) block.\n",
- tp->nic_name, i, medianame[leaf->media], leaf->media,
- leaf->type < 6 ? block_name[leaf->type] : "UNKNOWN",
- leaf->type);
- }
- if (new_advertise)
- tp->sym_advertise = new_advertise;
- }
- }
-
-
- /*********************************************************************/
- /* tulip_init_ring - setup the tx and rx descriptors */
- /*********************************************************************/
- static void tulip_init_ring(struct nic *nic __unused)
- {
- int i;
-
- #ifdef TULIP_DEBUG_WHERE
- whereami("tulip_init_ring\n");
- #endif
-
- tp->cur_rx = 0;
-
- for (i = 0; i < RX_RING_SIZE; i++) {
- rx_ring[i].status = cpu_to_le32(0x80000000);
- rx_ring[i].length = cpu_to_le32(BUFLEN);
- rx_ring[i].buffer1 = virt_to_le32desc(&rxb[i * BUFLEN]);
- rx_ring[i].buffer2 = virt_to_le32desc(&rx_ring[i+1]);
- }
- /* Mark the last entry as wrapping the ring. */
- rx_ring[i-1].length = cpu_to_le32(DESC_RING_WRAP | BUFLEN);
- rx_ring[i-1].buffer2 = virt_to_le32desc(&rx_ring[0]);
-
- /* We only use 1 transmit buffer, but we use 2 descriptors so
- transmit engines have somewhere to point to if they feel the need */
-
- tx_ring[0].status = 0x00000000;
- tx_ring[0].buffer1 = virt_to_le32desc(&txb[0]);
- tx_ring[0].buffer2 = virt_to_le32desc(&tx_ring[1]);
-
- /* this descriptor should never get used, since it will never be owned
- by the machine (status will always == 0) */
- tx_ring[1].status = 0x00000000;
- tx_ring[1].buffer1 = virt_to_le32desc(&txb[0]);
- tx_ring[1].buffer2 = virt_to_le32desc(&tx_ring[0]);
-
- /* Mark the last entry as wrapping the ring, though this should never happen */
- tx_ring[1].length = cpu_to_le32(DESC_RING_WRAP | BUFLEN);
- }
-
-
- static void set_rx_mode(struct nic *nic __unused) {
- int csr6 = inl(ioaddr + CSR6) & ~0x00D5;
-
- tp->csr6 &= ~0x00D5;
-
- /* !IFF_PROMISC */
- tp->csr6 |= AcceptAllMulticast;
- csr6 |= AcceptAllMulticast;
-
- outl(csr6, ioaddr + CSR6);
-
-
-
- }
-
- /*********************************************************************/
- /* eth_reset - Reset adapter */
- /*********************************************************************/
- static void tulip_reset(struct nic *nic)
- {
- int i;
- unsigned long to;
-
- #ifdef TULIP_DEBUG_WHERE
- whereami("tulip_reset\n");
- #endif
-
- /* Stop Tx and RX */
- outl(inl(ioaddr + CSR6) & ~0x00002002, ioaddr + CSR6);
-
- /* On some chip revs we must set the MII/SYM port before the reset!? */
- if (tp->mii_cnt || (tp->mtable && tp->mtable->has_mii)) {
- outl(0x814C0000, ioaddr + CSR6);
- }
-
- /* Reset the chip, holding bit 0 set at least 50 PCI cycles. */
- outl(0x00000001, ioaddr + CSR0);
- tulip_wait(1);
-
- /* turn off reset and set cache align=16lword, burst=unlimit */
- outl(tp->csr0, ioaddr + CSR0);
-
- /* Wait the specified 50 PCI cycles after a reset */
- tulip_wait(1);
-
- /* set up transmit and receive descriptors */
- tulip_init_ring(nic);
-
- if (tp->chip_id == PNIC2) {
- u32 addr_high = (nic->node_addr[1]<<8) + (nic->node_addr[0]<<0);
- /* This address setting does not appear to impact chip operation?? */
- outl((nic->node_addr[5]<<8) + nic->node_addr[4] +
- (nic->node_addr[3]<<24) + (nic->node_addr[2]<<16),
- ioaddr + 0xB0);
- outl(addr_high + (addr_high<<16), ioaddr + 0xB8);
- }
-
- /* MC_HASH_ONLY boards don't support setup packets */
- if (tp->flags & MC_HASH_ONLY) {
- u32 addr_low = cpu_to_le32(get_unaligned((u32 *)nic->node_addr));
- u32 addr_high = cpu_to_le32(get_unaligned((u16 *)(nic->node_addr+4)));
-
- /* clear multicast hash filters and setup MAC address filters */
- if (tp->flags & IS_ASIX) {
- outl(0, ioaddr + CSR13);
- outl(addr_low, ioaddr + CSR14);
- outl(1, ioaddr + CSR13);
- outl(addr_high, ioaddr + CSR14);
- outl(2, ioaddr + CSR13);
- outl(0, ioaddr + CSR14);
- outl(3, ioaddr + CSR13);
- outl(0, ioaddr + CSR14);
- } else if (tp->chip_id == COMET) {
- outl(addr_low, ioaddr + 0xA4);
- outl(addr_high, ioaddr + 0xA8);
- outl(0, ioaddr + 0xAC);
- outl(0, ioaddr + 0xB0);
- }
- } else {
- /* for other boards we send a setup packet to initialize
- the filters */
- u32 tx_flags = 0x08000000 | 192;
-
- /* construct perfect filter frame with mac address as first match
- and broadcast address for all others */
- for (i=0; i<192; i++)
- txb[i] = 0xFF;
- txb[0] = nic->node_addr[0];
- txb[1] = nic->node_addr[1];
- txb[4] = nic->node_addr[2];
- txb[5] = nic->node_addr[3];
- txb[8] = nic->node_addr[4];
- txb[9] = nic->node_addr[5];
-
- tx_ring[0].length = cpu_to_le32(tx_flags);
- tx_ring[0].buffer1 = virt_to_le32desc(&txb[0]);
- tx_ring[0].status = cpu_to_le32(0x80000000);
- }
-
- /* Point to rx and tx descriptors */
- outl(virt_to_le32desc(&rx_ring[0]), ioaddr + CSR3);
- outl(virt_to_le32desc(&tx_ring[0]), ioaddr + CSR4);
-
- init_media(nic);
-
- /* set the chip's operating mode (but don't turn on xmit and recv yet) */
- outl((tp->csr6 & ~0x00002002), ioaddr + CSR6);
-
- /* send setup packet for cards that support it */
- if (!(tp->flags & MC_HASH_ONLY)) {
- /* enable transmit wait for completion */
- outl(tp->csr6 | 0x00002000, ioaddr + CSR6);
- /* immediate transmit demand */
- outl(0, ioaddr + CSR1);
-
- to = currticks() + TX_TIME_OUT;
- while ((tx_ring[0].status & 0x80000000) && (currticks() < to))
- /* wait */ ;
-
- if (currticks() >= to) {
- printf ("%s: TX Setup Timeout.\n", tp->nic_name);
- }
- }
-
- if (tp->chip_id == LC82C168)
- tulip_check_duplex(nic);
-
- set_rx_mode(nic);
-
- /* enable transmit and receive */
- outl(tp->csr6 | 0x00002002, ioaddr + CSR6);
- }
-
-
- /*********************************************************************/
- /* eth_transmit - Transmit a frame */
- /*********************************************************************/
- static void tulip_transmit(struct nic *nic, const char *d, unsigned int t,
- unsigned int s, const char *p)
- {
- u16 nstype;
- u32 to;
- u32 csr6 = inl(ioaddr + CSR6);
-
- #ifdef TULIP_DEBUG_WHERE
- whereami("tulip_transmit\n");
- #endif
-
- /* Disable Tx */
- outl(csr6 & ~0x00002000, ioaddr + CSR6);
-
- memcpy(txb, d, ETH_ALEN);
- memcpy(txb + ETH_ALEN, nic->node_addr, ETH_ALEN);
- nstype = htons((u16) t);
- memcpy(txb + 2 * ETH_ALEN, (u8 *)&nstype, 2);
- memcpy(txb + ETH_HLEN, p, s);
-
- s += ETH_HLEN;
- s &= 0x0FFF;
-
- /* pad to minimum packet size */
- while (s < ETH_ZLEN)
- txb[s++] = '\0';
-
- #ifdef TULIP_DEBUG
- if (tulip_debug > 1)
- printf("%s: sending %d bytes ethtype %hX\n", tp->nic_name, s, t);
- #endif
-
- /* setup the transmit descriptor */
- /* 0x60000000 = no interrupt on completion */
- tx_ring[0].length = cpu_to_le32(0x60000000 | s);
- tx_ring[0].status = cpu_to_le32(0x80000000);
-
- /* Point to transmit descriptor */
- outl(virt_to_le32desc(&tx_ring[0]), ioaddr + CSR4);
-
- /* Enable Tx */
- outl(csr6 | 0x00002000, ioaddr + CSR6);
- /* immediate transmit demand */
- outl(0, ioaddr + CSR1);
-
- to = currticks() + TX_TIME_OUT;
- while ((tx_ring[0].status & 0x80000000) && (currticks() < to))
- /* wait */ ;
-
- if (currticks() >= to) {
- printf ("TX Timeout!\n");
- }
-
- /* Disable Tx */
- outl(csr6 & ~0x00002000, ioaddr + CSR6);
- }
-
- /*********************************************************************/
- /* eth_poll - Wait for a frame */
- /*********************************************************************/
- static int tulip_poll(struct nic *nic, int retrieve)
- {
-
- #ifdef TULIP_DEBUG_WHERE
- whereami("tulip_poll\n");
- #endif
-
- /* no packet waiting. packet still owned by NIC */
- if (rx_ring[tp->cur_rx].status & 0x80000000)
- return 0;
-
- if ( ! retrieve ) return 1;
-
- #ifdef TULIP_DEBUG_WHERE
- whereami("tulip_poll got one\n");
- #endif
-
- nic->packetlen = (rx_ring[tp->cur_rx].status & 0x3FFF0000) >> 16;
-
- /* if we get a corrupted packet. throw it away and move on */
- if (rx_ring[tp->cur_rx].status & 0x00008000) {
- /* return the descriptor and buffer to receive ring */
- rx_ring[tp->cur_rx].status = 0x80000000;
- tp->cur_rx = (++tp->cur_rx) % RX_RING_SIZE;
- return 0;
- }
-
- /* copy packet to working buffer */
- memcpy(nic->packet, rxb + tp->cur_rx * BUFLEN, nic->packetlen);
-
- /* return the descriptor and buffer to receive ring */
- rx_ring[tp->cur_rx].status = 0x80000000;
- tp->cur_rx = (++tp->cur_rx) % RX_RING_SIZE;
-
- return 1;
- }
-
- /*********************************************************************/
- /* eth_disable - Disable the interface */
- /*********************************************************************/
- static void tulip_disable ( struct nic *nic ) {
-
- #ifdef TULIP_DEBUG_WHERE
- whereami("tulip_disable\n");
- #endif
-
- tulip_reset(nic);
-
- /* disable interrupts */
- outl(0x00000000, ioaddr + CSR7);
-
- /* Stop the chip's Tx and Rx processes. */
- outl(inl(ioaddr + CSR6) & ~0x00002002, ioaddr + CSR6);
-
- /* Clear the missed-packet counter. */
- (volatile unsigned long)inl(ioaddr + CSR8);
- }
-
- /*********************************************************************/
- /*IRQ - Enable, Disable, or Force interrupts */
- /*********************************************************************/
- static void tulip_irq(struct nic *nic __unused, irq_action_t action __unused)
- {
- switch ( action ) {
- case DISABLE :
- break;
- case ENABLE :
- break;
- case FORCE :
- break;
- }
- }
-
- static struct nic_operations tulip_operations = {
- .connect = dummy_connect,
- .poll = tulip_poll,
- .transmit = tulip_transmit,
- .irq = tulip_irq,
-
- };
-
- /*********************************************************************/
- /* eth_probe - Look for an adapter */
- /*********************************************************************/
- static int tulip_probe ( struct nic *nic, struct pci_device *pci ) {
-
- u32 i;
- u8 chip_rev;
- u8 ee_data[EEPROM_SIZE];
- unsigned short sum;
- int chip_idx;
- static unsigned char last_phys_addr[ETH_ALEN] = {0x00, 'L', 'i', 'n', 'u', 'x'};
-
- if (pci->ioaddr == 0)
- return 0;
-
- ioaddr = pci->ioaddr;
- nic->ioaddr = pci->ioaddr & ~3;
- nic->irqno = 0;
-
- /* point to private storage */
- tp = &tulip_bss.tpx;
-
- tp->vendor_id = pci->vendor;
- tp->dev_id = pci->device;
- tp->nic_name = pci->driver_name;
-
- tp->if_port = 0;
- tp->default_port = 0;
-
- adjust_pci_device(pci);
-
- /* disable interrupts */
- outl(0x00000000, ioaddr + CSR7);
-
- /* Stop the chip's Tx and Rx processes. */
- outl(inl(ioaddr + CSR6) & ~0x00002002, ioaddr + CSR6);
-
- /* Clear the missed-packet counter. */
- (volatile unsigned long)inl(ioaddr + CSR8);
-
- printf("\n"); /* so we start on a fresh line */
- #ifdef TULIP_DEBUG_WHERE
- whereami("tulip_probe\n");
- #endif
-
- #ifdef TULIP_DEBUG
- if (tulip_debug > 1)
- printf ("%s: Looking for Tulip Chip: Vendor=%hX Device=%hX\n", tp->nic_name,
- tp->vendor, tp->dev_id);
- #endif
-
- /* Figure out which chip we're dealing with */
- i = 0;
- chip_idx = -1;
-
- while (pci_id_tbl[i].name) {
- if ( (((u32) tp->dev_id << 16) | tp->vendor_id) ==
- (pci_id_tbl[i].id.pci & pci_id_tbl[i].id.pci_mask) ) {
- chip_idx = pci_id_tbl[i].drv_flags;
- break;
- }
- i++;
- }
-
- if (chip_idx == -1) {
- printf ("%s: Unknown Tulip Chip: Vendor=%hX Device=%hX\n", tp->nic_name,
- tp->vendor_id, tp->dev_id);
- return 0;
- }
-
- tp->pci_id_idx = i;
- tp->flags = tulip_tbl[chip_idx].flags;
-
- #ifdef TULIP_DEBUG
- if (tulip_debug > 1) {
- printf ("%s: tp->pci_id_idx == %d, name == %s\n", tp->nic_name,
- tp->pci_id_idx, pci_id_tbl[tp->pci_id_idx].name);
- printf ("%s: chip_idx == %d, name == %s\n", tp->nic_name, chip_idx,
- tulip_tbl[chip_idx].chip_name);
- }
- #endif
-
- /* Bring the 21041/21143 out of sleep mode.
- Caution: Snooze mode does not work with some boards! */
- if (tp->flags & HAS_PWRDWN)
- pci_write_config_dword(pci, 0x40, 0x00000000);
-
- if (inl(ioaddr + CSR5) == 0xFFFFFFFF) {
- printf("%s: The Tulip chip at %X is not functioning.\n",
- tp->nic_name, (unsigned int) ioaddr);
- return 0;
- }
-
- pci_read_config_byte(pci, PCI_REVISION, &chip_rev);
-
- printf("%s: [chip: %s] rev %d at %hX\n", tp->nic_name,
- tulip_tbl[chip_idx].chip_name, chip_rev, (unsigned int) ioaddr);
- printf("%s: Vendor=%hX Device=%hX", tp->nic_name, tp->vendor_id, tp->dev_id);
-
- if (chip_idx == DC21041 && inl(ioaddr + CSR9) & 0x8000) {
- printf(" 21040 compatible mode.");
- chip_idx = DC21040;
- }
-
- printf("\n");
-
- /* The SROM/EEPROM interface varies dramatically. */
- sum = 0;
- if (chip_idx == DC21040) {
- outl(0, ioaddr + CSR9); /* Reset the pointer with a dummy write. */
- for (i = 0; i < ETH_ALEN; i++) {
- int value, boguscnt = 100000;
- do
- value = inl(ioaddr + CSR9);
- while (value < 0 && --boguscnt > 0);
- nic->node_addr[i] = value;
- sum += value & 0xff;
- }
- } else if (chip_idx == LC82C168) {
- for (i = 0; i < 3; i++) {
- int value, boguscnt = 100000;
- outl(0x600 | i, ioaddr + 0x98);
- do
- value = inl(ioaddr + CSR9);
- while (value < 0 && --boguscnt > 0);
- put_unaligned(le16_to_cpu(value), ((u16*)nic->node_addr) + i);
- sum += value & 0xffff;
- }
- } else if (chip_idx == COMET) {
- /* No need to read the EEPROM. */
- put_unaligned(inl(ioaddr + 0xA4), (u32 *)nic->node_addr);
- put_unaligned(inl(ioaddr + 0xA8), (u16 *)(nic->node_addr + 4));
- for (i = 0; i < ETH_ALEN; i ++)
- sum += nic->node_addr[i];
- } else {
- /* A serial EEPROM interface, we read now and sort it out later. */
- int sa_offset = 0;
- int ee_addr_size = read_eeprom(ioaddr, 0xff, 8) & 0x40000 ? 8 : 6;
-
- for (i = 0; i < sizeof(ee_data)/2; i++)
- ((u16 *)ee_data)[i] =
- le16_to_cpu(read_eeprom(ioaddr, i, ee_addr_size));
-
- /* DEC now has a specification (see Notes) but early board makers
- just put the address in the first EEPROM locations. */
- /* This does memcmp(eedata, eedata+16, 8) */
- for (i = 0; i < 8; i ++)
- if (ee_data[i] != ee_data[16+i])
- sa_offset = 20;
- if (ee_data[0] == 0xff && ee_data[1] == 0xff && ee_data[2] == 0) {
- sa_offset = 2; /* Grrr, damn Matrox boards. */
- }
- for (i = 0; i < ETH_ALEN; i ++) {
- nic->node_addr[i] = ee_data[i + sa_offset];
- sum += ee_data[i + sa_offset];
- }
- }
- /* Lite-On boards have the address byte-swapped. */
- if ((nic->node_addr[0] == 0xA0 || nic->node_addr[0] == 0xC0)
- && nic->node_addr[1] == 0x00)
- for (i = 0; i < ETH_ALEN; i+=2) {
- char tmp = nic->node_addr[i];
- nic->node_addr[i] = nic->node_addr[i+1];
- nic->node_addr[i+1] = tmp;
- }
-
- if (sum == 0 || sum == ETH_ALEN*0xff) {
- printf("%s: EEPROM not present!\n", tp->nic_name);
- for (i = 0; i < ETH_ALEN-1; i++)
- nic->node_addr[i] = last_phys_addr[i];
- nic->node_addr[i] = last_phys_addr[i] + 1;
- }
-
- for (i = 0; i < ETH_ALEN; i++)
- last_phys_addr[i] = nic->node_addr[i];
-
- DBG ( "%s: %s at ioaddr %hX\n", tp->nic_name, eth_ntoa ( nic->node_addr ),
- (unsigned int) ioaddr );
-
- tp->chip_id = chip_idx;
- tp->revision = chip_rev;
- tp->csr0 = csr0;
-
- /* BugFixes: The 21143-TD hangs with PCI Write-and-Invalidate cycles.
- And the ASIX must have a burst limit or horrible things happen. */
- if (chip_idx == DC21143 && chip_rev == 65)
- tp->csr0 &= ~0x01000000;
- else if (tp->flags & IS_ASIX)
- tp->csr0 |= 0x2000;
-
- if (media_cap[tp->default_port] & MediaIsMII) {
- static const u16 media2advert[] = { 0x20, 0x40, 0x03e0, 0x60,
- 0x80, 0x100, 0x200 };
- tp->mii_advertise = media2advert[tp->default_port - 9];
- tp->mii_advertise |= (tp->flags & HAS_8023X); /* Matching bits! */
- }
-
- /* This is logically part of the probe routine, but too complex
- to write inline. */
- if (tp->flags & HAS_MEDIA_TABLE) {
- memcpy(tp->eeprom, ee_data, sizeof(tp->eeprom));
- parse_eeprom(nic);
- }
-
- start_link(nic);
-
- /* reset the device and make ready for tx and rx of packets */
- tulip_reset(nic);
- nic->nic_op = &tulip_operations;
-
- /* give the board a chance to reset before returning */
- tulip_wait(4*TICKS_PER_SEC);
-
- return 1;
- }
-
- static void start_link(struct nic *nic)
- {
- int i;
-
- #ifdef TULIP_DEBUG_WHERE
- whereami("start_link\n");
- #endif
-
- if ((tp->flags & ALWAYS_CHECK_MII) ||
- (tp->mtable && tp->mtable->has_mii) ||
- ( ! tp->mtable && (tp->flags & HAS_MII))) {
- unsigned int phy, phy_idx;
- if (tp->mtable && tp->mtable->has_mii) {
- for (i = 0; i < tp->mtable->leafcount; i++)
- if (tp->mtable->mleaf[i].media == 11) {
- tp->cur_index = i;
- tp->saved_if_port = tp->if_port;
- select_media(nic, 2);
- tp->if_port = tp->saved_if_port;
- break;
- }
- }
-
- /* Find the connected MII xcvrs. */
- for (phy = 0, phy_idx = 0; phy < 32 && phy_idx < sizeof(tp->phys);
- phy++) {
- int mii_status = mdio_read(nic, phy, 1);
- if ((mii_status & 0x8301) == 0x8001 ||
- ((mii_status & 0x8000) == 0 && (mii_status & 0x7800) != 0)) {
- int mii_reg0 = mdio_read(nic, phy, 0);
- int mii_advert = mdio_read(nic, phy, 4);
- int to_advert;
-
- if (tp->mii_advertise)
- to_advert = tp->mii_advertise;
- else if (tp->advertising[phy_idx])
- to_advert = tp->advertising[phy_idx];
- else /* Leave unchanged. */
- tp->mii_advertise = to_advert = mii_advert;
-
- tp->phys[phy_idx++] = phy;
- printf("%s: MII transceiver %d config %hX status %hX advertising %hX.\n",
- tp->nic_name, phy, mii_reg0, mii_status, mii_advert);
- /* Fixup for DLink with miswired PHY. */
- if (mii_advert != to_advert) {
- printf("%s: Advertising %hX on PHY %d previously advertising %hX.\n",
- tp->nic_name, to_advert, phy, mii_advert);
- mdio_write(nic, phy, 4, to_advert);
- }
- /* Enable autonegotiation: some boards default to off. */
- mdio_write(nic, phy, 0, mii_reg0 |
- (tp->full_duplex ? 0x1100 : 0x1000) |
- (media_cap[tp->default_port]&MediaIs100 ? 0x2000:0));
- }
- }
- tp->mii_cnt = phy_idx;
- if (tp->mtable && tp->mtable->has_mii && phy_idx == 0) {
- printf("%s: ***WARNING***: No MII transceiver found!\n",
- tp->nic_name);
- tp->phys[0] = 1;
- }
- }
-
- /* Reset the xcvr interface and turn on heartbeat. */
- switch (tp->chip_id) {
- case DC21040:
- outl(0x00000000, ioaddr + CSR13);
- outl(0x00000004, ioaddr + CSR13);
- break;
- case DC21041:
- /* This is nway_start(). */
- if (tp->sym_advertise == 0)
- tp->sym_advertise = 0x0061;
- outl(0x00000000, ioaddr + CSR13);
- outl(0xFFFFFFFF, ioaddr + CSR14);
- outl(0x00000008, ioaddr + CSR15); /* Listen on AUI also. */
- outl(inl(ioaddr + CSR6) | 0x0200, ioaddr + CSR6);
- outl(0x0000EF01, ioaddr + CSR13);
- break;
- case DC21140: default:
- if (tp->mtable)
- outl(tp->mtable->csr12dir | 0x100, ioaddr + CSR12);
- break;
- case DC21142:
- case PNIC2:
- if (tp->mii_cnt || media_cap[tp->if_port] & MediaIsMII) {
- outl(0x82020000, ioaddr + CSR6);
- outl(0x0000, ioaddr + CSR13);
- outl(0x0000, ioaddr + CSR14);
- outl(0x820E0000, ioaddr + CSR6);
- } else
- nway_start(nic);
- break;
- case LC82C168:
- if ( ! tp->mii_cnt) {
- tp->nway = 1;
- tp->nwayset = 0;
- outl(0x00420000, ioaddr + CSR6);
- outl(0x30, ioaddr + CSR12);
- outl(0x0001F078, ioaddr + 0xB8);
- outl(0x0201F078, ioaddr + 0xB8); /* Turn on autonegotiation. */
- }
- break;
- case MX98713: case COMPEX9881:
- outl(0x00000000, ioaddr + CSR6);
- outl(0x000711C0, ioaddr + CSR14); /* Turn on NWay. */
- outl(0x00000001, ioaddr + CSR13);
- break;
- case MX98715: case MX98725:
- outl(0x01a80000, ioaddr + CSR6);
- outl(0xFFFFFFFF, ioaddr + CSR14);
- outl(0x00001000, ioaddr + CSR12);
- break;
- case COMET:
- /* No initialization necessary. */
- break;
- }
- }
-
- static void nway_start(struct nic *nic __unused)
- {
- int csr14 = ((tp->sym_advertise & 0x0780) << 9) |
- ((tp->sym_advertise&0x0020)<<1) | 0xffbf;
-
- #ifdef TULIP_DEBUG_WHERE
- whereami("nway_start\n");
- #endif
-
- tp->if_port = 0;
- tp->nway = tp->mediasense = 1;
- tp->nwayset = tp->lpar = 0;
- if (tp->chip_id == PNIC2) {
- tp->csr6 = 0x01000000 | (tp->sym_advertise & 0x0040 ? 0x0200 : 0);
- return;
- }
- #ifdef TULIP_DEBUG
- if (tulip_debug > 1)
- printf("%s: Restarting internal NWay autonegotiation, %X.\n",
- tp->nic_name, csr14);
- #endif
- outl(0x0001, ioaddr + CSR13);
- outl(csr14, ioaddr + CSR14);
- tp->csr6 = 0x82420000 | (tp->sym_advertise & 0x0040 ? 0x0200 : 0);
- outl(tp->csr6, ioaddr + CSR6);
- if (tp->mtable && tp->mtable->csr15dir) {
- outl(tp->mtable->csr15dir, ioaddr + CSR15);
- outl(tp->mtable->csr15val, ioaddr + CSR15);
- } else if (tp->chip_id != PNIC2)
- outw(0x0008, ioaddr + CSR15);
- if (tp->chip_id == DC21041) /* Trigger NWAY. */
- outl(0xEF01, ioaddr + CSR12);
- else
- outl(0x1301, ioaddr + CSR12);
- }
-
- static void init_media(struct nic *nic)
- {
- int i;
-
- #ifdef TULIP_DEBUG_WHERE
- whereami("init_media\n");
- #endif
-
- tp->saved_if_port = tp->if_port;
- if (tp->if_port == 0)
- tp->if_port = tp->default_port;
-
- /* Allow selecting a default media. */
- i = 0;
- if (tp->mtable == NULL)
- goto media_picked;
- if (tp->if_port) {
- int looking_for = media_cap[tp->if_port] & MediaIsMII ? 11 :
- (tp->if_port == 12 ? 0 : tp->if_port);
- for (i = 0; i < tp->mtable->leafcount; i++)
- if (tp->mtable->mleaf[i].media == looking_for) {
- printf("%s: Using user-specified media %s.\n",
- tp->nic_name, medianame[tp->if_port]);
- goto media_picked;
- }
- }
- if ((tp->mtable->defaultmedia & 0x0800) == 0) {
- int looking_for = tp->mtable->defaultmedia & 15;
- for (i = 0; i < tp->mtable->leafcount; i++)
- if (tp->mtable->mleaf[i].media == looking_for) {
- printf("%s: Using EEPROM-set media %s.\n",
- tp->nic_name, medianame[looking_for]);
- goto media_picked;
- }
- }
- /* Start sensing first non-full-duplex media. */
- for (i = tp->mtable->leafcount - 1;
- (media_cap[tp->mtable->mleaf[i].media] & MediaAlwaysFD) && i > 0; i--)
- ;
- media_picked:
-
- tp->csr6 = 0;
- tp->cur_index = i;
- tp->nwayset = 0;
-
- if (tp->if_port) {
- if (tp->chip_id == DC21143 && media_cap[tp->if_port] & MediaIsMII) {
- /* We must reset the media CSRs when we force-select MII mode. */
- outl(0x0000, ioaddr + CSR13);
- outl(0x0000, ioaddr + CSR14);
- outl(0x0008, ioaddr + CSR15);
- }
- select_media(nic, 1);
- return;
- }
- switch(tp->chip_id) {
- case DC21041:
- /* tp->nway = 1;*/
- nway_start(nic);
- break;
- case DC21142:
- if (tp->mii_cnt) {
- select_media(nic, 1);
- #ifdef TULIP_DEBUG
- if (tulip_debug > 1)
- printf("%s: Using MII transceiver %d, status %hX.\n",
- tp->nic_name, tp->phys[0], mdio_read(nic, tp->phys[0], 1));
- #endif
- outl(0x82020000, ioaddr + CSR6);
- tp->csr6 = 0x820E0000;
- tp->if_port = 11;
- outl(0x0000, ioaddr + CSR13);
- outl(0x0000, ioaddr + CSR14);
- } else
- nway_start(nic);
- break;
- case PNIC2:
- nway_start(nic);
- break;
- case LC82C168:
- if (tp->mii_cnt) {
- tp->if_port = 11;
- tp->csr6 = 0x814C0000 | (tp->full_duplex ? 0x0200 : 0);
- outl(0x0001, ioaddr + CSR15);
- } else if (inl(ioaddr + CSR5) & TPLnkPass)
- pnic_do_nway(nic);
- else {
- /* Start with 10mbps to do autonegotiation. */
- outl(0x32, ioaddr + CSR12);
- tp->csr6 = 0x00420000;
- outl(0x0001B078, ioaddr + 0xB8);
- outl(0x0201B078, ioaddr + 0xB8);
- }
- break;
- case MX98713: case COMPEX9881:
- tp->if_port = 0;
- tp->csr6 = 0x01880000 | (tp->full_duplex ? 0x0200 : 0);
- outl(0x0f370000 | inw(ioaddr + 0x80), ioaddr + 0x80);
- break;
- case MX98715: case MX98725:
- /* Provided by BOLO, Macronix - 12/10/1998. */
- tp->if_port = 0;
- tp->csr6 = 0x01a80200;
- outl(0x0f370000 | inw(ioaddr + 0x80), ioaddr + 0x80);
- outl(0x11000 | inw(ioaddr + 0xa0), ioaddr + 0xa0);
- break;
- case COMET:
- /* Enable automatic Tx underrun recovery */
- outl(inl(ioaddr + 0x88) | 1, ioaddr + 0x88);
- tp->if_port = 0;
- tp->csr6 = 0x00040000;
- break;
- case AX88140: case AX88141:
- tp->csr6 = tp->mii_cnt ? 0x00040100 : 0x00000100;
- break;
- default:
- select_media(nic, 1);
- }
- }
-
- static void pnic_do_nway(struct nic *nic __unused)
- {
- u32 phy_reg = inl(ioaddr + 0xB8);
- u32 new_csr6 = tp->csr6 & ~0x40C40200;
-
- #ifdef TULIP_DEBUG_WHERE
- whereami("pnic_do_nway\n");
- #endif
-
- if (phy_reg & 0x78000000) { /* Ignore baseT4 */
- if (phy_reg & 0x20000000) tp->if_port = 5;
- else if (phy_reg & 0x40000000) tp->if_port = 3;
- else if (phy_reg & 0x10000000) tp->if_port = 4;
- else if (phy_reg & 0x08000000) tp->if_port = 0;
- tp->nwayset = 1;
- new_csr6 = (tp->if_port & 1) ? 0x01860000 : 0x00420000;
- outl(0x32 | (tp->if_port & 1), ioaddr + CSR12);
- if (tp->if_port & 1)
- outl(0x1F868, ioaddr + 0xB8);
- if (phy_reg & 0x30000000) {
- tp->full_duplex = 1;
- new_csr6 |= 0x00000200;
- }
- #ifdef TULIP_DEBUG
- if (tulip_debug > 1)
- printf("%s: PNIC autonegotiated status %X, %s.\n",
- tp->nic_name, phy_reg, medianame[tp->if_port]);
- #endif
- if (tp->csr6 != new_csr6) {
- tp->csr6 = new_csr6;
- outl(tp->csr6 | 0x0002, ioaddr + CSR6); /* Restart Tx */
- outl(tp->csr6 | 0x2002, ioaddr + CSR6);
- }
- }
- }
-
- /* Set up the transceiver control registers for the selected media type. */
- static void select_media(struct nic *nic, int startup)
- {
- struct mediatable *mtable = tp->mtable;
- u32 new_csr6;
- int i;
-
- #ifdef TULIP_DEBUG_WHERE
- whereami("select_media\n");
- #endif
-
- if (mtable) {
- struct medialeaf *mleaf = &mtable->mleaf[tp->cur_index];
- unsigned char *p = mleaf->leafdata;
- switch (mleaf->type) {
- case 0: /* 21140 non-MII xcvr. */
- #ifdef TULIP_DEBUG
- if (tulip_debug > 1)
- printf("%s: Using a 21140 non-MII transceiver"
- " with control setting %hhX.\n",
- tp->nic_name, p[1]);
- #endif
- tp->if_port = p[0];
- if (startup)
- outl(mtable->csr12dir | 0x100, ioaddr + CSR12);
- outl(p[1], ioaddr + CSR12);
- new_csr6 = 0x02000000 | ((p[2] & 0x71) << 18);
- break;
- case 2: case 4: {
- u16 setup[5];
- u32 csr13val, csr14val, csr15dir, csr15val;
- for (i = 0; i < 5; i++)
- setup[i] = get_u16(&p[i*2 + 1]);
-
- tp->if_port = p[0] & 15;
- if (media_cap[tp->if_port] & MediaAlwaysFD)
- tp->full_duplex = 1;
-
- if (startup && mtable->has_reset) {
- struct medialeaf *rleaf = &mtable->mleaf[mtable->has_reset];
- unsigned char *rst = rleaf->leafdata;
- #ifdef TULIP_DEBUG
- if (tulip_debug > 1)
- printf("%s: Resetting the transceiver.\n",
- tp->nic_name);
- #endif
- for (i = 0; i < rst[0]; i++)
- outl(get_u16(rst + 1 + (i<<1)) << 16, ioaddr + CSR15);
- }
- #ifdef TULIP_DEBUG
- if (tulip_debug > 1)
- printf("%s: 21143 non-MII %s transceiver control "
- "%hX/%hX.\n",
- tp->nic_name, medianame[tp->if_port], setup[0], setup[1]);
- #endif
- if (p[0] & 0x40) { /* SIA (CSR13-15) setup values are provided. */
- csr13val = setup[0];
- csr14val = setup[1];
- csr15dir = (setup[3]<<16) | setup[2];
- csr15val = (setup[4]<<16) | setup[2];
- outl(0, ioaddr + CSR13);
- outl(csr14val, ioaddr + CSR14);
- outl(csr15dir, ioaddr + CSR15); /* Direction */
- outl(csr15val, ioaddr + CSR15); /* Data */
- outl(csr13val, ioaddr + CSR13);
- } else {
- csr13val = 1;
- csr14val = 0x0003FF7F;
- csr15dir = (setup[0]<<16) | 0x0008;
- csr15val = (setup[1]<<16) | 0x0008;
- if (tp->if_port <= 4)
- csr14val = t21142_csr14[tp->if_port];
- if (startup) {
- outl(0, ioaddr + CSR13);
- outl(csr14val, ioaddr + CSR14);
- }
- outl(csr15dir, ioaddr + CSR15); /* Direction */
- outl(csr15val, ioaddr + CSR15); /* Data */
- if (startup) outl(csr13val, ioaddr + CSR13);
- }
- #ifdef TULIP_DEBUG
- if (tulip_debug > 1)
- printf("%s: Setting CSR15 to %X/%X.\n",
- tp->nic_name, csr15dir, csr15val);
- #endif
- if (mleaf->type == 4)
- new_csr6 = 0x82020000 | ((setup[2] & 0x71) << 18);
- else
- new_csr6 = 0x82420000;
- break;
- }
- case 1: case 3: {
- int phy_num = p[0];
- int init_length = p[1];
- u16 *misc_info;
-
- tp->if_port = 11;
- new_csr6 = 0x020E0000;
- if (mleaf->type == 3) { /* 21142 */
- u16 *init_sequence = (u16*)(p+2);
- u16 *reset_sequence = &((u16*)(p+3))[init_length];
- int reset_length = p[2 + init_length*2];
- misc_info = reset_sequence + reset_length;
- if (startup)
- for (i = 0; i < reset_length; i++)
- outl(get_u16(&reset_sequence[i]) << 16, ioaddr + CSR15);
- for (i = 0; i < init_length; i++)
- outl(get_u16(&init_sequence[i]) << 16, ioaddr + CSR15);
- } else {
- u8 *init_sequence = p + 2;
- u8 *reset_sequence = p + 3 + init_length;
- int reset_length = p[2 + init_length];
- misc_info = (u16*)(reset_sequence + reset_length);
- if (startup) {
- outl(mtable->csr12dir | 0x100, ioaddr + CSR12);
- for (i = 0; i < reset_length; i++)
- outl(reset_sequence[i], ioaddr + CSR12);
- }
- for (i = 0; i < init_length; i++)
- outl(init_sequence[i], ioaddr + CSR12);
- }
- tp->advertising[phy_num] = get_u16(&misc_info[1]) | 1;
- if (startup < 2) {
- if (tp->mii_advertise == 0)
- tp->mii_advertise = tp->advertising[phy_num];
- #ifdef TULIP_DEBUG
- if (tulip_debug > 1)
- printf("%s: Advertising %hX on MII %d.\n",
- tp->nic_name, tp->mii_advertise, tp->phys[phy_num]);
- #endif
- mdio_write(nic, tp->phys[phy_num], 4, tp->mii_advertise);
- }
- break;
- }
- default:
- printf("%s: Invalid media table selection %d.\n",
- tp->nic_name, mleaf->type);
- new_csr6 = 0x020E0000;
- }
- #ifdef TULIP_DEBUG
- if (tulip_debug > 1)
- printf("%s: Using media type %s, CSR12 is %hhX.\n",
- tp->nic_name, medianame[tp->if_port],
- inl(ioaddr + CSR12) & 0xff);
- #endif
- } else if (tp->chip_id == DC21041) {
- int port = tp->if_port <= 4 ? tp->if_port : 0;
- #ifdef TULIP_DEBUG
- if (tulip_debug > 1)
- printf("%s: 21041 using media %s, CSR12 is %hX.\n",
- tp->nic_name, medianame[port == 3 ? 12: port],
- inl(ioaddr + CSR12));
- #endif
- outl(0x00000000, ioaddr + CSR13); /* Reset the serial interface */
- outl(t21041_csr14[port], ioaddr + CSR14);
- outl(t21041_csr15[port], ioaddr + CSR15);
- outl(t21041_csr13[port], ioaddr + CSR13);
- new_csr6 = 0x80020000;
- } else if (tp->chip_id == LC82C168) {
- if (startup && ! tp->medialock)
- tp->if_port = tp->mii_cnt ? 11 : 0;
- #ifdef TULIP_DEBUG
- if (tulip_debug > 1)
- printf("%s: PNIC PHY status is %hX, media %s.\n",
- tp->nic_name, inl(ioaddr + 0xB8), medianame[tp->if_port]);
- #endif
- if (tp->mii_cnt) {
- new_csr6 = 0x810C0000;
- outl(0x0001, ioaddr + CSR15);
- outl(0x0201B07A, ioaddr + 0xB8);
- } else if (startup) {
- /* Start with 10mbps to do autonegotiation. */
- outl(0x32, ioaddr + CSR12);
- new_csr6 = 0x00420000;
- outl(0x0001B078, ioaddr + 0xB8);
- outl(0x0201B078, ioaddr + 0xB8);
- } else if (tp->if_port == 3 || tp->if_port == 5) {
- outl(0x33, ioaddr + CSR12);
- new_csr6 = 0x01860000;
- /* Trigger autonegotiation. */
- outl(startup ? 0x0201F868 : 0x0001F868, ioaddr + 0xB8);
- } else {
- outl(0x32, ioaddr + CSR12);
- new_csr6 = 0x00420000;
- outl(0x1F078, ioaddr + 0xB8);
- }
- } else if (tp->chip_id == DC21040) { /* 21040 */
- /* Turn on the xcvr interface. */
- #ifdef TULIP_DEBUG
- int csr12 = inl(ioaddr + CSR12);
- if (tulip_debug > 1)
- printf("%s: 21040 media type is %s, CSR12 is %hhX.\n",
- tp->nic_name, medianame[tp->if_port], csr12);
- #endif
- if (media_cap[tp->if_port] & MediaAlwaysFD)
- tp->full_duplex = 1;
- new_csr6 = 0x20000;
- /* Set the full duplux match frame. */
- outl(FULL_DUPLEX_MAGIC, ioaddr + CSR11);
- outl(0x00000000, ioaddr + CSR13); /* Reset the serial interface */
- if (t21040_csr13[tp->if_port] & 8) {
- outl(0x0705, ioaddr + CSR14);
- outl(0x0006, ioaddr + CSR15);
- } else {
- outl(0xffff, ioaddr + CSR14);
- outl(0x0000, ioaddr + CSR15);
- }
- outl(0x8f01 | t21040_csr13[tp->if_port], ioaddr + CSR13);
- } else { /* Unknown chip type with no media table. */
- if (tp->default_port == 0)
- tp->if_port = tp->mii_cnt ? 11 : 3;
- if (media_cap[tp->if_port] & MediaIsMII) {
- new_csr6 = 0x020E0000;
- } else if (media_cap[tp->if_port] & MediaIsFx) {
- new_csr6 = 0x028600000;
- } else
- new_csr6 = 0x038600000;
- #ifdef TULIP_DEBUG
- if (tulip_debug > 1)
- printf("%s: No media description table, assuming "
- "%s transceiver, CSR12 %hhX.\n",
- tp->nic_name, medianame[tp->if_port],
- inl(ioaddr + CSR12));
- #endif
- }
-
- tp->csr6 = new_csr6 | (tp->csr6 & 0xfdff) | (tp->full_duplex ? 0x0200 : 0);
- return;
- }
-
- /*
- Check the MII negotiated duplex and change the CSR6 setting if
- required.
- Return 0 if everything is OK.
- Return < 0 if the transceiver is missing or has no link beat.
- */
- static int tulip_check_duplex(struct nic *nic)
- {
- unsigned int bmsr, lpa, negotiated, new_csr6;
-
- bmsr = mdio_read(nic, tp->phys[0], 1);
- lpa = mdio_read(nic, tp->phys[0], 5);
-
- #ifdef TULIP_DEBUG
- if (tulip_debug > 1)
- printf("%s: MII status %#x, Link partner report "
- "%#x.\n", tp->nic_name, bmsr, lpa);
- #endif
-
- if (bmsr == 0xffff)
- return -2;
- if ((bmsr & 4) == 0) {
- int new_bmsr = mdio_read(nic, tp->phys[0], 1);
- if ((new_bmsr & 4) == 0) {
- #ifdef TULIP_DEBUG
- if (tulip_debug > 1)
- printf("%s: No link beat on the MII interface,"
- " status %#x.\n", tp->nic_name,
- new_bmsr);
- #endif
- return -1;
- }
- }
- tp->full_duplex = lpa & 0x140;
-
- new_csr6 = tp->csr6;
- negotiated = lpa & tp->advertising[0];
-
- if(negotiated & 0x380) new_csr6 &= ~0x400000;
- else new_csr6 |= 0x400000;
- if (tp->full_duplex) new_csr6 |= 0x200;
- else new_csr6 &= ~0x200;
-
- if (new_csr6 != tp->csr6) {
- tp->csr6 = new_csr6;
-
- #ifdef TULIP_DEBUG
- if (tulip_debug > 0)
- printf("%s: Setting %s-duplex based on MII"
- "#%d link partner capability of %#x.\n",
- tp->nic_name,
- tp->full_duplex ? "full" : "half",
- tp->phys[0], lpa);
- #endif
- return 1;
- }
-
- return 0;
- }
-
- static struct pci_device_id tulip_nics[] = {
- PCI_ROM(0x1011, 0x0002, "dc21040", "Digital Tulip"),
- PCI_ROM(0x1011, 0x0009, "ds21140", "Digital Tulip Fast"),
- PCI_ROM(0x1011, 0x0014, "dc21041", "Digital Tulip+"),
- PCI_ROM(0x1011, 0x0019, "ds21142", "Digital Tulip 21142"),
- PCI_ROM(0x10b7, 0x9300, "3csoho100b-tx","3ComSOHO100B-TX"),
- PCI_ROM(0x10b9, 0x5261, "ali1563", "ALi 1563 integrated ethernet"),
- PCI_ROM(0x10d9, 0x0512, "mx98713", "Macronix MX987x3"),
- PCI_ROM(0x10d9, 0x0531, "mx98715", "Macronix MX987x5"),
- PCI_ROM(0x1113, 0x1217, "mxic-98715", "Macronix MX987x5"),
- PCI_ROM(0x11ad, 0xc115, "lc82c115", "LinkSys LNE100TX"),
- PCI_ROM(0x11ad, 0x0002, "82c168", "Netgear FA310TX"),
- PCI_ROM(0x1282, 0x9100, "dm9100", "Davicom 9100"),
- PCI_ROM(0x1282, 0x9102, "dm9102", "Davicom 9102"),
- PCI_ROM(0x1282, 0x9009, "dm9009", "Davicom 9009"),
- PCI_ROM(0x1282, 0x9132, "dm9132", "Davicom 9132"),
- PCI_ROM(0x1317, 0x0985, "centaur-p", "ADMtek Centaur-P"),
- PCI_ROM(0x1317, 0x0981, "an981", "ADMtek AN981 Comet"), /* ADMTek Centaur-P (stmicro) */
- PCI_ROM(0x1113, 0x1216, "an983", "ADMTek AN983 Comet"),
- PCI_ROM(0x1317, 0x9511, "an983b", "ADMTek Comet 983b"),
- PCI_ROM(0x1317, 0x1985, "centaur-c", "ADMTek Centaur-C"),
- PCI_ROM(0x8086, 0x0039, "intel21145", "Intel Tulip"),
- PCI_ROM(0x125b, 0x1400, "ax88140", "ASIX AX88140"),
- PCI_ROM(0x11f6, 0x9881, "rl100tx", "Compex RL100-TX"),
- PCI_ROM(0x115d, 0x0003, "xircomtulip", "Xircom Tulip"),
- PCI_ROM(0x104a, 0x0981, "tulip-0981", "Tulip 0x104a 0x0981"),
- PCI_ROM(0x104a, 0x2774, "SGThomson-STE10100A", "Tulip 0x104a 0x2774"), /*Modified by Ramesh Chander*/
- PCI_ROM(0x1113, 0x9511, "tulip-9511", "Tulip 0x1113 0x9511"),
- PCI_ROM(0x1186, 0x1561, "tulip-1561", "Tulip 0x1186 0x1561"),
- PCI_ROM(0x1259, 0xa120, "tulip-a120", "Tulip 0x1259 0xa120"),
- PCI_ROM(0x13d1, 0xab02, "tulip-ab02", "Tulip 0x13d1 0xab02"),
- PCI_ROM(0x13d1, 0xab03, "tulip-ab03", "Tulip 0x13d1 0xab03"),
- PCI_ROM(0x13d1, 0xab08, "tulip-ab08", "Tulip 0x13d1 0xab08"),
- PCI_ROM(0x14f1, 0x1803, "lanfinity", "Conexant LANfinity"),
- PCI_ROM(0x1626, 0x8410, "tulip-8410", "Tulip 0x1626 0x8410"),
- PCI_ROM(0x1737, 0xab08, "tulip-1737-ab08","Tulip 0x1737 0xab08"),
- PCI_ROM(0x1737, 0xab09, "tulip-ab09", "Tulip 0x1737 0xab09"),
- };
-
- PCI_DRIVER ( tulip_driver, tulip_nics, PCI_NO_CLASS );
-
- DRIVER ( "Tulip", nic_driver, pci_driver, tulip_driver,
- tulip_probe, tulip_disable );
-
- /*
- * Local variables:
- * c-basic-offset: 8
- * c-indent-level: 8
- * tab-width: 8
- * End:
- */
|