⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 eth_drv.c

📁 移植到WLIT项目的redboot源代码
💻 C
字号:
//==========================================================================////      eth_drv.c////      Stand-alone hardware independent networking support for RedBoot////==========================================================================//####COPYRIGHTBEGIN####//                                                                          // -------------------------------------------                              // The contents of this file are subject to the Red Hat eCos Public License // Version 1.1 (the "License"); you may not use this file except in         // compliance with the License.  You may obtain a copy of the License at    // http://www.redhat.com/                                                   //                                                                          // Software distributed under the License is distributed on an "AS IS"      // basis, WITHOUT WARRANTY OF ANY KIND, either express or implied.  See the // License for the specific language governing rights and limitations under // the License.                                                             //                                                                          // The Original Code is eCos - Embedded Configurable Operating System,      // released September 30, 1998.                                             //                                                                          // The Initial Developer of the Original Code is Red Hat.                   // Portions created by Red Hat are                                          // Copyright (C) 1998, 1999, 2000 Red Hat, Inc.                             // All Rights Reserved.                                                     // -------------------------------------------                              //                                                                          //####COPYRIGHTEND####//==========================================================================//#####DESCRIPTIONBEGIN####//// Author(s):    gthomas// Contributors: gthomas// Date:         2000-07-14// Purpose:      // Description:  //              // This code is part of RedBoot (tm).////####DESCRIPTIONEND####////==========================================================================#include <pkgconf/system.h>#include <pkgconf/io_eth_drivers.h>#include <cyg/infra/cyg_type.h>#include <cyg/hal/hal_arch.h>#include <cyg/infra/diag.h>#include <cyg/hal/drv_api.h>#include <cyg/hal/hal_if.h>#include <eth_drv.h>#include <netdev.h>// High-level ethernet driver// Define this to allow packets seen by this layer to be passed on to// the previous logical layer, i.e. when stand-alone processing replaces// system [eCos] processing.//// CAUTION! Observations indicate that this cannot work.  When TCP packets// destined for the stand-alone layer are passed up, the eCos stack treats// them as evil and sends "disconnect" responses.//#undef ETH_DRV_PASS_PACKETS#define ETH_DRV_PASS_PACKETS// Interfaces exported to driversstatic void eth_drv_init(struct eth_drv_sc *sc, unsigned char *enaddr);static void eth_drv_recv(struct eth_drv_sc *sc, int total_len);static void eth_drv_tx_done(struct eth_drv_sc *sc, CYG_ADDRWORD key, int status);struct eth_drv_funs eth_drv_funs = {eth_drv_init, eth_drv_recv, eth_drv_tx_done};int net_debug;  // FIXMEunsigned char      __local_enet_addr[ETHER_ADDR_LEN+2];struct eth_drv_sc *__local_enet_sc;#ifdef ETH_DRV_PASS_PACKETS//// Horrible hack: In order to allow the stand-alone networking code to work// alongside eCos (or any other stack), separate IP addresses must be used.// When a packet arrives at the interface, we check to see which IP address// it corresponds to and only pass it "up" if it's not for the stand-alone// layer.//// tres degolas :-(//extern char __local_ip_addr[4]; #endif//// Buffer 'get' support.  The problem is that this function only gets// called when some data is required, but packets may arrive on the device// at any time.  More particularly, on some devices when data arrive, all// of that data needs to be consumed immediately or be lost.  This process// is driven by interrupts, which in the stand-along case are simulated by// calling the "poll" interface.//// Thus there will be a pool of buffers, some free and some full, to try// and manage this.//#define MAX_ETH_MSG 1540#define NUM_ETH_MSG CYGNUM_IO_ETH_DRIVERS_NUM_PKTstruct eth_msg {    struct eth_msg *next, *prev;    int len;   // Actual number of bytes in message    unsigned char data[MAX_ETH_MSG];};struct eth_msg_hdr {    struct eth_msg *first, *last;};static struct eth_msg_hdr eth_msg_free, eth_msg_full;static struct eth_msg eth_msgs[NUM_ETH_MSG];// Prototypes for functions used in this modulestatic void eth_drv_start(struct eth_drv_sc *sc);extern int  start_console(void);extern void end_console(int);extern void printf(char *fmt, ...);extern void dump_buf(CYG_ADDRWORD, int);// Simple queue management functionsstatic voideth_drv_msg_put(struct eth_msg_hdr *hdr, struct eth_msg *msg){    if (hdr->first != (struct eth_msg *)hdr) {        // Something already in queue        hdr->last->next = msg;        msg->prev = hdr->last;        msg->next = (struct eth_msg *)hdr;        hdr->last = msg;    } else {        hdr->first = hdr->last = msg;        msg->next = msg->prev = (struct eth_msg *)hdr;    }}static struct eth_msg *eth_drv_msg_get(struct eth_msg_hdr *hdr){    struct eth_msg *msg;    if (hdr->first != (struct eth_msg *)hdr) {        msg = hdr->first;        hdr->first = msg->next;        msg->next->prev = (struct eth_msg *)hdr;    } else {        msg = (struct eth_msg *)NULL;    }    return msg;}voideth_drv_buffers_init(void){    int i;    struct eth_msg *msg = eth_msgs;    eth_msg_full.first = eth_msg_full.last = (struct eth_msg *)&eth_msg_full;    eth_msg_free.first = eth_msg_free.last = (struct eth_msg *)&eth_msg_free;    for (i = 0;  i < NUM_ETH_MSG;  i++, msg++) {        eth_drv_msg_put(&eth_msg_free, msg);    }}//// This function is called during system initialization to register a// network interface with the system.//static voideth_drv_init(struct eth_drv_sc *sc, unsigned char *enaddr){    // Set up hardware address    memcpy(&sc->sc_arpcom.esa, enaddr, ETHER_ADDR_LEN);    memcpy(__local_enet_addr, enaddr, ETHER_ADDR_LEN);    __local_enet_sc = sc;    eth_drv_start(sc);}#if 0 // Not currently used.  Left in case it's needed in the future//// This [internal] function will be called to stop activity on an interface.//static voideth_drv_stop(struct eth_drv_sc *sc){    (sc->funs->stop)(sc);}#endif//// This [internal] function will be called to start activity on an interface.//static voideth_drv_start(struct eth_drv_sc *sc){    // Perform any hardware initialization    (sc->funs->start)(sc, (unsigned char *)&sc->sc_arpcom.esa, 0);}//// Send a packet of data to the hardware//static int packet_sent;voideth_drv_write(char *eth_hdr, char *buf, int len){    struct eth_drv_sg sg_list[MAX_ETH_DRV_SG];    struct eth_drv_sc *sc = __local_enet_sc;    int sg_len = 2;    void *dbg = CYGACC_CALL_IF_DBG_DATA();    int old_state;    int wait_time = 5;  // Timeout before giving up    if (dbg) {        sc = (struct eth_drv_sc *)dbg;  // Use control from installed driver        sc->funs->eth_drv_old = sc->funs->eth_drv;        sc->funs->eth_drv = &eth_drv_funs;    // Substitute stand-alone driver        old_state = sc->state;        if (!old_state & ETH_DRV_STATE_ACTIVE) {            // This interface not fully initialized, do it now            (sc->funs->start)(sc, (unsigned char *)&__local_enet_addr, 0);            sc->state |= ETH_DRV_STATE_ACTIVE;        }    }    while (!(sc->funs->can_send)(sc)) {        // Give driver a chance to service hardware        (sc->funs->poll)(sc);        CYGACC_CALL_IF_DELAY_US(2*100000);        if (--wait_time <= 0)            goto reset_and_out;  // Give up on sending packet    }    sg_list[0].buf = (CYG_ADDRESS)eth_hdr;    sg_list[0].len = 14;  // FIXME    sg_list[1].buf = (CYG_ADDRESS)buf;    sg_list[1].len = len;    packet_sent = 0;    (sc->funs->send)(sc, sg_list, sg_len, len+14, (CYG_ADDRWORD)&packet_sent);    if (net_debug) {        int old_console;        old_console = start_console();        printf("Ethernet send:\n");        dump_buf((CYG_ADDRWORD)eth_hdr, 14);        dump_buf((CYG_ADDRWORD)buf, len);        end_console(old_console);    }    while (!packet_sent) {        (sc->funs->poll)(sc);    } reset_and_out:       if (dbg) {//        if (!old_state & ETH_DRV_STATE_ACTIVE) {//            // This interface was not fully initialized, shut it back down//            (sc->funs->stop)(sc);//        }        sc->funs->eth_drv = sc->funs->eth_drv_old;        sc->funs->eth_drv_old = (struct eth_drv_funs *)0;    }}//// This function is called from the hardware driver when an output operation// has completed - i.e. the packet has been sent.//static voideth_drv_tx_done(struct eth_drv_sc *sc, CYG_ADDRWORD key, int status){    if ((int *)key == &packet_sent) {        *(int *)key = 1;    } else {        // It's possible that this acknowledgement is for a different        // [logical] driver.  Try and pass it on.#ifdef CYGSEM_IO_ETH_DRIVERS_DEBUG        int old_console;        old_console = start_console();        printf("tx_done for other key: %x\n", key);        end_console(old_console);#endif        (sc->funs->eth_drv_old->tx_done)(sc, key, status);    }}//// Receive one packet of data from the hardware, if available//inteth_drv_read(char *eth_hdr, char *buf, int len){    struct eth_drv_sc *sc = __local_enet_sc;    struct eth_msg *msg;    int res;    void *dbg = CYGACC_CALL_IF_DBG_DATA();    int old_state;    if (dbg) {        sc = (struct eth_drv_sc *)dbg;  // Use control from installed driver        sc->funs->eth_drv_old = sc->funs->eth_drv;        sc->funs->eth_drv = &eth_drv_funs;    // Substitute stand-alone driver        old_state = sc->state;        if (!old_state & ETH_DRV_STATE_ACTIVE) {            // This interface not fully initialized, do it now            (sc->funs->start)(sc, (unsigned char *)&__local_enet_addr, 0);            sc->state |= ETH_DRV_STATE_ACTIVE;        }    }    (sc->funs->poll)(sc);  // Give the driver a chance to fetch packets    msg = eth_drv_msg_get(&eth_msg_full);    if (msg) {        memcpy(eth_hdr, msg->data, 14);        memcpy(buf, &msg->data[14], msg->len-14);        eth_drv_msg_put(&eth_msg_free, msg);        res = msg->len;    } else {        res = 0;  // No packets available    }       if (dbg) {        sc->funs->eth_drv = sc->funs->eth_drv_old;        sc->funs->eth_drv_old = (struct eth_drv_funs *)0;//        if (!old_state & ETH_DRV_STATE_ACTIVE) {//            // This interface was not fully initialized, shut it back down//            (sc->funs->stop)(sc);//        }    }    return res;}#ifdef ETH_DRV_PASS_PACKETS//// This function is called to copy a message up to the next level.// It is only used when this driver has usurped the processing of// network functions.//static unsigned char *eth_drv_copy_recv_buf;static void eth_drv_copy_recv(struct eth_drv_sc *sc,                  struct eth_drv_sg *sg_list,                  int sg_len){    int i;       for (i = 0;  i < sg_len;  i++) {        memcpy((unsigned char *)sg_list[i].buf,                eth_drv_copy_recv_buf, sg_list[i].len);        eth_drv_copy_recv_buf += sg_list[i].len;    }}#endif//// This function is called from a hardware driver to indicate that an input// packet has arrived.  The routine will set up appropriate network resources// to hold the data and call back into the driver to retrieve the data.//static voideth_drv_recv(struct eth_drv_sc *sc, int total_len){    struct eth_drv_sg sg_list[MAX_ETH_DRV_SG];    int               sg_len = 0;    struct eth_msg *msg;    unsigned char *buf;    msg = eth_drv_msg_get(&eth_msg_free);    if (msg) {        buf = msg->data;    } else {#ifdef CYGSEM_IO_ETH_DRIVERS_WARN        int old_console;        old_console = start_console();        printf("%s: packet of %d bytes dropped\n", __FUNCTION__, total_len);        end_console(old_console);#endif        buf = (char *)0;  // Drivers know this means "the bit bucket"    }    sg_list[0].buf = (CYG_ADDRESS)buf;    sg_list[0].len = total_len;    sg_len = 1;    (sc->funs->recv)(sc, sg_list, sg_len);    if (net_debug) {        int old_console;        old_console = start_console();        printf("Ethernet recv:\n");        dump_buf((CYG_ADDRWORD)buf, 14);        dump_buf((CYG_ADDRWORD)buf+14, total_len-14);        end_console(old_console);    }#ifdef ETH_DRV_PASS_PACKETS    if (sc->funs->eth_drv_old != (struct eth_drv_funs *)0) {        void (*hold_recv)(struct eth_drv_sc *sc,                          struct eth_drv_sg *sg_list,                          int sg_len);        // See if this packet was for us.  If not, pass it upwards        // This is a major layering violation!!        if (memcmp(&__local_ip_addr, &buf[14+16], 4)) {            hold_recv = sc->funs->recv;            sc->funs->recv = eth_drv_copy_recv;            eth_drv_copy_recv_buf = buf;            // This calls into the 'other' driver, giving it a chance to            // do something with this data (since it wasn't for us)            (sc->funs->eth_drv_old->recv)(sc, total_len);            sc->funs->recv = hold_recv;        }    }#endif    if (msg) {        msg->len = total_len;        eth_drv_msg_put(&eth_msg_full, msg);#ifdef CYGSEM_IO_ETH_DRIVERS_WARN    } else {        dump_buf(sg_list[0].buf, sg_list[0].len);#endif    }}//// Determine the interrupt vector used by an interface//inteth_drv_int_vector(void){    struct eth_drv_sc *sc = __local_enet_sc;    return sc->funs->int_vector(sc);}void eth_drv_dsr(cyg_vector_t vector,                 cyg_ucount32 count,                 cyg_addrword_t data){    printf( "eth_drv_dsr should not be called: vector %d, data %x\n",            vector, data );}// EOF src/stand_alone/eth_drv.c

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -