cregit-Linux how code gets into the kernel

Release 4.16 drivers/net/wireless/quantenna/qtnfmac/core.c

/*
 * Copyright (c) 2015-2016 Quantenna Communications, Inc.
 * All rights reserved.
 *
 * This program 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
 * of the License, or (at your option) any later version.
 *
 * This program 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.
 *
 */

#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/if_ether.h>

#include "core.h"
#include "bus.h"
#include "trans.h"
#include "commands.h"
#include "cfg80211.h"
#include "event.h"
#include "util.h"


#define QTNF_DMP_MAX_LEN 48

#define QTNF_PRIMARY_VIF_IDX	0


struct qtnf_frame_meta_info {
	
u8 magic_s;
	
u8 ifidx;
	
u8 macid;
	
u8 magic_e;

} __packed;


struct qtnf_wmac *qtnf_core_get_mac(const struct qtnf_bus *bus, u8 macid) { struct qtnf_wmac *mac = NULL; if (unlikely(macid >= QTNF_MAX_MAC)) { pr_err("invalid MAC index %u\n", macid); return NULL; } mac = bus->mac[macid]; if (unlikely(!mac)) { pr_err("MAC%u: not initialized\n", macid); return NULL; } return mac; }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko76100.00%1100.00%
Total76100.00%1100.00%

