| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783 |
- /*
- * Copyright (C) 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/
- *
- */
- /*
- * $Log$
- * Revision 1.9 2009/02/06 15:37:39 haraldkipp
- * Added stack space multiplier and addend. Adjusted stack space.
- *
- * Revision 1.8 2009/01/17 11:26:37 haraldkipp
- * Getting rid of two remaining BSD types in favor of stdint.
- * Replaced 'u_int' by 'unsinged int' and 'uptr_t' by 'uintptr_t'.
- *
- * Revision 1.7 2008/08/28 11:12:15 haraldkipp
- * Added interface flags, which will be required to implement Ethernet ioctl
- * functions.
- *
- * Revision 1.6 2008/08/11 06:59:05 haraldkipp
- * BSD types replaced by stdint types (feature request #1282721).
- *
- * Revision 1.5 2007/05/02 11:28:47 haraldkipp
- * Added multicast table entry.
- *
- * Revision 1.4 2006/08/31 19:19:55 haraldkipp
- * No time to write comments. ;-)
- *
- * Revision 1.3 2006/07/18 14:01:38 haraldkipp
- * Transmitter buffer handling was buggy and delayed any second packet.
- * Added handling of receiver overflow events.
- *
- * Revision 1.2 2006/07/15 11:11:44 haraldkipp
- * PHY initialization disabled user reset. Packet receiver routine filled
- * wrong buffer and always returned an error.
- * Many thanks to Andras Albert for fixing this.
- *
- * Revision 1.1 2006/07/05 07:38:44 haraldkipp
- * New Ethernet driver for the AT91SAM7X EMAC and the Davicom DM9161A.
- * This driver is not yet finished. Ethernet link auto-negotiation works
- * and receive interrupts are generated when sending packets to the
- * board. But transmitting packets fails, nothing is sent out.
- *
- */
- #include <cfg/os.h>
- #include <arch/arm.h>
- #include <string.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 <netinet/if_ether.h>
- #include <net/ether.h>
- #include <net/if_var.h>
- #include <dev/irqreg.h>
- #include <dev/phy.h>
- #include <dev/at91sam7x_emac.h>
- #define NUTDEBUG
- /* WARNING: Variadic macros are C99 and may fail with C89 compilers. */
- #ifdef NUTDEBUG
- #include <stdio.h>
- #define EMPRINTF(args,...) printf(args,##__VA_ARGS__)
- #else
- #define EMPRINTF(args,...)
- #endif
- #ifndef NUT_THREAD_NICRXSTACK
- #define NUT_THREAD_NICRXSTACK 768
- #endif
- #ifndef EMAC_RX_BUFFERS
- #define EMAC_RX_BUFFERS 32
- #endif
- #define EMAC_RX_BUFSIZ 128
- #define EMAC_TX_BUFFERS 2
- #ifndef EMAC_TX_BUFSIZ
- #define EMAC_TX_BUFSIZ 1536
- #endif
- #define NIC_PHY_ADDR 31
- #define PHY_TXCLK_ISOLATE_BIT 0
- #define PHY_REFCLK_XT2_BIT 0
- #define PHY_TXEN_BIT 1
- #define PHY_TXD0_BIT 2
- #define PHY_TXD1_BIT 3
- #define PHY_CRS_AD4_BIT 4
- #define PHY_RXD0_AD0_BIT 5
- #define PHY_RXD1_AD1_BIT 6
- #define PHY_RXER_RXD4_RPTR_BIT 7
- #define PHY_MDC_BIT 8
- #define PHY_MDIO_BIT 9
- #define PHY_TXD2_BIT 10
- #define PHY_TXD3_BIT 11
- #define PHY_TXER_TXD4_BIT 12
- #define PHY_RXD2_AD2_BIT 13
- #define PHY_RXD3_AD3_BIT 14
- #define PHY_RXDV_TESTMODE_BIT 15
- #define PHY_COL_RMII_BIT 16
- #define PHY_RXCLK_10BTSER_BIT 17
- #define PHY_PWRDN_BIT 18
- #define PHY_MDINTR_BIT 26
- #define PHY_MII_PINS _BV(PHY_REFCLK_XT2_BIT) \
- | _BV(PHY_TXEN_BIT) \
- | _BV(PHY_TXD0_BIT) \
- | _BV(PHY_TXD1_BIT) \
- | _BV(PHY_CRS_AD4_BIT) \
- | _BV(PHY_RXD0_AD0_BIT) \
- | _BV(PHY_RXD1_AD1_BIT) \
- | _BV(PHY_RXER_RXD4_RPTR_BIT) \
- | _BV(PHY_MDC_BIT) \
- | _BV(PHY_MDIO_BIT) \
- | _BV(PHY_TXD2_BIT) \
- | _BV(PHY_TXD3_BIT) \
- | _BV(PHY_TXER_TXD4_BIT) \
- | _BV(PHY_RXD2_AD2_BIT) \
- | _BV(PHY_RXD3_AD3_BIT) \
- | _BV(PHY_RXDV_TESTMODE_BIT) \
- | _BV(PHY_COL_RMII_BIT) \
- | _BV(PHY_RXCLK_10BTSER_BIT)
- /*!
- * \brief Network interface controller information structure.
- */
- struct _EMACINFO {
- #ifdef NUT_PERFMON
- uint32_t ni_rx_packets; /*!< Number of packets received. */
- uint32_t ni_tx_packets; /*!< Number of packets sent. */
- 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. */
- #endif
- HANDLE volatile ni_rx_rdy; /*!< Receiver event queue. */
- HANDLE volatile ni_tx_rdy; /*!< Transmitter event queue. */
- HANDLE ni_mutex; /*!< Access mutex semaphore. */
- volatile int ni_tx_queued; /*!< Number of packets in transmission queue. */
- volatile int ni_tx_quelen; /*!< Number of bytes in transmission queue not sent. */
- volatile int ni_insane; /*!< Set by error detection. */
- int ni_iomode; /*!< 8 or 16 bit access. 32 bit is not supported. */
- };
- /*!
- * \brief Network interface controller information type.
- */
- typedef struct _EMACINFO EMACINFO;
- /*
- * TODO: Buffers and their descriptors should be part of the EMACINFO
- * structure. Actually there will be no dual Ethernet chip (sure?),
- * but just to keep the code clean.
- */
- typedef struct _BufDescriptor {
- unsigned int addr;
- unsigned int stat;
- } BufDescriptor;
- static volatile BufDescriptor txBufTab[2];
- static volatile uint8_t txBuf[EMAC_TX_BUFFERS * EMAC_TX_BUFSIZ] NUT_ALIGNED_TYPE(8);
- static unsigned int txBufIdx;
- static volatile BufDescriptor rxBufTab[EMAC_RX_BUFFERS];
- static volatile uint8_t rxBuf[EMAC_RX_BUFFERS * EMAC_RX_BUFSIZ] NUT_ALIGNED_TYPE(8);
- static unsigned int rxBufIdx;
- #define RXBUF_OWNERSHIP 0x00000001
- #define RXBUF_WRAP 0x00000002
- #define RXBUF_ADDRMASK 0xFFFFFFFC
- #define RXS_BROADCAST_ADDR 0x80000000 /*!< \brief Broadcast address detected. */
- #define RXS_MULTICAST_HASH 0x40000000 /*!< \brief Multicast hash match. */
- #define RXS_UNICAST_HASH 0x20000000 /*!< \brief Unicast hash match. */
- #define RXS_EXTERNAL_ADDR 0x10000000 /*!< \brief External address match. */
- #define RXS_SA1_ADDR 0x04000000 /*!< \brief Specific address register 1 match. */
- #define RXS_SA2_ADDR 0x02000000 /*!< \brief Specific address register 2 match. */
- #define RXS_SA3_ADDR 0x01000000 /*!< \brief Specific address register 3 match. */
- #define RXS_SA4_ADDR 0x00800000 /*!< \brief Specific address register 4 match. */
- #define RXS_TYPE_ID 0x00400000 /*!< \brief Type ID match. */
- #define RXS_VLAN_TAG 0x00200000 /*!< \brief VLAN tag detected. */
- #define RXS_PRIORITY_TAG 0x00100000 /*!< \brief Priority tag detected. */
- #define RXS_VLAN_PRIORITY 0x000E0000 /*!< \brief VLAN priority. */
- #define RXS_CFI_IND 0x00010000 /*!< \brief Concatenation format indicator. */
- #define RXS_EOF 0x00008000 /*!< \brief End of frame. */
- #define RXS_SOF 0x00004000 /*!< \brief Start of frame. */
- #define RXS_RBF_OFFSET 0x00003000 /*!< \brief Receive buffer offset mask. */
- #define RXS_LENGTH_FRAME 0x000007FF /*!< \brief Length of frame including FCS. */
- #define TXS_USED 0x80000000 /*!< \brief Used buffer. */
- #define TXS_WRAP 0x40000000 /*!< \brief Last descriptor. */
- #define TXS_ERROR 0x20000000 /*!< \brief Retry limit exceeded. */
- #define TXS_UNDERRUN 0x10000000 /*!< \brief Transmit underrun. */
- #define TXS_NO_BUFFER 0x08000000 /*!< \brief Buffer exhausted. */
- #define TXS_NO_CRC 0x00010000 /*!< \brief CRC not appended. */
- #define TXS_LAST_BUFF 0x00008000 /*!< \brief Last buffer of frame. */
- /*!
- * \addtogroup xgEmacSam7x
- */
- /*@{*/
- /*!
- * \brief Read contents of PHY register.
- *
- * \param reg PHY register number.
- *
- * \return Contents of the specified register.
- */
- static int phy_inw(uint8_t reg, uint16_t *val)
- {
- /* PHY read command. */
- outr(EMAC_MAN, _BV(30) | _BV(29) | _BV(17) | ((NIC_PHY_ADDR) << 23) | ((reg & 0x1F) << 18));
- /* Wait until PHY logic completed. */
- while ((inr(EMAC_NSR) & EMAC_IDLE) == 0);
- /* Get data from PHY maintenance register. */
- *val = (uint16_t) inr(EMAC_MAN);
- return 0;
- }
- /*!
- * \brief Write value to PHY register.
- *
- * \param reg PHY register number.
- * \param val Value to write.
- */
- static int phy_outw(uint8_t reg, uint16_t *val)
- {
- /* PHY write command. */
- outr(EMAC_MAN, _BV(30) | _BV(28) | _BV(17) | ((NIC_PHY_ADDR) << 23) | ((reg & 0x1F) << 18) | *val);
- /* Wait until PHY logic completed. */
- while ((inr(EMAC_NSR) & EMAC_IDLE) == 0);
- }
- /*!
- * \brief Reset the Ethernet controller.
- *
- * \return 0 on success, -1 otherwise.
- */
- static int EmacReset(void)
- {
- int rc = 0;
- uint16_t phy = 0;
- int link_wait;
- outr(PMC_PCER, _BV(PIOA_ID));
- outr(PMC_PCER, _BV(PIOB_ID));
- outr(PMC_PCER, _BV(EMAC_ID));
- /* Disable RMII and TESTMODE by disabling pull-ups. */
- outr(PIOB_PUDR, _BV(PHY_COL_RMII_BIT) | _BV(PHY_RXDV_TESTMODE_BIT));
- /* Disable PHY power down. */
- outr(PIOB_PER, _BV(PHY_PWRDN_BIT));
- outr(PIOB_OER, _BV(PHY_PWRDN_BIT));
- outr(PIOB_CODR, _BV(PHY_PWRDN_BIT));
- /* Toggle external hardware reset pin. */
- outr(RSTC_MR, RSTC_KEY | 0x00000100 | RSTC_URSTEN);
- outr(RSTC_CR, RSTC_KEY | RSTC_EXTRST);
- while ((inr(RSTC_SR) & RSTC_NRSTL) == 0);
- /* Configure MII port. */
- outr(PIOB_ASR, PHY_MII_PINS);
- outr(PIOB_BSR, 0);
- outr(PIOB_PDR, PHY_MII_PINS);
- /* Enable management port. */
- outr(EMAC_NCR, inr(EMAC_NCR) | EMAC_MPE);
- outr(EMAC_NCFGR, inr(EMAC_NCFGR) | EMAC_CLK_HCLK_32);
- /* Wait for PHY ready. */
- NutDelay(255);
- /* Register PHY */
- rc = NutRegisterPhy( 1, phy_outw, phy_inw);
- EMPRINTF("EMRPHY rc = %d\n", rc);
- /* Clear MII isolate. */
- NutPhyCtl(PHY_CTL_ISOLATE, &phy);
- /* Wait for auto negotiation completed. */
- /* Wait for link. */
- for (link_wait = 20;; link_wait--) {
- NutPhyCtl(PHY_GET_LINK, &phy);
- if (phy==0)
- break;
- }
- if (link_wait == 0) {
- return -1;
- }
- NutSleep(200);
- }
- /* Disable management port. */
- outr(EMAC_NCR, inr(EMAC_NCR) & ~EMAC_MPE);
- /* Enable receive and transmit clocks. */
- outr(EMAC_USRIO, EMAC_CLKEN);
- return rc;
- }
- /*
- * NIC interrupt entry.
- */
- static void EmacInterrupt(void *arg)
- {
- unsigned int isr;
- EMACINFO *ni = (EMACINFO *) ((NUTDEVICE *) arg)->dev_dcb;
- /* Read interrupt status and disable interrupts. */
- isr = inr(EMAC_ISR);
- /* Receiver interrupt. */
- if ((isr & EMAC_RCOMP) != 0 || (isr & EMAC_ROVR) != 0 || (inr(EMAC_RSR) & EMAC_REC) != 0) {
- outr(EMAC_RSR, EMAC_REC);
- NutEventPostFromIrq(&ni->ni_rx_rdy);
- }
- /* Transmitter interrupt. */
- if ((isr & EMAC_TCOMP) != 0 || (inr(EMAC_TSR) & EMAC_COMP) != 0) {
- outr(EMAC_TSR, EMAC_COMP);
- NutEventPostFromIrq(&ni->ni_tx_rdy);
- }
- }
- /*!
- * \brief Fetch the next packet out of the receive buffers.
- *
- * \return 0 on success, -1 otherwise.
- */
- static int EmacGetPacket(EMACINFO * ni, NETBUF ** nbp)
- {
- int rc = -1;
- uint16_t fbc = 0;
- unsigned int i;
- *nbp = NULL;
- /*
- * Search the next frame start. Release any fragment.
- */
- while ((rxBufTab[rxBufIdx].addr & RXBUF_OWNERSHIP) != 0 && (rxBufTab[rxBufIdx].stat & RXS_SOF) == 0) {
- rxBufTab[rxBufIdx].addr &= ~(RXBUF_OWNERSHIP);
- rxBufIdx++;
- if (rxBufIdx >= EMAC_RX_BUFFERS) {
- rxBufIdx = 0;
- }
- }
- /*
- * Determine the size of the next frame.
- */
- i = rxBufIdx;
- while (rxBufTab[i].addr & RXBUF_OWNERSHIP) {
- if ((fbc = rxBufTab[i].stat & RXS_LENGTH_FRAME) != 0) {
- break;
- }
- i++;
- if (i >= EMAC_RX_BUFFERS) {
- i = 0;
- }
- }
- if (fbc) {
- /*
- * Receiving long packets is unexpected. Let's declare the
- * chip insane. Short packets will be handled by the caller.
- */
- if (fbc > EMAC_TX_BUFSIZ) {
- ni->ni_insane = 1;
- } else {
- *nbp = NutNetBufAlloc(0, NBAF_DATALINK, fbc);
- if (*nbp != NULL) {
- uint8_t *bp = (uint8_t *) (* nbp)->nb_dl.vp;
- unsigned int len;
- while (fbc) {
- if (fbc > EMAC_RX_BUFSIZ) {
- len = EMAC_RX_BUFSIZ;
- } else {
- len = fbc;
- }
- memcpy(bp, (void *) (rxBufTab[rxBufIdx].addr & RXBUF_ADDRMASK), len);
- rxBufTab[rxBufIdx].addr &= ~RXBUF_OWNERSHIP;
- rxBufIdx++;
- if (rxBufIdx >= EMAC_RX_BUFFERS) {
- rxBufIdx = 0;
- }
- fbc -= len;
- bp += len;
- }
- rc = 0;
- }
- }
- }
- return rc;
- }
- /*!
- * \brief Load a packet into the nic's transmit ring buffer.
- *
- * \todo This routine simply does not work. Any idea?
- *
- * \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 EmacPutPacket(int bufnum, EMACINFO * ni, NETBUF * nb)
- {
- int rc = -1;
- unsigned int sz;
- uint8_t *buf;
- /*
- * 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;
- }
- sz += nb->nb_dl.sz;
- if (sz & 1) {
- sz++;
- }
- /* Disable EMAC interrupts. */
- NutIrqDisable(&sig_EMAC);
- /* TODO: Check for link. */
- if (ni->ni_insane == 0) {
- buf = (uint8_t *) txBufTab[bufnum].addr;
- memcpy(buf, nb->nb_dl.vp, nb->nb_dl.sz);
- buf += nb->nb_dl.sz;
- memcpy(buf, nb->nb_nw.vp, nb->nb_nw.sz);
- buf += nb->nb_nw.sz;
- memcpy(buf, nb->nb_tp.vp, nb->nb_tp.sz);
- buf += nb->nb_tp.sz;
- memcpy(buf, nb->nb_ap.vp, nb->nb_ap.sz);
- sz |= TXS_LAST_BUFF;
- if (bufnum) {
- sz |= TXS_WRAP;
- }
- txBufTab[bufnum].stat = sz;
- outr(EMAC_NCR, inr(EMAC_NCR) | EMAC_TSTART);
- rc = 0;
- #ifdef NUT_PERFMON
- ni->ni_tx_packets++;
- #endif
- }
- /* Enable EMAC interrupts. */
- NutIrqEnable(&sig_EMAC);
- return rc;
- }
- /*!
- * \brief Fire up the network interface.
- *
- * NIC interrupts must be disabled when calling this function.
- *
- * \param mac Six byte unique MAC address.
- */
- static int EmacStart(const uint8_t * mac)
- {
- int i;
- if (EmacReset()) {
- return -1;
- }
- /* Set local MAC address. */
- outr(EMAC_SA1L, (mac[3] << 24) | (mac[2] << 16) | (mac[1] << 8) | mac[0]);
- outr(EMAC_SA1H, (mac[5] << 8) | mac[4]);
- /* Initialize receive buffer descriptors. */
- for (i = 0; i < EMAC_RX_BUFFERS - 1; i++) {
- rxBufTab[i].addr = (unsigned int) (&rxBuf[i * EMAC_RX_BUFSIZ]) & RXBUF_ADDRMASK;
- }
- rxBufTab[i].addr = ((unsigned int) (&rxBuf[i * EMAC_RX_BUFSIZ]) & RXBUF_ADDRMASK) | RXBUF_WRAP;
- outr(EMAC_RBQP, (unsigned int) rxBufTab);
- /* Initialize transmit buffer descriptors. */
- txBufTab[0].addr = (unsigned int) (&txBuf[0]);
- txBufTab[0].stat = TXS_USED;
- txBufTab[1].addr = (unsigned int) (&txBuf[EMAC_TX_BUFSIZ]);
- txBufTab[1].stat = TXS_USED | TXS_WRAP;
- outr(EMAC_TBQP, (unsigned int) txBufTab);
- /* Clear receiver status. */
- outr(EMAC_RSR, EMAC_OVR | EMAC_REC | EMAC_BNA);
- /* Copy all frames and discard FCS. */
- outr(EMAC_NCFGR, inr(EMAC_NCFGR) | EMAC_CAF | EMAC_DRFCS);
- /* Enable receiver, transmitter and statistics. */
- outr(EMAC_NCR, inr(EMAC_NCR) | EMAC_TE | EMAC_RE | EMAC_WESTAT);
- return 0;
- }
- /*! \fn EmacRxThread(void *arg)
- * \brief NIC receiver thread.
- *
- */
- THREAD(EmacRxThread, arg)
- {
- NUTDEVICE *dev;
- IFNET *ifn;
- EMACINFO *ni;
- NETBUF *nb;
- dev = arg;
- ifn = (IFNET *) dev->dev_icb;
- ni = (EMACINFO *) 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 (EmacStart(ifn->if_mac)) {
- NutSleep(1000);
- }
- /* Initialize the access mutex. */
- NutEventPost(&ni->ni_mutex);
- /* Run at high priority. */
- NutThreadSetPriority(9);
- /* Enable receive interrupts. */
- outr(EMAC_IER, EMAC_RCOMP | EMAC_TCOMP | EMAC_ROVR);
- NutIrqEnable(&sig_EMAC);
- for (;;) {
- /*
- * Wait for the arrival of new packets or poll the receiver
- * every two seconds.
- */
- NutEventWait(&ni->ni_rx_rdy, 2000);
- /*
- * Fetch all packets from the NIC's internal buffer and pass
- * them to the registered handler.
- */
- while (EmacGetPacket(ni, &nb) == 0) {
- /* Discard short packets. */
- if (nb->nb_dl.sz < 60) {
- NutNetBufFree(nb);
- } else {
- (*ifn->if_recv) (dev, nb);
- }
- }
- /* We got a weird chip, try to restart it. */
- while (ni->ni_insane) {
- if (EmacStart(ifn->if_mac) == 0) {
- ni->ni_insane = 0;
- ni->ni_tx_queued = 0;
- ni->ni_tx_quelen = 0;
- NutIrqEnable(&sig_EMAC);
- } else {
- NutSleep(1000);
- }
- }
- }
- }
- /*!
- * \brief Send Ethernet packet.
- *
- * \todo This routine does not work.
- *
- * \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 EmacOutput(NUTDEVICE * dev, NETBUF * nb)
- {
- static uint32_t mx_wait = 5000;
- int rc = -1;
- EMACINFO *ni = (EMACINFO *) dev->dev_dcb;
- /*
- * After initialization we are waiting for a long time to give
- * the PHY a chance to establish an Ethernet link.
- */
- while (rc) {
- if (ni->ni_insane) {
- break;
- }
- if (NutEventWait(&ni->ni_mutex, mx_wait)) {
- break;
- }
- /* Check for packet queue space. */
- if ((txBufTab[txBufIdx].stat & TXS_USED) == 0) {
- if (NutEventWait(&ni->ni_tx_rdy, 500) && (txBufTab[txBufIdx].stat & TXS_USED) == 0) {
- /* No queue space. Release the lock and give up. */
- NutEventPost(&ni->ni_mutex);
- break;
- }
- } else {
- if ((rc = EmacPutPacket(txBufIdx, ni, nb)) == 0) {
- txBufIdx++;
- txBufIdx &= 1;
- }
- }
- NutEventPost(&ni->ni_mutex);
- }
- /*
- * Probably no Ethernet link. Significantly reduce the waiting
- * time, so following transmission will soon return an error.
- */
- if (rc) {
- mx_wait = 500;
- } else {
- /* Ethernet works. Set a long waiting time in case we
- temporarly lose the link next time. */
- mx_wait = 5000;
- }
- return rc;
- }
- /*!
- * \brief Initialize Ethernet hardware.
- *
- * Applications should do not directly call this function. It is
- * automatically executed during during device registration by
- * NutRegisterDevice().
- *
- * \param dev Identifies the device to initialize.
- */
- int EmacInit(NUTDEVICE * dev)
- {
- EMACINFO *ni = (EMACINFO *) dev->dev_dcb;
- /* Reset the controller. */
- if (EmacReset()) {
- return -1;
- }
- /* Clear EMACINFO structure. */
- memset(ni, 0, sizeof(EMACINFO));
- /* Register interrupt handler. */
- if (NutRegisterIrqHandler(&sig_EMAC, EmacInterrupt, dev)) {
- return -1;
- }
- /* Start the receiver thread. */
- if (NutThreadCreate("emacrx", EmacRxThread, dev,
- (NUT_THREAD_NICRXSTACK * NUT_THREAD_STACK_MULT) + NUT_THREAD_STACK_ADD) == NULL) {
- return -1;
- }
- return 0;
- }
- static EMACINFO dcb_eth0;
- /*!
- * \brief Network interface information structure.
- *
- * Used to call.
- */
- static IFNET ifn_eth0 = {
- IFT_ETHER, /*!< \brief Interface type, if_type. */
- 0, /*!< \brief Interface flags, if_flags. */
- {0, 0, 0, 0, 0, 0}, /*!< \brief Hardware net address, if_mac. */
- 0, /*!< \brief IP address, if_local_ip. */
- 0, /*!< \brief Remote IP address for point to point, if_remote_ip. */
- 0, /*!< \brief IP network mask, if_mask. */
- ETHERMTU, /*!< \brief Maximum size of a transmission unit, if_mtu. */
- 0, /*!< \brief Packet identifier, if_pkt_id. */
- 0, /*!< \brief Linked list of arp entries, arpTable. */
- 0, /*!< \brief Linked list of multicast address entries, if_mcast. */
- NutEtherInput, /*!< \brief Routine to pass received data to, if_recv(). */
- EmacOutput, /*!< \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 devAt91Emac = {
- 0, /*!< \brief Pointer to next device. */
- {'e', 't', 'h', '0', 0, 0, 0, 0, 0}, /*!< \brief Unique device name. */
- IFTYP_NET, /*!< \brief Type of device. */
- 0, /*!< \brief Base address. */
- 0, /*!< \brief First interrupt number. */
- &ifn_eth0, /*!< \brief Interface control block. */
- &dcb_eth0, /*!< \brief Driver control block. */
- EmacInit, /*!< \brief Driver initialization routine. */
- 0, /*!< \brief Driver specific control function. */
- 0, /*!< \brief Read from device. */
- 0, /*!< \brief Write to device. */
- #ifdef __HARVARD_ARCH__
- 0, /*!< \brief Write from program space data to device. */
- #endif
- 0, /*!< \brief Open a device or file. */
- 0, /*!< \brief Close a device or file. */
- 0, /*!< \brief Request file size. */
- 0, /*!< \brief Select function, optional, not yet implemented */
- };
- /*@}*/
|