| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405 |
- /*
- * Copyright (C) 2003-2006 by egnite Software GmbH. All rights reserved.
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions
- * are met:
- *
- * 1. Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- * 2. Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
- * 3. Neither the name of the copyright holders nor the names of
- * contributors may be used to endorse or promote products derived
- * from this software without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
- * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
- * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
- * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
- * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
- * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
- * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
- * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
- * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
- * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
- * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
- * SUCH DAMAGE.
- *
- * For additional information see http://www.ethernut.de/
- *
- */
- /*!
- * \file arch/avr/dev/lanc111.c
- * \brief AVR network driver for SMSC LAN91C111.
- *
- * \verbatim
- * $Id: lanc111.c 5472 2013-12-06 00:16:28Z olereinhardt $
- * \endverbatim
- */
- #include <cfg/os.h>
- #include <cfg/arch/avr.h>
- #include <sys/atom.h>
- #include <sys/heap.h>
- #include <sys/thread.h>
- #include <sys/event.h>
- #include <sys/timer.h>
- #include <sys/confnet.h>
- #include <stdlib.h>
- #include <string.h>
- #include <netinet/if_ether.h>
- #include <net/ether.h>
- #include <net/if_var.h>
- #include <dev/irqreg.h>
- #include <dev/lanc111.h>
- #include <stdlib.h>
- #include <string.h>
- #ifdef NUTDEBUG
- #include <stdio.h>
- #endif
- #ifndef NUT_THREAD_LANCRXSTACK
- #if defined(__GNUC__)
- /* avr-gcc size optimized code used 76 bytes.
- Sigh! 144 bytes are reported using avr-gcc 4.3.3 and Nut/OS 4.9.10. */
- #define NUT_THREAD_LANCRXSTACK 256
- #else
- /* icc-avr v7.19 used 200 bytes. */
- #define NUT_THREAD_LANCRXSTACK 384
- #endif
- #endif
- /*
- * Determine ports, which had not been explicitely configured.
- */
- #ifndef LANC111_BASE_ADDR
- #define LANC111_BASE_ADDR 0xC000
- #endif
- #ifndef LANC111_SIGNAL_IRQ
- #define LANC111_SIGNAL_IRQ INT5
- #endif
- #ifdef LANC111_RESET_BIT
- #if (LANC111_RESET_AVRPORT == AVRPORTB)
- #define LANC111_RESET_PORT PORTB
- #define LANC111_RESET_DDR DDRB
- #elif (LANC111_RESET_AVRPORT == AVRPORTD)
- #define LANC111_RESET_PORT PORTD
- #define LANC111_RESET_DDR DDRD
- #elif (LANC111_RESET_AVRPORT == AVRPORTE)
- #define LANC111_RESET_PORT PORTE
- #define LANC111_RESET_DDR DDRE
- #elif (LANC111_RESET_AVRPORT == AVRPORTF)
- #define LANC111_RESET_PORT PORTF
- #define LANC111_RESET_DDR DDRF
- #endif /* LANC111_RESET_AVRPORT */
- #endif /* LANC111_RESET_BIT */
- /*
- * Determine interrupt settings.
- */
- #if (LANC111_SIGNAL_IRQ == INT0)
- #define LANC111_SIGNAL sig_INTERRUPT0
- #define LANC111_SIGNAL_MODE() sbi(EICRA, ISC00); sbi(EICRA, ISC01)
- #elif (LANC111_SIGNAL_IRQ == INT1)
- #define LANC111_SIGNAL sig_INTERRUPT1
- #define LANC111_SIGNAL_MODE() sbi(EICRA, ISC10); sbi(EICRA, ISC11)
- #elif (LANC111_SIGNAL_IRQ == INT2)
- #define LANC111_SIGNAL sig_INTERRUPT2
- #define LANC111_SIGNAL_MODE() sbi(EICRA, ISC20); sbi(EICRA, ISC21)
- #elif (LANC111_SIGNAL_IRQ == INT3)
- #define LANC111_SIGNAL sig_INTERRUPT3
- #define LANC111_SIGNAL_MODE() sbi(EICRA, ISC30); sbi(EICRA, ISC31)
- #elif (LANC111_SIGNAL_IRQ == INT4)
- #define LANC111_SIGNAL sig_INTERRUPT4
- #define LANC111_SIGNAL_MODE() sbi(EICR, ISC40); sbi(EICR, ISC41)
- #elif (LANC111_SIGNAL_IRQ == INT6)
- #define LANC111_SIGNAL sig_INTERRUPT6
- #define LANC111_SIGNAL_MODE() sbi(EICR, ISC60); sbi(EICR, ISC61)
- #elif (LANC111_SIGNAL_IRQ == INT7)
- #define LANC111_SIGNAL sig_INTERRUPT7
- #define LANC111_SIGNAL_MODE() sbi(EICR, ISC70); sbi(EICR, ISC71)
- #else
- #define LANC111_SIGNAL sig_INTERRUPT5
- #define LANC111_SIGNAL_MODE() sbi(EICR, ISC50); sbi(EICR, ISC51)
- #endif
- /*!
- * \addtogroup xgSmscRegs
- */
- /*@{*/
- /*!
- * \brief Bank select register.
- */
- #define NIC_BSR (LANC111_BASE_ADDR + 0x0E)
- /*!
- * \brief Bank 0 - Transmit control register.
- */
- #define NIC_TCR (LANC111_BASE_ADDR + 0x00)
- #define TCR_SWFDUP 0x8000 /*!< \ref NIC_TCR bit mask, enables full duplex. */
- #define TCR_EPH_LOOP 0x2000 /*!< \ref NIC_TCR bit mask, enables internal loopback. */
- #define TCR_STP_SQET 0x1000 /*!< \ref NIC_TCR bit mask, enables transmission stop on SQET error. */
- #define TCR_FDUPLX 0x0800 /*!< \ref NIC_TCR bit mask, enables receiving own frames. */
- #define TCR_MON_CSN 0x0400 /*!< \ref NIC_TCR bit mask, enables carrier monitoring. */
- #define TCR_NOCRC 0x0100 /*!< \ref NIC_TCR bit mask, disables CRC transmission. */
- #define TCR_PAD_EN 0x0080 /*!< \ref NIC_TCR bit mask, enables automatic padding. */
- #define TCR_FORCOL 0x0004 /*!< \ref NIC_TCR bit mask, forces collision. */
- #define TCR_LOOP 0x0002 /*!< \ref NIC_TCR bit mask, enables PHY loopback. */
- #define TCR_TXENA 0x0001 /*!< \ref NIC_TCR bit mask, enables transmitter. */
- /*!
- * \brief Bank 0 - EPH status register.
- */
- #define NIC_EPHSR (LANC111_BASE_ADDR + 0x02)
- /*!
- * \brief Bank 0 - Receive control register.
- */
- #define NIC_RCR (LANC111_BASE_ADDR + 0x04)
- #define RCR_SOFT_RST 0x8000 /*!< \ref NIC_RCR bit mask, activates software reset. */
- #define RCR_FILT_CAR 0x4000 /*!< \ref NIC_RCR bit mask, enables carrier filter. */
- #define RCR_ABORT_ENB 0x2000 /*!< \ref NIC_RCR bit mask, enables receive abort on collision. */
- #define RCR_STRIP_CRC 0x0200 /*!< \ref NIC_RCR bit mask, strips CRC. */
- #define RCR_RXEN 0x0100 /*!< \ref NIC_RCR bit mask, enables receiver. */
- #define RCR_ALMUL 0x0004 /*!< \ref NIC_RCR bit mask, multicast frames accepted when set. */
- #define RCR_PRMS 0x0002 /*!< \ref NIC_RCR bit mask, enables promiscuous mode. */
- #define RCR_RX_ABORT 0x0001 /*!< \ref NIC_RCR bit mask, set when receive was aborted. */
- /*!
- * \brief Bank 0 - Counter register.
- */
- #define NIC_ECR (LANC111_BASE_ADDR + 0x06)
- /*!
- * \brief Bank 0 - Memory information register.
- */
- #define NIC_MIR (LANC111_BASE_ADDR + 0x08)
- /*!
- * \brief Bank 0 - Receive / PHY control register.
- */
- #define NIC_RPCR (LANC111_BASE_ADDR + 0x0A)
- #define RPCR_SPEED 0x2000 /*!< \ref NIC_RPCR bit mask, PHY operates at 100 Mbps. */
- #define RPCR_DPLX 0x1000 /*!< \ref NIC_RPCR bit mask, PHY operates at full duplex mode. */
- #define RPCR_ANEG 0x0800 /*!< \ref NIC_RPCR bit mask, sets PHY in auto-negotiation mode. */
- #define RPCR_LEDA_PAT 0x0000 /*!< \ref NIC_RPCR bit mask for LEDA mode. */
- #define RPCR_LEDB_PAT 0x0010 /*!< \ref NIC_RPCR bit mask for LEDB mode. */
- /*!
- * \brief Bank 1 - Configuration register.
- */
- #define NIC_CR (LANC111_BASE_ADDR + 0x00)
- #define CR_EPH_EN 0x8000 /*!< \ref NIC_CR bit mask, . */
- /*!
- * \brief Bank 1 - Base address register.
- */
- #define NIC_BAR (LANC111_BASE_ADDR + 0x02)
- /*!
- * \brief Bank 1 - Individual address register.
- */
- #define NIC_IAR (LANC111_BASE_ADDR + 0x04)
- /*!
- * \brief Bank 1 - General purpose register.
- */
- #define NIC_GPR (LANC111_BASE_ADDR + 0x0A)
- /*!
- * \brief Bank 1 - Control register.
- */
- #define NIC_CTR (LANC111_BASE_ADDR + 0x0C)
- #define CTR_RCV_BAD 0x4000 /*!< \ref NIC_CTR bit mask. */
- #define CTR_AUTO_RELEASE 0x0800 /*!< \ref NIC_CTR bit mask, transmit packets automatically released. */
- /*!
- * \brief Bank 2 - MMU command register.
- */
- #define NIC_MMUCR (LANC111_BASE_ADDR + 0x00)
- #define MMUCR_BUSY 0x0001
- #define MMU_NOP 0
- #define MMU_ALO (1<<5)
- #define MMU_RST (2<<5)
- #define MMU_REM (3<<5)
- #define MMU_TOP (4<<5)
- #define MMU_PKT (5<<5)
- #define MMU_ENQ (6<<5)
- #define MMU_RTX (7<<5)
- /*!
- * \brief Bank 2 - Packet number register.
- *
- * This byte register specifies the accessible transmit packet number.
- */
- #define NIC_PNR (LANC111_BASE_ADDR + 0x02)
- /*!
- * \brief Bank 2 - Allocation result register.
- *
- * This byte register is updated upon a MMU_ALO command.
- */
- #define NIC_ARR (LANC111_BASE_ADDR + 0x03)
- #define ARR_FAILED 0x80
- /*!
- * \brief Bank 2 - FIFO ports register.
- */
- #define NIC_FIFO (LANC111_BASE_ADDR + 0x04)
- /*!
- * \brief Bank 2 - Pointer register.
- */
- #define NIC_PTR (LANC111_BASE_ADDR + 0x06)
- #define PTR_RCV 0x8000 /*! \ref NIC_PTR bit mask, specifies receive or transmit buffer. */
- #define PTR_AUTO_INCR 0x4000 /*! \ref NIC_PTR bit mask, enables automatic pointer increment. */
- #define PTR_READ 0x2000 /*! \ref NIC_PTR bit mask, indicates type of access. */
- #define PTR_ETEN 0x1000 /*! \ref NIC_PTR bit mask, enables early transmit underrun detection. */
- #define PTR_NOT_EMPTY 0x0800 /*! \ref NIC_PTR bit mask, set when write data fifo is not empty. */
- /*!
- * \brief Bank 2 - Data register.
- */
- #define NIC_DATA (LANC111_BASE_ADDR + 0x08)
- /*!
- * \brief Bank 2 - Interrupt status register.
- */
- #define NIC_IST (LANC111_BASE_ADDR + 0x0C)
- /*!
- * \brief Bank 2 - Interrupt acknowledge register.
- */
- #define NIC_ACK (LANC111_BASE_ADDR + 0x0C)
- /*!
- * \brief Bank 2 - Interrupt mask register.
- */
- #define NIC_MSK (LANC111_BASE_ADDR + 0x0D)
- #define INT_MD 0x80 /*!< \brief PHY state change interrupt bit mask. */
- #define INT_ERCV 0x40 /*!< \brief Early receive interrupt bit mask. */
- #define INT_EPH 0x20 /*!< \brief Ethernet protocol interrupt bit mask. */
- #define INT_RX_OVRN 0x10 /*!< \brief Receive overrun interrupt bit mask. */
- #define INT_ALLOC 0x08 /*!< \brief Transmit allocation interrupt bit mask. */
- #define INT_TX_EMPTY 0x04 /*!< \brief Transmitter empty interrupt bit mask. */
- #define INT_TX 0x02 /*!< \brief Transmit complete interrupt bit mask. */
- #define INT_RCV 0x01 /*!< \brief Receive interrupt bit mask. */
- /*!
- * \brief Bank 3 - Multicast table register.
- */
- #define NIC_MT (LANC111_BASE_ADDR + 0x00)
- /*!
- * \brief Bank 3 - Management interface register.
- */
- #define NIC_MGMT (LANC111_BASE_ADDR + 0x08)
- #define MGMT_MDOE 0x08 /*!< \ref NIC_MGMT bit mask, enables MDO pin. */
- #define MGMT_MCLK 0x04 /*!< \ref NIC_MGMT bit mask, drives MDCLK pin. */
- #define MGMT_MDI 0x02 /*!< \ref NIC_MGMT bit mask, reflects MDI pin status. */
- #define MGMT_MDO 0x01 /*!< \ref NIC_MGMT bit mask, drives MDO pin. */
- /*!
- * \brief Bank 3 - Revision register.
- */
- #define NIC_REV (LANC111_BASE_ADDR + 0x0A)
- /*!
- * \brief Bank 3 - Early RCV register.
- */
- #define NIC_ERCV (LANC111_BASE_ADDR + 0x0C)
- /*!
- * \brief PHY control register.
- */
- #define NIC_PHYCR 0
- #define PHYCR_RST 0x8000 /*!< \ref NIC_PHYCR bit mask, resets PHY. */
- #define PHYCR_LPBK 0x4000 /*!< \ref NIC_PHYCR bit mask, . */
- #define PHYCR_SPEED 0x2000 /*!< \ref NIC_PHYCR bit mask, . */
- #define PHYCR_ANEG_EN 0x1000 /*!< \ref NIC_PHYCR bit mask, . */
- #define PHYCR_PDN 0x0800 /*!< \ref NIC_PHYCR bit mask, . */
- #define PHYCR_MII_DIS 0x0400 /*!< \ref NIC_PHYCR bit mask, . */
- #define PHYCR_ANEG_RST 0x0200 /*!< \ref NIC_PHYCR bit mask, . */
- #define PHYCR_DPLX 0x0100 /*!< \ref NIC_PHYCR bit mask, . */
- #define PHYCR_COLST 0x0080 /*!< \ref NIC_PHYCR bit mask, . */
- /*!
- * \brief PHY status register.
- */
- #define NIC_PHYSR 1
- #define PHYSR_CAP_T4 0x8000 /*!< \ref NIC_PHYSR bit mask, indicates 100BASE-T4 capability. */
- #define PHYSR_CAP_TXF 0x4000 /*!< \ref NIC_PHYSR bit mask, indicates 100BASE-TX full duplex capability. */
- #define PHYSR_CAP_TXH 0x2000 /*!< \ref NIC_PHYSR bit mask, indicates 100BASE-TX half duplex capability. */
- #define PHYSR_CAP_TF 0x1000 /*!< \ref NIC_PHYSR bit mask, indicates 10BASE-T full duplex capability. */
- #define PHYSR_CAP_TH 0x0800 /*!< \ref NIC_PHYSR bit mask, indicates 10BASE-T half duplex capability. */
- #define PHYSR_CAP_SUPR 0x0040 /*!< \ref NIC_PHYSR bit mask, indicates preamble suppression capability. */
- #define PHYSR_ANEG_ACK 0x0020 /*!< \ref NIC_PHYSR bit mask, auto-negotiation completed. */
- #define PHYSR_REM_FLT 0x0010 /*!< \ref NIC_PHYSR bit mask, remote fault detected. */
- #define PHYSR_CAP_ANEG 0x0008 /*!< \ref NIC_PHYSR bit mask, indicates auto-negotiation capability. */
- #define PHYSR_LINK 0x0004 /*!< \ref NIC_PHYSR bit mask, valid link status. */
- #define PHYSR_JAB 0x0002 /*!< \ref NIC_PHYSR bit mask, jabber collision detected. */
- #define PHYSR_EXREG 0x0001 /*!< \ref NIC_PHYSR bit mask, extended capabilities available. */
- /*!
- * \brief PHY identifier register 1.
- */
- #define NIC_PHYID1 2
- /*!
- * \brief PHY identifier register 1.
- */
- #define NIC_PHYID2 3
- /*!
- * \brief PHY auto-negotiation advertisement register.
- */
- #define NIC_PHYANAD 4
- #define PHYANAD_NP 0x8000 /*!< \ref NIC_PHYANAD bit mask, exchanging next page information. */
- #define PHYANAD_ACK 0x4000 /*!< \ref NIC_PHYANAD bit mask, acknowledged. */
- #define PHYANAD_RF 0x2000 /*!< \ref NIC_PHYANAD bit mask, remote fault. */
- #define PHYANAD_T4 0x0200 /*!< \ref NIC_PHYANAD bit mask, indicates 100BASE-T4 capability. */
- #define PHYANAD_TX_FDX 0x0100 /*!< \ref NIC_PHYANAD bit mask, indicates 100BASE-TX full duplex capability. */
- #define PHYANAD_TX_HDX 0x0080 /*!< \ref NIC_PHYANAD bit mask, indicates 100BASE-TX half duplex capability. */
- #define PHYANAD_10FDX 0x0040 /*!< \ref NIC_PHYANAD bit mask, indicates 10BASE-T full duplex capability. */
- #define PHYANAD_10_HDX 0x0020 /*!< \ref NIC_PHYANAD bit mask, indicates 10BASE-T half duplex capability. */
- #define PHYANAD_CSMA 0x0001 /*!< \ref NIC_PHYANAD bit mask, indicates 802.3 CSMA capability. */
- /*!
- * \brief PHY auto-negotiation remote end capability register.
- */
- #define NIC_PHYANRC 5
- /*!
- * \brief PHY configuration register 1.
- */
- #define NIC_PHYCFR1 16
- /*!
- * \brief PHY configuration register 2.
- */
- #define NIC_PHYCFR2 17
- /*!
- * \brief PHY status output register.
- */
- #define NIC_PHYSOR 18
- #define PHYSOR_INT 0x8000 /*!< \ref NIC_PHYSOR bit mask, interrupt bits changed. */
- #define PHYSOR_LNKFAIL 0x4000 /*!< \ref NIC_PHYSOR bit mask, link failure detected. */
- #define PHYSOR_LOSSSYNC 0x2000 /*!< \ref NIC_PHYSOR bit mask, descrambler sync lost detected. */
- #define PHYSOR_CWRD 0x1000 /*!< \ref NIC_PHYSOR bit mask, code word error detected. */
- #define PHYSOR_SSD 0x0800 /*!< \ref NIC_PHYSOR bit mask, start of stream error detected. */
- #define PHYSOR_ESD 0x0400 /*!< \ref NIC_PHYSOR bit mask, end of stream error detected. */
- #define PHYSOR_RPOL 0x0200 /*!< \ref NIC_PHYSOR bit mask, reverse polarity detected. */
- #define PHYSOR_JAB 0x0100 /*!< \ref NIC_PHYSOR bit mask, jabber detected. */
- #define PHYSOR_SPDDET 0x0080 /*!< \ref NIC_PHYSOR bit mask, 100/10 speed detected. */
- #define PHYSOR_DPLXDET 0x0040 /*!< \ref NIC_PHYSOR bit mask, duplex detected. */
- /*!
- * \brief PHY mask register.
- */
- #define NIC_PHYMSK 19
- #define PHYMSK_MINT 0x8000 /*!< \ref NIC_PHYMSK bit mask, enables \ref PHYSOR_INT interrupt. */
- #define PHYMSK_MLNKFAIL 0x4000 /*!< \ref NIC_PHYMSK bit mask, enables \ref PHYSOR_LNKFAIL interrupt. */
- #define PHYMSK_MLOSSSYN 0x2000 /*!< \ref NIC_PHYMSK bit mask, enables \ref PHYSOR_LOSSSYNC interrupt. */
- #define PHYMSK_MCWRD 0x1000 /*!< \ref NIC_PHYMSK bit mask, enables \ref PHYSOR_CWRD interrupt. */
- #define PHYMSK_MSSD 0x0800 /*!< \ref NIC_PHYMSK bit mask, enables \ref PHYSOR_SSD interrupt. */
- #define PHYMSK_MESD 0x0400 /*!< \ref NIC_PHYMSK bit mask, enables \ref PHYSOR_ESD interrupt. */
- #define PHYMSK_MRPOL 0x0200 /*!< \ref NIC_PHYMSK bit mask, enables \ref PHYSOR_RPOL interrupt. */
- #define PHYMSK_MJAB 0x0100 /*!< \ref NIC_PHYMSK bit mask, enables \ref PHYSOR_JAB interrupt. */
- #define PHYMSK_MSPDDT 0x0080 /*!< \ref NIC_PHYMSK bit mask, enables \ref PHYSOR_SPDDET interrupt. */
- #define PHYMSK_MDPLDT 0x0040 /*!< \ref NIC_PHYMSK bit mask, enables \ref PHYSOR_DPLXDET interrupt. */
- #define MSBV(bit) (1 << ((bit) - 8))
- #define nic_outlb(addr, val) (*(volatile uint8_t *)(addr) = (val))
- #define nic_outhb(addr, val) (*(volatile uint8_t *)((addr) + 1) = (val))
- #define nic_outwx(addr, val) (*(volatile uint16_t *)(addr) = (val))
- #define nic_outw(addr, val) { \
- *(volatile uint8_t *)(addr) = (uint8_t)(val); \
- *((volatile uint8_t *)(addr) + 1) = (uint8_t)((val) >> 8); \
- }
- #define nic_inlb(addr) (*(volatile uint8_t *)(addr))
- #define nic_inhb(addr) (*(volatile uint8_t *)((addr) + 1))
- #define nic_inw(addr) (*(volatile uint16_t *)(addr))
- #define nic_bs(bank) nic_outlb(NIC_BSR, bank)
- /*!
- * \struct _NICINFO lanc111.h dev/lanc111.h
- * \brief Network interface controller information structure.
- */
- struct _NICINFO {
- HANDLE volatile ni_rx_rdy; /*!< Receiver event queue. */
- uint16_t ni_tx_cnt; /*!< Number of bytes in transmission queue. */
- uint32_t ni_rx_packets; /*!< Number of packets received. */
- uint32_t ni_tx_packets; /*!< Number of packets sent. */
- uint32_t ni_interrupts; /*!< Number of interrupts. */
- uint32_t ni_overruns; /*!< Number of packet overruns. */
- uint32_t ni_rx_frame_errors; /*!< Number of frame errors. */
- uint32_t ni_rx_crc_errors; /*!< Number of CRC errors. */
- uint32_t ni_rx_missed_errors; /*!< Number of missed packets. */
- uint8_t ni_mar[8]; /*!< Multicast Address Register. */
- };
- /*!
- * \brief Network interface controller information type.
- */
- typedef struct _NICINFO NICINFO;
- /*@}*/
- /*!
- * \addtogroup xgNicLanc111
- */
- /*@{*/
- static HANDLE mutex;
- static HANDLE maq;
- /*!
- * \brief Select specified PHY register for reading or writing.
- *
- * \note NIC interrupts must have been disabled before calling this routine.
- *
- * \param reg PHY register number.
- * \param we Indicates type of access, 1 for write and 0 for read.
- *
- * \return Contents of the PHY interface rgister.
- */
- static uint8_t NicPhyRegSelect(uint8_t reg, uint8_t we)
- {
- uint8_t rs;
- uint8_t msk;
- uint8_t i;
- nic_bs(3);
- rs = (nic_inlb(NIC_MGMT) & ~(MGMT_MCLK | MGMT_MDO)) | MGMT_MDOE;
- /* Send idle pattern. */
- for (i = 0; i < 33; i++) {
- nic_outlb(NIC_MGMT, rs | MGMT_MDO);
- nic_outlb(NIC_MGMT, rs | MGMT_MDO | MGMT_MCLK);
- }
- /* Send start sequence. */
- nic_outlb(NIC_MGMT, rs);
- nic_outlb(NIC_MGMT, rs | MGMT_MCLK);
- nic_outlb(NIC_MGMT, rs | MGMT_MDO);
- nic_outlb(NIC_MGMT, rs | MGMT_MDO | MGMT_MCLK);
- /* Write or read mode. */
- if (we) {
- nic_outlb(NIC_MGMT, rs);
- nic_outlb(NIC_MGMT, rs | MGMT_MCLK);
- nic_outlb(NIC_MGMT, rs | MGMT_MDO);
- nic_outlb(NIC_MGMT, rs | MGMT_MDO | MGMT_MCLK);
- } else {
- nic_outlb(NIC_MGMT, rs | MGMT_MDO);
- nic_outlb(NIC_MGMT, rs | MGMT_MDO | MGMT_MCLK);
- nic_outlb(NIC_MGMT, rs);
- nic_outlb(NIC_MGMT, rs | MGMT_MCLK);
- }
- /* Send PHY address. Zero is used for the internal PHY. */
- for (i = 0; i < 5; i++) {
- nic_outlb(NIC_MGMT, rs);
- nic_outlb(NIC_MGMT, rs | MGMT_MCLK);
- }
- /* Send PHY register number. */
- for (msk = 0x10; msk; msk >>= 1) {
- if (reg & msk) {
- nic_outlb(NIC_MGMT, rs | MGMT_MDO);
- nic_outlb(NIC_MGMT, rs | MGMT_MDO | MGMT_MCLK);
- } else {
- nic_outlb(NIC_MGMT, rs);
- nic_outlb(NIC_MGMT, rs | MGMT_MCLK);
- }
- }
- nic_outlb(NIC_MGMT, rs);
- return rs;
- }
- /*!
- * \brief Read contents of PHY register.
- *
- * \note NIC interrupts must have been disabled before calling this routine.
- *
- * \param reg PHY register number.
- *
- * \return Contents of the specified register.
- */
- static uint16_t NicPhyRead(uint8_t reg)
- {
- uint16_t rc = 0;
- uint8_t rs;
- uint8_t i;
- /* Select register for reading. */
- rs = NicPhyRegSelect(reg, 0);
- /* Switch data direction. */
- rs &= ~MGMT_MDOE;
- nic_outlb(NIC_MGMT, rs);
- nic_outlb(NIC_MGMT, rs | MGMT_MCLK);
- /* Clock data in. */
- for (i = 0; i < 16; i++) {
- nic_outlb(NIC_MGMT, rs);
- nic_outlb(NIC_MGMT, rs | MGMT_MCLK);
- rc <<= 1;
- rc |= (nic_inlb(NIC_MGMT) & MGMT_MDI) != 0;
- }
- /* This will set the clock line to low. */
- nic_outlb(NIC_MGMT, rs);
- return rc;
- }
- /*!
- * \brief Write value to PHY register.
- *
- * \note NIC interrupts must have been disabled before calling this routine.
- *
- * \param reg PHY register number.
- * \param val Value to write.
- */
- static void NicPhyWrite(uint8_t reg, uint16_t val)
- {
- uint16_t msk;
- uint8_t rs;
- /* Select register for writing. */
- rs = NicPhyRegSelect(reg, 1);
- /* Switch data direction dummy. */
- nic_outlb(NIC_MGMT, rs | MGMT_MDO);
- nic_outlb(NIC_MGMT, rs | MGMT_MDO | MGMT_MCLK);
- nic_outlb(NIC_MGMT, rs);
- nic_outlb(NIC_MGMT, rs | MGMT_MCLK);
- /* Clock data out. */
- for (msk = 0x8000; msk; msk >>= 1) {
- if (val & msk) {
- nic_outlb(NIC_MGMT, rs | MGMT_MDO);
- nic_outlb(NIC_MGMT, rs | MGMT_MDO | MGMT_MCLK);
- } else {
- nic_outlb(NIC_MGMT, rs);
- nic_outlb(NIC_MGMT, rs | MGMT_MCLK);
- }
- }
- /* Set clock line low and output line int z-state. */
- nic_outlb(NIC_MGMT, rs & ~MGMT_MDOE);
- }
- /*!
- * \brief Configure the internal PHY.
- *
- * Reset the PHY and initiate auto-negotiation.
- */
- static int NicPhyConfig(void)
- {
- uint16_t phy_sr;
- uint16_t phy_to;
- uint16_t mode;
- /*
- * Reset the PHY and wait until this self clearing bit
- * becomes zero. We sleep 63 ms before each poll and
- * give up after 3 retries.
- */
- //printf("Reset PHY..");
- NicPhyWrite(NIC_PHYCR, PHYCR_RST);
- for (phy_to = 0;; phy_to++) {
- NutSleep(63);
- if ((NicPhyRead(NIC_PHYCR) & PHYCR_RST) == 0)
- break;
- if (phy_to > 3)
- return -1;
- }
- //printf("OK\n");
- /* Store PHY status output. */
- NicPhyRead(NIC_PHYSOR);
- /* Enable PHY interrupts. */
- NicPhyWrite(NIC_PHYMSK, PHYMSK_MLOSSSYN | PHYMSK_MCWRD | PHYMSK_MSSD |
- PHYMSK_MESD | PHYMSK_MRPOL | PHYMSK_MJAB | PHYMSK_MSPDDT | PHYMSK_MDPLDT);
- /* Set RPC register. */
- mode = RPCR_ANEG | RPCR_LEDA_PAT | RPCR_LEDB_PAT;
- nic_bs(0);
- nic_outw(NIC_RPCR, mode);
- #ifdef NIC_FIXED
- /* Disable link. */
- phy_sr = NicPhyRead(NIC_PHYCFR1);
- NicPhyWrite(NIC_PHYCFR1, phy_sr | 0x8000);
- NutSleep(63);
- /* Set fixed capabilities. */
- NicPhyWrite(NIC_PHYCR, NIC_FIXED);
- nic_bs(0);
- nic_outw(NIC_RPCR, mode);
- /* Enable link. */
- phy_sr = NicPhyRead(NIC_PHYCFR1);
- NicPhyWrite(NIC_PHYCFR1, phy_sr & ~0x8000);
- phy_sr = NicPhyRead(NIC_PHYCFR1);
- #else
- /*
- * Advertise our capabilities, initiate auto negotiation
- * and wait until this has been completed.
- */
- //printf("Negotiate..");
- NicPhyWrite(NIC_PHYANAD, PHYANAD_TX_FDX | PHYANAD_TX_HDX | PHYANAD_10FDX | PHYANAD_10_HDX | PHYANAD_CSMA);
- NutSleep(63);
- for (phy_to = 0, phy_sr = 0;; phy_to++) {
- /* Give up after 10 seconds. */
- if (phy_to >= 1024)
- return -1;
- /* Restart auto negotiation every 4 seconds or on failures. */
- if ((phy_to & 127) == 0 /* || (phy_sr & PHYSR_REM_FLT) != 0 */ ) {
- NicPhyWrite(NIC_PHYCR, PHYCR_ANEG_EN | PHYCR_ANEG_RST);
- //printf("Restart..");
- NutSleep(63);
- }
- /* Check if we are done. */
- phy_sr = NicPhyRead(NIC_PHYSR);
- //printf("[SR %04X]", phy_sr);
- if (phy_sr & PHYSR_ANEG_ACK)
- break;
- NutSleep(63);
- }
- //printf("OK\n");
- #endif
- return 0;
- }
- /*!
- * \brief Wait until MMU is ready.
- *
- * Poll the MMU command register until \ref MMUCR_BUSY
- * is cleared.
- *
- * \param tmo Timeout in milliseconds.
- *
- * \return 0 on success or -1 on timeout.
- */
- static NUT_INLINE_FUNC int NicMmuWait(uint16_t tmo)
- {
- while (tmo--) {
- if ((nic_inlb(NIC_MMUCR) & MMUCR_BUSY) == 0)
- break;
- NutDelay(1);
- }
- return tmo ? 0 : -1;
- }
- /*!
- * \brief Reset the Ethernet controller.
- *
- * \return 0 on success, -1 otherwise.
- */
- static int NicReset(void)
- {
- #ifdef LANC111_RESET_BIT
- sbi(LANC111_RESET_DDR, LANC111_RESET_BIT);
- sbi(LANC111_RESET_PORT, LANC111_RESET_BIT);
- NutDelay(WAIT100);
- cbi(LANC111_RESET_PORT, LANC111_RESET_BIT);
- NutDelay(WAIT250);
- NutDelay(WAIT250);
- #endif
- /* Disable all interrupts. */
- nic_outlb(NIC_MSK, 0);
- /* MAC and PHY software reset. */
- nic_bs(0);
- nic_outw(NIC_RCR, RCR_SOFT_RST);
- /* Enable Ethernet protocol handler. */
- nic_bs(1);
- nic_outw(NIC_CR, CR_EPH_EN);
- NutDelay(10);
- /* Disable transmit and receive. */
- nic_bs(0);
- nic_outw(NIC_RCR, 0);
- nic_outw(NIC_TCR, 0);
- /* Enable auto release. */
- nic_bs(1);
- nic_outw(NIC_CTR, CTR_AUTO_RELEASE);
- /* Reset MMU. */
- nic_bs(2);
- nic_outlb(NIC_MMUCR, MMU_RST);
- if (NicMmuWait(1000))
- return -1;
- return 0;
- }
- /*!
- * \brief Update the multicast register.
- *
- * \param Network interface controller information.
- */
- static void NicUpdateMCHardware(NICINFO * ni)
- {
- int i;
- /* Set multicast address register */
- nic_bs(3);
- for (i = 0; i < 7; i++) {
- nic_outlb(NIC_MT + i, ni->ni_mar[i]);
- }
- }
- /*
- * Fires up the network interface. NIC interrupts
- * should have been disabled when calling this
- * function.
- *
- * \param mac Six byte unique MAC address.
- */
- static int NicStart(const uint8_t * mac, NICINFO * ni)
- {
- uint8_t i;
- if (NicReset())
- return -1;
- /* Enable receiver. */
- nic_bs(3);
- nic_outlb(NIC_ERCV, 7);
- nic_bs(0);
- nic_outw(NIC_RCR, RCR_RXEN);
- /* Enable transmitter and padding. */
- nic_outw(NIC_TCR, TCR_PAD_EN | TCR_TXENA);
- /* Configure the PHY. */
- if (NicPhyConfig())
- return -1;
- /* Set MAC address. */
- //printf("Set MAC %02X%02X%02X%02X%02X%02X", mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]);
- nic_bs(1);
- for (i = 0; i < 6; i++)
- nic_outlb(NIC_IAR + i, mac[i]);
- //printf("OK\n");
- /* Set multicast address register */
- NicUpdateMCHardware(ni);
- /* Enable interrupts. */
- nic_bs(2);
- nic_outlb(NIC_MSK, INT_ERCV | INT_RCV | INT_RX_OVRN);
- return 0;
- }
- /*
- * NIC interrupt entry.
- */
- static void NicInterrupt(void *arg)
- {
- uint8_t isr;
- uint8_t imr;
- NICINFO *ni = (NICINFO *) ((NUTDEVICE *) arg)->dev_dcb;
- ni->ni_interrupts++;
- /* Read the interrupt mask and disable all interrupts. */
- nic_bs(2);
- imr = nic_inlb(NIC_MSK);
- nic_outlb(NIC_MSK, 0);
- /* Read the interrupt status and acknowledge all interrupts. */
- isr = nic_inlb(NIC_IST);
- //printf("\n!%02X-%02X ", isr, imr);
- isr &= imr;
- /*
- * If this is a transmit interrupt, then a packet has been sent.
- * So we can clear the transmitter busy flag and wake up the
- * transmitter thread.
- */
- if (isr & INT_TX_EMPTY) {
- nic_outlb(NIC_ACK, INT_TX_EMPTY);
- imr &= ~INT_TX_EMPTY;
- }
- /* Transmit error. */
- else if (isr & INT_TX) {
- /* re-enable transmit */
- nic_bs(0);
- nic_outw(NIC_TCR, nic_inlb(NIC_TCR) | TCR_TXENA);
- nic_bs(2);
- nic_outlb(NIC_ACK, INT_TX);
- /* kill the packet */
- nic_outlb(NIC_MMUCR, MMU_PKT);
- }
- /*
- * If this is a receive interrupt, then wake up the receiver
- * thread.
- */
- if (isr & INT_RX_OVRN) {
- nic_outlb(NIC_ACK, INT_RX_OVRN);
- //nic_outlb(NIC_MMUCR, MMU_TOP);
- NutEventPostFromIrq(&ni->ni_rx_rdy);
- }
- if (isr & INT_ERCV) {
- nic_outlb(NIC_ACK, INT_ERCV);
- NutEventPostFromIrq(&ni->ni_rx_rdy);
- }
- if (isr & INT_RCV) {
- nic_outlb(NIC_ACK, INT_RCV);
- imr &= ~INT_RCV;
- NutEventPostFromIrq(&ni->ni_rx_rdy);
- }
- if (isr & INT_ALLOC) {
- imr &= ~INT_ALLOC;
- NutEventPostFromIrq(&maq);
- }
- //printf(" -%02X-%02X- ", nic_inlb(NIC_IST), inb(PINE) & 0x20);
- nic_outlb(NIC_MSK, imr);
- }
- /*
- * Write data block to the NIC.
- */
- static void NicWrite(uint8_t * buf, uint16_t len)
- {
- register uint16_t l = len - 1;
- register uint8_t ih = (uint16_t) l >> 8;
- register uint8_t il = (uint8_t) l;
- if (!len)
- return;
- do {
- do {
- nic_outlb(NIC_DATA, *buf++);
- } while (il-- != 0);
- } while (ih-- != 0);
- }
- /*
- * Read data block from the NIC.
- */
- static void NicRead(uint8_t * buf, uint16_t len)
- {
- register uint16_t l = len - 1;
- register uint8_t ih = (uint16_t) l >> 8;
- register uint8_t il = (uint8_t) l;
- if (!len)
- return;
- do {
- do {
- *buf++ = nic_inlb(NIC_DATA);
- } while (il-- != 0);
- } while (ih-- != 0);
- }
- /*!
- * \brief Fetch the next packet out of the receive ring buffer.
- *
- * Nic interrupts must be disabled when calling this funtion.
- *
- * \return Pointer to an allocated ::NETBUF. If there is no
- * no data available, then the function returns a
- * null pointer. If the NIC's buffer seems to be
- * corrupted, a pointer to 0xFFFF is returned.
- */
- static NETBUF *NicGetPacket(void)
- {
- NETBUF *nb = 0;
- //uint8_t *buf;
- uint16_t fsw;
- uint16_t fbc;
- /* Check the fifo empty bit. If it is set, then there is
- nothing in the receiver fifo. */
- nic_bs(2);
- if (nic_inw(NIC_FIFO) & 0x8000) {
- return 0;
- }
- /* Inialize pointer register. */
- nic_outw(NIC_PTR, PTR_READ | PTR_RCV | PTR_AUTO_INCR);
- _NOP();
- _NOP();
- _NOP();
- _NOP();
- /* Read status word and byte count. */
- fsw = nic_inw(NIC_DATA);
- fbc = nic_inw(NIC_DATA);
- //printf("[SW=%04X,BC=%04X]", fsw, fbc);
- /* Check for frame errors. */
- if (fsw & 0xAC00) {
- nb = (NETBUF *) 0xFFFF;
- }
- /* Check the byte count. */
- else if (fbc < 66 || fbc > 1524) {
- nb = (NETBUF *) 0xFFFF;
- }
- else {
- /*
- * Allocate a NETBUF.
- * Hack alert: Rev A chips never set the odd frame indicator.
- */
- fbc -= 3;
- nb = NutNetBufAlloc(0, NBAF_DATALINK, fbc);
- /* Perform the read. */
- if (nb)
- NicRead(nb->nb_dl.vp, fbc);
- }
- /* Release the packet. */
- nic_outlb(NIC_MMUCR, MMU_TOP);
- return nb;
- }
- /*!
- * \brief Load a packet into the nic's transmit ring buffer.
- *
- * Interupts must have been disabled when calling this function.
- *
- * \param nb Network buffer structure containing the packet to be sent.
- * The structure must have been allocated by a previous
- * call NutNetBufAlloc(). This routine will automatically
- * release the buffer in case of an error.
- *
- * \return 0 on success, -1 in case of any errors. Errors
- * will automatically release the network buffer
- * structure.
- */
- static int NicPutPacket(NETBUF * nb)
- {
- uint16_t sz;
- uint8_t odd = 0;
- uint8_t imsk;
- //printf("[P]");
- /*
- * Calculate the number of bytes to be send. Do not send packets
- * larger than the Ethernet maximum transfer unit. The MTU
- * consist of 1500 data bytes plus the 14 byte Ethernet header
- * plus 4 bytes CRC. We check the data bytes only.
- */
- if ((sz = nb->nb_nw.sz + nb->nb_tp.sz + nb->nb_ap.sz) > ETHERMTU)
- return -1;
- /* Disable all interrupts. */
- imsk = nic_inlb(NIC_MSK);
- nic_outlb(NIC_MSK, 0);
- /* Allocate packet buffer space. */
- nic_bs(2);
- nic_outlb(NIC_MMUCR, MMU_ALO);
- if (NicMmuWait(100))
- return -1;
- /* Enable interrupts including allocation success. */
- nic_outlb(NIC_MSK, imsk | INT_ALLOC);
- /* The MMU needs some time. Use it to calculate the byte count. */
- sz += nb->nb_dl.sz;
- sz += 6;
- if (sz & 1) {
- sz++;
- odd++;
- }
- /* Wait for allocation success. */
- while ((nic_inlb(NIC_IST) & INT_ALLOC) == 0) {
- if (NutEventWait(&maq, 125)) {
- nic_outlb(NIC_MMUCR, MMU_RST);
- NicMmuWait(1000);
- nic_outlb(NIC_MMUCR, MMU_ALO);
- if (NicMmuWait(100) || (nic_inlb(NIC_IST) & INT_ALLOC) == 0) {
- if (NutEventWait(&maq, 125)) {
- return -1;
- }
- }
- }
- }
- /* Disable interrupts. */
- imsk = nic_inlb(NIC_MSK);
- nic_outlb(NIC_MSK, 0);
- nic_outlb(NIC_PNR, nic_inhb(NIC_PNR));
- nic_outw(NIC_PTR, 0x4000);
- /* Transfer control word. */
- nic_outlb(NIC_DATA, 0);
- nic_outlb(NIC_DATA, 0);
- /* Transfer the byte count. */
- nic_outw(NIC_DATA, sz);
- /* Transfer the Ethernet frame. */
- NicWrite(nb->nb_dl.vp, nb->nb_dl.sz);
- NicWrite(nb->nb_nw.vp, nb->nb_nw.sz);
- NicWrite(nb->nb_tp.vp, nb->nb_tp.sz);
- NicWrite(nb->nb_ap.vp, nb->nb_ap.sz);
- if (odd)
- nic_outlb(NIC_DATA, 0);
- /* Transfer the control word. */
- nic_outw(NIC_DATA, 0);
- /* Enqueue packet. */
- if (NicMmuWait(100))
- return -1;
- nic_outlb(NIC_MMUCR, MMU_ENQ);
- /* Enable interrupts. */
- imsk |= INT_TX | INT_TX_EMPTY;
- nic_outlb(NIC_MSK, imsk);
- return 0;
- }
- /*! \fn NicRxLanc(void *arg)
- * \brief NIC receiver thread.
- *
- */
- THREAD(NicRxLanc, arg)
- {
- NUTDEVICE *dev;
- IFNET *ifn;
- NICINFO *ni;
- NETBUF *nb;
- uint8_t imsk;
- dev = arg;
- ifn = (IFNET *) dev->dev_icb;
- ni = (NICINFO *) dev->dev_dcb;
- /*
- * This is a temporary hack. Due to a change in initialization,
- * we may not have got a MAC address yet. Wait until a valid one
- * has been set.
- */
- while (!ETHER_IS_UNICAST(ifn->if_mac)) {
- NutSleep(10);
- }
- /*
- * Do not continue unless we managed to start the NIC. We are
- * trapped here if the Ethernet link cannot be established.
- * This happens, for example, if no Ethernet cable is plugged
- * in.
- */
- while (NicStart(ifn->if_mac, ni)) {
- NutSleep(1000);
- }
- LANC111_SIGNAL_MODE();
- sbi(EIMSK, LANC111_SIGNAL_IRQ);
- NutEventPost(&mutex);
- /* Run at high priority. */
- NutThreadSetPriority(9);
- for (;;) {
- /*
- * Wait for the arrival of new packets or
- * check the receiver every two second.
- */
- NutEventWait(&ni->ni_rx_rdy, 2000);
- /*
- * Fetch all packets from the NIC's internal
- * buffer and pass them to the registered handler.
- */
- imsk = nic_inlb(NIC_MSK);
- nic_outlb(NIC_MSK, 0);
- while ((nb = NicGetPacket()) != 0) {
- if (nb != (NETBUF *) 0xFFFF) {
- ni->ni_rx_packets++;
- (*ifn->if_recv) (dev, nb);
- }
- }
- nic_outlb(NIC_MSK, imsk | INT_RCV | INT_ERCV);
- }
- }
- /*!
- * \brief Send Ethernet packet.
- *
- * \param dev Identifies the device to use.
- * \param nb Network buffer structure containing the packet to be sent.
- * The structure must have been allocated by a previous
- * call NutNetBufAlloc().
- *
- * \return 0 on success, -1 in case of any errors.
- */
- int LancOutput(NUTDEVICE * dev, NETBUF * nb)
- {
- static uint32_t mx_wait = 5000;
- int rc = -1;
- NICINFO *ni;
- /*
- * After initialization we are waiting for a long time to give
- * the PHY a chance to establish an Ethernet link.
- */
- if (NutEventWait(&mutex, mx_wait) == 0) {
- ni = (NICINFO *) dev->dev_dcb;
- if (NicPutPacket(nb) == 0) {
- ni->ni_tx_packets++;
- rc = 0;
- /* Ethernet works. Set a long waiting time in case we
- temporarly lose the link next time. */
- mx_wait = 5000;
- }
- NutEventPost(&mutex);
- }
- /*
- * Probably no Ethernet link. Significantly reduce the waiting
- * time, so following transmission will soon return an error.
- */
- else {
- mx_wait = 500;
- }
- return rc;
- }
- /*!
- * \brief Initialize Ethernet hardware.
- *
- * Resets the LAN91C111 Ethernet controller, initializes all required
- * hardware registers and starts a background thread for incoming
- * Ethernet traffic.
- *
- * Applications should do not directly call this function. It is
- * automatically executed during during device registration by
- * NutRegisterDevice().
- *
- * If the network configuration hasn't been set by the application
- * before registering the specified device, this function will
- * call NutNetLoadConfig() to get the MAC address.
- *
- * \param dev Identifies the device to initialize.
- */
- int LancInit(NUTDEVICE * dev)
- {
- /* Disable NIC interrupt and clear NICINFO structure. */
- cbi(EIMSK, LANC111_SIGNAL_IRQ);
- memset(dev->dev_dcb, 0, sizeof(NICINFO));
- /* Register interrupt handler and enable interrupts. */
- if (NutRegisterIrqHandler(&LANC111_SIGNAL, NicInterrupt, dev))
- return -1;
- /*
- * Start the receiver thread.
- *
- * avr-gcc size optimized code used 76 bytes.
- */
- NutThreadCreate("rxi5", NicRxLanc, dev, (NUT_THREAD_LANCRXSTACK * NUT_THREAD_STACK_MULT) + NUT_THREAD_STACK_ADD);
- //NutSleep(500);
- return 0;
- }
- static int LancIOCtl(NUTDEVICE * dev, int req, void *conf)
- {
- int rc = 0;
- IFNET *nif = (IFNET *) dev->dev_icb;
- NICINFO *ni = (NICINFO *) dev->dev_dcb;
- uint32_t i;
- MCASTENTRY *mcast;
- uint8_t mac[6];
- switch (req) {
- /* Set interface address */
- case SIOCSIFADDR:
- /* Set interface hardware address. */
- memcpy(nif->if_mac, conf, sizeof(nif->if_mac));
- break;
- /* Add multicast address */
- case SIOCADDMULTI:
- mac[0] = 0x01;
- mac[1] = 0x00;
- mac[2] = 0x5E;
- mac[3] = ((uint8_t *) conf)[1] & 0x7f;
- mac[4] = ((uint8_t *) conf)[2];
- mac[5] = ((uint8_t *) conf)[3];
- mcast = malloc(sizeof(MCASTENTRY));
- if (mcast != NULL) {
- /*
- * HACK ALERT (MF):
- * I do not know the correct algorithm. The algorithm
- * which works for the dm9000 does not work here.
- * Therefore set all bits to 1.
- */
- for (i = 0; i < 7; i++) {
- ni->ni_mar[i] = 0xFF;
- }
- /* Add new mcast to the mcast list */
- memcpy(mcast->mca_ha, mac, 6);
- mcast->mca_ip = *((uint32_t *) conf);
- mcast->mca_next = nif->if_mcast;
- nif->if_mcast = mcast;
- /* Update the MC hardware */
- NicUpdateMCHardware(ni);
- } else {
- rc = -1;
- }
- break;
- /* Delete multicast address */
- case SIOCDELMULTI:
- /* Will be implemented later */
- rc = -1;
- break;
- default:
- rc = -1;
- break;
- }
- return rc;
- }
- static NICINFO dcb_eth0;
- /*!
- * \brief Network interface information structure.
- *
- * Used to call.
- */
- static IFNET ifn_eth0 = {
- IFT_ETHER, /*!< \brief Interface type. */
- 0, /*!< \brief Interface flags, if_flags. */
- {0, 0, 0, 0, 0, 0}, /*!< \brief Hardware net address. */
- 0, /*!< \brief IP address. */
- 0, /*!< \brief Remote IP address for point to point. */
- 0, /*!< \brief IP network mask. */
- ETHERMTU, /*!< \brief Maximum size of a transmission unit. */
- 0, /*!< \brief Packet identifier. */
- 0, /*!< \brief Linked list of arp entries. */
- 0, /*!< \brief Linked list of multicast address entries, if_mcast. */
- NutEtherInput, /*!< \brief Routine to pass received data to, if_recv(). */
- LancOutput, /*!< \brief Driver output routine, if_send(). */
- NutEtherOutput, /*!< \brief Media output routine, if_output(). */
- NULL /*!< \brief Interface specific control function, if_ioctl(). */
- #ifdef NUT_PERFMON
- , 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
- #endif
- };
- /*!
- * \brief Device information structure.
- *
- * A pointer to this structure must be passed to NutRegisterDevice()
- * to bind this Ethernet device driver to the Nut/OS kernel.
- * An application may then call NutNetIfConfig() with the name \em eth0
- * of this driver to initialize the network interface.
- *
- */
- NUTDEVICE devSmsc111 = {
- 0, /* Pointer to next device. */
- {'e', 't', 'h', '0', 0, 0, 0, 0, 0}, /* Unique device name. */
- IFTYP_NET, /* Type of device. */
- 0, /* Base address. */
- 0, /* First interrupt number. */
- &ifn_eth0, /* Interface control block. */
- &dcb_eth0, /* Driver control block. */
- LancInit, /* Driver initialization routine. */
- LancIOCtl, /* Driver specific control function. */
- 0, /* Read from device. */
- 0, /* Write to device. */
- 0, /* Write from program space data to device. */
- 0, /* Open a device or file. */
- 0, /* Close a device or file. */
- 0, /* Request file size. */
- 0, /* Select function, optional, not yet implemented */
- };
- /*@}*/
|