/* Netdev handler for open. */
static int qtnf_netdev_open(struct net_device *ndev) { netif_carrier_off(ndev); qtnf_netdev_updown(ndev, 1); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko26100.00%1100.00%
Total26100.00%1100.00%

/* Netdev handler for close. */
static int qtnf_netdev_close(struct net_device *ndev) { netif_carrier_off(ndev); qtnf_virtual_intf_cleanup(ndev); qtnf_netdev_updown(ndev, 0); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko31100.00%1100.00%
Total31100.00%1100.00%

/* Netdev handler for data transmission. */
static int qtnf_netdev_hard_start_xmit(struct sk_buff *skb, struct net_device *ndev) { struct qtnf_vif *vif; struct qtnf_wmac *mac; vif = qtnf_netdev_get_priv(ndev); if (unlikely(skb->dev != ndev)) { pr_err_ratelimited("invalid skb->dev"); dev_kfree_skb_any(skb); return 0; } if (unlikely(vif->wdev.iftype == NL80211_IFTYPE_UNSPECIFIED)) { pr_err_ratelimited("%s: VIF not initialized\n", ndev->name); dev_kfree_skb_any(skb); return 0; } mac = vif->mac; if (unlikely(!mac)) { pr_err_ratelimited("%s: NULL mac pointer", ndev->name); dev_kfree_skb_any(skb); return 0; } if (!skb->len || (skb->len > ETH_FRAME_LEN)) { pr_err_ratelimited("%s: invalid skb len %d\n", ndev->name, skb->len); dev_kfree_skb_any(skb); ndev->stats.tx_dropped++; return 0; } /* tx path is enabled: reset vif timeout */ vif->cons_tx_timeout_cnt = 0; return qtnf_bus_data_tx(mac->bus, skb); }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko186100.00%1100.00%
Total186100.00%1100.00%

/* Netdev handler for getting stats. */
static void qtnf_netdev_get_stats64(struct net_device *ndev, struct rtnl_link_stats64 *stats) { struct qtnf_vif *vif = qtnf_netdev_get_priv(ndev); unsigned int start; int cpu; netdev_stats_to_stats64(stats, &ndev->stats); if (!vif->stats64) return; for_each_possible_cpu(cpu) { struct pcpu_sw_netstats *stats64; u64 rx_packets, rx_bytes; u64 tx_packets, tx_bytes; stats64 = per_cpu_ptr(vif->stats64, cpu); do { start = u64_stats_fetch_begin_irq(&stats64->syncp); rx_packets = stats64->rx_packets; rx_bytes = stats64->rx_bytes; tx_packets = stats64->tx_packets; tx_bytes = stats64->tx_bytes; } while (u64_stats_fetch_retry_irq(&stats64->syncp, start)); stats->rx_packets += rx_packets; stats->rx_bytes += rx_bytes; stats->tx_packets += tx_packets; stats->tx_bytes += tx_bytes; } }

Contributors

PersonTokensPropCommitsCommitProp
Vasily Ulyanov14692.99%150.00%
Igor Mitsyanko117.01%150.00%
Total157100.00%2100.00%

/* Netdev handler for transmission timeout. */
static void qtnf_netdev_tx_timeout(struct net_device *ndev) { struct qtnf_vif *vif = qtnf_netdev_get_priv(ndev); struct qtnf_wmac *mac; struct qtnf_bus *bus; if (unlikely(!vif || !vif->mac || !vif->mac->bus)) return; mac = vif->mac; bus = mac->bus; pr_warn("VIF%u.%u: Tx timeout- %lu\n", mac->macid, vif->vifid, jiffies); qtnf_bus_data_tx_timeout(bus, ndev); ndev->stats.tx_errors++; if (++vif->cons_tx_timeout_cnt > QTNF_TX_TIMEOUT_TRSHLD) { pr_err("Tx timeout threshold exceeded !\n"); pr_err("schedule interface %s reset !\n", netdev_name(ndev)); queue_work(bus->workqueue, &vif->reset_work); } }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko131100.00%1100.00%
Total131100.00%1100.00%

/* Network device ops handlers */ const struct net_device_ops qtnf_netdev_ops = { .ndo_open = qtnf_netdev_open, .ndo_stop = qtnf_netdev_close, .ndo_start_xmit = qtnf_netdev_hard_start_xmit, .ndo_tx_timeout = qtnf_netdev_tx_timeout, .ndo_get_stats64 = qtnf_netdev_get_stats64, };
static int qtnf_mac_init_single_band(struct wiphy *wiphy, struct qtnf_wmac *mac, enum nl80211_band band) { int ret; wiphy->bands[band] = kzalloc(sizeof(*wiphy->bands[band]), GFP_KERNEL); if (!wiphy->bands[band]) return -ENOMEM; wiphy->bands[band]->band = band; ret = qtnf_cmd_band_info_get(mac, wiphy->bands[band]); if (ret) { pr_err("MAC%u: band %u: failed to get chans info: %d\n", mac->macid, band, ret); return ret; } qtnf_band_init_rates(wiphy->bands[band]); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko120100.00%2100.00%
Total120100.00%2100.00%


static int qtnf_mac_init_bands(struct qtnf_wmac *mac) { struct wiphy *wiphy = priv_to_wiphy(mac); int ret = 0; if (mac->macinfo.bands_cap & QLINK_BAND_2GHZ) { ret = qtnf_mac_init_single_band(wiphy, mac, NL80211_BAND_2GHZ); if (ret) goto out; } if (mac->macinfo.bands_cap & QLINK_BAND_5GHZ) { ret = qtnf_mac_init_single_band(wiphy, mac, NL80211_BAND_5GHZ); if (ret) goto out; } if (mac->macinfo.bands_cap & QLINK_BAND_60GHZ) ret = qtnf_mac_init_single_band(wiphy, mac, NL80211_BAND_60GHZ); out: return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko112100.00%1100.00%
Total112100.00%1100.00%


struct qtnf_vif *qtnf_mac_get_free_vif(struct qtnf_wmac *mac) { struct qtnf_vif *vif; int i; for (i = 0; i < QTNF_MAX_INTF; i++) { vif = &mac->iflist[i]; if (vif->wdev.iftype == NL80211_IFTYPE_UNSPECIFIED) return vif; } return NULL; }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko61100.00%1100.00%
Total61100.00%1100.00%


struct qtnf_vif *qtnf_mac_get_base_vif(struct qtnf_wmac *mac) { struct qtnf_vif *vif; vif = &mac->iflist[QTNF_PRIMARY_VIF_IDX]; if (vif->wdev.iftype == NL80211_IFTYPE_UNSPECIFIED) return NULL; return vif; }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko43100.00%1100.00%
Total43100.00%1100.00%


void qtnf_mac_iface_comb_free(struct qtnf_wmac *mac) { struct ieee80211_iface_combination *comb; int i; if (mac->macinfo.if_comb) { for (i = 0; i < mac->macinfo.n_if_comb; i++) { comb = &mac->macinfo.if_comb[i]; kfree(comb->limits); comb->limits = NULL; } kfree(mac->macinfo.if_comb); mac->macinfo.if_comb = NULL; } }

Contributors

PersonTokensPropCommitsCommitProp
Sergey Matyukevich89100.00%1100.00%
Total89100.00%1100.00%


static void qtnf_vif_reset_handler(struct work_struct *work) { struct qtnf_vif *vif = container_of(work, struct qtnf_vif, reset_work); rtnl_lock(); if (vif->wdev.iftype == NL80211_IFTYPE_UNSPECIFIED) { rtnl_unlock(); return; } /* stop tx completely */ netif_tx_stop_all_queues(vif->netdev); if (netif_carrier_ok(vif->netdev)) netif_carrier_off(vif->netdev); qtnf_cfg80211_vif_reset(vif); rtnl_unlock(); }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko77100.00%1100.00%
Total77100.00%1100.00%


static void qtnf_mac_init_primary_intf(struct qtnf_wmac *mac) { struct qtnf_vif *vif = &mac->iflist[QTNF_PRIMARY_VIF_IDX]; vif->wdev.iftype = NL80211_IFTYPE_STATION; vif->bss_priority = QTNF_DEF_BSS_PRIORITY; vif->wdev.wiphy = priv_to_wiphy(mac); INIT_WORK(&vif->reset_work, qtnf_vif_reset_handler); vif->cons_tx_timeout_cnt = 0; }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko65100.00%2100.00%
Total65100.00%2100.00%


static void qtnf_mac_scan_finish(struct qtnf_wmac *mac, bool aborted) { struct cfg80211_scan_info info = { .aborted = aborted, }; mutex_lock(&mac->mac_lock); if (mac->scan_req) { cfg80211_scan_done(mac->scan_req, &info); mac->scan_req = NULL; } mutex_unlock(&mac->mac_lock); }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko65100.00%1100.00%
Total65100.00%1100.00%


void qtnf_scan_done(struct qtnf_wmac *mac, bool aborted) { cancel_delayed_work_sync(&mac->scan_timeout); qtnf_mac_scan_finish(mac, aborted); }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko28100.00%1100.00%
Total28100.00%1100.00%


static void qtnf_mac_scan_timeout(struct work_struct *work) { struct qtnf_wmac *mac = container_of(work, struct qtnf_wmac, scan_timeout.work); pr_warn("MAC%d: scan timed out\n", mac->macid); qtnf_mac_scan_finish(mac, true); }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko44100.00%1100.00%
Total44100.00%1100.00%


static struct qtnf_wmac *qtnf_core_mac_alloc(struct qtnf_bus *bus, unsigned int macid) { struct wiphy *wiphy; struct qtnf_wmac *mac; unsigned int i; wiphy = qtnf_wiphy_allocate(bus); if (!wiphy) return ERR_PTR(-ENOMEM); mac = wiphy_priv(wiphy); mac->macid = macid; mac->bus = bus; for (i = 0; i < QTNF_MAX_INTF; i++) { memset(&mac->iflist[i], 0, sizeof(struct qtnf_vif)); mac->iflist[i].wdev.iftype = NL80211_IFTYPE_UNSPECIFIED; mac->iflist[i].mac = mac; mac->iflist[i].vifid = i; qtnf_sta_list_init(&mac->iflist[i].sta_list); mutex_init(&mac->mac_lock); INIT_DELAYED_WORK(&mac->scan_timeout, qtnf_mac_scan_timeout); mac->iflist[i].stats64 = netdev_alloc_pcpu_stats(struct pcpu_sw_netstats); if (!mac->iflist[i].stats64) pr_warn("VIF%u.%u: per cpu stats allocation failed\n", macid, i); } qtnf_mac_init_primary_intf(mac); bus->mac[macid] = mac; return mac; }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko17076.58%240.00%
Vasily Ulyanov3616.22%120.00%
Sergey Matyukevich156.76%120.00%
Kees Cook10.45%120.00%
Total222100.00%5100.00%

static const struct ethtool_ops qtnf_ethtool_ops = { .get_drvinfo = cfg80211_get_drvinfo, };
int qtnf_core_net_attach(struct qtnf_wmac *mac, struct qtnf_vif *vif, const char *name, unsigned char name_assign_type) { struct wiphy *wiphy = priv_to_wiphy(mac); struct net_device *dev; void *qdev_vif; int ret; dev = alloc_netdev_mqs(sizeof(struct qtnf_vif *), name, name_assign_type, ether_setup, 1, 1); if (!dev) { memset(&vif->wdev, 0, sizeof(vif->wdev)); vif->wdev.iftype = NL80211_IFTYPE_UNSPECIFIED; return -ENOMEM; } vif->netdev = dev; dev->netdev_ops = &qtnf_netdev_ops; dev->needs_free_netdev = true; dev_net_set(dev, wiphy_net(wiphy)); dev->ieee80211_ptr = &vif->wdev; ether_addr_copy(dev->dev_addr, vif->mac_addr); SET_NETDEV_DEV(dev, wiphy_dev(wiphy)); dev->flags |= IFF_BROADCAST | IFF_MULTICAST; dev->watchdog_timeo = QTNF_DEF_WDOG_TIMEOUT; dev->tx_queue_len = 100; dev->ethtool_ops = &qtnf_ethtool_ops; qdev_vif = netdev_priv(dev); *((void **)qdev_vif) = vif; SET_NETDEV_DEV(dev, mac->bus->dev); ret = register_netdevice(dev); if (ret) { free_netdev(dev); vif->wdev.iftype = NL80211_IFTYPE_UNSPECIFIED; } return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko24096.39%133.33%
Vasily Ulyanov72.81%133.33%
David S. Miller20.80%133.33%
Total249100.00%3100.00%


static void qtnf_core_mac_detach(struct qtnf_bus *bus, unsigned int macid) { struct qtnf_wmac *mac; struct wiphy *wiphy; struct qtnf_vif *vif; unsigned int i; enum nl80211_band band; mac = bus->mac[macid]; if (!mac) return; wiphy = priv_to_wiphy(mac); for (i = 0; i < QTNF_MAX_INTF; i++) { vif = &mac->iflist[i]; rtnl_lock(); if (vif->netdev && vif->wdev.iftype != NL80211_IFTYPE_UNSPECIFIED) { qtnf_virtual_intf_cleanup(vif->netdev); qtnf_del_virtual_intf(wiphy, &vif->wdev); } rtnl_unlock(); qtnf_sta_list_free(&vif->sta_list); free_percpu(vif->stats64); } if (mac->wiphy_registered) wiphy_unregister(wiphy); for (band = NL80211_BAND_2GHZ; band < NUM_NL80211_BANDS; ++band) { if (!wiphy->bands[band]) continue; kfree(wiphy->bands[band]->channels); wiphy->bands[band]->n_channels = 0; kfree(wiphy->bands[band]); wiphy->bands[band] = NULL; } qtnf_mac_iface_comb_free(mac); kfree(mac->macinfo.extended_capabilities); kfree(mac->macinfo.extended_capabilities_mask); wiphy_free(wiphy); bus->mac[macid] = NULL; }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko23190.59%125.00%
Vasily Ulyanov239.02%250.00%
Sergey Matyukevich10.39%125.00%
Total255100.00%4100.00%


static int qtnf_core_mac_attach(struct qtnf_bus *bus, unsigned int macid) { struct qtnf_wmac *mac; struct qtnf_vif *vif; int ret; if (!(bus->hw_info.mac_bitmap & BIT(macid))) { pr_info("MAC%u is not active in FW\n", macid); return 0; } mac = qtnf_core_mac_alloc(bus, macid); if (IS_ERR(mac)) { pr_err("MAC%u allocation failed\n", macid); return PTR_ERR(mac); } ret = qtnf_cmd_get_mac_info(mac); if (ret) { pr_err("MAC%u: failed to get info\n", macid); goto error; } vif = qtnf_mac_get_base_vif(mac); if (!vif) { pr_err("MAC%u: primary VIF is not ready\n", macid); ret = -EFAULT; goto error; } ret = qtnf_cmd_send_add_intf(vif, vif->wdev.iftype, vif->mac_addr); if (ret) { pr_err("MAC%u: failed to add VIF\n", macid); goto error; } ret = qtnf_cmd_send_get_phy_params(mac); if (ret) { pr_err("MAC%u: failed to get PHY settings\n", macid); goto error; } ret = qtnf_mac_init_bands(mac); if (ret) { pr_err("MAC%u: failed to init bands\n", macid); goto error; } ret = qtnf_wiphy_register(&bus->hw_info, mac); if (ret) { pr_err("MAC%u: wiphy registration failed\n", macid); goto error; } mac->wiphy_registered = 1; rtnl_lock(); ret = qtnf_core_net_attach(mac, vif, "wlan%d", NET_NAME_ENUM); rtnl_unlock(); if (ret) { pr_err("MAC%u: failed to attach netdev\n", macid); vif->wdev.iftype = NL80211_IFTYPE_UNSPECIFIED; vif->netdev = NULL; goto error; } pr_debug("MAC%u initialized\n", macid); return 0; error: qtnf_core_mac_detach(bus, macid); return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko323100.00%2100.00%
Total323100.00%2100.00%


int qtnf_core_attach(struct qtnf_bus *bus) { unsigned int i; int ret; qtnf_trans_init(bus); bus->fw_state = QTNF_FW_STATE_BOOT_DONE; qtnf_bus_data_rx_start(bus); bus->workqueue = alloc_ordered_workqueue("QTNF_BUS", 0); if (!bus->workqueue) { pr_err("failed to alloc main workqueue\n"); ret = -ENOMEM; goto error; } INIT_WORK(&bus->event_work, qtnf_event_work_handler); ret = qtnf_cmd_send_init_fw(bus); if (ret) { pr_err("failed to init FW: %d\n", ret); goto error; } bus->fw_state = QTNF_FW_STATE_ACTIVE; ret = qtnf_cmd_get_hw_info(bus); if (ret) { pr_err("failed to get HW info: %d\n", ret); goto error; } if (bus->hw_info.ql_proto_ver != QLINK_PROTO_VER) { pr_err("qlink version mismatch %u != %u\n", QLINK_PROTO_VER, bus->hw_info.ql_proto_ver); ret = -EPROTONOSUPPORT; goto error; } if (bus->hw_info.num_mac > QTNF_MAX_MAC) { pr_err("no support for number of MACs=%u\n", bus->hw_info.num_mac); ret = -ERANGE; goto error; } for (i = 0; i < bus->hw_info.num_mac; i++) { ret = qtnf_core_mac_attach(bus, i); if (ret) { pr_err("MAC%u: attach failed: %d\n", i, ret); goto error; } } return 0; error: qtnf_core_detach(bus); return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko251100.00%1100.00%
Total251100.00%1100.00%

EXPORT_SYMBOL_GPL(qtnf_core_attach);
void qtnf_core_detach(struct qtnf_bus *bus) { unsigned int macid; qtnf_bus_data_rx_stop(bus); for (macid = 0; macid < QTNF_MAX_MAC; macid++) qtnf_core_mac_detach(bus, macid); if (bus->fw_state == QTNF_FW_STATE_ACTIVE) qtnf_cmd_send_deinit_fw(bus); bus->fw_state = QTNF_FW_STATE_DEAD; if (bus->workqueue) { flush_workqueue(bus->workqueue); destroy_workqueue(bus->workqueue); } kfree(bus->hw_info.rd); bus->hw_info.rd = NULL; qtnf_trans_free(bus); }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko8583.33%150.00%
Sergey Matyukevich1716.67%150.00%
Total102100.00%2100.00%

EXPORT_SYMBOL_GPL(qtnf_core_detach);
static inline int qtnf_is_frame_meta_magic_valid(struct qtnf_frame_meta_info *m) { return m->magic_s == 0xAB && m->magic_e == 0xBA; }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko25100.00%1100.00%
Total25100.00%1100.00%


struct net_device *qtnf_classify_skb(struct qtnf_bus *bus, struct sk_buff *skb) { struct qtnf_frame_meta_info *meta; struct net_device *ndev = NULL; struct qtnf_wmac *mac; struct qtnf_vif *vif; meta = (struct qtnf_frame_meta_info *) (skb_tail_pointer(skb) - sizeof(*meta)); if (unlikely(!qtnf_is_frame_meta_magic_valid(meta))) { pr_err_ratelimited("invalid magic 0x%x:0x%x\n", meta->magic_s, meta->magic_e); goto out; } if (unlikely(meta->macid >= QTNF_MAX_MAC)) { pr_err_ratelimited("invalid mac(%u)\n", meta->macid); goto out; } if (unlikely(meta->ifidx >= QTNF_MAX_INTF)) { pr_err_ratelimited("invalid vif(%u)\n", meta->ifidx); goto out; } mac = bus->mac[meta->macid]; if (unlikely(!mac)) { pr_err_ratelimited("mac(%d) does not exist\n", meta->macid); goto out; } vif = &mac->iflist[meta->ifidx]; if (unlikely(vif->wdev.iftype == NL80211_IFTYPE_UNSPECIFIED)) { pr_err_ratelimited("vif(%u) does not exists\n", meta->ifidx); goto out; } ndev = vif->netdev; if (unlikely(!ndev)) { pr_err_ratelimited("netdev for wlan%u.%u does not exists\n", meta->macid, meta->ifidx); goto out; } __skb_trim(skb, skb->len - sizeof(*meta)); out: return ndev; }

Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko262100.00%1100.00%
Total262100.00%1100.00%

EXPORT_SYMBOL_GPL(qtnf_classify_skb);
void qtnf_wake_all_queues(struct net_device *ndev) { struct qtnf_vif *vif = qtnf_netdev_get_priv(ndev); struct qtnf_wmac *mac; struct qtnf_bus *bus; int macid; int i; if (unlikely(!vif || !vif->mac || !vif->mac->bus)) return; bus = vif->mac->bus; for (macid = 0; macid < QTNF_MAX_MAC; macid++) { if (!(bus->hw_info.mac_bitmap & BIT(macid))) continue; mac = bus->mac[macid]; for (i = 0; i < QTNF_MAX_INTF; i++) { vif = &mac->iflist[i]; if (vif->netdev && netif_queue_stopped(vif->netdev)) netif_tx_wake_all_queues(vif->netdev); } } }

Contributors

PersonTokensPropCommitsCommitProp
Sergey Matyukevich151100.00%1100.00%
Total151100.00%1100.00%

EXPORT_SYMBOL_GPL(qtnf_wake_all_queues);
void qtnf_update_rx_stats(struct net_device *ndev, const struct sk_buff *skb) { struct qtnf_vif *vif = qtnf_netdev_get_priv(ndev); struct pcpu_sw_netstats *stats64; if (unlikely(!vif || !vif->stats64)) { ndev->stats.rx_packets++; ndev->stats.rx_bytes += skb->len; return; } stats64 = this_cpu_ptr(vif->stats64); u64_stats_update_begin(&stats64->syncp); stats64->rx_packets++; stats64->rx_bytes += skb->len; u64_stats_update_end(&stats64->syncp); }

Contributors

PersonTokensPropCommitsCommitProp
Vasily Ulyanov102100.00%1100.00%
Total102100.00%1100.00%

EXPORT_SYMBOL_GPL(qtnf_update_rx_stats);
void qtnf_update_tx_stats(struct net_device *ndev, const struct sk_buff *skb) { struct qtnf_vif *vif = qtnf_netdev_get_priv(ndev); struct pcpu_sw_netstats *stats64; if (unlikely(!vif || !vif->stats64)) { ndev->stats.tx_packets++; ndev->stats.tx_bytes += skb->len; return; } stats64 = this_cpu_ptr(vif->stats64); u64_stats_update_begin(&stats64->syncp); stats64->tx_packets++; stats64->tx_bytes += skb->len; u64_stats_update_end(&stats64->syncp); }

Contributors

PersonTokensPropCommitsCommitProp
Vasily Ulyanov102100.00%1100.00%
Total102100.00%1100.00%

EXPORT_SYMBOL_GPL(qtnf_update_tx_stats); MODULE_AUTHOR("Quantenna Communications"); MODULE_DESCRIPTION("Quantenna 802.11 wireless LAN FullMAC driver."); MODULE_LICENSE("GPL");

Overall Contributors

PersonTokensPropCommitsCommitProp
Igor Mitsyanko278679.42%430.77%
Vasily Ulyanov44112.57%323.08%
Sergey Matyukevich2787.92%430.77%
David S. Miller20.06%17.69%
Kees Cook10.03%17.69%
Total3508100.00%13100.00%
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.