at v2.6.30 348 lines 11 kB view raw
1/**************************************************************************** 2 3 (c) SYSTEC electronic GmbH, D-07973 Greiz, August-Bebel-Str. 29 4 www.systec-electronic.com 5 6 Project: openPOWERLINK 7 8 Description: Virtual Ethernet Driver for Linux 9 10 License: 11 12 Redistribution and use in source and binary forms, with or without 13 modification, are permitted provided that the following conditions 14 are met: 15 16 1. Redistributions of source code must retain the above copyright 17 notice, this list of conditions and the following disclaimer. 18 19 2. Redistributions in binary form must reproduce the above copyright 20 notice, this list of conditions and the following disclaimer in the 21 documentation and/or other materials provided with the distribution. 22 23 3. Neither the name of SYSTEC electronic GmbH nor the names of its 24 contributors may be used to endorse or promote products derived 25 from this software without prior written permission. For written 26 permission, please contact info@systec-electronic.com. 27 28 THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 29 "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 30 LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 31 FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE 32 COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 33 INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 34 BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; 35 LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER 36 CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 37 LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN 38 ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 39 POSSIBILITY OF SUCH DAMAGE. 40 41 Severability Clause: 42 43 If a provision of this License is or becomes illegal, invalid or 44 unenforceable in any jurisdiction, that shall not affect: 45 1. the validity or enforceability in that jurisdiction of any other 46 provision of this License; or 47 2. the validity or enforceability in other jurisdictions of that or 48 any other provision of this License. 49 50 ------------------------------------------------------------------------- 51 52 $RCSfile: VirtualEthernetLinux.c,v $ 53 54 $Author: D.Krueger $ 55 56 $Revision: 1.8 $ $Date: 2008/11/20 17:06:51 $ 57 58 $State: Exp $ 59 60 Build Environment: 61 62 ------------------------------------------------------------------------- 63 64 Revision History: 65 66 2006/06/12 -ar: start of the implementation, version 1.00 67 68 2006/09/18 d.k.: integration into EPL DLLk module 69 70 ToDo: 71 72 void netif_carrier_off(struct net_device *dev); 73 void netif_carrier_on(struct net_device *dev); 74 75****************************************************************************/ 76 77#include <linux/module.h> 78#include <linux/netdevice.h> 79#include <linux/etherdevice.h> 80#include <linux/kernel.h> 81#include <linux/init.h> 82#include <linux/if_arp.h> 83#include <net/arp.h> 84 85#include <net/protocol.h> 86#include <net/pkt_sched.h> 87#include <linux/if_ether.h> 88#include <linux/in.h> 89#include <linux/ip.h> 90#include <linux/udp.h> 91#include <linux/mm.h> 92#include <linux/types.h> 93#include <linux/skbuff.h> /* for struct sk_buff */ 94 95#include "kernel/VirtualEthernet.h" 96#include "kernel/EplDllkCal.h" 97#include "kernel/EplDllk.h" 98 99#if(((EPL_MODULE_INTEGRATION) & (EPL_MODULE_VETH)) != 0) 100 101/***************************************************************************/ 102/* */ 103/* */ 104/* G L O B A L D E F I N I T I O N S */ 105/* */ 106/* */ 107/***************************************************************************/ 108 109//--------------------------------------------------------------------------- 110// const defines 111//--------------------------------------------------------------------------- 112 113#ifndef EPL_VETH_TX_TIMEOUT 114//#define EPL_VETH_TX_TIMEOUT (2*HZ) 115#define EPL_VETH_TX_TIMEOUT 0 // d.k.: we use no timeout 116#endif 117 118//--------------------------------------------------------------------------- 119// local types 120//--------------------------------------------------------------------------- 121 122//--------------------------------------------------------------------------- 123// modul globale vars 124//--------------------------------------------------------------------------- 125 126static struct net_device *pVEthNetDevice_g = NULL; 127 128//--------------------------------------------------------------------------- 129// local function prototypes 130//--------------------------------------------------------------------------- 131 132static int VEthOpen(struct net_device *pNetDevice_p); 133static int VEthClose(struct net_device *pNetDevice_p); 134static int VEthXmit(struct sk_buff *pSkb_p, struct net_device *pNetDevice_p); 135static struct net_device_stats *VEthGetStats(struct net_device *pNetDevice_p); 136static void VEthTimeout(struct net_device *pNetDevice_p); 137static tEplKernel VEthRecvFrame(tEplFrameInfo * pFrameInfo_p); 138 139//=========================================================================// 140// // 141// P U B L I C F U N C T I O N S // 142// // 143//=========================================================================// 144 145//--------------------------------------------------------------------------- 146// 147// Function: 148// 149// Description: 150// 151// 152// 153// Parameters: 154// 155// 156// Returns: 157// 158// 159// State: 160// 161//--------------------------------------------------------------------------- 162 163static int VEthOpen(struct net_device *pNetDevice_p) 164{ 165 tEplKernel Ret = kEplSuccessful; 166 167 //open the device 168// struct net_device_stats* pStats = netdev_priv(pNetDevice_p); 169 170 //start the interface queue for the network subsystem 171 netif_start_queue(pNetDevice_p); 172 173 // register callback function in DLL 174 Ret = EplDllkRegAsyncHandler(VEthRecvFrame); 175 176 EPL_DBGLVL_VETH_TRACE1 177 ("VEthOpen: EplDllkRegAsyncHandler returned 0x%02X\n", Ret); 178 179 return 0; 180} 181 182static int VEthClose(struct net_device *pNetDevice_p) 183{ 184 tEplKernel Ret = kEplSuccessful; 185 186 EPL_DBGLVL_VETH_TRACE0("VEthClose\n"); 187 188 Ret = EplDllkDeregAsyncHandler(VEthRecvFrame); 189 190 //stop the interface queue for the network subsystem 191 netif_stop_queue(pNetDevice_p); 192 return 0; 193} 194 195static int VEthXmit(struct sk_buff *pSkb_p, struct net_device *pNetDevice_p) 196{ 197 tEplKernel Ret = kEplSuccessful; 198 tEplFrameInfo FrameInfo; 199 200 //transmit function 201 struct net_device_stats *pStats = netdev_priv(pNetDevice_p); 202 203 //save timestemp 204 pNetDevice_p->trans_start = jiffies; 205 206 FrameInfo.m_pFrame = (tEplFrame *) pSkb_p->data; 207 FrameInfo.m_uiFrameSize = pSkb_p->len; 208 209 //call send fkt on DLL 210 Ret = EplDllkCalAsyncSend(&FrameInfo, kEplDllAsyncReqPrioGeneric); 211 if (Ret != kEplSuccessful) { 212 EPL_DBGLVL_VETH_TRACE1 213 ("VEthXmit: EplDllkCalAsyncSend returned 0x%02X\n", Ret); 214 netif_stop_queue(pNetDevice_p); 215 goto Exit; 216 } else { 217 EPL_DBGLVL_VETH_TRACE0("VEthXmit: frame passed to DLL\n"); 218 dev_kfree_skb(pSkb_p); 219 220 //set stats for the device 221 pStats->tx_packets++; 222 pStats->tx_bytes += FrameInfo.m_uiFrameSize; 223 } 224 225 Exit: 226 return 0; 227 228} 229 230static struct net_device_stats *VEthGetStats(struct net_device *pNetDevice_p) 231{ 232 EPL_DBGLVL_VETH_TRACE0("VEthGetStats\n"); 233 234 return netdev_priv(pNetDevice_p); 235} 236 237static void VEthTimeout(struct net_device *pNetDevice_p) 238{ 239 EPL_DBGLVL_VETH_TRACE0("VEthTimeout(\n"); 240 241 // $$$ d.k.: move to extra function, which is called by DLL when new space is available in TxFifo 242 if (netif_queue_stopped(pNetDevice_p)) { 243 netif_wake_queue(pNetDevice_p); 244 } 245} 246 247static tEplKernel VEthRecvFrame(tEplFrameInfo * pFrameInfo_p) 248{ 249 tEplKernel Ret = kEplSuccessful; 250 struct net_device *pNetDevice = pVEthNetDevice_g; 251 struct net_device_stats *pStats = netdev_priv(pNetDevice); 252 struct sk_buff *pSkb; 253 254 EPL_DBGLVL_VETH_TRACE1("VEthRecvFrame: FrameSize=%u\n", 255 pFrameInfo_p->m_uiFrameSize); 256 257 pSkb = dev_alloc_skb(pFrameInfo_p->m_uiFrameSize + 2); 258 if (pSkb == NULL) { 259 pStats->rx_dropped++; 260 goto Exit; 261 } 262 pSkb->dev = pNetDevice; 263 264 skb_reserve(pSkb, 2); 265 266 memcpy((void *)skb_put(pSkb, pFrameInfo_p->m_uiFrameSize), 267 pFrameInfo_p->m_pFrame, pFrameInfo_p->m_uiFrameSize); 268 269 pSkb->protocol = eth_type_trans(pSkb, pNetDevice); 270 pSkb->ip_summed = CHECKSUM_UNNECESSARY; 271 272 // call netif_rx with skb 273 netif_rx(pSkb); 274 275 EPL_DBGLVL_VETH_TRACE1("VEthRecvFrame: SrcMAC=0x%llx\n", 276 AmiGetQword48FromBe(pFrameInfo_p->m_pFrame-> 277 m_be_abSrcMac)); 278 279 // update receive statistics 280 pStats->rx_packets++; 281 pStats->rx_bytes += pFrameInfo_p->m_uiFrameSize; 282 283 Exit: 284 return Ret; 285} 286 287static const struct net_device_ops epl_netdev_ops = { 288 .ndo_open = VEthOpen, 289 .ndo_stop = VEthClose, 290 .ndo_get_stats = VEthGetStats, 291 .ndo_start_xmit = VEthXmit, 292 .ndo_tx_timeout = VEthTimeout, 293 .ndo_change_mtu = eth_change_mtu, 294 .ndo_set_mac_address = eth_mac_addr, 295 .ndo_validate_addr = eth_validate_addr, 296}; 297 298tEplKernel VEthAddInstance(tEplDllkInitParam *pInitParam_p) 299{ 300 tEplKernel Ret = kEplSuccessful; 301 302 // allocate net device structure with priv pointing to stats structure 303 pVEthNetDevice_g = 304 alloc_netdev(sizeof(struct net_device_stats), EPL_VETH_NAME, 305 ether_setup); 306// pVEthNetDevice_g = alloc_etherdev(sizeof (struct net_device_stats)); 307 308 if (pVEthNetDevice_g == NULL) { 309 Ret = kEplNoResource; 310 goto Exit; 311 } 312 313 pVEthNetDevice_g->netdev_ops = &epl_netdev_ops; 314 pVEthNetDevice_g->watchdog_timeo = EPL_VETH_TX_TIMEOUT; 315 pVEthNetDevice_g->destructor = free_netdev; 316 317 // copy own MAC address to net device structure 318 memcpy(pVEthNetDevice_g->dev_addr, pInitParam_p->m_be_abSrcMac, 6); 319 320 //register VEth to the network subsystem 321 if (register_netdev(pVEthNetDevice_g)) { 322 EPL_DBGLVL_VETH_TRACE0 323 ("VEthAddInstance: Could not register VEth...\n"); 324 } else { 325 EPL_DBGLVL_VETH_TRACE0 326 ("VEthAddInstance: Register VEth successfull...\n"); 327 } 328 329 Exit: 330 return Ret; 331} 332 333tEplKernel VEthDelInstance(void) 334{ 335 tEplKernel Ret = kEplSuccessful; 336 337 if (pVEthNetDevice_g != NULL) { 338 //unregister VEth from the network subsystem 339 unregister_netdev(pVEthNetDevice_g); 340 // destructor was set to free_netdev, 341 // so we do not need to call free_netdev here 342 pVEthNetDevice_g = NULL; 343 } 344 345 return Ret; 346} 347 348#endif // (((EPL_MODULE_INTEGRATION) & (EPL_MODULE_VETH)) != 0)