URL
https://opencores.org/ocsvn/or1k/or1k/trunk
Subversion Repositories or1k
[/] [or1k/] [trunk/] [ecos-2.0/] [packages/] [devs/] [eth/] [powerpc/] [adder/] [v2_0/] [src/] [adder_eth.c] - Rev 1254
Go to most recent revision | Compare with Previous | Blame | View Log
//========================================================================== // // adder_eth.c // // Ethernet device driver specifics for Analogue & Micro Adder (PPC850) // //========================================================================== //####ECOSGPLCOPYRIGHTBEGIN#### // ------------------------------------------- // This file is part of eCos, the Embedded Configurable Operating System. // Copyright (C) 1998, 1999, 2000, 2001, 2002 Red Hat, Inc. // Copyright (C) 2002 Gary Thomas // // eCos is free software; you can redistribute it and/or modify it under // the terms of the GNU General Public License as published by the Free // Software Foundation; either version 2 or (at your option) any later version. // // eCos is distributed in the hope that it will be useful, but WITHOUT ANY // WARRANTY; without even the implied warranty of MERCHANTABILITY or // FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License // for more details. // // You should have received a copy of the GNU General Public License along // with eCos; if not, write to the Free Software Foundation, Inc., // 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA. // // As a special exception, if other files instantiate templates or use macros // or inline functions from this file, or you compile this file and link it // with other works to produce a work based on this file, this file does not // by itself cause the resulting work to be covered by the GNU General Public // License. However the source code for this file must still be made available // in accordance with section (3) of the GNU General Public License. // // This exception does not invalidate any other reasons why a work based on // this file might be covered by the GNU General Public License. // // Alternative licenses for eCos may be arranged by contacting Red Hat, Inc. // at http://sources.redhat.com/ecos/ecos-license/ // ------------------------------------------- //####ECOSGPLCOPYRIGHTEND#### //####BSDCOPYRIGHTBEGIN#### // // ------------------------------------------- // // Portions of this software may have been derived from OpenBSD or other sources, // and are covered by the appropriate copyright disclaimers included herein. // // ------------------------------------------- // //####BSDCOPYRIGHTEND#### //========================================================================== //#####DESCRIPTIONBEGIN#### // // Author(s): gthomas // Contributors: gthomas // Date: 2002-11-25 // Purpose: // Description: platform driver specifics for A&M Adder // // //####DESCRIPTIONEND#### // //========================================================================== // Ethernet device driver support for PHY on Adder/MPC850 #include <pkgconf/system.h> #include <cyg/infra/cyg_type.h> #include <cyg/infra/diag.h> #include <cyg/hal/hal_arch.h> #include <cyg/hal/hal_cache.h> #include <cyg/hal/hal_if.h> #include <cyg/hal/drv_api.h> #include CYGDAT_DEVS_QUICC_ETH_INL // Platform specifics #include <cyg/hal/quicc/ppc8xx.h> // QUICC structure definitions // MII interface #define MII_Start 0x40000000 #define MII_Read 0x20000000 #define MII_Write 0x10000000 #define MII_Cmd 0x30000000 #define MII_Phy(phy) (phy << 23) #define MII_Reg(reg) (reg << 18) #define MII_TA 0x00020000 // Transceiver mode #define PHY_BMCR 0x00 // Register number #define PHY_BMCR_RESET 0x8000 #define PHY_BMCR_LOOPBACK 0x4000 #define PHY_BMCR_100MB 0x2000 #define PHY_BMCR_AUTO_NEG 0x1000 #define PHY_BMCR_POWER_DOWN 0x0800 #define PHY_BMCR_ISOLATE 0x0400 #define PHY_BMCR_RESTART 0x0200 #define PHY_BMCR_FULL_DUPLEX 0x0100 #define PHY_BMCR_COLL_TEST 0x0080 #define PHY_BMSR 0x01 // Status register #define PHY_BMSR_AUTO_NEG 0x0020 #define PHY_BMSR_LINK 0x0004 // Bits in port D - used for 2 wire MII interface #define MII_DATA 0x1000 #define MII_CLOCK 0x0800 #define MII_SET_DATA(val) \ if (val) { \ eppc->pio_pddat |= MII_DATA; \ } else { \ eppc->pio_pddat &= ~MII_DATA; \ } #define MII_GET_DATA() \ ((eppc->pio_pddat & MII_DATA) != 0) #define MII_SET_CLOCK(val) \ if (val) { \ eppc->pio_pddat |= MII_CLOCK; \ } else { \ eppc->pio_pddat &= ~MII_CLOCK; \ } static cyg_uint32 phy_cmd(cyg_uint32 cmd) { volatile EPPC *eppc = (volatile EPPC *)eppc_base(); cyg_uint32 retval; int i, off; bool is_read = ((cmd & MII_Cmd) == MII_Read); // Set both bits as output eppc->pio_pddir |= MII_DATA | MII_CLOCK; // Preamble for (i = 0; i < 32; i++) { MII_SET_CLOCK(0); MII_SET_DATA(1); CYGACC_CALL_IF_DELAY_US(1); MII_SET_CLOCK(1); CYGACC_CALL_IF_DELAY_US(1); } // Command/data for (i = 0, off = 31; i < (is_read ? 14 : 32); i++, --off) { MII_SET_CLOCK(0); MII_SET_DATA((cmd >> off) & 0x00000001); CYGACC_CALL_IF_DELAY_US(1); MII_SET_CLOCK(1); CYGACC_CALL_IF_DELAY_US(1); } retval = cmd; // If read, fetch data register if (is_read) { retval >>= 16; MII_SET_CLOCK(0); eppc->pio_pddir &= ~MII_DATA; // Data bit is now input CYGACC_CALL_IF_DELAY_US(1); MII_SET_CLOCK(1); CYGACC_CALL_IF_DELAY_US(1); MII_SET_CLOCK(0); CYGACC_CALL_IF_DELAY_US(1); for (i = 0, off = 15; i < 16; i++, off--) { MII_SET_CLOCK(1); retval <<= 1; retval |= MII_GET_DATA(); CYGACC_CALL_IF_DELAY_US(1); MII_SET_CLOCK(0); CYGACC_CALL_IF_DELAY_US(1); } } // Set both bits as output eppc->pio_pddir |= MII_DATA | MII_CLOCK; // Postamble for (i = 0; i < 32; i++) { MII_SET_CLOCK(0); MII_SET_DATA(1); CYGACC_CALL_IF_DELAY_US(1); MII_SET_CLOCK(1); CYGACC_CALL_IF_DELAY_US(1); } return retval; } // // PHY unit access (via MII channel) // static void phy_write(int reg, int addr, unsigned short data) { phy_cmd(MII_Start | MII_Write | MII_Phy(addr) | MII_Reg(reg) | MII_TA | data); } static bool phy_read(int reg, int addr, unsigned short *val) { cyg_uint32 ret; ret = phy_cmd(MII_Start | MII_Read | MII_Phy(addr) | MII_Reg(reg) | MII_TA); *val = ret; return true; } bool _adder_reset_phy(void) { volatile EPPC *eppc = (volatile EPPC *)eppc_base(); int phy_timeout = 5*1000; // Wait 5 seconds max for link to clear bool phy_ok; unsigned short phy_state = 0; int phy_unit = -1; int i; // Reset PHY (transceiver) eppc->pip_pbdat &= ~0x00004000; // Reset PHY chip CYGACC_CALL_IF_DELAY_US(15000); // > 10ms eppc->pip_pbdat |= 0x00004000; // Enable PHY chip phy_ok = false; // Try and discover how this PHY is wired for (i = 0; i < 0x20; i++) { phy_read(PHY_BMCR, i, &phy_state); if ((phy_state & PHY_BMCR_RESET) == 0) { phy_unit = i; break; } } if (phy_unit < 0) { diag_printf("QUICC ETH - Can't locate PHY\n"); return false; } else { #if 0 diag_printf("QUICC ETH - using PHY %d\n", phy_unit); #endif } if (phy_read(PHY_BMSR, phy_unit, &phy_state)) { if ((phy_state & PHY_BMSR_LINK) != PHY_BMSR_LINK) { unsigned short reset_mode; phy_write(PHY_BMCR, phy_unit, PHY_BMCR_RESET); for (i = 0; i < 10; i++) { phy_ok = phy_read(PHY_BMCR, phy_unit, &phy_state); if (!phy_ok) break; if (!(phy_state & PHY_BMCR_RESET)) break; } if (!phy_ok || (phy_state & PHY_BMCR_RESET)) { diag_printf("QUICC/ETH: Can't get PHY unit to soft reset: %x\n", phy_state); return false; } reset_mode = PHY_BMCR_RESTART | PHY_BMCR_AUTO_NEG | PHY_BMCR_FULL_DUPLEX; phy_write(PHY_BMCR, phy_unit, reset_mode); while (phy_timeout-- >= 0) { phy_ok = phy_read(PHY_BMSR, phy_unit, &phy_state); if (phy_ok && (phy_state & PHY_BMSR_LINK)) { break; } else { CYGACC_CALL_IF_DELAY_US(10000); // 10ms } } if (phy_timeout <= 0) { diag_printf("** QUICC/ETH Warning: PHY LINK UP failed\n"); } } else { diag_printf("** QUICC/ETH Info: PHY LINK already UP \n"); } } return phy_ok; }
Go to most recent revision | Compare with Previous | Blame | View Log