fp@1619: /****************************************************************************** fp@1619: * fp@1619: * $Id$ fp@1619: * fp@1619: * Copyright (C) 2006 Florian Pose, Ingenieurgemeinschaft IgH fp@1619: * fp@1619: * This file is part of the IgH EtherCAT Master. fp@1619: * fp@1619: * The IgH EtherCAT Master is free software; you can redistribute it fp@1619: * and/or modify it under the terms of the GNU General Public License fp@1619: * as published by the Free Software Foundation; either version 2 of the fp@1619: * License, or (at your option) any later version. fp@1619: * fp@1619: * The IgH EtherCAT Master is distributed in the hope that it will be fp@1619: * useful, but WITHOUT ANY WARRANTY; without even the implied warranty of fp@1619: * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the fp@1619: * GNU General Public License for more details. fp@1619: * fp@1619: * You should have received a copy of the GNU General Public License fp@1619: * along with the IgH EtherCAT Master; if not, write to the Free Software fp@1619: * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA fp@1619: * fp@1619: * The right to use EtherCAT Technology is granted and comes free of fp@1619: * charge under condition of compatibility of product made by fp@1619: * Licensee. People intending to distribute/sell products based on the fp@1619: * code, have to sign an agreement to guarantee that products using fp@1619: * software based on IgH EtherCAT master stay compatible with the actual fp@1619: * EtherCAT specification (which are released themselves as an open fp@1619: * standard) as the (only) precondition to have the right to use EtherCAT fp@1619: * Technology, IP and trade marks. fp@1619: * fp@1619: *****************************************************************************/ fp@1619: fp@1619: /** fp@1619: \file fp@1619: Ethernet interface for debugging purposes. fp@1619: */ fp@1619: fp@1619: /*****************************************************************************/ fp@1619: fp@1624: #include fp@1619: #include fp@1619: fp@1619: #include "globals.h" fp@1619: #include "debug.h" fp@1619: fp@1619: /*****************************************************************************/ fp@1619: fp@1619: // net_device functions fp@1619: int ec_dbgdev_open(struct net_device *); fp@1619: int ec_dbgdev_stop(struct net_device *); fp@1619: struct net_device_stats *ec_dbgdev_stats(struct net_device *); fp@1619: fp@1619: /*****************************************************************************/ fp@1619: fp@1619: /** fp@1619: Debug constructor. fp@1619: Initializes the debug object, creates a net_device and registeres it. fp@1619: */ fp@1619: fp@1619: int ec_debug_init(ec_debug_t *dbg /**< debug object */) fp@1619: { fp@1619: int result; fp@1619: fp@1619: dbg->opened = 0; fp@1619: memset(&dbg->stats, 0, sizeof(struct net_device_stats)); fp@1619: fp@1619: if (!(dbg->dev = fp@1619: alloc_netdev(sizeof(ec_debug_t *), "ec%d", ether_setup))) { fp@1619: EC_ERR("Unable to allocate net_device for debug object!\n"); fp@1619: goto out_return; fp@1619: } fp@1619: fp@1619: // initialize net_device fp@1619: dbg->dev->open = ec_dbgdev_open; fp@1619: dbg->dev->stop = ec_dbgdev_stop; fp@1619: dbg->dev->get_stats = ec_dbgdev_stats; fp@1619: fp@1619: // initialize private data fp@1619: *((ec_debug_t **) netdev_priv(dbg->dev)) = dbg; fp@1619: fp@1619: // connect the net_device to the kernel fp@1619: if ((result = register_netdev(dbg->dev))) { fp@1619: EC_ERR("Unable to register net_device: error %i\n", result); fp@1619: goto out_free; fp@1619: } fp@1619: fp@1619: return 0; fp@1619: fp@1619: out_free: fp@1619: free_netdev(dbg->dev); fp@1619: dbg->dev = NULL; fp@1619: out_return: fp@1619: return -1; fp@1619: } fp@1619: fp@1619: /*****************************************************************************/ fp@1619: fp@1619: /** fp@1619: Debug destructor. fp@1619: Unregisteres the net_device and frees allocated memory. fp@1619: */ fp@1619: fp@1619: void ec_debug_clear(ec_debug_t *dbg /**< debug object */) fp@1619: { fp@1619: if (dbg->dev) { fp@1619: unregister_netdev(dbg->dev); fp@1619: free_netdev(dbg->dev); fp@1619: } fp@1619: } fp@1619: fp@1619: /*****************************************************************************/ fp@1619: fp@1619: /** fp@1619: Sends frame data to the interface. fp@1619: */ fp@1619: fp@1619: void ec_debug_send(ec_debug_t *dbg, /**< debug object */ fp@1619: const uint8_t *data, /**< frame data */ fp@1619: size_t size /**< size of the frame data */ fp@1619: ) fp@1619: { fp@1619: struct sk_buff *skb; fp@1619: fp@1619: if (!dbg->opened) return; fp@1619: fp@1619: // allocate socket buffer fp@1619: if (!(skb = dev_alloc_skb(size))) { fp@1619: dbg->stats.rx_dropped++; fp@1619: return; fp@1619: } fp@1619: fp@1619: // copy frame contents into socket buffer fp@1619: memcpy(skb_put(skb, size), data, size); fp@1619: fp@1619: // update device statistics fp@1619: dbg->stats.rx_packets++; fp@1619: dbg->stats.rx_bytes += size; fp@1619: fp@1619: // pass socket buffer to network stack fp@1619: skb->dev = dbg->dev; fp@1619: skb->protocol = eth_type_trans(skb, dbg->dev); fp@1619: skb->ip_summed = CHECKSUM_UNNECESSARY; fp@1619: netif_rx(skb); fp@1619: } fp@1619: fp@1619: /****************************************************************************** fp@1619: * NET_DEVICE functions fp@1619: *****************************************************************************/ fp@1619: fp@1619: /** fp@1619: Opens the virtual network device. fp@1619: */ fp@1619: fp@1619: int ec_dbgdev_open(struct net_device *dev /**< debug net_device */) fp@1619: { fp@1619: ec_debug_t *dbg = *((ec_debug_t **) netdev_priv(dev)); fp@1619: dbg->opened = 1; fp@1619: EC_INFO("debug interface %s opened.\n", dev->name); fp@1619: return 0; fp@1619: } fp@1619: fp@1619: /*****************************************************************************/ fp@1619: fp@1619: /** fp@1619: Stops the virtual network device. fp@1619: */ fp@1619: fp@1619: int ec_dbgdev_stop(struct net_device *dev /**< debug net_device */) fp@1619: { fp@1619: ec_debug_t *dbg = *((ec_debug_t **) netdev_priv(dev)); fp@1619: dbg->opened = 0; fp@1619: EC_INFO("debug interface %s stopped.\n", dev->name); fp@1619: return 0; fp@1619: } fp@1619: fp@1619: /*****************************************************************************/ fp@1619: fp@1619: /** fp@1619: Gets statistics about the virtual network device. fp@1619: */ fp@1619: fp@1619: struct net_device_stats *ec_dbgdev_stats(struct net_device *dev fp@1619: /**< debug net_device */) fp@1619: { fp@1619: ec_debug_t *dbg = *((ec_debug_t **) netdev_priv(dev)); fp@1619: return &dbg->stats; fp@1619: } fp@1619: fp@1619: /*****************************************************************************/