12439e4bfSJean-Christophe PLAGNIOL-VILLARD /*------------------------------------------------------------------------ 22439e4bfSJean-Christophe PLAGNIOL-VILLARD . smc91111.c 32439e4bfSJean-Christophe PLAGNIOL-VILLARD . This is a driver for SMSC's 91C111 single-chip Ethernet device. 42439e4bfSJean-Christophe PLAGNIOL-VILLARD . 52439e4bfSJean-Christophe PLAGNIOL-VILLARD . (C) Copyright 2002 62439e4bfSJean-Christophe PLAGNIOL-VILLARD . Sysgo Real-Time Solutions, GmbH <www.elinos.com> 72439e4bfSJean-Christophe PLAGNIOL-VILLARD . Rolf Offermanns <rof@sysgo.de> 82439e4bfSJean-Christophe PLAGNIOL-VILLARD . 92439e4bfSJean-Christophe PLAGNIOL-VILLARD . Copyright (C) 2001 Standard Microsystems Corporation (SMSC) 102439e4bfSJean-Christophe PLAGNIOL-VILLARD . Developed by Simple Network Magic Corporation (SNMC) 112439e4bfSJean-Christophe PLAGNIOL-VILLARD . Copyright (C) 1996 by Erik Stahlman (ES) 122439e4bfSJean-Christophe PLAGNIOL-VILLARD . 132439e4bfSJean-Christophe PLAGNIOL-VILLARD . This program is free software; you can redistribute it and/or modify 142439e4bfSJean-Christophe PLAGNIOL-VILLARD . it under the terms of the GNU General Public License as published by 152439e4bfSJean-Christophe PLAGNIOL-VILLARD . the Free Software Foundation; either version 2 of the License, or 162439e4bfSJean-Christophe PLAGNIOL-VILLARD . (at your option) any later version. 172439e4bfSJean-Christophe PLAGNIOL-VILLARD . 182439e4bfSJean-Christophe PLAGNIOL-VILLARD . This program is distributed in the hope that it will be useful, 192439e4bfSJean-Christophe PLAGNIOL-VILLARD . but WITHOUT ANY WARRANTY; without even the implied warranty of 202439e4bfSJean-Christophe PLAGNIOL-VILLARD . MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 212439e4bfSJean-Christophe PLAGNIOL-VILLARD . GNU General Public License for more details. 222439e4bfSJean-Christophe PLAGNIOL-VILLARD . 232439e4bfSJean-Christophe PLAGNIOL-VILLARD . You should have received a copy of the GNU General Public License 242439e4bfSJean-Christophe PLAGNIOL-VILLARD . along with this program; if not, write to the Free Software 252439e4bfSJean-Christophe PLAGNIOL-VILLARD . Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA 262439e4bfSJean-Christophe PLAGNIOL-VILLARD . 272439e4bfSJean-Christophe PLAGNIOL-VILLARD . Information contained in this file was obtained from the LAN91C111 282439e4bfSJean-Christophe PLAGNIOL-VILLARD . manual from SMC. To get a copy, if you really want one, you can find 292439e4bfSJean-Christophe PLAGNIOL-VILLARD . information under www.smsc.com. 302439e4bfSJean-Christophe PLAGNIOL-VILLARD . 312439e4bfSJean-Christophe PLAGNIOL-VILLARD . 322439e4bfSJean-Christophe PLAGNIOL-VILLARD . "Features" of the SMC chip: 332439e4bfSJean-Christophe PLAGNIOL-VILLARD . Integrated PHY/MAC for 10/100BaseT Operation 342439e4bfSJean-Christophe PLAGNIOL-VILLARD . Supports internal and external MII 352439e4bfSJean-Christophe PLAGNIOL-VILLARD . Integrated 8K packet memory 362439e4bfSJean-Christophe PLAGNIOL-VILLARD . EEPROM interface for configuration 372439e4bfSJean-Christophe PLAGNIOL-VILLARD . 382439e4bfSJean-Christophe PLAGNIOL-VILLARD . Arguments: 392439e4bfSJean-Christophe PLAGNIOL-VILLARD . io = for the base address 402439e4bfSJean-Christophe PLAGNIOL-VILLARD . irq = for the IRQ 412439e4bfSJean-Christophe PLAGNIOL-VILLARD . 422439e4bfSJean-Christophe PLAGNIOL-VILLARD . author: 432439e4bfSJean-Christophe PLAGNIOL-VILLARD . Erik Stahlman ( erik@vt.edu ) 442439e4bfSJean-Christophe PLAGNIOL-VILLARD . Daris A Nevil ( dnevil@snmc.com ) 452439e4bfSJean-Christophe PLAGNIOL-VILLARD . 462439e4bfSJean-Christophe PLAGNIOL-VILLARD . 472439e4bfSJean-Christophe PLAGNIOL-VILLARD . Hardware multicast code from Peter Cammaert ( pc@denkart.be ) 482439e4bfSJean-Christophe PLAGNIOL-VILLARD . 492439e4bfSJean-Christophe PLAGNIOL-VILLARD . Sources: 502439e4bfSJean-Christophe PLAGNIOL-VILLARD . o SMSC LAN91C111 databook (www.smsc.com) 512439e4bfSJean-Christophe PLAGNIOL-VILLARD . o smc9194.c by Erik Stahlman 522439e4bfSJean-Christophe PLAGNIOL-VILLARD . o skeleton.c by Donald Becker ( becker@cesdis.gsfc.nasa.gov ) 532439e4bfSJean-Christophe PLAGNIOL-VILLARD . 542439e4bfSJean-Christophe PLAGNIOL-VILLARD . History: 552439e4bfSJean-Christophe PLAGNIOL-VILLARD . 06/19/03 Richard Woodruff Made u-boot environment aware and added mac addr checks. 562439e4bfSJean-Christophe PLAGNIOL-VILLARD . 10/17/01 Marco Hasewinkel Modify for DNP/1110 572439e4bfSJean-Christophe PLAGNIOL-VILLARD . 07/25/01 Woojung Huh Modify for ADS Bitsy 582439e4bfSJean-Christophe PLAGNIOL-VILLARD . 04/25/01 Daris A Nevil Initial public release through SMSC 592439e4bfSJean-Christophe PLAGNIOL-VILLARD . 03/16/01 Daris A Nevil Modified smc9194.c for use with LAN91C111 602439e4bfSJean-Christophe PLAGNIOL-VILLARD ----------------------------------------------------------------------------*/ 612439e4bfSJean-Christophe PLAGNIOL-VILLARD 622439e4bfSJean-Christophe PLAGNIOL-VILLARD #include <common.h> 632439e4bfSJean-Christophe PLAGNIOL-VILLARD #include <command.h> 642439e4bfSJean-Christophe PLAGNIOL-VILLARD #include <config.h> 652439e4bfSJean-Christophe PLAGNIOL-VILLARD #include "smc91111.h" 662439e4bfSJean-Christophe PLAGNIOL-VILLARD #include <net.h> 672439e4bfSJean-Christophe PLAGNIOL-VILLARD 682439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Use power-down feature of the chip */ 692439e4bfSJean-Christophe PLAGNIOL-VILLARD #define POWER_DOWN 0 702439e4bfSJean-Christophe PLAGNIOL-VILLARD 712439e4bfSJean-Christophe PLAGNIOL-VILLARD #define NO_AUTOPROBE 722439e4bfSJean-Christophe PLAGNIOL-VILLARD 732439e4bfSJean-Christophe PLAGNIOL-VILLARD #define SMC_DEBUG 0 742439e4bfSJean-Christophe PLAGNIOL-VILLARD 752439e4bfSJean-Christophe PLAGNIOL-VILLARD #if SMC_DEBUG > 1 762439e4bfSJean-Christophe PLAGNIOL-VILLARD static const char version[] = 772439e4bfSJean-Christophe PLAGNIOL-VILLARD "smc91111.c:v1.0 04/25/01 by Daris A Nevil (dnevil@snmc.com)\n"; 782439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 792439e4bfSJean-Christophe PLAGNIOL-VILLARD 802439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Autonegotiation timeout in seconds */ 812439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifndef CONFIG_SMC_AUTONEG_TIMEOUT 822439e4bfSJean-Christophe PLAGNIOL-VILLARD #define CONFIG_SMC_AUTONEG_TIMEOUT 10 832439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 842439e4bfSJean-Christophe PLAGNIOL-VILLARD 852439e4bfSJean-Christophe PLAGNIOL-VILLARD /*------------------------------------------------------------------------ 862439e4bfSJean-Christophe PLAGNIOL-VILLARD . 872439e4bfSJean-Christophe PLAGNIOL-VILLARD . Configuration options, for the experienced user to change. 882439e4bfSJean-Christophe PLAGNIOL-VILLARD . 892439e4bfSJean-Christophe PLAGNIOL-VILLARD -------------------------------------------------------------------------*/ 902439e4bfSJean-Christophe PLAGNIOL-VILLARD 912439e4bfSJean-Christophe PLAGNIOL-VILLARD /* 922439e4bfSJean-Christophe PLAGNIOL-VILLARD . Wait time for memory to be free. This probably shouldn't be 932439e4bfSJean-Christophe PLAGNIOL-VILLARD . tuned that much, as waiting for this means nothing else happens 942439e4bfSJean-Christophe PLAGNIOL-VILLARD . in the system 952439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 962439e4bfSJean-Christophe PLAGNIOL-VILLARD #define MEMORY_WAIT_TIME 16 972439e4bfSJean-Christophe PLAGNIOL-VILLARD 982439e4bfSJean-Christophe PLAGNIOL-VILLARD 992439e4bfSJean-Christophe PLAGNIOL-VILLARD #if (SMC_DEBUG > 2 ) 1002439e4bfSJean-Christophe PLAGNIOL-VILLARD #define PRINTK3(args...) printf(args) 1012439e4bfSJean-Christophe PLAGNIOL-VILLARD #else 1022439e4bfSJean-Christophe PLAGNIOL-VILLARD #define PRINTK3(args...) 1032439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 1042439e4bfSJean-Christophe PLAGNIOL-VILLARD 1052439e4bfSJean-Christophe PLAGNIOL-VILLARD #if SMC_DEBUG > 1 1062439e4bfSJean-Christophe PLAGNIOL-VILLARD #define PRINTK2(args...) printf(args) 1072439e4bfSJean-Christophe PLAGNIOL-VILLARD #else 1082439e4bfSJean-Christophe PLAGNIOL-VILLARD #define PRINTK2(args...) 1092439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 1102439e4bfSJean-Christophe PLAGNIOL-VILLARD 1112439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifdef SMC_DEBUG 1122439e4bfSJean-Christophe PLAGNIOL-VILLARD #define PRINTK(args...) printf(args) 1132439e4bfSJean-Christophe PLAGNIOL-VILLARD #else 1142439e4bfSJean-Christophe PLAGNIOL-VILLARD #define PRINTK(args...) 1152439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 1162439e4bfSJean-Christophe PLAGNIOL-VILLARD 1172439e4bfSJean-Christophe PLAGNIOL-VILLARD 1182439e4bfSJean-Christophe PLAGNIOL-VILLARD /*------------------------------------------------------------------------ 1192439e4bfSJean-Christophe PLAGNIOL-VILLARD . 1202439e4bfSJean-Christophe PLAGNIOL-VILLARD . The internal workings of the driver. If you are changing anything 1212439e4bfSJean-Christophe PLAGNIOL-VILLARD . here with the SMC stuff, you should have the datasheet and know 1222439e4bfSJean-Christophe PLAGNIOL-VILLARD . what you are doing. 1232439e4bfSJean-Christophe PLAGNIOL-VILLARD . 1242439e4bfSJean-Christophe PLAGNIOL-VILLARD -------------------------------------------------------------------------*/ 1252439e4bfSJean-Christophe PLAGNIOL-VILLARD #define CARDNAME "LAN91C111" 1262439e4bfSJean-Christophe PLAGNIOL-VILLARD 1272439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Memory sizing constant */ 1282439e4bfSJean-Christophe PLAGNIOL-VILLARD #define LAN91C111_MEMORY_MULTIPLIER (1024*2) 1292439e4bfSJean-Christophe PLAGNIOL-VILLARD 1302439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifndef CONFIG_SMC91111_BASE 1312439e4bfSJean-Christophe PLAGNIOL-VILLARD #define CONFIG_SMC91111_BASE 0x20000300 1322439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 1332439e4bfSJean-Christophe PLAGNIOL-VILLARD 1342439e4bfSJean-Christophe PLAGNIOL-VILLARD #define SMC_BASE_ADDRESS CONFIG_SMC91111_BASE 1352439e4bfSJean-Christophe PLAGNIOL-VILLARD 1362439e4bfSJean-Christophe PLAGNIOL-VILLARD #define SMC_DEV_NAME "SMC91111" 1372439e4bfSJean-Christophe PLAGNIOL-VILLARD #define SMC_PHY_ADDR 0x0000 1382439e4bfSJean-Christophe PLAGNIOL-VILLARD #define SMC_ALLOC_MAX_TRY 5 1392439e4bfSJean-Christophe PLAGNIOL-VILLARD #define SMC_TX_TIMEOUT 30 1402439e4bfSJean-Christophe PLAGNIOL-VILLARD 1412439e4bfSJean-Christophe PLAGNIOL-VILLARD #define SMC_PHY_CLOCK_DELAY 1000 1422439e4bfSJean-Christophe PLAGNIOL-VILLARD 1432439e4bfSJean-Christophe PLAGNIOL-VILLARD #define ETH_ZLEN 60 1442439e4bfSJean-Christophe PLAGNIOL-VILLARD 1452439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifdef CONFIG_SMC_USE_32_BIT 1462439e4bfSJean-Christophe PLAGNIOL-VILLARD #define USE_32_BIT 1 1472439e4bfSJean-Christophe PLAGNIOL-VILLARD #else 1482439e4bfSJean-Christophe PLAGNIOL-VILLARD #undef USE_32_BIT 1492439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 1502439e4bfSJean-Christophe PLAGNIOL-VILLARD /*----------------------------------------------------------------- 1512439e4bfSJean-Christophe PLAGNIOL-VILLARD . 1522439e4bfSJean-Christophe PLAGNIOL-VILLARD . The driver can be entered at any of the following entry points. 1532439e4bfSJean-Christophe PLAGNIOL-VILLARD . 1542439e4bfSJean-Christophe PLAGNIOL-VILLARD .------------------------------------------------------------------ */ 1552439e4bfSJean-Christophe PLAGNIOL-VILLARD 1562439e4bfSJean-Christophe PLAGNIOL-VILLARD extern int eth_init(bd_t *bd); 1572439e4bfSJean-Christophe PLAGNIOL-VILLARD extern void eth_halt(void); 1582439e4bfSJean-Christophe PLAGNIOL-VILLARD extern int eth_rx(void); 1592439e4bfSJean-Christophe PLAGNIOL-VILLARD extern int eth_send(volatile void *packet, int length); 1602439e4bfSJean-Christophe PLAGNIOL-VILLARD 1612439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifdef SHARED_RESOURCES 1622439e4bfSJean-Christophe PLAGNIOL-VILLARD extern void swap_to(int device_id); 1632439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 1642439e4bfSJean-Christophe PLAGNIOL-VILLARD 1652439e4bfSJean-Christophe PLAGNIOL-VILLARD /* 1662439e4bfSJean-Christophe PLAGNIOL-VILLARD . This is called by register_netdev(). It is responsible for 1672439e4bfSJean-Christophe PLAGNIOL-VILLARD . checking the portlist for the SMC9000 series chipset. If it finds 1682439e4bfSJean-Christophe PLAGNIOL-VILLARD . one, then it will initialize the device, find the hardware information, 1692439e4bfSJean-Christophe PLAGNIOL-VILLARD . and sets up the appropriate device parameters. 1702439e4bfSJean-Christophe PLAGNIOL-VILLARD . NOTE: Interrupts are *OFF* when this procedure is called. 1712439e4bfSJean-Christophe PLAGNIOL-VILLARD . 1722439e4bfSJean-Christophe PLAGNIOL-VILLARD . NB:This shouldn't be static since it is referred to externally. 1732439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 1742439e4bfSJean-Christophe PLAGNIOL-VILLARD int smc_init(void); 1752439e4bfSJean-Christophe PLAGNIOL-VILLARD 1762439e4bfSJean-Christophe PLAGNIOL-VILLARD /* 1772439e4bfSJean-Christophe PLAGNIOL-VILLARD . This is called by unregister_netdev(). It is responsible for 1782439e4bfSJean-Christophe PLAGNIOL-VILLARD . cleaning up before the driver is finally unregistered and discarded. 1792439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 1802439e4bfSJean-Christophe PLAGNIOL-VILLARD void smc_destructor(void); 1812439e4bfSJean-Christophe PLAGNIOL-VILLARD 1822439e4bfSJean-Christophe PLAGNIOL-VILLARD /* 1832439e4bfSJean-Christophe PLAGNIOL-VILLARD . The kernel calls this function when someone wants to use the device, 1842439e4bfSJean-Christophe PLAGNIOL-VILLARD . typically 'ifconfig ethX up'. 1852439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 1862439e4bfSJean-Christophe PLAGNIOL-VILLARD static int smc_open(bd_t *bd); 1872439e4bfSJean-Christophe PLAGNIOL-VILLARD 1882439e4bfSJean-Christophe PLAGNIOL-VILLARD 1892439e4bfSJean-Christophe PLAGNIOL-VILLARD /* 1902439e4bfSJean-Christophe PLAGNIOL-VILLARD . This is called by the kernel in response to 'ifconfig ethX down'. It 1912439e4bfSJean-Christophe PLAGNIOL-VILLARD . is responsible for cleaning up everything that the open routine 1922439e4bfSJean-Christophe PLAGNIOL-VILLARD . does, and maybe putting the card into a powerdown state. 1932439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 1942439e4bfSJean-Christophe PLAGNIOL-VILLARD static int smc_close(void); 1952439e4bfSJean-Christophe PLAGNIOL-VILLARD 1962439e4bfSJean-Christophe PLAGNIOL-VILLARD /* 1972439e4bfSJean-Christophe PLAGNIOL-VILLARD . Configures the PHY through the MII Management interface 1982439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 1992439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifndef CONFIG_SMC91111_EXT_PHY 2002439e4bfSJean-Christophe PLAGNIOL-VILLARD static void smc_phy_configure(void); 2012439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif /* !CONFIG_SMC91111_EXT_PHY */ 2022439e4bfSJean-Christophe PLAGNIOL-VILLARD 2032439e4bfSJean-Christophe PLAGNIOL-VILLARD /* 2042439e4bfSJean-Christophe PLAGNIOL-VILLARD . This is a separate procedure to handle the receipt of a packet, to 2052439e4bfSJean-Christophe PLAGNIOL-VILLARD . leave the interrupt code looking slightly cleaner 2062439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 2072439e4bfSJean-Christophe PLAGNIOL-VILLARD static int smc_rcv(void); 2082439e4bfSJean-Christophe PLAGNIOL-VILLARD 2092439e4bfSJean-Christophe PLAGNIOL-VILLARD /* See if a MAC address is defined in the current environment. If so use it. If not 2102439e4bfSJean-Christophe PLAGNIOL-VILLARD . print a warning and set the environment and other globals with the default. 2112439e4bfSJean-Christophe PLAGNIOL-VILLARD . If an EEPROM is present it really should be consulted. 2122439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 2132439e4bfSJean-Christophe PLAGNIOL-VILLARD int smc_get_ethaddr(bd_t *bd); 2142439e4bfSJean-Christophe PLAGNIOL-VILLARD int get_rom_mac(uchar *v_rom_mac); 2152439e4bfSJean-Christophe PLAGNIOL-VILLARD 2162439e4bfSJean-Christophe PLAGNIOL-VILLARD /* 2172439e4bfSJean-Christophe PLAGNIOL-VILLARD ------------------------------------------------------------ 2182439e4bfSJean-Christophe PLAGNIOL-VILLARD . 2192439e4bfSJean-Christophe PLAGNIOL-VILLARD . Internal routines 2202439e4bfSJean-Christophe PLAGNIOL-VILLARD . 2212439e4bfSJean-Christophe PLAGNIOL-VILLARD ------------------------------------------------------------ 2222439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 2232439e4bfSJean-Christophe PLAGNIOL-VILLARD 2242439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifdef CONFIG_SMC_USE_IOFUNCS 2252439e4bfSJean-Christophe PLAGNIOL-VILLARD /* 2262439e4bfSJean-Christophe PLAGNIOL-VILLARD * input and output functions 2272439e4bfSJean-Christophe PLAGNIOL-VILLARD * 2282439e4bfSJean-Christophe PLAGNIOL-VILLARD * Implemented due to inx,outx macros accessing the device improperly 2292439e4bfSJean-Christophe PLAGNIOL-VILLARD * and putting the device into an unkown state. 2302439e4bfSJean-Christophe PLAGNIOL-VILLARD * 2312439e4bfSJean-Christophe PLAGNIOL-VILLARD * For instance, on Sharp LPD7A400 SDK, affects were chip memory 2322439e4bfSJean-Christophe PLAGNIOL-VILLARD * could not be free'd (hence the alloc failures), duplicate packets, 2332439e4bfSJean-Christophe PLAGNIOL-VILLARD * packets being corrupt (shifted) on the wire, etc. Switching to the 2342439e4bfSJean-Christophe PLAGNIOL-VILLARD * inx,outx functions fixed this problem. 2352439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 2362439e4bfSJean-Christophe PLAGNIOL-VILLARD static inline word SMC_inw(dword offset); 2372439e4bfSJean-Christophe PLAGNIOL-VILLARD static inline void SMC_outw(word value, dword offset); 2382439e4bfSJean-Christophe PLAGNIOL-VILLARD static inline byte SMC_inb(dword offset); 2392439e4bfSJean-Christophe PLAGNIOL-VILLARD static inline void SMC_outb(byte value, dword offset); 2402439e4bfSJean-Christophe PLAGNIOL-VILLARD static inline void SMC_insw(dword offset, volatile uchar* buf, dword len); 2412439e4bfSJean-Christophe PLAGNIOL-VILLARD static inline void SMC_outsw(dword offset, uchar* buf, dword len); 2422439e4bfSJean-Christophe PLAGNIOL-VILLARD 2432439e4bfSJean-Christophe PLAGNIOL-VILLARD #define barrier() __asm__ __volatile__("": : :"memory") 2442439e4bfSJean-Christophe PLAGNIOL-VILLARD 2452439e4bfSJean-Christophe PLAGNIOL-VILLARD static inline word SMC_inw(dword offset) 2462439e4bfSJean-Christophe PLAGNIOL-VILLARD { 2472439e4bfSJean-Christophe PLAGNIOL-VILLARD word v; 2482439e4bfSJean-Christophe PLAGNIOL-VILLARD v = *((volatile word*)(SMC_BASE_ADDRESS+offset)); 2492439e4bfSJean-Christophe PLAGNIOL-VILLARD barrier(); *(volatile u32*)(0xc0000000); 2502439e4bfSJean-Christophe PLAGNIOL-VILLARD return v; 2512439e4bfSJean-Christophe PLAGNIOL-VILLARD } 2522439e4bfSJean-Christophe PLAGNIOL-VILLARD 2532439e4bfSJean-Christophe PLAGNIOL-VILLARD static inline void SMC_outw(word value, dword offset) 2542439e4bfSJean-Christophe PLAGNIOL-VILLARD { 2552439e4bfSJean-Christophe PLAGNIOL-VILLARD *((volatile word*)(SMC_BASE_ADDRESS+offset)) = value; 2562439e4bfSJean-Christophe PLAGNIOL-VILLARD barrier(); *(volatile u32*)(0xc0000000); 2572439e4bfSJean-Christophe PLAGNIOL-VILLARD } 2582439e4bfSJean-Christophe PLAGNIOL-VILLARD 2592439e4bfSJean-Christophe PLAGNIOL-VILLARD static inline byte SMC_inb(dword offset) 2602439e4bfSJean-Christophe PLAGNIOL-VILLARD { 2612439e4bfSJean-Christophe PLAGNIOL-VILLARD word _w; 2622439e4bfSJean-Christophe PLAGNIOL-VILLARD 2632439e4bfSJean-Christophe PLAGNIOL-VILLARD _w = SMC_inw(offset & ~((dword)1)); 2642439e4bfSJean-Christophe PLAGNIOL-VILLARD return (offset & 1) ? (byte)(_w >> 8) : (byte)(_w); 2652439e4bfSJean-Christophe PLAGNIOL-VILLARD } 2662439e4bfSJean-Christophe PLAGNIOL-VILLARD 2672439e4bfSJean-Christophe PLAGNIOL-VILLARD static inline void SMC_outb(byte value, dword offset) 2682439e4bfSJean-Christophe PLAGNIOL-VILLARD { 2692439e4bfSJean-Christophe PLAGNIOL-VILLARD word _w; 2702439e4bfSJean-Christophe PLAGNIOL-VILLARD 2712439e4bfSJean-Christophe PLAGNIOL-VILLARD _w = SMC_inw(offset & ~((dword)1)); 2722439e4bfSJean-Christophe PLAGNIOL-VILLARD if (offset & 1) 2732439e4bfSJean-Christophe PLAGNIOL-VILLARD *((volatile word*)(SMC_BASE_ADDRESS+(offset & ~((dword)1)))) = (value<<8) | (_w & 0x00ff); 2742439e4bfSJean-Christophe PLAGNIOL-VILLARD else 2752439e4bfSJean-Christophe PLAGNIOL-VILLARD *((volatile word*)(SMC_BASE_ADDRESS+offset)) = value | (_w & 0xff00); 2762439e4bfSJean-Christophe PLAGNIOL-VILLARD } 2772439e4bfSJean-Christophe PLAGNIOL-VILLARD 2782439e4bfSJean-Christophe PLAGNIOL-VILLARD static inline void SMC_insw(dword offset, volatile uchar* buf, dword len) 2792439e4bfSJean-Christophe PLAGNIOL-VILLARD { 2802439e4bfSJean-Christophe PLAGNIOL-VILLARD volatile word *p = (volatile word *)buf; 2812439e4bfSJean-Christophe PLAGNIOL-VILLARD 2822439e4bfSJean-Christophe PLAGNIOL-VILLARD while (len-- > 0) { 2832439e4bfSJean-Christophe PLAGNIOL-VILLARD *p++ = SMC_inw(offset); 2842439e4bfSJean-Christophe PLAGNIOL-VILLARD barrier(); 2852439e4bfSJean-Christophe PLAGNIOL-VILLARD *((volatile u32*)(0xc0000000)); 2862439e4bfSJean-Christophe PLAGNIOL-VILLARD } 2872439e4bfSJean-Christophe PLAGNIOL-VILLARD } 2882439e4bfSJean-Christophe PLAGNIOL-VILLARD 2892439e4bfSJean-Christophe PLAGNIOL-VILLARD static inline void SMC_outsw(dword offset, uchar* buf, dword len) 2902439e4bfSJean-Christophe PLAGNIOL-VILLARD { 2912439e4bfSJean-Christophe PLAGNIOL-VILLARD volatile word *p = (volatile word *)buf; 2922439e4bfSJean-Christophe PLAGNIOL-VILLARD 2932439e4bfSJean-Christophe PLAGNIOL-VILLARD while (len-- > 0) { 2942439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw(*p++, offset); 2952439e4bfSJean-Christophe PLAGNIOL-VILLARD barrier(); 2962439e4bfSJean-Christophe PLAGNIOL-VILLARD *(volatile u32*)(0xc0000000); 2972439e4bfSJean-Christophe PLAGNIOL-VILLARD } 2982439e4bfSJean-Christophe PLAGNIOL-VILLARD } 2992439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif /* CONFIG_SMC_USE_IOFUNCS */ 3002439e4bfSJean-Christophe PLAGNIOL-VILLARD 3012439e4bfSJean-Christophe PLAGNIOL-VILLARD static char unsigned smc_mac_addr[6] = {0x02, 0x80, 0xad, 0x20, 0x31, 0xb8}; 3022439e4bfSJean-Christophe PLAGNIOL-VILLARD 3032439e4bfSJean-Christophe PLAGNIOL-VILLARD /* 3042439e4bfSJean-Christophe PLAGNIOL-VILLARD * This function must be called before smc_open() if you want to override 3052439e4bfSJean-Christophe PLAGNIOL-VILLARD * the default mac address. 3062439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 3072439e4bfSJean-Christophe PLAGNIOL-VILLARD 3082439e4bfSJean-Christophe PLAGNIOL-VILLARD void smc_set_mac_addr(const unsigned char *addr) { 3092439e4bfSJean-Christophe PLAGNIOL-VILLARD int i; 3102439e4bfSJean-Christophe PLAGNIOL-VILLARD 3112439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i=0; i < sizeof(smc_mac_addr); i++){ 3122439e4bfSJean-Christophe PLAGNIOL-VILLARD smc_mac_addr[i] = addr[i]; 3132439e4bfSJean-Christophe PLAGNIOL-VILLARD } 3142439e4bfSJean-Christophe PLAGNIOL-VILLARD } 3152439e4bfSJean-Christophe PLAGNIOL-VILLARD 3162439e4bfSJean-Christophe PLAGNIOL-VILLARD /* 3172439e4bfSJean-Christophe PLAGNIOL-VILLARD * smc_get_macaddr is no longer used. If you want to override the default 3182439e4bfSJean-Christophe PLAGNIOL-VILLARD * mac address, call smc_get_mac_addr as a part of the board initialization. 3192439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 3202439e4bfSJean-Christophe PLAGNIOL-VILLARD 3212439e4bfSJean-Christophe PLAGNIOL-VILLARD #if 0 3222439e4bfSJean-Christophe PLAGNIOL-VILLARD void smc_get_macaddr( byte *addr ) { 3232439e4bfSJean-Christophe PLAGNIOL-VILLARD /* MAC ADDRESS AT FLASHBLOCK 1 / OFFSET 0x10 */ 3242439e4bfSJean-Christophe PLAGNIOL-VILLARD unsigned char *dnp1110_mac = (unsigned char *) (0xE8000000 + 0x20010); 3252439e4bfSJean-Christophe PLAGNIOL-VILLARD int i; 3262439e4bfSJean-Christophe PLAGNIOL-VILLARD 3272439e4bfSJean-Christophe PLAGNIOL-VILLARD 3282439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i=0; i<6; i++) { 3292439e4bfSJean-Christophe PLAGNIOL-VILLARD addr[0] = *(dnp1110_mac+0); 3302439e4bfSJean-Christophe PLAGNIOL-VILLARD addr[1] = *(dnp1110_mac+1); 3312439e4bfSJean-Christophe PLAGNIOL-VILLARD addr[2] = *(dnp1110_mac+2); 3322439e4bfSJean-Christophe PLAGNIOL-VILLARD addr[3] = *(dnp1110_mac+3); 3332439e4bfSJean-Christophe PLAGNIOL-VILLARD addr[4] = *(dnp1110_mac+4); 3342439e4bfSJean-Christophe PLAGNIOL-VILLARD addr[5] = *(dnp1110_mac+5); 3352439e4bfSJean-Christophe PLAGNIOL-VILLARD } 3362439e4bfSJean-Christophe PLAGNIOL-VILLARD } 3372439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif /* 0 */ 3382439e4bfSJean-Christophe PLAGNIOL-VILLARD 3392439e4bfSJean-Christophe PLAGNIOL-VILLARD /*********************************************** 3402439e4bfSJean-Christophe PLAGNIOL-VILLARD * Show available memory * 3412439e4bfSJean-Christophe PLAGNIOL-VILLARD ***********************************************/ 3422439e4bfSJean-Christophe PLAGNIOL-VILLARD void dump_memory_info(void) 3432439e4bfSJean-Christophe PLAGNIOL-VILLARD { 3442439e4bfSJean-Christophe PLAGNIOL-VILLARD word mem_info; 3452439e4bfSJean-Christophe PLAGNIOL-VILLARD word old_bank; 3462439e4bfSJean-Christophe PLAGNIOL-VILLARD 3472439e4bfSJean-Christophe PLAGNIOL-VILLARD old_bank = SMC_inw(BANK_SELECT)&0xF; 3482439e4bfSJean-Christophe PLAGNIOL-VILLARD 3492439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK(0); 3502439e4bfSJean-Christophe PLAGNIOL-VILLARD mem_info = SMC_inw( MIR_REG ); 3512439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK2("Memory: %4d available\n", (mem_info >> 8)*2048); 3522439e4bfSJean-Christophe PLAGNIOL-VILLARD 3532439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK(old_bank); 3542439e4bfSJean-Christophe PLAGNIOL-VILLARD } 3552439e4bfSJean-Christophe PLAGNIOL-VILLARD /* 3562439e4bfSJean-Christophe PLAGNIOL-VILLARD . A rather simple routine to print out a packet for debugging purposes. 3572439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 3582439e4bfSJean-Christophe PLAGNIOL-VILLARD #if SMC_DEBUG > 2 3592439e4bfSJean-Christophe PLAGNIOL-VILLARD static void print_packet( byte *, int ); 3602439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 3612439e4bfSJean-Christophe PLAGNIOL-VILLARD 3622439e4bfSJean-Christophe PLAGNIOL-VILLARD #define tx_done(dev) 1 3632439e4bfSJean-Christophe PLAGNIOL-VILLARD 3642439e4bfSJean-Christophe PLAGNIOL-VILLARD 3652439e4bfSJean-Christophe PLAGNIOL-VILLARD /* this does a soft reset on the device */ 3662439e4bfSJean-Christophe PLAGNIOL-VILLARD static void smc_reset( void ); 3672439e4bfSJean-Christophe PLAGNIOL-VILLARD 3682439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Enable Interrupts, Receive, and Transmit */ 3692439e4bfSJean-Christophe PLAGNIOL-VILLARD static void smc_enable( void ); 3702439e4bfSJean-Christophe PLAGNIOL-VILLARD 3712439e4bfSJean-Christophe PLAGNIOL-VILLARD /* this puts the device in an inactive state */ 3722439e4bfSJean-Christophe PLAGNIOL-VILLARD static void smc_shutdown( void ); 3732439e4bfSJean-Christophe PLAGNIOL-VILLARD 3742439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Routines to Read and Write the PHY Registers across the 3752439e4bfSJean-Christophe PLAGNIOL-VILLARD MII Management Interface 3762439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 3772439e4bfSJean-Christophe PLAGNIOL-VILLARD 3782439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifndef CONFIG_SMC91111_EXT_PHY 3792439e4bfSJean-Christophe PLAGNIOL-VILLARD static word smc_read_phy_register(byte phyreg); 3802439e4bfSJean-Christophe PLAGNIOL-VILLARD static void smc_write_phy_register(byte phyreg, word phydata); 3812439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif /* !CONFIG_SMC91111_EXT_PHY */ 3822439e4bfSJean-Christophe PLAGNIOL-VILLARD 3832439e4bfSJean-Christophe PLAGNIOL-VILLARD 3842439e4bfSJean-Christophe PLAGNIOL-VILLARD static int poll4int (byte mask, int timeout) 3852439e4bfSJean-Christophe PLAGNIOL-VILLARD { 3866d0f6bcfSJean-Christophe PLAGNIOL-VILLARD int tmo = get_timer (0) + timeout * CONFIG_SYS_HZ; 3872439e4bfSJean-Christophe PLAGNIOL-VILLARD int is_timeout = 0; 3882439e4bfSJean-Christophe PLAGNIOL-VILLARD word old_bank = SMC_inw (BSR_REG); 3892439e4bfSJean-Christophe PLAGNIOL-VILLARD 3902439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK2 ("Polling...\n"); 3912439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK (2); 3922439e4bfSJean-Christophe PLAGNIOL-VILLARD while ((SMC_inw (SMC91111_INT_REG) & mask) == 0) { 3932439e4bfSJean-Christophe PLAGNIOL-VILLARD if (get_timer (0) >= tmo) { 3942439e4bfSJean-Christophe PLAGNIOL-VILLARD is_timeout = 1; 3952439e4bfSJean-Christophe PLAGNIOL-VILLARD break; 3962439e4bfSJean-Christophe PLAGNIOL-VILLARD } 3972439e4bfSJean-Christophe PLAGNIOL-VILLARD } 3982439e4bfSJean-Christophe PLAGNIOL-VILLARD 3992439e4bfSJean-Christophe PLAGNIOL-VILLARD /* restore old bank selection */ 4002439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK (old_bank); 4012439e4bfSJean-Christophe PLAGNIOL-VILLARD 4022439e4bfSJean-Christophe PLAGNIOL-VILLARD if (is_timeout) 4032439e4bfSJean-Christophe PLAGNIOL-VILLARD return 1; 4042439e4bfSJean-Christophe PLAGNIOL-VILLARD else 4052439e4bfSJean-Christophe PLAGNIOL-VILLARD return 0; 4062439e4bfSJean-Christophe PLAGNIOL-VILLARD } 4072439e4bfSJean-Christophe PLAGNIOL-VILLARD 4082439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Only one release command at a time, please */ 4092439e4bfSJean-Christophe PLAGNIOL-VILLARD static inline void smc_wait_mmu_release_complete (void) 4102439e4bfSJean-Christophe PLAGNIOL-VILLARD { 4112439e4bfSJean-Christophe PLAGNIOL-VILLARD int count = 0; 4122439e4bfSJean-Christophe PLAGNIOL-VILLARD 4132439e4bfSJean-Christophe PLAGNIOL-VILLARD /* assume bank 2 selected */ 4142439e4bfSJean-Christophe PLAGNIOL-VILLARD while (SMC_inw (MMU_CMD_REG) & MC_BUSY) { 4152439e4bfSJean-Christophe PLAGNIOL-VILLARD udelay (1); /* Wait until not busy */ 4162439e4bfSJean-Christophe PLAGNIOL-VILLARD if (++count > 200) 4172439e4bfSJean-Christophe PLAGNIOL-VILLARD break; 4182439e4bfSJean-Christophe PLAGNIOL-VILLARD } 4192439e4bfSJean-Christophe PLAGNIOL-VILLARD } 4202439e4bfSJean-Christophe PLAGNIOL-VILLARD 4212439e4bfSJean-Christophe PLAGNIOL-VILLARD /* 4222439e4bfSJean-Christophe PLAGNIOL-VILLARD . Function: smc_reset( void ) 4232439e4bfSJean-Christophe PLAGNIOL-VILLARD . Purpose: 4242439e4bfSJean-Christophe PLAGNIOL-VILLARD . This sets the SMC91111 chip to its normal state, hopefully from whatever 4252439e4bfSJean-Christophe PLAGNIOL-VILLARD . mess that any other DOS driver has put it in. 4262439e4bfSJean-Christophe PLAGNIOL-VILLARD . 4272439e4bfSJean-Christophe PLAGNIOL-VILLARD . Maybe I should reset more registers to defaults in here? SOFTRST should 4282439e4bfSJean-Christophe PLAGNIOL-VILLARD . do that for me. 4292439e4bfSJean-Christophe PLAGNIOL-VILLARD . 4302439e4bfSJean-Christophe PLAGNIOL-VILLARD . Method: 4312439e4bfSJean-Christophe PLAGNIOL-VILLARD . 1. send a SOFT RESET 4322439e4bfSJean-Christophe PLAGNIOL-VILLARD . 2. wait for it to finish 4332439e4bfSJean-Christophe PLAGNIOL-VILLARD . 3. enable autorelease mode 4342439e4bfSJean-Christophe PLAGNIOL-VILLARD . 4. reset the memory management unit 4352439e4bfSJean-Christophe PLAGNIOL-VILLARD . 5. clear all interrupts 4362439e4bfSJean-Christophe PLAGNIOL-VILLARD . 4372439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 4382439e4bfSJean-Christophe PLAGNIOL-VILLARD static void smc_reset (void) 4392439e4bfSJean-Christophe PLAGNIOL-VILLARD { 4402439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK2 ("%s: smc_reset\n", SMC_DEV_NAME); 4412439e4bfSJean-Christophe PLAGNIOL-VILLARD 4422439e4bfSJean-Christophe PLAGNIOL-VILLARD /* This resets the registers mostly to defaults, but doesn't 4432439e4bfSJean-Christophe PLAGNIOL-VILLARD affect EEPROM. That seems unnecessary */ 4442439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK (0); 4452439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (RCR_SOFTRST, RCR_REG); 4462439e4bfSJean-Christophe PLAGNIOL-VILLARD 4472439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Setup the Configuration Register */ 4482439e4bfSJean-Christophe PLAGNIOL-VILLARD /* This is necessary because the CONFIG_REG is not affected */ 4492439e4bfSJean-Christophe PLAGNIOL-VILLARD /* by a soft reset */ 4502439e4bfSJean-Christophe PLAGNIOL-VILLARD 4512439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK (1); 4522439e4bfSJean-Christophe PLAGNIOL-VILLARD #if defined(CONFIG_SMC91111_EXT_PHY) 4532439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (CONFIG_DEFAULT | CONFIG_EXT_PHY, CONFIG_REG); 4542439e4bfSJean-Christophe PLAGNIOL-VILLARD #else 4552439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (CONFIG_DEFAULT, CONFIG_REG); 4562439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 4572439e4bfSJean-Christophe PLAGNIOL-VILLARD 4582439e4bfSJean-Christophe PLAGNIOL-VILLARD 4592439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Release from possible power-down state */ 4602439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Configuration register is not affected by Soft Reset */ 4612439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (SMC_inw (CONFIG_REG) | CONFIG_EPH_POWER_EN, CONFIG_REG); 4622439e4bfSJean-Christophe PLAGNIOL-VILLARD 4632439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK (0); 4642439e4bfSJean-Christophe PLAGNIOL-VILLARD 4652439e4bfSJean-Christophe PLAGNIOL-VILLARD /* this should pause enough for the chip to be happy */ 4662439e4bfSJean-Christophe PLAGNIOL-VILLARD udelay (10); 4672439e4bfSJean-Christophe PLAGNIOL-VILLARD 4682439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Disable transmit and receive functionality */ 4692439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (RCR_CLEAR, RCR_REG); 4702439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (TCR_CLEAR, TCR_REG); 4712439e4bfSJean-Christophe PLAGNIOL-VILLARD 4722439e4bfSJean-Christophe PLAGNIOL-VILLARD /* set the control register */ 4732439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK (1); 4742439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (CTL_DEFAULT, CTL_REG); 4752439e4bfSJean-Christophe PLAGNIOL-VILLARD 4762439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Reset the MMU */ 4772439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK (2); 4782439e4bfSJean-Christophe PLAGNIOL-VILLARD smc_wait_mmu_release_complete (); 4792439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (MC_RESET, MMU_CMD_REG); 4802439e4bfSJean-Christophe PLAGNIOL-VILLARD while (SMC_inw (MMU_CMD_REG) & MC_BUSY) 4812439e4bfSJean-Christophe PLAGNIOL-VILLARD udelay (1); /* Wait until not busy */ 4822439e4bfSJean-Christophe PLAGNIOL-VILLARD 4832439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Note: It doesn't seem that waiting for the MMU busy is needed here, 4842439e4bfSJean-Christophe PLAGNIOL-VILLARD but this is a place where future chipsets _COULD_ break. Be wary 4852439e4bfSJean-Christophe PLAGNIOL-VILLARD of issuing another MMU command right after this */ 4862439e4bfSJean-Christophe PLAGNIOL-VILLARD 4872439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Disable all interrupts */ 4882439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outb (0, IM_REG); 4892439e4bfSJean-Christophe PLAGNIOL-VILLARD } 4902439e4bfSJean-Christophe PLAGNIOL-VILLARD 4912439e4bfSJean-Christophe PLAGNIOL-VILLARD /* 4922439e4bfSJean-Christophe PLAGNIOL-VILLARD . Function: smc_enable 4932439e4bfSJean-Christophe PLAGNIOL-VILLARD . Purpose: let the chip talk to the outside work 4942439e4bfSJean-Christophe PLAGNIOL-VILLARD . Method: 4952439e4bfSJean-Christophe PLAGNIOL-VILLARD . 1. Enable the transmitter 4962439e4bfSJean-Christophe PLAGNIOL-VILLARD . 2. Enable the receiver 4972439e4bfSJean-Christophe PLAGNIOL-VILLARD . 3. Enable interrupts 4982439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 4992439e4bfSJean-Christophe PLAGNIOL-VILLARD static void smc_enable() 5002439e4bfSJean-Christophe PLAGNIOL-VILLARD { 5012439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK2("%s: smc_enable\n", SMC_DEV_NAME); 5022439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK( 0 ); 5032439e4bfSJean-Christophe PLAGNIOL-VILLARD /* see the header file for options in TCR/RCR DEFAULT*/ 5042439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw( TCR_DEFAULT, TCR_REG ); 5052439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw( RCR_DEFAULT, RCR_REG ); 5062439e4bfSJean-Christophe PLAGNIOL-VILLARD 5072439e4bfSJean-Christophe PLAGNIOL-VILLARD /* clear MII_DIS */ 5082439e4bfSJean-Christophe PLAGNIOL-VILLARD /* smc_write_phy_register(PHY_CNTL_REG, 0x0000); */ 5092439e4bfSJean-Christophe PLAGNIOL-VILLARD } 5102439e4bfSJean-Christophe PLAGNIOL-VILLARD 5112439e4bfSJean-Christophe PLAGNIOL-VILLARD /* 5122439e4bfSJean-Christophe PLAGNIOL-VILLARD . Function: smc_shutdown 5132439e4bfSJean-Christophe PLAGNIOL-VILLARD . Purpose: closes down the SMC91xxx chip. 5142439e4bfSJean-Christophe PLAGNIOL-VILLARD . Method: 5152439e4bfSJean-Christophe PLAGNIOL-VILLARD . 1. zero the interrupt mask 5162439e4bfSJean-Christophe PLAGNIOL-VILLARD . 2. clear the enable receive flag 5172439e4bfSJean-Christophe PLAGNIOL-VILLARD . 3. clear the enable xmit flags 5182439e4bfSJean-Christophe PLAGNIOL-VILLARD . 5192439e4bfSJean-Christophe PLAGNIOL-VILLARD . TODO: 5202439e4bfSJean-Christophe PLAGNIOL-VILLARD . (1) maybe utilize power down mode. 5212439e4bfSJean-Christophe PLAGNIOL-VILLARD . Why not yet? Because while the chip will go into power down mode, 5222439e4bfSJean-Christophe PLAGNIOL-VILLARD . the manual says that it will wake up in response to any I/O requests 5232439e4bfSJean-Christophe PLAGNIOL-VILLARD . in the register space. Empirical results do not show this working. 5242439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 5252439e4bfSJean-Christophe PLAGNIOL-VILLARD static void smc_shutdown() 5262439e4bfSJean-Christophe PLAGNIOL-VILLARD { 5272439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK2(CARDNAME ": smc_shutdown\n"); 5282439e4bfSJean-Christophe PLAGNIOL-VILLARD 5292439e4bfSJean-Christophe PLAGNIOL-VILLARD /* no more interrupts for me */ 5302439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK( 2 ); 5312439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outb( 0, IM_REG ); 5322439e4bfSJean-Christophe PLAGNIOL-VILLARD 5332439e4bfSJean-Christophe PLAGNIOL-VILLARD /* and tell the card to stay away from that nasty outside world */ 5342439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK( 0 ); 5352439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outb( RCR_CLEAR, RCR_REG ); 5362439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outb( TCR_CLEAR, TCR_REG ); 5372439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifdef SHARED_RESOURCES 5382439e4bfSJean-Christophe PLAGNIOL-VILLARD swap_to(FLASH); 5392439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 5402439e4bfSJean-Christophe PLAGNIOL-VILLARD } 5412439e4bfSJean-Christophe PLAGNIOL-VILLARD 5422439e4bfSJean-Christophe PLAGNIOL-VILLARD 5432439e4bfSJean-Christophe PLAGNIOL-VILLARD /* 5442439e4bfSJean-Christophe PLAGNIOL-VILLARD . Function: smc_hardware_send_packet(struct net_device * ) 5452439e4bfSJean-Christophe PLAGNIOL-VILLARD . Purpose: 5462439e4bfSJean-Christophe PLAGNIOL-VILLARD . This sends the actual packet to the SMC9xxx chip. 5472439e4bfSJean-Christophe PLAGNIOL-VILLARD . 5482439e4bfSJean-Christophe PLAGNIOL-VILLARD . Algorithm: 5492439e4bfSJean-Christophe PLAGNIOL-VILLARD . First, see if a saved_skb is available. 5502439e4bfSJean-Christophe PLAGNIOL-VILLARD . ( this should NOT be called if there is no 'saved_skb' 5512439e4bfSJean-Christophe PLAGNIOL-VILLARD . Now, find the packet number that the chip allocated 5522439e4bfSJean-Christophe PLAGNIOL-VILLARD . Point the data pointers at it in memory 5532439e4bfSJean-Christophe PLAGNIOL-VILLARD . Set the length word in the chip's memory 5542439e4bfSJean-Christophe PLAGNIOL-VILLARD . Dump the packet to chip memory 5552439e4bfSJean-Christophe PLAGNIOL-VILLARD . Check if a last byte is needed ( odd length packet ) 5562439e4bfSJean-Christophe PLAGNIOL-VILLARD . if so, set the control flag right 5572439e4bfSJean-Christophe PLAGNIOL-VILLARD . Tell the card to send it 5582439e4bfSJean-Christophe PLAGNIOL-VILLARD . Enable the transmit interrupt, so I know if it failed 5592439e4bfSJean-Christophe PLAGNIOL-VILLARD . Free the kernel data if I actually sent it. 5602439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 5612439e4bfSJean-Christophe PLAGNIOL-VILLARD static int smc_send_packet (volatile void *packet, int packet_length) 5622439e4bfSJean-Christophe PLAGNIOL-VILLARD { 5632439e4bfSJean-Christophe PLAGNIOL-VILLARD byte packet_no; 5642439e4bfSJean-Christophe PLAGNIOL-VILLARD unsigned long ioaddr; 5652439e4bfSJean-Christophe PLAGNIOL-VILLARD byte *buf; 5662439e4bfSJean-Christophe PLAGNIOL-VILLARD int length; 5672439e4bfSJean-Christophe PLAGNIOL-VILLARD int numPages; 5682439e4bfSJean-Christophe PLAGNIOL-VILLARD int try = 0; 5692439e4bfSJean-Christophe PLAGNIOL-VILLARD int time_out; 5702439e4bfSJean-Christophe PLAGNIOL-VILLARD byte status; 5712439e4bfSJean-Christophe PLAGNIOL-VILLARD byte saved_pnr; 5722439e4bfSJean-Christophe PLAGNIOL-VILLARD word saved_ptr; 5732439e4bfSJean-Christophe PLAGNIOL-VILLARD 5742439e4bfSJean-Christophe PLAGNIOL-VILLARD /* save PTR and PNR registers before manipulation */ 5752439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK (2); 5762439e4bfSJean-Christophe PLAGNIOL-VILLARD saved_pnr = SMC_inb( PN_REG ); 5772439e4bfSJean-Christophe PLAGNIOL-VILLARD saved_ptr = SMC_inw( PTR_REG ); 5782439e4bfSJean-Christophe PLAGNIOL-VILLARD 5792439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK3 ("%s: smc_hardware_send_packet\n", SMC_DEV_NAME); 5802439e4bfSJean-Christophe PLAGNIOL-VILLARD 5812439e4bfSJean-Christophe PLAGNIOL-VILLARD length = ETH_ZLEN < packet_length ? packet_length : ETH_ZLEN; 5822439e4bfSJean-Christophe PLAGNIOL-VILLARD 5832439e4bfSJean-Christophe PLAGNIOL-VILLARD /* allocate memory 5842439e4bfSJean-Christophe PLAGNIOL-VILLARD ** The MMU wants the number of pages to be the number of 256 bytes 5852439e4bfSJean-Christophe PLAGNIOL-VILLARD ** 'pages', minus 1 ( since a packet can't ever have 0 pages :) ) 5862439e4bfSJean-Christophe PLAGNIOL-VILLARD ** 5872439e4bfSJean-Christophe PLAGNIOL-VILLARD ** The 91C111 ignores the size bits, but the code is left intact 5882439e4bfSJean-Christophe PLAGNIOL-VILLARD ** for backwards and future compatibility. 5892439e4bfSJean-Christophe PLAGNIOL-VILLARD ** 5902439e4bfSJean-Christophe PLAGNIOL-VILLARD ** Pkt size for allocating is data length +6 (for additional status 5912439e4bfSJean-Christophe PLAGNIOL-VILLARD ** words, length and ctl!) 5922439e4bfSJean-Christophe PLAGNIOL-VILLARD ** 5932439e4bfSJean-Christophe PLAGNIOL-VILLARD ** If odd size then last byte is included in this header. 5942439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 5952439e4bfSJean-Christophe PLAGNIOL-VILLARD numPages = ((length & 0xfffe) + 6); 5962439e4bfSJean-Christophe PLAGNIOL-VILLARD numPages >>= 8; /* Divide by 256 */ 5972439e4bfSJean-Christophe PLAGNIOL-VILLARD 5982439e4bfSJean-Christophe PLAGNIOL-VILLARD if (numPages > 7) { 5992439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("%s: Far too big packet error. \n", SMC_DEV_NAME); 6002439e4bfSJean-Christophe PLAGNIOL-VILLARD return 0; 6012439e4bfSJean-Christophe PLAGNIOL-VILLARD } 6022439e4bfSJean-Christophe PLAGNIOL-VILLARD 6032439e4bfSJean-Christophe PLAGNIOL-VILLARD /* now, try to allocate the memory */ 6042439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK (2); 6052439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (MC_ALLOC | numPages, MMU_CMD_REG); 6062439e4bfSJean-Christophe PLAGNIOL-VILLARD 6072439e4bfSJean-Christophe PLAGNIOL-VILLARD /* FIXME: the ALLOC_INT bit never gets set * 6082439e4bfSJean-Christophe PLAGNIOL-VILLARD * so the following will always give a * 6092439e4bfSJean-Christophe PLAGNIOL-VILLARD * memory allocation error. * 6102439e4bfSJean-Christophe PLAGNIOL-VILLARD * same code works in armboot though * 6112439e4bfSJean-Christophe PLAGNIOL-VILLARD * -ro 6122439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 6132439e4bfSJean-Christophe PLAGNIOL-VILLARD 6142439e4bfSJean-Christophe PLAGNIOL-VILLARD again: 6152439e4bfSJean-Christophe PLAGNIOL-VILLARD try++; 6162439e4bfSJean-Christophe PLAGNIOL-VILLARD time_out = MEMORY_WAIT_TIME; 6172439e4bfSJean-Christophe PLAGNIOL-VILLARD do { 6182439e4bfSJean-Christophe PLAGNIOL-VILLARD status = SMC_inb (SMC91111_INT_REG); 6192439e4bfSJean-Christophe PLAGNIOL-VILLARD if (status & IM_ALLOC_INT) { 6202439e4bfSJean-Christophe PLAGNIOL-VILLARD /* acknowledge the interrupt */ 6212439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outb (IM_ALLOC_INT, SMC91111_INT_REG); 6222439e4bfSJean-Christophe PLAGNIOL-VILLARD break; 6232439e4bfSJean-Christophe PLAGNIOL-VILLARD } 6242439e4bfSJean-Christophe PLAGNIOL-VILLARD } while (--time_out); 6252439e4bfSJean-Christophe PLAGNIOL-VILLARD 6262439e4bfSJean-Christophe PLAGNIOL-VILLARD if (!time_out) { 6272439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK2 ("%s: memory allocation, try %d failed ...\n", 6282439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_DEV_NAME, try); 6292439e4bfSJean-Christophe PLAGNIOL-VILLARD if (try < SMC_ALLOC_MAX_TRY) 6302439e4bfSJean-Christophe PLAGNIOL-VILLARD goto again; 6312439e4bfSJean-Christophe PLAGNIOL-VILLARD else 6322439e4bfSJean-Christophe PLAGNIOL-VILLARD return 0; 6332439e4bfSJean-Christophe PLAGNIOL-VILLARD } 6342439e4bfSJean-Christophe PLAGNIOL-VILLARD 6352439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK2 ("%s: memory allocation, try %d succeeded ...\n", 6362439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_DEV_NAME, try); 6372439e4bfSJean-Christophe PLAGNIOL-VILLARD 6382439e4bfSJean-Christophe PLAGNIOL-VILLARD /* I can send the packet now.. */ 6392439e4bfSJean-Christophe PLAGNIOL-VILLARD 6402439e4bfSJean-Christophe PLAGNIOL-VILLARD ioaddr = SMC_BASE_ADDRESS; 6412439e4bfSJean-Christophe PLAGNIOL-VILLARD 6422439e4bfSJean-Christophe PLAGNIOL-VILLARD buf = (byte *) packet; 6432439e4bfSJean-Christophe PLAGNIOL-VILLARD 6442439e4bfSJean-Christophe PLAGNIOL-VILLARD /* If I get here, I _know_ there is a packet slot waiting for me */ 6452439e4bfSJean-Christophe PLAGNIOL-VILLARD packet_no = SMC_inb (AR_REG); 6462439e4bfSJean-Christophe PLAGNIOL-VILLARD if (packet_no & AR_FAILED) { 6472439e4bfSJean-Christophe PLAGNIOL-VILLARD /* or isn't there? BAD CHIP! */ 6482439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("%s: Memory allocation failed. \n", SMC_DEV_NAME); 6492439e4bfSJean-Christophe PLAGNIOL-VILLARD return 0; 6502439e4bfSJean-Christophe PLAGNIOL-VILLARD } 6512439e4bfSJean-Christophe PLAGNIOL-VILLARD 6522439e4bfSJean-Christophe PLAGNIOL-VILLARD /* we have a packet address, so tell the card to use it */ 6532439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifndef CONFIG_XAENIAX 6542439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outb (packet_no, PN_REG); 6552439e4bfSJean-Christophe PLAGNIOL-VILLARD #else 6562439e4bfSJean-Christophe PLAGNIOL-VILLARD /* On Xaeniax board, we can't use SMC_outb here because that way 6572439e4bfSJean-Christophe PLAGNIOL-VILLARD * the Allocate MMU command will end up written to the command register 6582439e4bfSJean-Christophe PLAGNIOL-VILLARD * as well, which will lead to a problem. 6592439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 6602439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outl (packet_no << 16, 0); 6612439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 6622439e4bfSJean-Christophe PLAGNIOL-VILLARD /* do not write new ptr value if Write data fifo not empty */ 6632439e4bfSJean-Christophe PLAGNIOL-VILLARD while ( saved_ptr & PTR_NOTEMPTY ) 6642439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("Write data fifo not empty!\n"); 6652439e4bfSJean-Christophe PLAGNIOL-VILLARD 6662439e4bfSJean-Christophe PLAGNIOL-VILLARD /* point to the beginning of the packet */ 6672439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (PTR_AUTOINC, PTR_REG); 6682439e4bfSJean-Christophe PLAGNIOL-VILLARD 6692439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK3 ("%s: Trying to xmit packet of length %x\n", 6702439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_DEV_NAME, length); 6712439e4bfSJean-Christophe PLAGNIOL-VILLARD 6722439e4bfSJean-Christophe PLAGNIOL-VILLARD #if SMC_DEBUG > 2 6732439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("Transmitting Packet\n"); 6742439e4bfSJean-Christophe PLAGNIOL-VILLARD print_packet (buf, length); 6752439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 6762439e4bfSJean-Christophe PLAGNIOL-VILLARD 6772439e4bfSJean-Christophe PLAGNIOL-VILLARD /* send the packet length ( +6 for status, length and ctl byte ) 6782439e4bfSJean-Christophe PLAGNIOL-VILLARD and the status word ( set to zeros ) */ 6792439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifdef USE_32_BIT 6802439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outl ((length + 6) << 16, SMC91111_DATA_REG); 6812439e4bfSJean-Christophe PLAGNIOL-VILLARD #else 6822439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (0, SMC91111_DATA_REG); 6832439e4bfSJean-Christophe PLAGNIOL-VILLARD /* send the packet length ( +6 for status words, length, and ctl */ 6842439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw ((length + 6), SMC91111_DATA_REG); 6852439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 6862439e4bfSJean-Christophe PLAGNIOL-VILLARD 6872439e4bfSJean-Christophe PLAGNIOL-VILLARD /* send the actual data 6882439e4bfSJean-Christophe PLAGNIOL-VILLARD . I _think_ it's faster to send the longs first, and then 6892439e4bfSJean-Christophe PLAGNIOL-VILLARD . mop up by sending the last word. It depends heavily 6902439e4bfSJean-Christophe PLAGNIOL-VILLARD . on alignment, at least on the 486. Maybe it would be 6912439e4bfSJean-Christophe PLAGNIOL-VILLARD . a good idea to check which is optimal? But that could take 6922439e4bfSJean-Christophe PLAGNIOL-VILLARD . almost as much time as is saved? 6932439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 6942439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifdef USE_32_BIT 6952439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outsl (SMC91111_DATA_REG, buf, length >> 2); 6962439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifndef CONFIG_XAENIAX 6972439e4bfSJean-Christophe PLAGNIOL-VILLARD if (length & 0x2) 6982439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (*((word *) (buf + (length & 0xFFFFFFFC))), 6992439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC91111_DATA_REG); 7002439e4bfSJean-Christophe PLAGNIOL-VILLARD #else 7012439e4bfSJean-Christophe PLAGNIOL-VILLARD /* On XANEIAX, we can only use 32-bit writes, so we need to handle 7022439e4bfSJean-Christophe PLAGNIOL-VILLARD * unaligned tail part specially. The standard code doesn't work. 7032439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 7042439e4bfSJean-Christophe PLAGNIOL-VILLARD if ((length & 3) == 3) { 7052439e4bfSJean-Christophe PLAGNIOL-VILLARD u16 * ptr = (u16*) &buf[length-3]; 7062439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outl((*ptr) | ((0x2000 | buf[length-1]) << 16), 7072439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC91111_DATA_REG); 7082439e4bfSJean-Christophe PLAGNIOL-VILLARD } else if ((length & 2) == 2) { 7092439e4bfSJean-Christophe PLAGNIOL-VILLARD u16 * ptr = (u16*) &buf[length-2]; 7102439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outl(*ptr, SMC91111_DATA_REG); 7112439e4bfSJean-Christophe PLAGNIOL-VILLARD } else if (length & 1) { 7122439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outl((0x2000 | buf[length-1]), SMC91111_DATA_REG); 7132439e4bfSJean-Christophe PLAGNIOL-VILLARD } else { 7142439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outl(0, SMC91111_DATA_REG); 7152439e4bfSJean-Christophe PLAGNIOL-VILLARD } 7162439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 7172439e4bfSJean-Christophe PLAGNIOL-VILLARD #else 7182439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outsw (SMC91111_DATA_REG, buf, (length) >> 1); 7192439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif /* USE_32_BIT */ 7202439e4bfSJean-Christophe PLAGNIOL-VILLARD 7212439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifndef CONFIG_XAENIAX 7222439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Send the last byte, if there is one. */ 7232439e4bfSJean-Christophe PLAGNIOL-VILLARD if ((length & 1) == 0) { 7242439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (0, SMC91111_DATA_REG); 7252439e4bfSJean-Christophe PLAGNIOL-VILLARD } else { 7262439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (buf[length - 1] | 0x2000, SMC91111_DATA_REG); 7272439e4bfSJean-Christophe PLAGNIOL-VILLARD } 7282439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 7292439e4bfSJean-Christophe PLAGNIOL-VILLARD 7302439e4bfSJean-Christophe PLAGNIOL-VILLARD /* and let the chipset deal with it */ 7312439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (MC_ENQUEUE, MMU_CMD_REG); 7322439e4bfSJean-Christophe PLAGNIOL-VILLARD 7332439e4bfSJean-Christophe PLAGNIOL-VILLARD /* poll for TX INT */ 7342439e4bfSJean-Christophe PLAGNIOL-VILLARD /* if (poll4int (IM_TX_INT, SMC_TX_TIMEOUT)) { */ 7352439e4bfSJean-Christophe PLAGNIOL-VILLARD /* poll for TX_EMPTY INT - autorelease enabled */ 7362439e4bfSJean-Christophe PLAGNIOL-VILLARD if (poll4int(IM_TX_EMPTY_INT, SMC_TX_TIMEOUT)) { 7372439e4bfSJean-Christophe PLAGNIOL-VILLARD /* sending failed */ 7382439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK2 ("%s: TX timeout, sending failed...\n", SMC_DEV_NAME); 7392439e4bfSJean-Christophe PLAGNIOL-VILLARD 7402439e4bfSJean-Christophe PLAGNIOL-VILLARD /* release packet */ 7412439e4bfSJean-Christophe PLAGNIOL-VILLARD /* no need to release, MMU does that now */ 7422439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifdef CONFIG_XAENIAX 7432439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (MC_FREEPKT, MMU_CMD_REG); 7442439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 7452439e4bfSJean-Christophe PLAGNIOL-VILLARD 7462439e4bfSJean-Christophe PLAGNIOL-VILLARD /* wait for MMU getting ready (low) */ 7472439e4bfSJean-Christophe PLAGNIOL-VILLARD while (SMC_inw (MMU_CMD_REG) & MC_BUSY) { 7482439e4bfSJean-Christophe PLAGNIOL-VILLARD udelay (10); 7492439e4bfSJean-Christophe PLAGNIOL-VILLARD } 7502439e4bfSJean-Christophe PLAGNIOL-VILLARD 7512439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK2 ("MMU ready\n"); 7522439e4bfSJean-Christophe PLAGNIOL-VILLARD 7532439e4bfSJean-Christophe PLAGNIOL-VILLARD 7542439e4bfSJean-Christophe PLAGNIOL-VILLARD return 0; 7552439e4bfSJean-Christophe PLAGNIOL-VILLARD } else { 7562439e4bfSJean-Christophe PLAGNIOL-VILLARD /* ack. int */ 7572439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outb (IM_TX_EMPTY_INT, SMC91111_INT_REG); 7582439e4bfSJean-Christophe PLAGNIOL-VILLARD /* SMC_outb (IM_TX_INT, SMC91111_INT_REG); */ 7592439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK2 ("%s: Sent packet of length %d \n", SMC_DEV_NAME, 7602439e4bfSJean-Christophe PLAGNIOL-VILLARD length); 7612439e4bfSJean-Christophe PLAGNIOL-VILLARD 7622439e4bfSJean-Christophe PLAGNIOL-VILLARD /* release packet */ 7632439e4bfSJean-Christophe PLAGNIOL-VILLARD /* no need to release, MMU does that now */ 7642439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifdef CONFIG_XAENIAX 7652439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (MC_FREEPKT, MMU_CMD_REG); 7662439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 7672439e4bfSJean-Christophe PLAGNIOL-VILLARD 7682439e4bfSJean-Christophe PLAGNIOL-VILLARD /* wait for MMU getting ready (low) */ 7692439e4bfSJean-Christophe PLAGNIOL-VILLARD while (SMC_inw (MMU_CMD_REG) & MC_BUSY) { 7702439e4bfSJean-Christophe PLAGNIOL-VILLARD udelay (10); 7712439e4bfSJean-Christophe PLAGNIOL-VILLARD } 7722439e4bfSJean-Christophe PLAGNIOL-VILLARD 7732439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK2 ("MMU ready\n"); 7742439e4bfSJean-Christophe PLAGNIOL-VILLARD 7752439e4bfSJean-Christophe PLAGNIOL-VILLARD 7762439e4bfSJean-Christophe PLAGNIOL-VILLARD } 7772439e4bfSJean-Christophe PLAGNIOL-VILLARD 7782439e4bfSJean-Christophe PLAGNIOL-VILLARD /* restore previously saved registers */ 7792439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifndef CONFIG_XAENIAX 7802439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outb( saved_pnr, PN_REG ); 7812439e4bfSJean-Christophe PLAGNIOL-VILLARD #else 7822439e4bfSJean-Christophe PLAGNIOL-VILLARD /* On Xaeniax board, we can't use SMC_outb here because that way 7832439e4bfSJean-Christophe PLAGNIOL-VILLARD * the Allocate MMU command will end up written to the command register 7842439e4bfSJean-Christophe PLAGNIOL-VILLARD * as well, which will lead to a problem. 7852439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 7862439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outl(saved_pnr << 16, 0); 7872439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 7882439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw( saved_ptr, PTR_REG ); 7892439e4bfSJean-Christophe PLAGNIOL-VILLARD 7902439e4bfSJean-Christophe PLAGNIOL-VILLARD return length; 7912439e4bfSJean-Christophe PLAGNIOL-VILLARD } 7922439e4bfSJean-Christophe PLAGNIOL-VILLARD 7932439e4bfSJean-Christophe PLAGNIOL-VILLARD /*------------------------------------------------------------------------- 7942439e4bfSJean-Christophe PLAGNIOL-VILLARD | 7952439e4bfSJean-Christophe PLAGNIOL-VILLARD | smc_destructor( struct net_device * dev ) 7962439e4bfSJean-Christophe PLAGNIOL-VILLARD | Input parameters: 7972439e4bfSJean-Christophe PLAGNIOL-VILLARD | dev, pointer to the device structure 7982439e4bfSJean-Christophe PLAGNIOL-VILLARD | 7992439e4bfSJean-Christophe PLAGNIOL-VILLARD | Output: 8002439e4bfSJean-Christophe PLAGNIOL-VILLARD | None. 8012439e4bfSJean-Christophe PLAGNIOL-VILLARD | 8022439e4bfSJean-Christophe PLAGNIOL-VILLARD --------------------------------------------------------------------------- 8032439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 8042439e4bfSJean-Christophe PLAGNIOL-VILLARD void smc_destructor() 8052439e4bfSJean-Christophe PLAGNIOL-VILLARD { 8062439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK2(CARDNAME ": smc_destructor\n"); 8072439e4bfSJean-Christophe PLAGNIOL-VILLARD } 8082439e4bfSJean-Christophe PLAGNIOL-VILLARD 8092439e4bfSJean-Christophe PLAGNIOL-VILLARD 8102439e4bfSJean-Christophe PLAGNIOL-VILLARD /* 8112439e4bfSJean-Christophe PLAGNIOL-VILLARD * Open and Initialize the board 8122439e4bfSJean-Christophe PLAGNIOL-VILLARD * 8132439e4bfSJean-Christophe PLAGNIOL-VILLARD * Set up everything, reset the card, etc .. 8142439e4bfSJean-Christophe PLAGNIOL-VILLARD * 8152439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 8162439e4bfSJean-Christophe PLAGNIOL-VILLARD static int smc_open (bd_t * bd) 8172439e4bfSJean-Christophe PLAGNIOL-VILLARD { 8182439e4bfSJean-Christophe PLAGNIOL-VILLARD int i, err; 8192439e4bfSJean-Christophe PLAGNIOL-VILLARD 8202439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK2 ("%s: smc_open\n", SMC_DEV_NAME); 8212439e4bfSJean-Christophe PLAGNIOL-VILLARD 8222439e4bfSJean-Christophe PLAGNIOL-VILLARD /* reset the hardware */ 8232439e4bfSJean-Christophe PLAGNIOL-VILLARD smc_reset (); 8242439e4bfSJean-Christophe PLAGNIOL-VILLARD smc_enable (); 8252439e4bfSJean-Christophe PLAGNIOL-VILLARD 8262439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Configure the PHY */ 8272439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifndef CONFIG_SMC91111_EXT_PHY 8282439e4bfSJean-Christophe PLAGNIOL-VILLARD smc_phy_configure (); 8292439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 8302439e4bfSJean-Christophe PLAGNIOL-VILLARD 8312439e4bfSJean-Christophe PLAGNIOL-VILLARD /* conservative setting (10Mbps, HalfDuplex, no AutoNeg.) */ 8322439e4bfSJean-Christophe PLAGNIOL-VILLARD /* SMC_SELECT_BANK(0); */ 8332439e4bfSJean-Christophe PLAGNIOL-VILLARD /* SMC_outw(0, RPC_REG); */ 8342439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK (1); 8352439e4bfSJean-Christophe PLAGNIOL-VILLARD 8362439e4bfSJean-Christophe PLAGNIOL-VILLARD err = smc_get_ethaddr (bd); /* set smc_mac_addr, and sync it with u-boot globals */ 837*03f3d8d3SMike Frysinger if (err < 0) 838*03f3d8d3SMike Frysinger return -1; 8392439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifdef USE_32_BIT 8402439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i = 0; i < 6; i += 2) { 8412439e4bfSJean-Christophe PLAGNIOL-VILLARD word address; 8422439e4bfSJean-Christophe PLAGNIOL-VILLARD 8432439e4bfSJean-Christophe PLAGNIOL-VILLARD address = smc_mac_addr[i + 1] << 8; 8442439e4bfSJean-Christophe PLAGNIOL-VILLARD address |= smc_mac_addr[i]; 8452439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (address, (ADDR0_REG + i)); 8462439e4bfSJean-Christophe PLAGNIOL-VILLARD } 8472439e4bfSJean-Christophe PLAGNIOL-VILLARD #else 8482439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i = 0; i < 6; i++) 8492439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outb (smc_mac_addr[i], (ADDR0_REG + i)); 8502439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 8512439e4bfSJean-Christophe PLAGNIOL-VILLARD 8522439e4bfSJean-Christophe PLAGNIOL-VILLARD return 0; 8532439e4bfSJean-Christophe PLAGNIOL-VILLARD } 8542439e4bfSJean-Christophe PLAGNIOL-VILLARD 8552439e4bfSJean-Christophe PLAGNIOL-VILLARD /*------------------------------------------------------------- 8562439e4bfSJean-Christophe PLAGNIOL-VILLARD . 8572439e4bfSJean-Christophe PLAGNIOL-VILLARD . smc_rcv - receive a packet from the card 8582439e4bfSJean-Christophe PLAGNIOL-VILLARD . 8592439e4bfSJean-Christophe PLAGNIOL-VILLARD . There is ( at least ) a packet waiting to be read from 8602439e4bfSJean-Christophe PLAGNIOL-VILLARD . chip-memory. 8612439e4bfSJean-Christophe PLAGNIOL-VILLARD . 8622439e4bfSJean-Christophe PLAGNIOL-VILLARD . o Read the status 8632439e4bfSJean-Christophe PLAGNIOL-VILLARD . o If an error, record it 8642439e4bfSJean-Christophe PLAGNIOL-VILLARD . o otherwise, read in the packet 8652439e4bfSJean-Christophe PLAGNIOL-VILLARD -------------------------------------------------------------- 8662439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 8672439e4bfSJean-Christophe PLAGNIOL-VILLARD static int smc_rcv() 8682439e4bfSJean-Christophe PLAGNIOL-VILLARD { 8692439e4bfSJean-Christophe PLAGNIOL-VILLARD int packet_number; 8702439e4bfSJean-Christophe PLAGNIOL-VILLARD word status; 8712439e4bfSJean-Christophe PLAGNIOL-VILLARD word packet_length; 8722439e4bfSJean-Christophe PLAGNIOL-VILLARD int is_error = 0; 8732439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifdef USE_32_BIT 8742439e4bfSJean-Christophe PLAGNIOL-VILLARD dword stat_len; 8752439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 8762439e4bfSJean-Christophe PLAGNIOL-VILLARD byte saved_pnr; 8772439e4bfSJean-Christophe PLAGNIOL-VILLARD word saved_ptr; 8782439e4bfSJean-Christophe PLAGNIOL-VILLARD 8792439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK(2); 8802439e4bfSJean-Christophe PLAGNIOL-VILLARD /* save PTR and PTR registers */ 8812439e4bfSJean-Christophe PLAGNIOL-VILLARD saved_pnr = SMC_inb( PN_REG ); 8822439e4bfSJean-Christophe PLAGNIOL-VILLARD saved_ptr = SMC_inw( PTR_REG ); 8832439e4bfSJean-Christophe PLAGNIOL-VILLARD 8842439e4bfSJean-Christophe PLAGNIOL-VILLARD packet_number = SMC_inw( RXFIFO_REG ); 8852439e4bfSJean-Christophe PLAGNIOL-VILLARD 8862439e4bfSJean-Christophe PLAGNIOL-VILLARD if ( packet_number & RXFIFO_REMPTY ) { 8872439e4bfSJean-Christophe PLAGNIOL-VILLARD 8882439e4bfSJean-Christophe PLAGNIOL-VILLARD return 0; 8892439e4bfSJean-Christophe PLAGNIOL-VILLARD } 8902439e4bfSJean-Christophe PLAGNIOL-VILLARD 8912439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK3("%s: smc_rcv\n", SMC_DEV_NAME); 8922439e4bfSJean-Christophe PLAGNIOL-VILLARD /* start reading from the start of the packet */ 8932439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw( PTR_READ | PTR_RCV | PTR_AUTOINC, PTR_REG ); 8942439e4bfSJean-Christophe PLAGNIOL-VILLARD 8952439e4bfSJean-Christophe PLAGNIOL-VILLARD /* First two words are status and packet_length */ 8962439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifdef USE_32_BIT 8972439e4bfSJean-Christophe PLAGNIOL-VILLARD stat_len = SMC_inl(SMC91111_DATA_REG); 8982439e4bfSJean-Christophe PLAGNIOL-VILLARD status = stat_len & 0xffff; 8992439e4bfSJean-Christophe PLAGNIOL-VILLARD packet_length = stat_len >> 16; 9002439e4bfSJean-Christophe PLAGNIOL-VILLARD #else 9012439e4bfSJean-Christophe PLAGNIOL-VILLARD status = SMC_inw( SMC91111_DATA_REG ); 9022439e4bfSJean-Christophe PLAGNIOL-VILLARD packet_length = SMC_inw( SMC91111_DATA_REG ); 9032439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 9042439e4bfSJean-Christophe PLAGNIOL-VILLARD 9052439e4bfSJean-Christophe PLAGNIOL-VILLARD packet_length &= 0x07ff; /* mask off top bits */ 9062439e4bfSJean-Christophe PLAGNIOL-VILLARD 9072439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK2("RCV: STATUS %4x LENGTH %4x\n", status, packet_length ); 9082439e4bfSJean-Christophe PLAGNIOL-VILLARD 9092439e4bfSJean-Christophe PLAGNIOL-VILLARD if ( !(status & RS_ERRORS ) ){ 9102439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Adjust for having already read the first two words */ 9112439e4bfSJean-Christophe PLAGNIOL-VILLARD packet_length -= 4; /*4; */ 9122439e4bfSJean-Christophe PLAGNIOL-VILLARD 9132439e4bfSJean-Christophe PLAGNIOL-VILLARD 9142439e4bfSJean-Christophe PLAGNIOL-VILLARD /* set odd length for bug in LAN91C111, */ 9152439e4bfSJean-Christophe PLAGNIOL-VILLARD /* which never sets RS_ODDFRAME */ 9162439e4bfSJean-Christophe PLAGNIOL-VILLARD /* TODO ? */ 9172439e4bfSJean-Christophe PLAGNIOL-VILLARD 9182439e4bfSJean-Christophe PLAGNIOL-VILLARD 9192439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifdef USE_32_BIT 9202439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK3(" Reading %d dwords (and %d bytes) \n", 9212439e4bfSJean-Christophe PLAGNIOL-VILLARD packet_length >> 2, packet_length & 3 ); 9222439e4bfSJean-Christophe PLAGNIOL-VILLARD /* QUESTION: Like in the TX routine, do I want 9232439e4bfSJean-Christophe PLAGNIOL-VILLARD to send the DWORDs or the bytes first, or some 9242439e4bfSJean-Christophe PLAGNIOL-VILLARD mixture. A mixture might improve already slow PIO 9252439e4bfSJean-Christophe PLAGNIOL-VILLARD performance */ 9262439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_insl( SMC91111_DATA_REG , NetRxPackets[0], packet_length >> 2 ); 9272439e4bfSJean-Christophe PLAGNIOL-VILLARD /* read the left over bytes */ 9282439e4bfSJean-Christophe PLAGNIOL-VILLARD if (packet_length & 3) { 9292439e4bfSJean-Christophe PLAGNIOL-VILLARD int i; 9302439e4bfSJean-Christophe PLAGNIOL-VILLARD 9312439e4bfSJean-Christophe PLAGNIOL-VILLARD byte *tail = (byte *)(NetRxPackets[0] + (packet_length & ~3)); 9322439e4bfSJean-Christophe PLAGNIOL-VILLARD dword leftover = SMC_inl(SMC91111_DATA_REG); 9332439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i=0; i<(packet_length & 3); i++) 9342439e4bfSJean-Christophe PLAGNIOL-VILLARD *tail++ = (byte) (leftover >> (8*i)) & 0xff; 9352439e4bfSJean-Christophe PLAGNIOL-VILLARD } 9362439e4bfSJean-Christophe PLAGNIOL-VILLARD #else 9372439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK3(" Reading %d words and %d byte(s) \n", 9382439e4bfSJean-Christophe PLAGNIOL-VILLARD (packet_length >> 1 ), packet_length & 1 ); 9392439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_insw(SMC91111_DATA_REG , NetRxPackets[0], packet_length >> 1); 9402439e4bfSJean-Christophe PLAGNIOL-VILLARD 9412439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif /* USE_32_BIT */ 9422439e4bfSJean-Christophe PLAGNIOL-VILLARD 9432439e4bfSJean-Christophe PLAGNIOL-VILLARD #if SMC_DEBUG > 2 9442439e4bfSJean-Christophe PLAGNIOL-VILLARD printf("Receiving Packet\n"); 9452439e4bfSJean-Christophe PLAGNIOL-VILLARD print_packet( NetRxPackets[0], packet_length ); 9462439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 9472439e4bfSJean-Christophe PLAGNIOL-VILLARD } else { 9482439e4bfSJean-Christophe PLAGNIOL-VILLARD /* error ... */ 9492439e4bfSJean-Christophe PLAGNIOL-VILLARD /* TODO ? */ 9502439e4bfSJean-Christophe PLAGNIOL-VILLARD is_error = 1; 9512439e4bfSJean-Christophe PLAGNIOL-VILLARD } 9522439e4bfSJean-Christophe PLAGNIOL-VILLARD 9532439e4bfSJean-Christophe PLAGNIOL-VILLARD while ( SMC_inw( MMU_CMD_REG ) & MC_BUSY ) 9542439e4bfSJean-Christophe PLAGNIOL-VILLARD udelay(1); /* Wait until not busy */ 9552439e4bfSJean-Christophe PLAGNIOL-VILLARD 9562439e4bfSJean-Christophe PLAGNIOL-VILLARD /* error or good, tell the card to get rid of this packet */ 9572439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw( MC_RELEASE, MMU_CMD_REG ); 9582439e4bfSJean-Christophe PLAGNIOL-VILLARD 9592439e4bfSJean-Christophe PLAGNIOL-VILLARD while ( SMC_inw( MMU_CMD_REG ) & MC_BUSY ) 9602439e4bfSJean-Christophe PLAGNIOL-VILLARD udelay(1); /* Wait until not busy */ 9612439e4bfSJean-Christophe PLAGNIOL-VILLARD 9622439e4bfSJean-Christophe PLAGNIOL-VILLARD /* restore saved registers */ 9632439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifndef CONFIG_XAENIAX 9642439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outb( saved_pnr, PN_REG ); 9652439e4bfSJean-Christophe PLAGNIOL-VILLARD #else 9662439e4bfSJean-Christophe PLAGNIOL-VILLARD /* On Xaeniax board, we can't use SMC_outb here because that way 9672439e4bfSJean-Christophe PLAGNIOL-VILLARD * the Allocate MMU command will end up written to the command register 9682439e4bfSJean-Christophe PLAGNIOL-VILLARD * as well, which will lead to a problem. 9692439e4bfSJean-Christophe PLAGNIOL-VILLARD */ 9702439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outl( saved_pnr << 16, 0); 9712439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 9722439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw( saved_ptr, PTR_REG ); 9732439e4bfSJean-Christophe PLAGNIOL-VILLARD 9742439e4bfSJean-Christophe PLAGNIOL-VILLARD if (!is_error) { 9752439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Pass the packet up to the protocol layers. */ 9762439e4bfSJean-Christophe PLAGNIOL-VILLARD NetReceive(NetRxPackets[0], packet_length); 9772439e4bfSJean-Christophe PLAGNIOL-VILLARD return packet_length; 9782439e4bfSJean-Christophe PLAGNIOL-VILLARD } else { 9792439e4bfSJean-Christophe PLAGNIOL-VILLARD return 0; 9802439e4bfSJean-Christophe PLAGNIOL-VILLARD } 9812439e4bfSJean-Christophe PLAGNIOL-VILLARD 9822439e4bfSJean-Christophe PLAGNIOL-VILLARD } 9832439e4bfSJean-Christophe PLAGNIOL-VILLARD 9842439e4bfSJean-Christophe PLAGNIOL-VILLARD 9852439e4bfSJean-Christophe PLAGNIOL-VILLARD /*---------------------------------------------------- 9862439e4bfSJean-Christophe PLAGNIOL-VILLARD . smc_close 9872439e4bfSJean-Christophe PLAGNIOL-VILLARD . 9882439e4bfSJean-Christophe PLAGNIOL-VILLARD . this makes the board clean up everything that it can 9892439e4bfSJean-Christophe PLAGNIOL-VILLARD . and not talk to the outside world. Caused by 9902439e4bfSJean-Christophe PLAGNIOL-VILLARD . an 'ifconfig ethX down' 9912439e4bfSJean-Christophe PLAGNIOL-VILLARD . 9922439e4bfSJean-Christophe PLAGNIOL-VILLARD -----------------------------------------------------*/ 9932439e4bfSJean-Christophe PLAGNIOL-VILLARD static int smc_close() 9942439e4bfSJean-Christophe PLAGNIOL-VILLARD { 9952439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK2("%s: smc_close\n", SMC_DEV_NAME); 9962439e4bfSJean-Christophe PLAGNIOL-VILLARD 9972439e4bfSJean-Christophe PLAGNIOL-VILLARD /* clear everything */ 9982439e4bfSJean-Christophe PLAGNIOL-VILLARD smc_shutdown(); 9992439e4bfSJean-Christophe PLAGNIOL-VILLARD 10002439e4bfSJean-Christophe PLAGNIOL-VILLARD return 0; 10012439e4bfSJean-Christophe PLAGNIOL-VILLARD } 10022439e4bfSJean-Christophe PLAGNIOL-VILLARD 10032439e4bfSJean-Christophe PLAGNIOL-VILLARD 10042439e4bfSJean-Christophe PLAGNIOL-VILLARD #if 0 10052439e4bfSJean-Christophe PLAGNIOL-VILLARD /*------------------------------------------------------------ 10062439e4bfSJean-Christophe PLAGNIOL-VILLARD . Modify a bit in the LAN91C111 register set 10072439e4bfSJean-Christophe PLAGNIOL-VILLARD .-------------------------------------------------------------*/ 10082439e4bfSJean-Christophe PLAGNIOL-VILLARD static word smc_modify_regbit(int bank, int ioaddr, int reg, 10092439e4bfSJean-Christophe PLAGNIOL-VILLARD unsigned int bit, int val) 10102439e4bfSJean-Christophe PLAGNIOL-VILLARD { 10112439e4bfSJean-Christophe PLAGNIOL-VILLARD word regval; 10122439e4bfSJean-Christophe PLAGNIOL-VILLARD 10132439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK( bank ); 10142439e4bfSJean-Christophe PLAGNIOL-VILLARD 10152439e4bfSJean-Christophe PLAGNIOL-VILLARD regval = SMC_inw( reg ); 10162439e4bfSJean-Christophe PLAGNIOL-VILLARD if (val) 10172439e4bfSJean-Christophe PLAGNIOL-VILLARD regval |= bit; 10182439e4bfSJean-Christophe PLAGNIOL-VILLARD else 10192439e4bfSJean-Christophe PLAGNIOL-VILLARD regval &= ~bit; 10202439e4bfSJean-Christophe PLAGNIOL-VILLARD 10212439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw( regval, 0 ); 10222439e4bfSJean-Christophe PLAGNIOL-VILLARD return(regval); 10232439e4bfSJean-Christophe PLAGNIOL-VILLARD } 10242439e4bfSJean-Christophe PLAGNIOL-VILLARD 10252439e4bfSJean-Christophe PLAGNIOL-VILLARD 10262439e4bfSJean-Christophe PLAGNIOL-VILLARD /*------------------------------------------------------------ 10272439e4bfSJean-Christophe PLAGNIOL-VILLARD . Retrieve a bit in the LAN91C111 register set 10282439e4bfSJean-Christophe PLAGNIOL-VILLARD .-------------------------------------------------------------*/ 10292439e4bfSJean-Christophe PLAGNIOL-VILLARD static int smc_get_regbit(int bank, int ioaddr, int reg, unsigned int bit) 10302439e4bfSJean-Christophe PLAGNIOL-VILLARD { 10312439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK( bank ); 10322439e4bfSJean-Christophe PLAGNIOL-VILLARD if ( SMC_inw( reg ) & bit) 10332439e4bfSJean-Christophe PLAGNIOL-VILLARD return(1); 10342439e4bfSJean-Christophe PLAGNIOL-VILLARD else 10352439e4bfSJean-Christophe PLAGNIOL-VILLARD return(0); 10362439e4bfSJean-Christophe PLAGNIOL-VILLARD } 10372439e4bfSJean-Christophe PLAGNIOL-VILLARD 10382439e4bfSJean-Christophe PLAGNIOL-VILLARD 10392439e4bfSJean-Christophe PLAGNIOL-VILLARD /*------------------------------------------------------------ 10402439e4bfSJean-Christophe PLAGNIOL-VILLARD . Modify a LAN91C111 register (word access only) 10412439e4bfSJean-Christophe PLAGNIOL-VILLARD .-------------------------------------------------------------*/ 10422439e4bfSJean-Christophe PLAGNIOL-VILLARD static void smc_modify_reg(int bank, int ioaddr, int reg, word val) 10432439e4bfSJean-Christophe PLAGNIOL-VILLARD { 10442439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK( bank ); 10452439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw( val, reg ); 10462439e4bfSJean-Christophe PLAGNIOL-VILLARD } 10472439e4bfSJean-Christophe PLAGNIOL-VILLARD 10482439e4bfSJean-Christophe PLAGNIOL-VILLARD 10492439e4bfSJean-Christophe PLAGNIOL-VILLARD /*------------------------------------------------------------ 10502439e4bfSJean-Christophe PLAGNIOL-VILLARD . Retrieve a LAN91C111 register (word access only) 10512439e4bfSJean-Christophe PLAGNIOL-VILLARD .-------------------------------------------------------------*/ 10522439e4bfSJean-Christophe PLAGNIOL-VILLARD static int smc_get_reg(int bank, int ioaddr, int reg) 10532439e4bfSJean-Christophe PLAGNIOL-VILLARD { 10542439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK( bank ); 10552439e4bfSJean-Christophe PLAGNIOL-VILLARD return(SMC_inw( reg )); 10562439e4bfSJean-Christophe PLAGNIOL-VILLARD } 10572439e4bfSJean-Christophe PLAGNIOL-VILLARD 10582439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif /* 0 */ 10592439e4bfSJean-Christophe PLAGNIOL-VILLARD 10602439e4bfSJean-Christophe PLAGNIOL-VILLARD /*---PHY CONTROL AND CONFIGURATION----------------------------------------- */ 10612439e4bfSJean-Christophe PLAGNIOL-VILLARD 10622439e4bfSJean-Christophe PLAGNIOL-VILLARD #if (SMC_DEBUG > 2 ) 10632439e4bfSJean-Christophe PLAGNIOL-VILLARD 10642439e4bfSJean-Christophe PLAGNIOL-VILLARD /*------------------------------------------------------------ 10652439e4bfSJean-Christophe PLAGNIOL-VILLARD . Debugging function for viewing MII Management serial bitstream 10662439e4bfSJean-Christophe PLAGNIOL-VILLARD .-------------------------------------------------------------*/ 10672439e4bfSJean-Christophe PLAGNIOL-VILLARD static void smc_dump_mii_stream (byte * bits, int size) 10682439e4bfSJean-Christophe PLAGNIOL-VILLARD { 10692439e4bfSJean-Christophe PLAGNIOL-VILLARD int i; 10702439e4bfSJean-Christophe PLAGNIOL-VILLARD 10712439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("BIT#:"); 10722439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i = 0; i < size; ++i) { 10732439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("%d", i % 10); 10742439e4bfSJean-Christophe PLAGNIOL-VILLARD } 10752439e4bfSJean-Christophe PLAGNIOL-VILLARD 10762439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("\nMDOE:"); 10772439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i = 0; i < size; ++i) { 10782439e4bfSJean-Christophe PLAGNIOL-VILLARD if (bits[i] & MII_MDOE) 10792439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("1"); 10802439e4bfSJean-Christophe PLAGNIOL-VILLARD else 10812439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("0"); 10822439e4bfSJean-Christophe PLAGNIOL-VILLARD } 10832439e4bfSJean-Christophe PLAGNIOL-VILLARD 10842439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("\nMDO :"); 10852439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i = 0; i < size; ++i) { 10862439e4bfSJean-Christophe PLAGNIOL-VILLARD if (bits[i] & MII_MDO) 10872439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("1"); 10882439e4bfSJean-Christophe PLAGNIOL-VILLARD else 10892439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("0"); 10902439e4bfSJean-Christophe PLAGNIOL-VILLARD } 10912439e4bfSJean-Christophe PLAGNIOL-VILLARD 10922439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("\nMDI :"); 10932439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i = 0; i < size; ++i) { 10942439e4bfSJean-Christophe PLAGNIOL-VILLARD if (bits[i] & MII_MDI) 10952439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("1"); 10962439e4bfSJean-Christophe PLAGNIOL-VILLARD else 10972439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("0"); 10982439e4bfSJean-Christophe PLAGNIOL-VILLARD } 10992439e4bfSJean-Christophe PLAGNIOL-VILLARD 11002439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("\n"); 11012439e4bfSJean-Christophe PLAGNIOL-VILLARD } 11022439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 11032439e4bfSJean-Christophe PLAGNIOL-VILLARD 11042439e4bfSJean-Christophe PLAGNIOL-VILLARD /*------------------------------------------------------------ 11052439e4bfSJean-Christophe PLAGNIOL-VILLARD . Reads a register from the MII Management serial interface 11062439e4bfSJean-Christophe PLAGNIOL-VILLARD .-------------------------------------------------------------*/ 11072439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifndef CONFIG_SMC91111_EXT_PHY 11082439e4bfSJean-Christophe PLAGNIOL-VILLARD static word smc_read_phy_register (byte phyreg) 11092439e4bfSJean-Christophe PLAGNIOL-VILLARD { 11102439e4bfSJean-Christophe PLAGNIOL-VILLARD int oldBank; 11112439e4bfSJean-Christophe PLAGNIOL-VILLARD int i; 11122439e4bfSJean-Christophe PLAGNIOL-VILLARD byte mask; 11132439e4bfSJean-Christophe PLAGNIOL-VILLARD word mii_reg; 11142439e4bfSJean-Christophe PLAGNIOL-VILLARD byte bits[64]; 11152439e4bfSJean-Christophe PLAGNIOL-VILLARD int clk_idx = 0; 11162439e4bfSJean-Christophe PLAGNIOL-VILLARD int input_idx; 11172439e4bfSJean-Christophe PLAGNIOL-VILLARD word phydata; 11182439e4bfSJean-Christophe PLAGNIOL-VILLARD byte phyaddr = SMC_PHY_ADDR; 11192439e4bfSJean-Christophe PLAGNIOL-VILLARD 11202439e4bfSJean-Christophe PLAGNIOL-VILLARD /* 32 consecutive ones on MDO to establish sync */ 11212439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i = 0; i < 32; ++i) 11222439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = MII_MDOE | MII_MDO; 11232439e4bfSJean-Christophe PLAGNIOL-VILLARD 11242439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Start code <01> */ 11252439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = MII_MDOE; 11262439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = MII_MDOE | MII_MDO; 11272439e4bfSJean-Christophe PLAGNIOL-VILLARD 11282439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Read command <10> */ 11292439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = MII_MDOE | MII_MDO; 11302439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = MII_MDOE; 11312439e4bfSJean-Christophe PLAGNIOL-VILLARD 11322439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Output the PHY address, msb first */ 11332439e4bfSJean-Christophe PLAGNIOL-VILLARD mask = (byte) 0x10; 11342439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i = 0; i < 5; ++i) { 11352439e4bfSJean-Christophe PLAGNIOL-VILLARD if (phyaddr & mask) 11362439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = MII_MDOE | MII_MDO; 11372439e4bfSJean-Christophe PLAGNIOL-VILLARD else 11382439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = MII_MDOE; 11392439e4bfSJean-Christophe PLAGNIOL-VILLARD 11402439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Shift to next lowest bit */ 11412439e4bfSJean-Christophe PLAGNIOL-VILLARD mask >>= 1; 11422439e4bfSJean-Christophe PLAGNIOL-VILLARD } 11432439e4bfSJean-Christophe PLAGNIOL-VILLARD 11442439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Output the phy register number, msb first */ 11452439e4bfSJean-Christophe PLAGNIOL-VILLARD mask = (byte) 0x10; 11462439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i = 0; i < 5; ++i) { 11472439e4bfSJean-Christophe PLAGNIOL-VILLARD if (phyreg & mask) 11482439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = MII_MDOE | MII_MDO; 11492439e4bfSJean-Christophe PLAGNIOL-VILLARD else 11502439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = MII_MDOE; 11512439e4bfSJean-Christophe PLAGNIOL-VILLARD 11522439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Shift to next lowest bit */ 11532439e4bfSJean-Christophe PLAGNIOL-VILLARD mask >>= 1; 11542439e4bfSJean-Christophe PLAGNIOL-VILLARD } 11552439e4bfSJean-Christophe PLAGNIOL-VILLARD 11562439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Tristate and turnaround (2 bit times) */ 11572439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = 0; 11582439e4bfSJean-Christophe PLAGNIOL-VILLARD /*bits[clk_idx++] = 0; */ 11592439e4bfSJean-Christophe PLAGNIOL-VILLARD 11602439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Input starts at this bit time */ 11612439e4bfSJean-Christophe PLAGNIOL-VILLARD input_idx = clk_idx; 11622439e4bfSJean-Christophe PLAGNIOL-VILLARD 11632439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Will input 16 bits */ 11642439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i = 0; i < 16; ++i) 11652439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = 0; 11662439e4bfSJean-Christophe PLAGNIOL-VILLARD 11672439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Final clock bit */ 11682439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = 0; 11692439e4bfSJean-Christophe PLAGNIOL-VILLARD 11702439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Save the current bank */ 11712439e4bfSJean-Christophe PLAGNIOL-VILLARD oldBank = SMC_inw (BANK_SELECT); 11722439e4bfSJean-Christophe PLAGNIOL-VILLARD 11732439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Select bank 3 */ 11742439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK (3); 11752439e4bfSJean-Christophe PLAGNIOL-VILLARD 11762439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Get the current MII register value */ 11772439e4bfSJean-Christophe PLAGNIOL-VILLARD mii_reg = SMC_inw (MII_REG); 11782439e4bfSJean-Christophe PLAGNIOL-VILLARD 11792439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Turn off all MII Interface bits */ 11802439e4bfSJean-Christophe PLAGNIOL-VILLARD mii_reg &= ~(MII_MDOE | MII_MCLK | MII_MDI | MII_MDO); 11812439e4bfSJean-Christophe PLAGNIOL-VILLARD 11822439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Clock all 64 cycles */ 11832439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i = 0; i < sizeof bits; ++i) { 11842439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Clock Low - output data */ 11852439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (mii_reg | bits[i], MII_REG); 11862439e4bfSJean-Christophe PLAGNIOL-VILLARD udelay (SMC_PHY_CLOCK_DELAY); 11872439e4bfSJean-Christophe PLAGNIOL-VILLARD 11882439e4bfSJean-Christophe PLAGNIOL-VILLARD 11892439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Clock Hi - input data */ 11902439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (mii_reg | bits[i] | MII_MCLK, MII_REG); 11912439e4bfSJean-Christophe PLAGNIOL-VILLARD udelay (SMC_PHY_CLOCK_DELAY); 11922439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[i] |= SMC_inw (MII_REG) & MII_MDI; 11932439e4bfSJean-Christophe PLAGNIOL-VILLARD } 11942439e4bfSJean-Christophe PLAGNIOL-VILLARD 11952439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Return to idle state */ 11962439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Set clock to low, data to low, and output tristated */ 11972439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (mii_reg, MII_REG); 11982439e4bfSJean-Christophe PLAGNIOL-VILLARD udelay (SMC_PHY_CLOCK_DELAY); 11992439e4bfSJean-Christophe PLAGNIOL-VILLARD 12002439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Restore original bank select */ 12012439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK (oldBank); 12022439e4bfSJean-Christophe PLAGNIOL-VILLARD 12032439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Recover input data */ 12042439e4bfSJean-Christophe PLAGNIOL-VILLARD phydata = 0; 12052439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i = 0; i < 16; ++i) { 12062439e4bfSJean-Christophe PLAGNIOL-VILLARD phydata <<= 1; 12072439e4bfSJean-Christophe PLAGNIOL-VILLARD 12082439e4bfSJean-Christophe PLAGNIOL-VILLARD if (bits[input_idx++] & MII_MDI) 12092439e4bfSJean-Christophe PLAGNIOL-VILLARD phydata |= 0x0001; 12102439e4bfSJean-Christophe PLAGNIOL-VILLARD } 12112439e4bfSJean-Christophe PLAGNIOL-VILLARD 12122439e4bfSJean-Christophe PLAGNIOL-VILLARD #if (SMC_DEBUG > 2 ) 12132439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("smc_read_phy_register(): phyaddr=%x,phyreg=%x,phydata=%x\n", 12142439e4bfSJean-Christophe PLAGNIOL-VILLARD phyaddr, phyreg, phydata); 12152439e4bfSJean-Christophe PLAGNIOL-VILLARD smc_dump_mii_stream (bits, sizeof bits); 12162439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 12172439e4bfSJean-Christophe PLAGNIOL-VILLARD 12182439e4bfSJean-Christophe PLAGNIOL-VILLARD return (phydata); 12192439e4bfSJean-Christophe PLAGNIOL-VILLARD } 12202439e4bfSJean-Christophe PLAGNIOL-VILLARD 12212439e4bfSJean-Christophe PLAGNIOL-VILLARD 12222439e4bfSJean-Christophe PLAGNIOL-VILLARD /*------------------------------------------------------------ 12232439e4bfSJean-Christophe PLAGNIOL-VILLARD . Writes a register to the MII Management serial interface 12242439e4bfSJean-Christophe PLAGNIOL-VILLARD .-------------------------------------------------------------*/ 12252439e4bfSJean-Christophe PLAGNIOL-VILLARD static void smc_write_phy_register (byte phyreg, word phydata) 12262439e4bfSJean-Christophe PLAGNIOL-VILLARD { 12272439e4bfSJean-Christophe PLAGNIOL-VILLARD int oldBank; 12282439e4bfSJean-Christophe PLAGNIOL-VILLARD int i; 12292439e4bfSJean-Christophe PLAGNIOL-VILLARD word mask; 12302439e4bfSJean-Christophe PLAGNIOL-VILLARD word mii_reg; 12312439e4bfSJean-Christophe PLAGNIOL-VILLARD byte bits[65]; 12322439e4bfSJean-Christophe PLAGNIOL-VILLARD int clk_idx = 0; 12332439e4bfSJean-Christophe PLAGNIOL-VILLARD byte phyaddr = SMC_PHY_ADDR; 12342439e4bfSJean-Christophe PLAGNIOL-VILLARD 12352439e4bfSJean-Christophe PLAGNIOL-VILLARD /* 32 consecutive ones on MDO to establish sync */ 12362439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i = 0; i < 32; ++i) 12372439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = MII_MDOE | MII_MDO; 12382439e4bfSJean-Christophe PLAGNIOL-VILLARD 12392439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Start code <01> */ 12402439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = MII_MDOE; 12412439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = MII_MDOE | MII_MDO; 12422439e4bfSJean-Christophe PLAGNIOL-VILLARD 12432439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Write command <01> */ 12442439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = MII_MDOE; 12452439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = MII_MDOE | MII_MDO; 12462439e4bfSJean-Christophe PLAGNIOL-VILLARD 12472439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Output the PHY address, msb first */ 12482439e4bfSJean-Christophe PLAGNIOL-VILLARD mask = (byte) 0x10; 12492439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i = 0; i < 5; ++i) { 12502439e4bfSJean-Christophe PLAGNIOL-VILLARD if (phyaddr & mask) 12512439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = MII_MDOE | MII_MDO; 12522439e4bfSJean-Christophe PLAGNIOL-VILLARD else 12532439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = MII_MDOE; 12542439e4bfSJean-Christophe PLAGNIOL-VILLARD 12552439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Shift to next lowest bit */ 12562439e4bfSJean-Christophe PLAGNIOL-VILLARD mask >>= 1; 12572439e4bfSJean-Christophe PLAGNIOL-VILLARD } 12582439e4bfSJean-Christophe PLAGNIOL-VILLARD 12592439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Output the phy register number, msb first */ 12602439e4bfSJean-Christophe PLAGNIOL-VILLARD mask = (byte) 0x10; 12612439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i = 0; i < 5; ++i) { 12622439e4bfSJean-Christophe PLAGNIOL-VILLARD if (phyreg & mask) 12632439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = MII_MDOE | MII_MDO; 12642439e4bfSJean-Christophe PLAGNIOL-VILLARD else 12652439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = MII_MDOE; 12662439e4bfSJean-Christophe PLAGNIOL-VILLARD 12672439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Shift to next lowest bit */ 12682439e4bfSJean-Christophe PLAGNIOL-VILLARD mask >>= 1; 12692439e4bfSJean-Christophe PLAGNIOL-VILLARD } 12702439e4bfSJean-Christophe PLAGNIOL-VILLARD 12712439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Tristate and turnaround (2 bit times) */ 12722439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = 0; 12732439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = 0; 12742439e4bfSJean-Christophe PLAGNIOL-VILLARD 12752439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Write out 16 bits of data, msb first */ 12762439e4bfSJean-Christophe PLAGNIOL-VILLARD mask = 0x8000; 12772439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i = 0; i < 16; ++i) { 12782439e4bfSJean-Christophe PLAGNIOL-VILLARD if (phydata & mask) 12792439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = MII_MDOE | MII_MDO; 12802439e4bfSJean-Christophe PLAGNIOL-VILLARD else 12812439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = MII_MDOE; 12822439e4bfSJean-Christophe PLAGNIOL-VILLARD 12832439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Shift to next lowest bit */ 12842439e4bfSJean-Christophe PLAGNIOL-VILLARD mask >>= 1; 12852439e4bfSJean-Christophe PLAGNIOL-VILLARD } 12862439e4bfSJean-Christophe PLAGNIOL-VILLARD 12872439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Final clock bit (tristate) */ 12882439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[clk_idx++] = 0; 12892439e4bfSJean-Christophe PLAGNIOL-VILLARD 12902439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Save the current bank */ 12912439e4bfSJean-Christophe PLAGNIOL-VILLARD oldBank = SMC_inw (BANK_SELECT); 12922439e4bfSJean-Christophe PLAGNIOL-VILLARD 12932439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Select bank 3 */ 12942439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK (3); 12952439e4bfSJean-Christophe PLAGNIOL-VILLARD 12962439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Get the current MII register value */ 12972439e4bfSJean-Christophe PLAGNIOL-VILLARD mii_reg = SMC_inw (MII_REG); 12982439e4bfSJean-Christophe PLAGNIOL-VILLARD 12992439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Turn off all MII Interface bits */ 13002439e4bfSJean-Christophe PLAGNIOL-VILLARD mii_reg &= ~(MII_MDOE | MII_MCLK | MII_MDI | MII_MDO); 13012439e4bfSJean-Christophe PLAGNIOL-VILLARD 13022439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Clock all cycles */ 13032439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i = 0; i < sizeof bits; ++i) { 13042439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Clock Low - output data */ 13052439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (mii_reg | bits[i], MII_REG); 13062439e4bfSJean-Christophe PLAGNIOL-VILLARD udelay (SMC_PHY_CLOCK_DELAY); 13072439e4bfSJean-Christophe PLAGNIOL-VILLARD 13082439e4bfSJean-Christophe PLAGNIOL-VILLARD 13092439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Clock Hi - input data */ 13102439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (mii_reg | bits[i] | MII_MCLK, MII_REG); 13112439e4bfSJean-Christophe PLAGNIOL-VILLARD udelay (SMC_PHY_CLOCK_DELAY); 13122439e4bfSJean-Christophe PLAGNIOL-VILLARD bits[i] |= SMC_inw (MII_REG) & MII_MDI; 13132439e4bfSJean-Christophe PLAGNIOL-VILLARD } 13142439e4bfSJean-Christophe PLAGNIOL-VILLARD 13152439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Return to idle state */ 13162439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Set clock to low, data to low, and output tristated */ 13172439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (mii_reg, MII_REG); 13182439e4bfSJean-Christophe PLAGNIOL-VILLARD udelay (SMC_PHY_CLOCK_DELAY); 13192439e4bfSJean-Christophe PLAGNIOL-VILLARD 13202439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Restore original bank select */ 13212439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK (oldBank); 13222439e4bfSJean-Christophe PLAGNIOL-VILLARD 13232439e4bfSJean-Christophe PLAGNIOL-VILLARD #if (SMC_DEBUG > 2 ) 13242439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("smc_write_phy_register(): phyaddr=%x,phyreg=%x,phydata=%x\n", 13252439e4bfSJean-Christophe PLAGNIOL-VILLARD phyaddr, phyreg, phydata); 13262439e4bfSJean-Christophe PLAGNIOL-VILLARD smc_dump_mii_stream (bits, sizeof bits); 13272439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 13282439e4bfSJean-Christophe PLAGNIOL-VILLARD } 13292439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif /* !CONFIG_SMC91111_EXT_PHY */ 13302439e4bfSJean-Christophe PLAGNIOL-VILLARD 13312439e4bfSJean-Christophe PLAGNIOL-VILLARD 13322439e4bfSJean-Christophe PLAGNIOL-VILLARD /*------------------------------------------------------------ 13332439e4bfSJean-Christophe PLAGNIOL-VILLARD . Waits the specified number of milliseconds - kernel friendly 13342439e4bfSJean-Christophe PLAGNIOL-VILLARD .-------------------------------------------------------------*/ 13352439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifndef CONFIG_SMC91111_EXT_PHY 13362439e4bfSJean-Christophe PLAGNIOL-VILLARD static void smc_wait_ms(unsigned int ms) 13372439e4bfSJean-Christophe PLAGNIOL-VILLARD { 13382439e4bfSJean-Christophe PLAGNIOL-VILLARD udelay(ms*1000); 13392439e4bfSJean-Christophe PLAGNIOL-VILLARD } 13402439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif /* !CONFIG_SMC91111_EXT_PHY */ 13412439e4bfSJean-Christophe PLAGNIOL-VILLARD 13422439e4bfSJean-Christophe PLAGNIOL-VILLARD 13432439e4bfSJean-Christophe PLAGNIOL-VILLARD /*------------------------------------------------------------ 13442439e4bfSJean-Christophe PLAGNIOL-VILLARD . Configures the specified PHY using Autonegotiation. Calls 13452439e4bfSJean-Christophe PLAGNIOL-VILLARD . smc_phy_fixed() if the user has requested a certain config. 13462439e4bfSJean-Christophe PLAGNIOL-VILLARD .-------------------------------------------------------------*/ 13472439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifndef CONFIG_SMC91111_EXT_PHY 13482439e4bfSJean-Christophe PLAGNIOL-VILLARD static void smc_phy_configure () 13492439e4bfSJean-Christophe PLAGNIOL-VILLARD { 13502439e4bfSJean-Christophe PLAGNIOL-VILLARD int timeout; 13512439e4bfSJean-Christophe PLAGNIOL-VILLARD byte phyaddr; 13522439e4bfSJean-Christophe PLAGNIOL-VILLARD word my_phy_caps; /* My PHY capabilities */ 13532439e4bfSJean-Christophe PLAGNIOL-VILLARD word my_ad_caps; /* My Advertised capabilities */ 13542439e4bfSJean-Christophe PLAGNIOL-VILLARD word status = 0; /*;my status = 0 */ 13552439e4bfSJean-Christophe PLAGNIOL-VILLARD int failed = 0; 13562439e4bfSJean-Christophe PLAGNIOL-VILLARD 13572439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK3 ("%s: smc_program_phy()\n", SMC_DEV_NAME); 13582439e4bfSJean-Christophe PLAGNIOL-VILLARD 13592439e4bfSJean-Christophe PLAGNIOL-VILLARD 13602439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Get the detected phy address */ 13612439e4bfSJean-Christophe PLAGNIOL-VILLARD phyaddr = SMC_PHY_ADDR; 13622439e4bfSJean-Christophe PLAGNIOL-VILLARD 13632439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Reset the PHY, setting all other bits to zero */ 13642439e4bfSJean-Christophe PLAGNIOL-VILLARD smc_write_phy_register (PHY_CNTL_REG, PHY_CNTL_RST); 13652439e4bfSJean-Christophe PLAGNIOL-VILLARD 13662439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Wait for the reset to complete, or time out */ 13672439e4bfSJean-Christophe PLAGNIOL-VILLARD timeout = 6; /* Wait up to 3 seconds */ 13682439e4bfSJean-Christophe PLAGNIOL-VILLARD while (timeout--) { 13692439e4bfSJean-Christophe PLAGNIOL-VILLARD if (!(smc_read_phy_register (PHY_CNTL_REG) 13702439e4bfSJean-Christophe PLAGNIOL-VILLARD & PHY_CNTL_RST)) { 13712439e4bfSJean-Christophe PLAGNIOL-VILLARD /* reset complete */ 13722439e4bfSJean-Christophe PLAGNIOL-VILLARD break; 13732439e4bfSJean-Christophe PLAGNIOL-VILLARD } 13742439e4bfSJean-Christophe PLAGNIOL-VILLARD 13752439e4bfSJean-Christophe PLAGNIOL-VILLARD smc_wait_ms (500); /* wait 500 millisecs */ 13762439e4bfSJean-Christophe PLAGNIOL-VILLARD } 13772439e4bfSJean-Christophe PLAGNIOL-VILLARD 13782439e4bfSJean-Christophe PLAGNIOL-VILLARD if (timeout < 1) { 13792439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("%s:PHY reset timed out\n", SMC_DEV_NAME); 13802439e4bfSJean-Christophe PLAGNIOL-VILLARD goto smc_phy_configure_exit; 13812439e4bfSJean-Christophe PLAGNIOL-VILLARD } 13822439e4bfSJean-Christophe PLAGNIOL-VILLARD 13832439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Read PHY Register 18, Status Output */ 13842439e4bfSJean-Christophe PLAGNIOL-VILLARD /* lp->lastPhy18 = smc_read_phy_register(PHY_INT_REG); */ 13852439e4bfSJean-Christophe PLAGNIOL-VILLARD 13862439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Enable PHY Interrupts (for register 18) */ 13872439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Interrupts listed here are disabled */ 13882439e4bfSJean-Christophe PLAGNIOL-VILLARD smc_write_phy_register (PHY_MASK_REG, 0xffff); 13892439e4bfSJean-Christophe PLAGNIOL-VILLARD 13902439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Configure the Receive/Phy Control register */ 13912439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK (0); 13922439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (RPC_DEFAULT, RPC_REG); 13932439e4bfSJean-Christophe PLAGNIOL-VILLARD 13942439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Copy our capabilities from PHY_STAT_REG to PHY_AD_REG */ 13952439e4bfSJean-Christophe PLAGNIOL-VILLARD my_phy_caps = smc_read_phy_register (PHY_STAT_REG); 13962439e4bfSJean-Christophe PLAGNIOL-VILLARD my_ad_caps = PHY_AD_CSMA; /* I am CSMA capable */ 13972439e4bfSJean-Christophe PLAGNIOL-VILLARD 13982439e4bfSJean-Christophe PLAGNIOL-VILLARD if (my_phy_caps & PHY_STAT_CAP_T4) 13992439e4bfSJean-Christophe PLAGNIOL-VILLARD my_ad_caps |= PHY_AD_T4; 14002439e4bfSJean-Christophe PLAGNIOL-VILLARD 14012439e4bfSJean-Christophe PLAGNIOL-VILLARD if (my_phy_caps & PHY_STAT_CAP_TXF) 14022439e4bfSJean-Christophe PLAGNIOL-VILLARD my_ad_caps |= PHY_AD_TX_FDX; 14032439e4bfSJean-Christophe PLAGNIOL-VILLARD 14042439e4bfSJean-Christophe PLAGNIOL-VILLARD if (my_phy_caps & PHY_STAT_CAP_TXH) 14052439e4bfSJean-Christophe PLAGNIOL-VILLARD my_ad_caps |= PHY_AD_TX_HDX; 14062439e4bfSJean-Christophe PLAGNIOL-VILLARD 14072439e4bfSJean-Christophe PLAGNIOL-VILLARD if (my_phy_caps & PHY_STAT_CAP_TF) 14082439e4bfSJean-Christophe PLAGNIOL-VILLARD my_ad_caps |= PHY_AD_10_FDX; 14092439e4bfSJean-Christophe PLAGNIOL-VILLARD 14102439e4bfSJean-Christophe PLAGNIOL-VILLARD if (my_phy_caps & PHY_STAT_CAP_TH) 14112439e4bfSJean-Christophe PLAGNIOL-VILLARD my_ad_caps |= PHY_AD_10_HDX; 14122439e4bfSJean-Christophe PLAGNIOL-VILLARD 14132439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Update our Auto-Neg Advertisement Register */ 14142439e4bfSJean-Christophe PLAGNIOL-VILLARD smc_write_phy_register (PHY_AD_REG, my_ad_caps); 14152439e4bfSJean-Christophe PLAGNIOL-VILLARD 14162439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Read the register back. Without this, it appears that when */ 14172439e4bfSJean-Christophe PLAGNIOL-VILLARD /* auto-negotiation is restarted, sometimes it isn't ready and */ 14182439e4bfSJean-Christophe PLAGNIOL-VILLARD /* the link does not come up. */ 14192439e4bfSJean-Christophe PLAGNIOL-VILLARD smc_read_phy_register(PHY_AD_REG); 14202439e4bfSJean-Christophe PLAGNIOL-VILLARD 14212439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK2 ("%s: phy caps=%x\n", SMC_DEV_NAME, my_phy_caps); 14222439e4bfSJean-Christophe PLAGNIOL-VILLARD PRINTK2 ("%s: phy advertised caps=%x\n", SMC_DEV_NAME, my_ad_caps); 14232439e4bfSJean-Christophe PLAGNIOL-VILLARD 14242439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Restart auto-negotiation process in order to advertise my caps */ 14252439e4bfSJean-Christophe PLAGNIOL-VILLARD smc_write_phy_register (PHY_CNTL_REG, 14262439e4bfSJean-Christophe PLAGNIOL-VILLARD PHY_CNTL_ANEG_EN | PHY_CNTL_ANEG_RST); 14272439e4bfSJean-Christophe PLAGNIOL-VILLARD 14282439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Wait for the auto-negotiation to complete. This may take from */ 14292439e4bfSJean-Christophe PLAGNIOL-VILLARD /* 2 to 3 seconds. */ 14302439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Wait for the reset to complete, or time out */ 14312439e4bfSJean-Christophe PLAGNIOL-VILLARD timeout = CONFIG_SMC_AUTONEG_TIMEOUT * 2; 14322439e4bfSJean-Christophe PLAGNIOL-VILLARD while (timeout--) { 14332439e4bfSJean-Christophe PLAGNIOL-VILLARD 14342439e4bfSJean-Christophe PLAGNIOL-VILLARD status = smc_read_phy_register (PHY_STAT_REG); 14352439e4bfSJean-Christophe PLAGNIOL-VILLARD if (status & PHY_STAT_ANEG_ACK) { 14362439e4bfSJean-Christophe PLAGNIOL-VILLARD /* auto-negotiate complete */ 14372439e4bfSJean-Christophe PLAGNIOL-VILLARD break; 14382439e4bfSJean-Christophe PLAGNIOL-VILLARD } 14392439e4bfSJean-Christophe PLAGNIOL-VILLARD 14402439e4bfSJean-Christophe PLAGNIOL-VILLARD smc_wait_ms (500); /* wait 500 millisecs */ 14412439e4bfSJean-Christophe PLAGNIOL-VILLARD 14422439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Restart auto-negotiation if remote fault */ 14432439e4bfSJean-Christophe PLAGNIOL-VILLARD if (status & PHY_STAT_REM_FLT) { 14442439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("%s: PHY remote fault detected\n", 14452439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_DEV_NAME); 14462439e4bfSJean-Christophe PLAGNIOL-VILLARD 14472439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Restart auto-negotiation */ 14482439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("%s: PHY restarting auto-negotiation\n", 14492439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_DEV_NAME); 14502439e4bfSJean-Christophe PLAGNIOL-VILLARD smc_write_phy_register (PHY_CNTL_REG, 14512439e4bfSJean-Christophe PLAGNIOL-VILLARD PHY_CNTL_ANEG_EN | 14522439e4bfSJean-Christophe PLAGNIOL-VILLARD PHY_CNTL_ANEG_RST | 14532439e4bfSJean-Christophe PLAGNIOL-VILLARD PHY_CNTL_SPEED | 14542439e4bfSJean-Christophe PLAGNIOL-VILLARD PHY_CNTL_DPLX); 14552439e4bfSJean-Christophe PLAGNIOL-VILLARD } 14562439e4bfSJean-Christophe PLAGNIOL-VILLARD } 14572439e4bfSJean-Christophe PLAGNIOL-VILLARD 14582439e4bfSJean-Christophe PLAGNIOL-VILLARD if (timeout < 1) { 14592439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("%s: PHY auto-negotiate timed out\n", SMC_DEV_NAME); 14602439e4bfSJean-Christophe PLAGNIOL-VILLARD failed = 1; 14612439e4bfSJean-Christophe PLAGNIOL-VILLARD } 14622439e4bfSJean-Christophe PLAGNIOL-VILLARD 14632439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Fail if we detected an auto-negotiate remote fault */ 14642439e4bfSJean-Christophe PLAGNIOL-VILLARD if (status & PHY_STAT_REM_FLT) { 14652439e4bfSJean-Christophe PLAGNIOL-VILLARD printf ("%s: PHY remote fault detected\n", SMC_DEV_NAME); 14662439e4bfSJean-Christophe PLAGNIOL-VILLARD failed = 1; 14672439e4bfSJean-Christophe PLAGNIOL-VILLARD } 14682439e4bfSJean-Christophe PLAGNIOL-VILLARD 14692439e4bfSJean-Christophe PLAGNIOL-VILLARD /* Re-Configure the Receive/Phy Control register */ 14702439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_outw (RPC_DEFAULT, RPC_REG); 14712439e4bfSJean-Christophe PLAGNIOL-VILLARD 14722439e4bfSJean-Christophe PLAGNIOL-VILLARD smc_phy_configure_exit: ; 14732439e4bfSJean-Christophe PLAGNIOL-VILLARD 14742439e4bfSJean-Christophe PLAGNIOL-VILLARD } 14752439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif /* !CONFIG_SMC91111_EXT_PHY */ 14762439e4bfSJean-Christophe PLAGNIOL-VILLARD 14772439e4bfSJean-Christophe PLAGNIOL-VILLARD 14782439e4bfSJean-Christophe PLAGNIOL-VILLARD #if SMC_DEBUG > 2 14792439e4bfSJean-Christophe PLAGNIOL-VILLARD static void print_packet( byte * buf, int length ) 14802439e4bfSJean-Christophe PLAGNIOL-VILLARD { 14812439e4bfSJean-Christophe PLAGNIOL-VILLARD int i; 14822439e4bfSJean-Christophe PLAGNIOL-VILLARD int remainder; 14832439e4bfSJean-Christophe PLAGNIOL-VILLARD int lines; 14842439e4bfSJean-Christophe PLAGNIOL-VILLARD 14852439e4bfSJean-Christophe PLAGNIOL-VILLARD printf("Packet of length %d \n", length ); 14862439e4bfSJean-Christophe PLAGNIOL-VILLARD 14872439e4bfSJean-Christophe PLAGNIOL-VILLARD #if SMC_DEBUG > 3 14882439e4bfSJean-Christophe PLAGNIOL-VILLARD lines = length / 16; 14892439e4bfSJean-Christophe PLAGNIOL-VILLARD remainder = length % 16; 14902439e4bfSJean-Christophe PLAGNIOL-VILLARD 14912439e4bfSJean-Christophe PLAGNIOL-VILLARD for ( i = 0; i < lines ; i ++ ) { 14922439e4bfSJean-Christophe PLAGNIOL-VILLARD int cur; 14932439e4bfSJean-Christophe PLAGNIOL-VILLARD 14942439e4bfSJean-Christophe PLAGNIOL-VILLARD for ( cur = 0; cur < 8; cur ++ ) { 14952439e4bfSJean-Christophe PLAGNIOL-VILLARD byte a, b; 14962439e4bfSJean-Christophe PLAGNIOL-VILLARD 14972439e4bfSJean-Christophe PLAGNIOL-VILLARD a = *(buf ++ ); 14982439e4bfSJean-Christophe PLAGNIOL-VILLARD b = *(buf ++ ); 14992439e4bfSJean-Christophe PLAGNIOL-VILLARD printf("%02x%02x ", a, b ); 15002439e4bfSJean-Christophe PLAGNIOL-VILLARD } 15012439e4bfSJean-Christophe PLAGNIOL-VILLARD printf("\n"); 15022439e4bfSJean-Christophe PLAGNIOL-VILLARD } 15032439e4bfSJean-Christophe PLAGNIOL-VILLARD for ( i = 0; i < remainder/2 ; i++ ) { 15042439e4bfSJean-Christophe PLAGNIOL-VILLARD byte a, b; 15052439e4bfSJean-Christophe PLAGNIOL-VILLARD 15062439e4bfSJean-Christophe PLAGNIOL-VILLARD a = *(buf ++ ); 15072439e4bfSJean-Christophe PLAGNIOL-VILLARD b = *(buf ++ ); 15082439e4bfSJean-Christophe PLAGNIOL-VILLARD printf("%02x%02x ", a, b ); 15092439e4bfSJean-Christophe PLAGNIOL-VILLARD } 15102439e4bfSJean-Christophe PLAGNIOL-VILLARD printf("\n"); 15112439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 15122439e4bfSJean-Christophe PLAGNIOL-VILLARD } 15132439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 15142439e4bfSJean-Christophe PLAGNIOL-VILLARD 15152439e4bfSJean-Christophe PLAGNIOL-VILLARD int eth_init(bd_t *bd) { 15162439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifdef SHARED_RESOURCES 15172439e4bfSJean-Christophe PLAGNIOL-VILLARD swap_to(ETHERNET); 15182439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 15192439e4bfSJean-Christophe PLAGNIOL-VILLARD return (smc_open(bd)); 15202439e4bfSJean-Christophe PLAGNIOL-VILLARD } 15212439e4bfSJean-Christophe PLAGNIOL-VILLARD 15222439e4bfSJean-Christophe PLAGNIOL-VILLARD void eth_halt() { 15232439e4bfSJean-Christophe PLAGNIOL-VILLARD smc_close(); 15242439e4bfSJean-Christophe PLAGNIOL-VILLARD } 15252439e4bfSJean-Christophe PLAGNIOL-VILLARD 15262439e4bfSJean-Christophe PLAGNIOL-VILLARD int eth_rx() { 15272439e4bfSJean-Christophe PLAGNIOL-VILLARD return smc_rcv(); 15282439e4bfSJean-Christophe PLAGNIOL-VILLARD } 15292439e4bfSJean-Christophe PLAGNIOL-VILLARD 15302439e4bfSJean-Christophe PLAGNIOL-VILLARD int eth_send(volatile void *packet, int length) { 15312439e4bfSJean-Christophe PLAGNIOL-VILLARD return smc_send_packet(packet, length); 15322439e4bfSJean-Christophe PLAGNIOL-VILLARD } 15332439e4bfSJean-Christophe PLAGNIOL-VILLARD 15342439e4bfSJean-Christophe PLAGNIOL-VILLARD int smc_get_ethaddr (bd_t * bd) 15352439e4bfSJean-Christophe PLAGNIOL-VILLARD { 1536*03f3d8d3SMike Frysinger uchar v_mac[6]; 15372439e4bfSJean-Christophe PLAGNIOL-VILLARD 1538*03f3d8d3SMike Frysinger if (!eth_getenv_enetaddr("ethaddr", v_mac)) { 1539*03f3d8d3SMike Frysinger /* get ROM mac value if any */ 1540*03f3d8d3SMike Frysinger if (!get_rom_mac(v_mac)) { 15412439e4bfSJean-Christophe PLAGNIOL-VILLARD printf("\n*** ERROR: ethaddr is NOT set !!\n"); 1542*03f3d8d3SMike Frysinger return -1; 15432439e4bfSJean-Christophe PLAGNIOL-VILLARD } 1544*03f3d8d3SMike Frysinger eth_setenv_enetaddr("ethaddr", v_mac); 15452439e4bfSJean-Christophe PLAGNIOL-VILLARD } 15462439e4bfSJean-Christophe PLAGNIOL-VILLARD 1547*03f3d8d3SMike Frysinger smc_set_mac_addr(v_mac); /* use old function to update smc default */ 1548*03f3d8d3SMike Frysinger PRINTK("Using MAC Address %pM\n", v_mac); 1549*03f3d8d3SMike Frysinger return 0; 15502439e4bfSJean-Christophe PLAGNIOL-VILLARD } 15512439e4bfSJean-Christophe PLAGNIOL-VILLARD 15522439e4bfSJean-Christophe PLAGNIOL-VILLARD int get_rom_mac (uchar *v_rom_mac) 15532439e4bfSJean-Christophe PLAGNIOL-VILLARD { 15542439e4bfSJean-Christophe PLAGNIOL-VILLARD #ifdef HARDCODE_MAC /* used for testing or to supress run time warnings */ 15552439e4bfSJean-Christophe PLAGNIOL-VILLARD char hw_mac_addr[] = { 0x02, 0x80, 0xad, 0x20, 0x31, 0xb8 }; 15562439e4bfSJean-Christophe PLAGNIOL-VILLARD 15572439e4bfSJean-Christophe PLAGNIOL-VILLARD memcpy (v_rom_mac, hw_mac_addr, 6); 15582439e4bfSJean-Christophe PLAGNIOL-VILLARD return (1); 15592439e4bfSJean-Christophe PLAGNIOL-VILLARD #else 15602439e4bfSJean-Christophe PLAGNIOL-VILLARD int i; 15612439e4bfSJean-Christophe PLAGNIOL-VILLARD int valid_mac = 0; 15622439e4bfSJean-Christophe PLAGNIOL-VILLARD 15632439e4bfSJean-Christophe PLAGNIOL-VILLARD SMC_SELECT_BANK (1); 15642439e4bfSJean-Christophe PLAGNIOL-VILLARD for (i=0; i<6; i++) 15652439e4bfSJean-Christophe PLAGNIOL-VILLARD { 15662439e4bfSJean-Christophe PLAGNIOL-VILLARD v_rom_mac[i] = SMC_inb ((ADDR0_REG + i)); 15672439e4bfSJean-Christophe PLAGNIOL-VILLARD valid_mac |= v_rom_mac[i]; 15682439e4bfSJean-Christophe PLAGNIOL-VILLARD } 15692439e4bfSJean-Christophe PLAGNIOL-VILLARD 15702439e4bfSJean-Christophe PLAGNIOL-VILLARD return (valid_mac ? 1 : 0); 15712439e4bfSJean-Christophe PLAGNIOL-VILLARD #endif 15722439e4bfSJean-Christophe PLAGNIOL-VILLARD } 1573