blob: 7b7cce1b36e8f31fcaad64d8b35f2113d537efab [file] [log] [blame]
Daniel Krueger9d7164c2008-12-19 11:41:57 -08001/****************************************************************************
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
Daniel Krueger9d7164c2008-12-19 11:41:57 -080062 -------------------------------------------------------------------------
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
Daniel Krueger9d7164c2008-12-19 11:41:57 -080077#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>
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -080093#include <linux/skbuff.h> /* for struct sk_buff */
Daniel Krueger9d7164c2008-12-19 11:41:57 -080094
95#include "kernel/VirtualEthernet.h"
96#include "kernel/EplDllkCal.h"
97#include "kernel/EplDllk.h"
98
Daniel Krueger9d7164c2008-12-19 11:41:57 -080099#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
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800109//---------------------------------------------------------------------------
110// const defines
111//---------------------------------------------------------------------------
112
113#ifndef EPL_VETH_TX_TIMEOUT
114//#define EPL_VETH_TX_TIMEOUT (2*HZ)
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800115#define EPL_VETH_TX_TIMEOUT 0 // d.k.: we use no timeout
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800116#endif
117
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800118//---------------------------------------------------------------------------
119// local types
120//---------------------------------------------------------------------------
121
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800122//---------------------------------------------------------------------------
123// modul globale vars
124//---------------------------------------------------------------------------
125
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800126static struct net_device *pVEthNetDevice_g = NULL;
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800127
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);
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800135static struct net_device_stats *VEthGetStats(struct net_device *pNetDevice_p);
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800136static void VEthTimeout(struct net_device *pNetDevice_p);
137static tEplKernel VEthRecvFrame(tEplFrameInfo * pFrameInfo_p);
138
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800139//=========================================================================//
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{
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800165 tEplKernel Ret = kEplSuccessful;
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800166
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800167 //open the device
Greg Kroah-Hartman4cf7c4c2009-01-05 15:45:21 -0800168// struct net_device_stats* pStats = netdev_priv(pNetDevice_p);
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800169
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800170 //start the interface queue for the network subsystem
171 netif_start_queue(pNetDevice_p);
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800172
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800173 // register callback function in DLL
174 Ret = EplDllkRegAsyncHandler(VEthRecvFrame);
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800175
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800176 EPL_DBGLVL_VETH_TRACE1
177 ("VEthOpen: EplDllkRegAsyncHandler returned 0x%02X\n", Ret);
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800178
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800179 return 0;
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800180}
181
182static int VEthClose(struct net_device *pNetDevice_p)
183{
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800184 tEplKernel Ret = kEplSuccessful;
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800185
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800186 EPL_DBGLVL_VETH_TRACE0("VEthClose\n");
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800187
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800188 Ret = EplDllkDeregAsyncHandler(VEthRecvFrame);
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800189
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800190 //stop the interface queue for the network subsystem
191 netif_stop_queue(pNetDevice_p);
192 return 0;
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800193}
194
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800195static int VEthXmit(struct sk_buff *pSkb_p, struct net_device *pNetDevice_p)
196{
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800197 tEplKernel Ret = kEplSuccessful;
198 tEplFrameInfo FrameInfo;
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800199
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800200 //transmit function
Greg Kroah-Hartman4cf7c4c2009-01-05 15:45:21 -0800201 struct net_device_stats *pStats = netdev_priv(pNetDevice_p);
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800202
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800203 //save timestemp
204 pNetDevice_p->trans_start = jiffies;
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800205
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800206 FrameInfo.m_pFrame = (tEplFrame *) pSkb_p->data;
207 FrameInfo.m_uiFrameSize = pSkb_p->len;
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800208
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800209 //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);
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800219
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800220 //set stats for the device
221 pStats->tx_packets++;
222 pStats->tx_bytes += FrameInfo.m_uiFrameSize;
223 }
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800224
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800225 Exit:
Patrick McHardy6ed10652009-06-23 06:03:08 +0000226 return NETDEV_TX_OK;
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800227
228}
229
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800230static struct net_device_stats *VEthGetStats(struct net_device *pNetDevice_p)
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800231{
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800232 EPL_DBGLVL_VETH_TRACE0("VEthGetStats\n");
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800233
Greg Kroah-Hartman4cf7c4c2009-01-05 15:45:21 -0800234 return netdev_priv(pNetDevice_p);
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800235}
236
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800237static void VEthTimeout(struct net_device *pNetDevice_p)
238{
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800239 EPL_DBGLVL_VETH_TRACE0("VEthTimeout(\n");
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800240
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800241 // $$$ 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 }
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800245}
246
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800247static tEplKernel VEthRecvFrame(tEplFrameInfo * pFrameInfo_p)
248{
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800249 tEplKernel Ret = kEplSuccessful;
250 struct net_device *pNetDevice = pVEthNetDevice_g;
Greg Kroah-Hartman4cf7c4c2009-01-05 15:45:21 -0800251 struct net_device_stats *pStats = netdev_priv(pNetDevice);
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800252 struct sk_buff *pSkb;
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800253
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800254 EPL_DBGLVL_VETH_TRACE1("VEthRecvFrame: FrameSize=%u\n",
255 pFrameInfo_p->m_uiFrameSize);
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800256
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800257 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;
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800263
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800264 skb_reserve(pSkb, 2);
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800265
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800266 memcpy((void *)skb_put(pSkb, pFrameInfo_p->m_uiFrameSize),
267 pFrameInfo_p->m_pFrame, pFrameInfo_p->m_uiFrameSize);
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800268
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800269 pSkb->protocol = eth_type_trans(pSkb, pNetDevice);
270 pSkb->ip_summed = CHECKSUM_UNNECESSARY;
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800271
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800272 // call netif_rx with skb
273 netif_rx(pSkb);
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800274
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800275 EPL_DBGLVL_VETH_TRACE1("VEthRecvFrame: SrcMAC=0x%llx\n",
276 AmiGetQword48FromBe(pFrameInfo_p->m_pFrame->
277 m_be_abSrcMac));
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800278
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800279 // update receive statistics
280 pStats->rx_packets++;
281 pStats->rx_bytes += pFrameInfo_p->m_uiFrameSize;
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800282
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800283 Exit:
284 return Ret;
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800285}
286
Alexander Beregalov0e46ff32009-03-29 19:24:32 +0400287static 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
Greg Kroah-Hartmand10f4692009-03-23 10:45:12 -0700298tEplKernel VEthAddInstance(tEplDllkInitParam *pInitParam_p)
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800299{
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800300 tEplKernel Ret = kEplSuccessful;
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800301
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800302 // 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);
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800306// pVEthNetDevice_g = alloc_etherdev(sizeof (struct net_device_stats));
307
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800308 if (pVEthNetDevice_g == NULL) {
309 Ret = kEplNoResource;
310 goto Exit;
311 }
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800312
Alexander Beregalov0e46ff32009-03-29 19:24:32 +0400313 pVEthNetDevice_g->netdev_ops = &epl_netdev_ops;
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800314 pVEthNetDevice_g->watchdog_timeo = EPL_VETH_TX_TIMEOUT;
315 pVEthNetDevice_g->destructor = free_netdev;
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800316
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800317 // copy own MAC address to net device structure
318 memcpy(pVEthNetDevice_g->dev_addr, pInitParam_p->m_be_abSrcMac, 6);
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800319
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800320 //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 }
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800328
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800329 Exit:
330 return Ret;
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800331}
332
Greg Kroah-Hartmand10f4692009-03-23 10:45:12 -0700333tEplKernel VEthDelInstance(void)
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800334{
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800335 tEplKernel Ret = kEplSuccessful;
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800336
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800337 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 }
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800344
Greg Kroah-Hartman833dfbe2008-12-19 17:11:52 -0800345 return Ret;
Daniel Krueger9d7164c2008-12-19 11:41:57 -0800346}
347
348#endif // (((EPL_MODULE_INTEGRATION) & (EPL_MODULE_VETH)) != 0)