diff options
author | Eric Dumazet <eric.dumazet@gmail.com> | 2010-07-07 17:58:56 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2010-07-07 17:58:56 -0400 |
commit | 28172739f0a276eb8d6ca917b3974c2edb036da3 (patch) | |
tree | b1dc00cfa20c209992e247c6f73601f609f9ca3b /drivers | |
parent | 217d32dc5f299c483ca0d3c8cc6811c72c0339c4 (diff) |
net: fix 64 bit counters on 32 bit arches
There is a small possibility that a reader gets incorrect values on 32
bit arches. SNMP applications could catch incorrect counters when a
32bit high part is changed by another stats consumer/provider.
One way to solve this is to add a rtnl_link_stats64 param to all
ndo_get_stats64() methods, and also add such a parameter to
dev_get_stats().
Rule is that we are not allowed to use dev->stats64 as a temporary
storage for 64bit stats, but a caller provided area (usually on stack)
Old drivers (only providing get_stats() method) need no changes.
Signed-off-by: Eric Dumazet <eric.dumazet@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'drivers')
-rw-r--r-- | drivers/net/bonding/bond_main.c | 64 | ||||
-rw-r--r-- | drivers/net/ixgbe/ixgbe_ethtool.c | 8 | ||||
-rw-r--r-- | drivers/net/loopback.c | 4 | ||||
-rw-r--r-- | drivers/net/macvlan.c | 6 | ||||
-rw-r--r-- | drivers/net/sfc/efx.c | 3 | ||||
-rw-r--r-- | drivers/net/sfc/ethtool.c | 3 | ||||
-rw-r--r-- | drivers/parisc/led.c | 3 | ||||
-rw-r--r-- | drivers/scsi/fcoe/fcoe.c | 3 | ||||
-rw-r--r-- | drivers/staging/batman-adv/hard-interface.c | 3 | ||||
-rw-r--r-- | drivers/usb/gadget/rndis.c | 3 |
10 files changed, 52 insertions, 48 deletions
diff --git a/drivers/net/bonding/bond_main.c b/drivers/net/bonding/bond_main.c index a95a41b74b4e..9bb9bfa225b6 100644 --- a/drivers/net/bonding/bond_main.c +++ b/drivers/net/bonding/bond_main.c | |||
@@ -3804,51 +3804,49 @@ static int bond_close(struct net_device *bond_dev) | |||
3804 | return 0; | 3804 | return 0; |
3805 | } | 3805 | } |
3806 | 3806 | ||
3807 | static struct rtnl_link_stats64 *bond_get_stats(struct net_device *bond_dev) | 3807 | static struct rtnl_link_stats64 *bond_get_stats(struct net_device *bond_dev, |
3808 | struct rtnl_link_stats64 *stats) | ||
3808 | { | 3809 | { |
3809 | struct bonding *bond = netdev_priv(bond_dev); | 3810 | struct bonding *bond = netdev_priv(bond_dev); |
3810 | struct rtnl_link_stats64 *stats = &bond_dev->stats64; | 3811 | struct rtnl_link_stats64 temp; |
3811 | struct rtnl_link_stats64 local_stats; | ||
3812 | struct slave *slave; | 3812 | struct slave *slave; |
3813 | int i; | 3813 | int i; |
3814 | 3814 | ||
3815 | memset(&local_stats, 0, sizeof(local_stats)); | 3815 | memset(stats, 0, sizeof(*stats)); |
3816 | 3816 | ||
3817 | read_lock_bh(&bond->lock); | 3817 | read_lock_bh(&bond->lock); |
3818 | 3818 | ||
3819 | bond_for_each_slave(bond, slave, i) { | 3819 | bond_for_each_slave(bond, slave, i) { |
3820 | const struct rtnl_link_stats64 *sstats = | 3820 | const struct rtnl_link_stats64 *sstats = |
3821 | dev_get_stats(slave->dev); | 3821 | dev_get_stats(slave->dev, &temp); |
3822 | 3822 | ||
3823 | local_stats.rx_packets += sstats->rx_packets; | 3823 | stats->rx_packets += sstats->rx_packets; |
3824 | local_stats.rx_bytes += sstats->rx_bytes; | 3824 | stats->rx_bytes += sstats->rx_bytes; |
3825 | local_stats.rx_errors += sstats->rx_errors; | 3825 | stats->rx_errors += sstats->rx_errors; |
3826 | local_stats.rx_dropped += sstats->rx_dropped; | 3826 | stats->rx_dropped += sstats->rx_dropped; |
3827 | 3827 | ||
3828 | local_stats.tx_packets += sstats->tx_packets; | 3828 | stats->tx_packets += sstats->tx_packets; |
3829 | local_stats.tx_bytes += sstats->tx_bytes; | 3829 | stats->tx_bytes += sstats->tx_bytes; |
3830 | local_stats.tx_errors += sstats->tx_errors; | 3830 | stats->tx_errors += sstats->tx_errors; |
3831 | local_stats.tx_dropped += sstats->tx_dropped; | 3831 | stats->tx_dropped += sstats->tx_dropped; |
3832 | 3832 | ||
3833 | local_stats.multicast += sstats->multicast; | 3833 | stats->multicast += sstats->multicast; |
3834 | local_stats.collisions += sstats->collisions; | 3834 | stats->collisions += sstats->collisions; |
3835 | 3835 | ||
3836 | local_stats.rx_length_errors += sstats->rx_length_errors; | 3836 | stats->rx_length_errors += sstats->rx_length_errors; |
3837 | local_stats.rx_over_errors += sstats->rx_over_errors; | 3837 | stats->rx_over_errors += sstats->rx_over_errors; |
3838 | local_stats.rx_crc_errors += sstats->rx_crc_errors; | 3838 | stats->rx_crc_errors += sstats->rx_crc_errors; |
3839 | local_stats.rx_frame_errors += sstats->rx_frame_errors; | 3839 | stats->rx_frame_errors += sstats->rx_frame_errors; |
3840 | local_stats.rx_fifo_errors += sstats->rx_fifo_errors; | 3840 | stats->rx_fifo_errors += sstats->rx_fifo_errors; |
3841 | local_stats.rx_missed_errors += sstats->rx_missed_errors; | 3841 | stats->rx_missed_errors += sstats->rx_missed_errors; |
3842 | 3842 | ||
3843 | local_stats.tx_aborted_errors += sstats->tx_aborted_errors; | 3843 | stats->tx_aborted_errors += sstats->tx_aborted_errors; |
3844 | local_stats.tx_carrier_errors += sstats->tx_carrier_errors; | 3844 | stats->tx_carrier_errors += sstats->tx_carrier_errors; |
3845 | local_stats.tx_fifo_errors += sstats->tx_fifo_errors; | 3845 | stats->tx_fifo_errors += sstats->tx_fifo_errors; |
3846 | local_stats.tx_heartbeat_errors += sstats->tx_heartbeat_errors; | 3846 | stats->tx_heartbeat_errors += sstats->tx_heartbeat_errors; |
3847 | local_stats.tx_window_errors += sstats->tx_window_errors; | 3847 | stats->tx_window_errors += sstats->tx_window_errors; |
3848 | } | 3848 | } |
3849 | 3849 | ||
3850 | memcpy(stats, &local_stats, sizeof(struct net_device_stats)); | ||
3851 | |||
3852 | read_unlock_bh(&bond->lock); | 3850 | read_unlock_bh(&bond->lock); |
3853 | 3851 | ||
3854 | return stats; | 3852 | return stats; |
diff --git a/drivers/net/ixgbe/ixgbe_ethtool.c b/drivers/net/ixgbe/ixgbe_ethtool.c index b35ef36741ef..da54b38bb480 100644 --- a/drivers/net/ixgbe/ixgbe_ethtool.c +++ b/drivers/net/ixgbe/ixgbe_ethtool.c | |||
@@ -55,7 +55,7 @@ struct ixgbe_stats { | |||
55 | offsetof(struct ixgbe_adapter, m) | 55 | offsetof(struct ixgbe_adapter, m) |
56 | #define IXGBE_NETDEV_STAT(m) NETDEV_STATS, \ | 56 | #define IXGBE_NETDEV_STAT(m) NETDEV_STATS, \ |
57 | sizeof(((struct net_device *)0)->m), \ | 57 | sizeof(((struct net_device *)0)->m), \ |
58 | offsetof(struct net_device, m) | 58 | offsetof(struct net_device, m) - offsetof(struct net_device, stats) |
59 | 59 | ||
60 | static struct ixgbe_stats ixgbe_gstrings_stats[] = { | 60 | static struct ixgbe_stats ixgbe_gstrings_stats[] = { |
61 | {"rx_packets", IXGBE_NETDEV_STAT(stats.rx_packets)}, | 61 | {"rx_packets", IXGBE_NETDEV_STAT(stats.rx_packets)}, |
@@ -998,16 +998,18 @@ static void ixgbe_get_ethtool_stats(struct net_device *netdev, | |||
998 | struct ixgbe_adapter *adapter = netdev_priv(netdev); | 998 | struct ixgbe_adapter *adapter = netdev_priv(netdev); |
999 | u64 *queue_stat; | 999 | u64 *queue_stat; |
1000 | int stat_count = sizeof(struct ixgbe_queue_stats) / sizeof(u64); | 1000 | int stat_count = sizeof(struct ixgbe_queue_stats) / sizeof(u64); |
1001 | struct rtnl_link_stats64 temp; | ||
1002 | const struct rtnl_link_stats64 *net_stats; | ||
1001 | int j, k; | 1003 | int j, k; |
1002 | int i; | 1004 | int i; |
1003 | char *p = NULL; | 1005 | char *p = NULL; |
1004 | 1006 | ||
1005 | ixgbe_update_stats(adapter); | 1007 | ixgbe_update_stats(adapter); |
1006 | dev_get_stats(netdev); | 1008 | net_stats = dev_get_stats(netdev, &temp); |
1007 | for (i = 0; i < IXGBE_GLOBAL_STATS_LEN; i++) { | 1009 | for (i = 0; i < IXGBE_GLOBAL_STATS_LEN; i++) { |
1008 | switch (ixgbe_gstrings_stats[i].type) { | 1010 | switch (ixgbe_gstrings_stats[i].type) { |
1009 | case NETDEV_STATS: | 1011 | case NETDEV_STATS: |
1010 | p = (char *) netdev + | 1012 | p = (char *) net_stats + |
1011 | ixgbe_gstrings_stats[i].stat_offset; | 1013 | ixgbe_gstrings_stats[i].stat_offset; |
1012 | break; | 1014 | break; |
1013 | case IXGBE_STATS: | 1015 | case IXGBE_STATS: |
diff --git a/drivers/net/loopback.c b/drivers/net/loopback.c index 4dd0510d7a99..9a0996795321 100644 --- a/drivers/net/loopback.c +++ b/drivers/net/loopback.c | |||
@@ -98,10 +98,10 @@ static netdev_tx_t loopback_xmit(struct sk_buff *skb, | |||
98 | return NETDEV_TX_OK; | 98 | return NETDEV_TX_OK; |
99 | } | 99 | } |
100 | 100 | ||
101 | static struct rtnl_link_stats64 *loopback_get_stats64(struct net_device *dev) | 101 | static struct rtnl_link_stats64 *loopback_get_stats64(struct net_device *dev, |
102 | struct rtnl_link_stats64 *stats) | ||
102 | { | 103 | { |
103 | const struct pcpu_lstats __percpu *pcpu_lstats; | 104 | const struct pcpu_lstats __percpu *pcpu_lstats; |
104 | struct rtnl_link_stats64 *stats = &dev->stats64; | ||
105 | u64 bytes = 0; | 105 | u64 bytes = 0; |
106 | u64 packets = 0; | 106 | u64 packets = 0; |
107 | u64 drops = 0; | 107 | u64 drops = 0; |
diff --git a/drivers/net/macvlan.c b/drivers/net/macvlan.c index e6d626e78515..6112f1498940 100644 --- a/drivers/net/macvlan.c +++ b/drivers/net/macvlan.c | |||
@@ -431,12 +431,12 @@ static void macvlan_uninit(struct net_device *dev) | |||
431 | free_percpu(vlan->rx_stats); | 431 | free_percpu(vlan->rx_stats); |
432 | } | 432 | } |
433 | 433 | ||
434 | static struct rtnl_link_stats64 *macvlan_dev_get_stats64(struct net_device *dev) | 434 | static struct rtnl_link_stats64 *macvlan_dev_get_stats64(struct net_device *dev, |
435 | struct rtnl_link_stats64 *stats) | ||
435 | { | 436 | { |
436 | struct rtnl_link_stats64 *stats = &dev->stats64; | ||
437 | struct macvlan_dev *vlan = netdev_priv(dev); | 437 | struct macvlan_dev *vlan = netdev_priv(dev); |
438 | 438 | ||
439 | dev_txq_stats_fold(dev, &dev->stats); | 439 | dev_txq_stats_fold(dev, (struct net_device_stats *)stats); |
440 | 440 | ||
441 | if (vlan->rx_stats) { | 441 | if (vlan->rx_stats) { |
442 | struct macvlan_rx_stats *p, accum = {0}; | 442 | struct macvlan_rx_stats *p, accum = {0}; |
diff --git a/drivers/net/sfc/efx.c b/drivers/net/sfc/efx.c index 35b3f2922e5c..ba674c5ca29e 100644 --- a/drivers/net/sfc/efx.c +++ b/drivers/net/sfc/efx.c | |||
@@ -1533,11 +1533,10 @@ static int efx_net_stop(struct net_device *net_dev) | |||
1533 | } | 1533 | } |
1534 | 1534 | ||
1535 | /* Context: process, dev_base_lock or RTNL held, non-blocking. */ | 1535 | /* Context: process, dev_base_lock or RTNL held, non-blocking. */ |
1536 | static struct rtnl_link_stats64 *efx_net_stats(struct net_device *net_dev) | 1536 | static struct rtnl_link_stats64 *efx_net_stats(struct net_device *net_dev, struct rtnl_link_stats64 *stats) |
1537 | { | 1537 | { |
1538 | struct efx_nic *efx = netdev_priv(net_dev); | 1538 | struct efx_nic *efx = netdev_priv(net_dev); |
1539 | struct efx_mac_stats *mac_stats = &efx->mac_stats; | 1539 | struct efx_mac_stats *mac_stats = &efx->mac_stats; |
1540 | struct rtnl_link_stats64 *stats = &net_dev->stats64; | ||
1541 | 1540 | ||
1542 | spin_lock_bh(&efx->stats_lock); | 1541 | spin_lock_bh(&efx->stats_lock); |
1543 | efx->type->update_stats(efx); | 1542 | efx->type->update_stats(efx); |
diff --git a/drivers/net/sfc/ethtool.c b/drivers/net/sfc/ethtool.c index 3b8b0a062749..fd19d6ab97a2 100644 --- a/drivers/net/sfc/ethtool.c +++ b/drivers/net/sfc/ethtool.c | |||
@@ -469,12 +469,13 @@ static void efx_ethtool_get_stats(struct net_device *net_dev, | |||
469 | struct efx_mac_stats *mac_stats = &efx->mac_stats; | 469 | struct efx_mac_stats *mac_stats = &efx->mac_stats; |
470 | struct efx_ethtool_stat *stat; | 470 | struct efx_ethtool_stat *stat; |
471 | struct efx_channel *channel; | 471 | struct efx_channel *channel; |
472 | struct rtnl_link_stats64 temp; | ||
472 | int i; | 473 | int i; |
473 | 474 | ||
474 | EFX_BUG_ON_PARANOID(stats->n_stats != EFX_ETHTOOL_NUM_STATS); | 475 | EFX_BUG_ON_PARANOID(stats->n_stats != EFX_ETHTOOL_NUM_STATS); |
475 | 476 | ||
476 | /* Update MAC and NIC statistics */ | 477 | /* Update MAC and NIC statistics */ |
477 | dev_get_stats(net_dev); | 478 | dev_get_stats(net_dev, &temp); |
478 | 479 | ||
479 | /* Fill detailed statistics buffer */ | 480 | /* Fill detailed statistics buffer */ |
480 | for (i = 0; i < EFX_ETHTOOL_NUM_STATS; i++) { | 481 | for (i = 0; i < EFX_ETHTOOL_NUM_STATS; i++) { |
diff --git a/drivers/parisc/led.c b/drivers/parisc/led.c index 188bc8496a26..18dff43b8bd2 100644 --- a/drivers/parisc/led.c +++ b/drivers/parisc/led.c | |||
@@ -355,12 +355,13 @@ static __inline__ int led_get_net_activity(void) | |||
355 | rcu_read_lock(); | 355 | rcu_read_lock(); |
356 | for_each_netdev_rcu(&init_net, dev) { | 356 | for_each_netdev_rcu(&init_net, dev) { |
357 | const struct net_device_stats *stats; | 357 | const struct net_device_stats *stats; |
358 | struct rtnl_link_stats64 temp; | ||
358 | struct in_device *in_dev = __in_dev_get_rcu(dev); | 359 | struct in_device *in_dev = __in_dev_get_rcu(dev); |
359 | if (!in_dev || !in_dev->ifa_list) | 360 | if (!in_dev || !in_dev->ifa_list) |
360 | continue; | 361 | continue; |
361 | if (ipv4_is_loopback(in_dev->ifa_list->ifa_local)) | 362 | if (ipv4_is_loopback(in_dev->ifa_list->ifa_local)) |
362 | continue; | 363 | continue; |
363 | stats = dev_get_stats(dev); | 364 | stats = dev_get_stats(dev, &temp); |
364 | rx_total += stats->rx_packets; | 365 | rx_total += stats->rx_packets; |
365 | tx_total += stats->tx_packets; | 366 | tx_total += stats->tx_packets; |
366 | } | 367 | } |
diff --git a/drivers/scsi/fcoe/fcoe.c b/drivers/scsi/fcoe/fcoe.c index 44a07593de56..1a429ed6da9d 100644 --- a/drivers/scsi/fcoe/fcoe.c +++ b/drivers/scsi/fcoe/fcoe.c | |||
@@ -2653,6 +2653,7 @@ static void fcoe_get_lesb(struct fc_lport *lport, | |||
2653 | u32 lfc, vlfc, mdac; | 2653 | u32 lfc, vlfc, mdac; |
2654 | struct fcoe_dev_stats *devst; | 2654 | struct fcoe_dev_stats *devst; |
2655 | struct fcoe_fc_els_lesb *lesb; | 2655 | struct fcoe_fc_els_lesb *lesb; |
2656 | struct rtnl_link_stats64 temp; | ||
2656 | struct net_device *netdev = fcoe_netdev(lport); | 2657 | struct net_device *netdev = fcoe_netdev(lport); |
2657 | 2658 | ||
2658 | lfc = 0; | 2659 | lfc = 0; |
@@ -2669,7 +2670,7 @@ static void fcoe_get_lesb(struct fc_lport *lport, | |||
2669 | lesb->lesb_link_fail = htonl(lfc); | 2670 | lesb->lesb_link_fail = htonl(lfc); |
2670 | lesb->lesb_vlink_fail = htonl(vlfc); | 2671 | lesb->lesb_vlink_fail = htonl(vlfc); |
2671 | lesb->lesb_miss_fka = htonl(mdac); | 2672 | lesb->lesb_miss_fka = htonl(mdac); |
2672 | lesb->lesb_fcs_error = htonl(dev_get_stats(netdev)->rx_crc_errors); | 2673 | lesb->lesb_fcs_error = htonl(dev_get_stats(netdev, &temp)->rx_crc_errors); |
2673 | } | 2674 | } |
2674 | 2675 | ||
2675 | /** | 2676 | /** |
diff --git a/drivers/staging/batman-adv/hard-interface.c b/drivers/staging/batman-adv/hard-interface.c index 5ede9c255094..96c86c873011 100644 --- a/drivers/staging/batman-adv/hard-interface.c +++ b/drivers/staging/batman-adv/hard-interface.c | |||
@@ -440,6 +440,7 @@ int batman_skb_recv(struct sk_buff *skb, struct net_device *dev, | |||
440 | struct batman_packet *batman_packet; | 440 | struct batman_packet *batman_packet; |
441 | struct batman_if *batman_if; | 441 | struct batman_if *batman_if; |
442 | struct net_device_stats *stats; | 442 | struct net_device_stats *stats; |
443 | struct rtnl_link_stats64 temp; | ||
443 | int ret; | 444 | int ret; |
444 | 445 | ||
445 | skb = skb_share_check(skb, GFP_ATOMIC); | 446 | skb = skb_share_check(skb, GFP_ATOMIC); |
@@ -468,7 +469,7 @@ int batman_skb_recv(struct sk_buff *skb, struct net_device *dev, | |||
468 | if (batman_if->if_status != IF_ACTIVE) | 469 | if (batman_if->if_status != IF_ACTIVE) |
469 | goto err_free; | 470 | goto err_free; |
470 | 471 | ||
471 | stats = (struct net_device_stats *)dev_get_stats(skb->dev); | 472 | stats = (struct net_device_stats *)dev_get_stats(skb->dev, &temp); |
472 | if (stats) { | 473 | if (stats) { |
473 | stats->rx_packets++; | 474 | stats->rx_packets++; |
474 | stats->rx_bytes += skb->len; | 475 | stats->rx_bytes += skb->len; |
diff --git a/drivers/usb/gadget/rndis.c b/drivers/usb/gadget/rndis.c index fb69b01c8f3a..020fa5a25fda 100644 --- a/drivers/usb/gadget/rndis.c +++ b/drivers/usb/gadget/rndis.c | |||
@@ -171,6 +171,7 @@ gen_ndis_query_resp (int configNr, u32 OID, u8 *buf, unsigned buf_len, | |||
171 | int i, count; | 171 | int i, count; |
172 | rndis_query_cmplt_type *resp; | 172 | rndis_query_cmplt_type *resp; |
173 | struct net_device *net; | 173 | struct net_device *net; |
174 | struct rtnl_link_stats64 temp; | ||
174 | const struct rtnl_link_stats64 *stats; | 175 | const struct rtnl_link_stats64 *stats; |
175 | 176 | ||
176 | if (!r) return -ENOMEM; | 177 | if (!r) return -ENOMEM; |
@@ -194,7 +195,7 @@ gen_ndis_query_resp (int configNr, u32 OID, u8 *buf, unsigned buf_len, | |||
194 | resp->InformationBufferOffset = cpu_to_le32 (16); | 195 | resp->InformationBufferOffset = cpu_to_le32 (16); |
195 | 196 | ||
196 | net = rndis_per_dev_params[configNr].dev; | 197 | net = rndis_per_dev_params[configNr].dev; |
197 | stats = dev_get_stats(net); | 198 | stats = dev_get_stats(net, &temp); |
198 | 199 | ||
199 | switch (OID) { | 200 | switch (OID) { |
200 | 201 | ||