syntheth.c
来自「eCos操作系统源码」· C语言 代码 · 共 462 行 · 第 1/2 页
C
462 行
synth_eth_recv(struct eth_drv_sc* sc, struct eth_drv_sg* sg_list, int sg_len){ synth_eth* eth = (synth_eth*)(sc->driver_private); unsigned char* buf = eth->rx_data; int len = eth->rx_len; int i; for (i = 0; i < sg_len; i++) { if (0 == sg_list[i].buf) { break; } else if (len <= sg_list[i].len) { memcpy((void*)sg_list[i].buf, buf, len); break; } else { memcpy((void*)sg_list[i].buf, buf, sg_list[i].len); buf += sg_list[i].len; len -= sg_list[i].len; } }}// The ISR does not have to do anything, the DSR does the real work.static cyg_uint32synth_eth_isr(cyg_vector_t vector, cyg_addrword_t data){ cyg_drv_interrupt_acknowledge(vector); return CYG_ISR_CALL_DSR;}// The DSR also does not have to do very much. The data argument is// actually the eth_drv_sc structure, which must match the vector.// Interrupts only go off when there are pending receives, so set the// rx_pending flag and call the generic DSR to do the real work.static voidsynth_eth_dsr(cyg_vector_t vector, cyg_ucount32 count, cyg_addrword_t data){ struct eth_drv_sc* sc = (struct eth_drv_sc*) data; synth_eth* eth = (synth_eth*)(sc->driver_private); CYG_ASSERT(eth->interrupt == vector, "Interrupt vectors cannot change during a run"); eth->rx_pending = 1; eth_drv_dsr(vector, count, data);}// ----------------------------------------------------------------------------// Delivery. This is invoked by a thread inside the TCP/IP stack, or by// the poll function.static voidsynth_eth_deliver(struct eth_drv_sc* sc){ synth_eth* eth = (synth_eth*)(sc->driver_private); if (eth->tx_done) { eth->tx_done = 0; (*sc->funs->eth_drv->tx_done)(sc, eth->tx_key, 1); } while (eth->rx_pending) { int more = 1; eth->rx_pending = 0; while (more && eth->up && synth_auxiliary_running) { synth_auxiliary_xchgmsg(eth->synth_id, SYNTH_ETH_RX, 0, 0, (void*) 0, 0, &more, eth->rx_data, &(eth->rx_len), ETHERNET_MAXTU); CYG_LOOP_INVARIANT(!more || (0 != eth->rx_len), "Auxiliary must send at least one packet if several are available"); if (eth->rx_len > 0) { CYG_ASSERT((eth->rx_len >= ETHERNET_MINTU) && (eth->rx_len <= ETHERNET_MAXTU), "Only normal-sized ethernet packets are supported"); // Inform higher-level code that data is available. // This should result in a call to recv() with a // suitable sg_list. If out of memory, recv() // will see a null pointer. (*sc->funs->eth_drv->recv)(sc, eth->rx_len); } }; }}// ----------------------------------------------------------------------------// Polling support. Transmits are relatively straightforward because// all the hard work is handled by send(). Receives are rather more// complicated because interrupts are disabled so we never know when// there is really pending data. However deliver() will do the right// thing even if there is no data, so simply faking up an interrupt// is enough. This does mean extra traffic between application and// auxiliary, but polling does rather imply that.static voidsynth_eth_poll(struct eth_drv_sc* sc){ synth_eth* eth = (synth_eth*)(sc->driver_private); if (synth_auxiliary_running && eth->up) { eth->rx_pending = 1; } synth_eth_deliver(sc);}static intsynth_eth_intvector(struct eth_drv_sc* sc){ synth_eth* eth = (synth_eth*)(sc->driver_private); return eth->interrupt;}// ----------------------------------------------------------------------------// ioctl()'s.//// SET_MAC_ADDRESS is not currently implemented, and probably should// not be implemented because the underlying ethernet device may not// support it.//// SET_MC_ALL is supported if the underlying hardware does. This is// needed for IPV6 support. More selective multicasting via// SET_MC_LIST is not supported, because it imposes too heavy a// requirement on the underlying Linux device. SET_MC_LIST can be// used to disable multicast support.//// GET_IF_STATS_UD and GET_IF_STATS are not currently implementedstatic intsynth_eth_ioctl(struct eth_drv_sc* sc, unsigned long key, void* data, int data_length){ synth_eth* eth = (synth_eth*)(sc->driver_private); int result = EINVAL; switch(key) { case ETH_DRV_SET_MC_ALL: { if (eth->multi_supported) { synth_auxiliary_xchgmsg(eth->synth_id, SYNTH_ETH_MULTIALL, 1, 0, (unsigned char*) 0, 0, (void*)0, (unsigned char*)0, (int*)0, 0); result = 0; } break; } case ETH_DRV_SET_MC_LIST: { struct eth_drv_mc_list* mcl = (struct eth_drv_mc_list*) data; if (eth->multi_supported && (0 == mcl->len)) { synth_auxiliary_xchgmsg(eth->synth_id, SYNTH_ETH_MULTIALL, 0, 0, (unsigned char*) 0, 0, (void*)0, (unsigned char*)0, (int*)0, 0); result = 0; } break; } default: break; } return result;}// ----------------------------------------------------------------------------// Starting and stopping an interface. This includes restarting in// promiscuous mode.static voidsynth_eth_start(struct eth_drv_sc* sc, unsigned char* enaddr, int flags){ synth_eth* eth = (synth_eth*)(sc->driver_private); eth->up = 0; eth->rx_pending = 0; if ((-1 != eth->synth_id) && synth_auxiliary_running) { synth_auxiliary_xchgmsg(eth->synth_id, SYNTH_ETH_START, flags & IFF_PROMISC, 0, (void*) 0, 0, (int*)0, (void*) 0, (int*) 0, 0); } eth->up = 1; if (enaddr != (unsigned char*)0) { memcpy(enaddr, eth->MAC, 6); }}static voidsynth_eth_stop(struct eth_drv_sc* sc){ synth_eth* eth = (synth_eth*)(sc->driver_private); eth->up = 0; if ((-1 != eth->synth_id) && synth_auxiliary_running) { synth_auxiliary_xchgmsg(eth->synth_id, SYNTH_ETH_STOP, 0, 0, (void*) 0, 0, (int*) 0, (void*) 0, (int*) 0, 0); } eth->rx_pending = 0;}// ----------------------------------------------------------------------------// Initialization.//// This requires instantiating a device of class ethernet with a specific// device name held in the eth_drv_sc structure. No additional device data// is needed. If a device can be instantiated then the interrupt vector// and the MAC address can be obtained, and an interrupt handler can be// installed.static boolsynth_eth_init(struct cyg_netdevtab_entry* ndp){ bool result = false; struct eth_drv_sc* sc = (struct eth_drv_sc*)(ndp->device_instance); struct synth_eth* eth = (struct synth_eth*)(sc->driver_private); if (synth_auxiliary_running) { eth->synth_id = synth_auxiliary_instantiate("devs/eth/synth/ecosynth", SYNTH_MAKESTRING(CYGPKG_DEVS_ETH_ECOSYNTH), "ethernet", sc->dev_name, (const char*) 0); if (-1 != eth->synth_id) { unsigned char data[7]; result = true; synth_auxiliary_xchgmsg(eth->synth_id, SYNTH_ETH_GETPARAMS, 0, 0, (const unsigned char*) 0, 0, &(eth->interrupt), data, 0, 7); memcpy(eth->MAC, data, 6); eth->multi_supported = data[6]; cyg_drv_interrupt_create(eth->interrupt, 0, (CYG_ADDRWORD) sc, &synth_eth_isr, &synth_eth_dsr, &(eth->interrupt_handle), &(eth->interrupt_data)); cyg_drv_interrupt_attach(eth->interrupt_handle); cyg_drv_interrupt_unmask(eth->interrupt); } } (*sc->funs->eth_drv->init)(sc, eth->MAC);#ifdef CYGPKG_NET if (eth->multi_supported) { sc->sc_arpcom.ac_if.if_flags |= IFF_ALLMULTI; }#endif return result;}
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?