syntheth.c

来自「eCos操作系统源码」· C语言 代码 · 共 462 行 · 第 1/2 页

C
462
字号
//==========================================================================////      syntheth.c////      Network device driver for the synthetic target////==========================================================================//####ECOSGPLCOPYRIGHTBEGIN####// -------------------------------------------// This file is part of eCos, the Embedded Configurable Operating System.// Copyright (C) 2002, 2003 Bart Veer//// 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 the// copyright holder(s).// -------------------------------------------//####ECOSGPLCOPYRIGHTEND####//==========================================================================//#####DESCRIPTIONBEGIN####//// Author(s):    bartv// Contributors: bartv// Date:         2002-08-07////####DESCRIPTIONEND####//==========================================================================#include <pkgconf/devs_eth_ecosynth.h>#include <cyg/infra/cyg_type.h>#include <cyg/infra/cyg_ass.h>#include <cyg/hal/hal_arch.h>#include <cyg/infra/diag.h>#include <cyg/hal/drv_api.h>#include <errno.h>#include <string.h>#define __ECOS 1#include <sys/types.h>#include <cyg/io/eth/netdev.h>#include <cyg/io/eth/eth_drv.h>#include <cyg/io/eth/eth_drv_stats.h>#ifdef CYGPKG_NET# include <net/if.h>#else# define IFF_PROMISC 0#endif#include <cyg/hal/hal_io.h>#include "protocol.h"// ----------------------------------------------------------------------------// Device instances. The synthetic target ethernet package can support// up to four ethernet devices, eth0 to eth3. A synth_eth structure// holds the data that is specific to a given device. Each device// needs an instance of this structure, followed by ETH_DRV_SC and// NETDEVTAB_ENTRY macros. Another macro SYNTH_ETH_INSTANCE takes// care of all that, to avoid unnecessary duplication of code here.//// NOTE: unfortunately this involves duplicating the eth_hwr_funs// structure. This could be eliminated but it would require bypassing// the ETH_DRV_SC macro.#define ETHERNET_MINTU 14#define ETHERNET_MAXTU 1514typedef struct synth_eth {    int             synth_id;           // Device id within the auxiliary    int             up;                 // Has there been a call to start()?    int             in_send;            // Detect recursive calls    int             tx_done;    unsigned long   tx_key;             // Allow mbuf's to be freed    volatile int    rx_pending;         // There is pending data.    int             rx_len;             // Length of buffered data.    unsigned char   MAC[6];             // Obtained from the underlying ethernet device    cyg_vector_t    interrupt;          // Interrupt number allocated by the auxiliary    int             multi_supported;    // Does the driver support multicasting?    cyg_handle_t    interrupt_handle;   // Allow the ISR and DSR to be installed.    cyg_interrupt   interrupt_data;    unsigned char   tx_data[ETHERNET_MAXTU];    unsigned char   rx_data[ETHERNET_MAXTU];} synth_eth;#define SYNTH_ETH_INSTANCE( _number_)           \static synth_eth synth_eth##_number_ = {        \    synth_id:   -1,                             \    up:          1,                             \    in_send:     0,                             \    tx_done:     0,                             \    tx_key:      0L,                            \    rx_pending:  0,                             \    rx_len:      0,                             \    MAC:         { 0, 0, 0, 0, 0, 0 },          \    interrupt:   0,                             \    interrupt_handle: 0                         \};                                              \ETH_DRV_SC(synth_eth_sc##_number_,              \           (void*) &synth_eth##_number_,        \           "eth" #_number_,                     \           synth_eth_start,                     \           synth_eth_stop,                      \           synth_eth_ioctl,                     \           synth_eth_can_send,                  \           synth_eth_send,                      \           synth_eth_recv,                      \           synth_eth_deliver,                   \           synth_eth_poll,                      \           synth_eth_intvector);                \NETDEVTAB_ENTRY(synth_eth_netdev##_number_,     \                "synth_eth" #_number_,          \                synth_eth_init,                 \                &synth_eth_sc##_number_);#ifdef CYGVAR_DEVS_ETH_ECOSYNTH_ETH0SYNTH_ETH_INSTANCE(0);#endif#ifdef CYGVAR_DEVS_ETH_ECOSYNTH_ETH1SYNTH_ETH_INSTANCE(1);#endif#ifdef CYGVAR_DEVS_ETH_ECOSYNTH_ETH2SYNTH_ETH_INSTANCE(2);#endif#ifdef CYGVAR_DEVS_ETH_ECOSYNTH_ETH3SYNTH_ETH_INSTANCE(3);#endif// ----------------------------------------------------------------------------// Data transmits.//// The eCos application will just send the data to the auxiliary,// which will in turn pass it on to the rawether utility. There is no// need for any response. Flow control is implicit: if the eCos// application tries to send ethernet packets too quickly then those// get passed on to the auxiliary, which in turn will pass them on to// the rawether process. If rawether is still busy with the previous// packet then the auxiliary will block on a pipe write, and in turn// the eCos application will block. As long as rawether manages to// complete its operations reasonably quickly these blocks should not// be noticeable to the user.//// So can_send() should always return true for an interface that is up// and running. The send operation needs to take the sg list, turn it// into a single buffer, and transmit it to the auxiliary. At that// point the transmission is already complete so eth_drv_dsr() should// be called to call deliver() and release the buffer.//// However there are some complications. The first is polled operation,// where eth_drv_dsr() is a no-op and should not really be called at// all because there are no interrupts going off. The second is that// calling eth_drv_dsr() directly will cause recursive operation:// send() -> dsr -> can_send()/send() -> ...// This is a bad idea, so can_send() has to check that we are not// already inside a send(). Data transmission will proceed merrily// once the send has returned.static intsynth_eth_can_send(struct eth_drv_sc* sc){    synth_eth*  eth = (synth_eth*)(sc->driver_private);    return synth_auxiliary_running && eth->up && !eth->in_send && !eth->tx_done;}static voidsynth_eth_send(struct eth_drv_sc* sc,               struct eth_drv_sg* sg_list, int sg_len, int total_len,               unsigned long key){    synth_eth*  eth = (synth_eth*)(sc->driver_private);    CYG_PRECONDITION((total_len >= ETHERNET_MINTU) && (total_len <= ETHERNET_MAXTU), "Only normal-sized ethernet packets are supported");    CYG_PRECONDITION(!eth->in_send && !eth->tx_done, "Ethernet device must not still be in use for transmits");    eth->in_send = 1;    eth->tx_key  = key;    if (synth_auxiliary_running && eth->up) {        int i;        unsigned char* buf = eth->tx_data;        for (i = 0; i < sg_len; i++) {            memcpy(buf, (void*) sg_list[i].buf, sg_list[i].len);            buf += sg_list[i].len;            CYG_LOOP_INVARIANT(buf <= &(eth->tx_data[ETHERNET_MAXTU]), "sg list must not exceed ethernet MTU");        }        CYG_POSTCONDITION(buf == &(eth->tx_data[total_len]), "sg list lengths should match total_len");        synth_auxiliary_xchgmsg(eth->synth_id, SYNTH_ETH_TX, 0, 0, eth->tx_data, total_len,                                (void*) 0, (unsigned char*)0, (int*)0, 0);    }    // The transfer has now completed, one way or another, so inform    // the higher-level code immediately.    eth->tx_done = 1;    eth_drv_dsr(eth->interrupt, 0, (cyg_addrword_t) sc);    eth->in_send = 0;}// ----------------------------------------------------------------------------// Receives.//// These are rather more complicated because there are real interrupts// involved, and polling needs to be supported as well. The actual// transfer of data from auxiliary to eCos happens inside deliver(),// and the data is buffered up in the synth_eth structure. All that// needs to be done here is scatter the existing data into the// sg_list. If higher-level code has run out of space then the// sg_list may contain null pointers.static void

⌨️ 快捷键说明

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