diff options
-rw-r--r-- | drivers/net/mv643xx_eth.c | 120 |
1 files changed, 66 insertions, 54 deletions
diff --git a/drivers/net/mv643xx_eth.c b/drivers/net/mv643xx_eth.c index ddd3e1148662..cc93af002013 100644 --- a/drivers/net/mv643xx_eth.c +++ b/drivers/net/mv643xx_eth.c | |||
@@ -325,7 +325,6 @@ struct mv643xx_eth_private { | |||
325 | 325 | ||
326 | struct net_device *dev; | 326 | struct net_device *dev; |
327 | struct napi_struct napi; | 327 | struct napi_struct napi; |
328 | struct net_device_stats stats; | ||
329 | struct mib_counters mib_counters; | 328 | struct mib_counters mib_counters; |
330 | spinlock_t lock; | 329 | spinlock_t lock; |
331 | /* Size of Tx Ring per queue */ | 330 | /* Size of Tx Ring per queue */ |
@@ -885,55 +884,59 @@ static void update_mib_counters(struct mv643xx_eth_private *mp) | |||
885 | struct mv643xx_eth_stats { | 884 | struct mv643xx_eth_stats { |
886 | char stat_string[ETH_GSTRING_LEN]; | 885 | char stat_string[ETH_GSTRING_LEN]; |
887 | int sizeof_stat; | 886 | int sizeof_stat; |
888 | int stat_offset; | 887 | int netdev_off; |
888 | int mp_off; | ||
889 | }; | 889 | }; |
890 | 890 | ||
891 | #define MV643XX_ETH_STAT(m) FIELD_SIZEOF(struct mv643xx_eth_private, m), \ | 891 | #define SSTAT(m) \ |
892 | offsetof(struct mv643xx_eth_private, m) | 892 | { #m, FIELD_SIZEOF(struct net_device_stats, m), \ |
893 | 893 | offsetof(struct net_device, stats.m), -1 } | |
894 | static const struct mv643xx_eth_stats mv643xx_eth_gstrings_stats[] = { | 894 | |
895 | { "rx_packets", MV643XX_ETH_STAT(stats.rx_packets) }, | 895 | #define MIBSTAT(m) \ |
896 | { "tx_packets", MV643XX_ETH_STAT(stats.tx_packets) }, | 896 | { #m, FIELD_SIZEOF(struct mib_counters, m), \ |
897 | { "rx_bytes", MV643XX_ETH_STAT(stats.rx_bytes) }, | 897 | -1, offsetof(struct mv643xx_eth_private, mib_counters.m) } |
898 | { "tx_bytes", MV643XX_ETH_STAT(stats.tx_bytes) }, | 898 | |
899 | { "rx_errors", MV643XX_ETH_STAT(stats.rx_errors) }, | 899 | static const struct mv643xx_eth_stats mv643xx_eth_stats[] = { |
900 | { "tx_errors", MV643XX_ETH_STAT(stats.tx_errors) }, | 900 | SSTAT(rx_packets), |
901 | { "rx_dropped", MV643XX_ETH_STAT(stats.rx_dropped) }, | 901 | SSTAT(tx_packets), |
902 | { "tx_dropped", MV643XX_ETH_STAT(stats.tx_dropped) }, | 902 | SSTAT(rx_bytes), |
903 | { "good_octets_received", MV643XX_ETH_STAT(mib_counters.good_octets_received) }, | 903 | SSTAT(tx_bytes), |
904 | { "bad_octets_received", MV643XX_ETH_STAT(mib_counters.bad_octets_received) }, | 904 | SSTAT(rx_errors), |
905 | { "internal_mac_transmit_err", MV643XX_ETH_STAT(mib_counters.internal_mac_transmit_err) }, | 905 | SSTAT(tx_errors), |
906 | { "good_frames_received", MV643XX_ETH_STAT(mib_counters.good_frames_received) }, | 906 | SSTAT(rx_dropped), |
907 | { "bad_frames_received", MV643XX_ETH_STAT(mib_counters.bad_frames_received) }, | 907 | SSTAT(tx_dropped), |
908 | { "broadcast_frames_received", MV643XX_ETH_STAT(mib_counters.broadcast_frames_received) }, | 908 | MIBSTAT(good_octets_received), |
909 | { "multicast_frames_received", MV643XX_ETH_STAT(mib_counters.multicast_frames_received) }, | 909 | MIBSTAT(bad_octets_received), |
910 | { "frames_64_octets", MV643XX_ETH_STAT(mib_counters.frames_64_octets) }, | 910 | MIBSTAT(internal_mac_transmit_err), |
911 | { "frames_65_to_127_octets", MV643XX_ETH_STAT(mib_counters.frames_65_to_127_octets) }, | 911 | MIBSTAT(good_frames_received), |
912 | { "frames_128_to_255_octets", MV643XX_ETH_STAT(mib_counters.frames_128_to_255_octets) }, | 912 | MIBSTAT(bad_frames_received), |
913 | { "frames_256_to_511_octets", MV643XX_ETH_STAT(mib_counters.frames_256_to_511_octets) }, | 913 | MIBSTAT(broadcast_frames_received), |
914 | { "frames_512_to_1023_octets", MV643XX_ETH_STAT(mib_counters.frames_512_to_1023_octets) }, | 914 | MIBSTAT(multicast_frames_received), |
915 | { "frames_1024_to_max_octets", MV643XX_ETH_STAT(mib_counters.frames_1024_to_max_octets) }, | 915 | MIBSTAT(frames_64_octets), |
916 | { "good_octets_sent", MV643XX_ETH_STAT(mib_counters.good_octets_sent) }, | 916 | MIBSTAT(frames_65_to_127_octets), |
917 | { "good_frames_sent", MV643XX_ETH_STAT(mib_counters.good_frames_sent) }, | 917 | MIBSTAT(frames_128_to_255_octets), |
918 | { "excessive_collision", MV643XX_ETH_STAT(mib_counters.excessive_collision) }, | 918 | MIBSTAT(frames_256_to_511_octets), |
919 | { "multicast_frames_sent", MV643XX_ETH_STAT(mib_counters.multicast_frames_sent) }, | 919 | MIBSTAT(frames_512_to_1023_octets), |
920 | { "broadcast_frames_sent", MV643XX_ETH_STAT(mib_counters.broadcast_frames_sent) }, | 920 | MIBSTAT(frames_1024_to_max_octets), |
921 | { "unrec_mac_control_received", MV643XX_ETH_STAT(mib_counters.unrec_mac_control_received) }, | 921 | MIBSTAT(good_octets_sent), |
922 | { "fc_sent", MV643XX_ETH_STAT(mib_counters.fc_sent) }, | 922 | MIBSTAT(good_frames_sent), |
923 | { "good_fc_received", MV643XX_ETH_STAT(mib_counters.good_fc_received) }, | 923 | MIBSTAT(excessive_collision), |
924 | { "bad_fc_received", MV643XX_ETH_STAT(mib_counters.bad_fc_received) }, | 924 | MIBSTAT(multicast_frames_sent), |
925 | { "undersize_received", MV643XX_ETH_STAT(mib_counters.undersize_received) }, | 925 | MIBSTAT(broadcast_frames_sent), |
926 | { "fragments_received", MV643XX_ETH_STAT(mib_counters.fragments_received) }, | 926 | MIBSTAT(unrec_mac_control_received), |
927 | { "oversize_received", MV643XX_ETH_STAT(mib_counters.oversize_received) }, | 927 | MIBSTAT(fc_sent), |
928 | { "jabber_received", MV643XX_ETH_STAT(mib_counters.jabber_received) }, | 928 | MIBSTAT(good_fc_received), |
929 | { "mac_receive_error", MV643XX_ETH_STAT(mib_counters.mac_receive_error) }, | 929 | MIBSTAT(bad_fc_received), |
930 | { "bad_crc_event", MV643XX_ETH_STAT(mib_counters.bad_crc_event) }, | 930 | MIBSTAT(undersize_received), |
931 | { "collision", MV643XX_ETH_STAT(mib_counters.collision) }, | 931 | MIBSTAT(fragments_received), |
932 | { "late_collision", MV643XX_ETH_STAT(mib_counters.late_collision) }, | 932 | MIBSTAT(oversize_received), |
933 | MIBSTAT(jabber_received), | ||
934 | MIBSTAT(mac_receive_error), | ||
935 | MIBSTAT(bad_crc_event), | ||
936 | MIBSTAT(collision), | ||
937 | MIBSTAT(late_collision), | ||
933 | }; | 938 | }; |
934 | 939 | ||
935 | #define MV643XX_ETH_STATS_LEN ARRAY_SIZE(mv643xx_eth_gstrings_stats) | ||
936 | |||
937 | static int mv643xx_eth_get_settings(struct net_device *dev, struct ethtool_cmd *cmd) | 940 | static int mv643xx_eth_get_settings(struct net_device *dev, struct ethtool_cmd *cmd) |
938 | { | 941 | { |
939 | struct mv643xx_eth_private *mp = netdev_priv(dev); | 942 | struct mv643xx_eth_private *mp = netdev_priv(dev); |
@@ -969,7 +972,7 @@ static void mv643xx_eth_get_drvinfo(struct net_device *netdev, | |||
969 | strncpy(drvinfo->version, mv643xx_eth_driver_version, 32); | 972 | strncpy(drvinfo->version, mv643xx_eth_driver_version, 32); |
970 | strncpy(drvinfo->fw_version, "N/A", 32); | 973 | strncpy(drvinfo->fw_version, "N/A", 32); |
971 | strncpy(drvinfo->bus_info, "mv643xx", 32); | 974 | strncpy(drvinfo->bus_info, "mv643xx", 32); |
972 | drvinfo->n_stats = MV643XX_ETH_STATS_LEN; | 975 | drvinfo->n_stats = ARRAY_SIZE(mv643xx_eth_stats); |
973 | } | 976 | } |
974 | 977 | ||
975 | static int mv643xx_eth_nway_restart(struct net_device *dev) | 978 | static int mv643xx_eth_nway_restart(struct net_device *dev) |
@@ -993,9 +996,9 @@ static void mv643xx_eth_get_strings(struct net_device *netdev, uint32_t stringse | |||
993 | 996 | ||
994 | switch(stringset) { | 997 | switch(stringset) { |
995 | case ETH_SS_STATS: | 998 | case ETH_SS_STATS: |
996 | for (i=0; i < MV643XX_ETH_STATS_LEN; i++) { | 999 | for (i=0; i < ARRAY_SIZE(mv643xx_eth_stats); i++) { |
997 | memcpy(data + i * ETH_GSTRING_LEN, | 1000 | memcpy(data + i * ETH_GSTRING_LEN, |
998 | mv643xx_eth_gstrings_stats[i].stat_string, | 1001 | mv643xx_eth_stats[i].stat_string, |
999 | ETH_GSTRING_LEN); | 1002 | ETH_GSTRING_LEN); |
1000 | } | 1003 | } |
1001 | break; | 1004 | break; |
@@ -1010,10 +1013,19 @@ static void mv643xx_eth_get_ethtool_stats(struct net_device *netdev, | |||
1010 | 1013 | ||
1011 | update_mib_counters(mp); | 1014 | update_mib_counters(mp); |
1012 | 1015 | ||
1013 | for (i = 0; i < MV643XX_ETH_STATS_LEN; i++) { | 1016 | for (i = 0; i < ARRAY_SIZE(mv643xx_eth_stats); i++) { |
1014 | char *p = (char *)mp+mv643xx_eth_gstrings_stats[i].stat_offset; | 1017 | const struct mv643xx_eth_stats *stat; |
1015 | data[i] = (mv643xx_eth_gstrings_stats[i].sizeof_stat == | 1018 | void *p; |
1016 | sizeof(uint64_t)) ? *(uint64_t *)p : *(uint32_t *)p; | 1019 | |
1020 | stat = mv643xx_eth_stats + i; | ||
1021 | |||
1022 | if (stat->netdev_off >= 0) | ||
1023 | p = ((void *)mp->dev) + stat->netdev_off; | ||
1024 | else | ||
1025 | p = ((void *)mp) + stat->mp_off; | ||
1026 | |||
1027 | data[i] = (stat->sizeof_stat == 8) ? | ||
1028 | *(uint64_t *)p : *(uint32_t *)p; | ||
1017 | } | 1029 | } |
1018 | } | 1030 | } |
1019 | 1031 | ||
@@ -1021,7 +1033,7 @@ static int mv643xx_eth_get_sset_count(struct net_device *netdev, int sset) | |||
1021 | { | 1033 | { |
1022 | switch (sset) { | 1034 | switch (sset) { |
1023 | case ETH_SS_STATS: | 1035 | case ETH_SS_STATS: |
1024 | return MV643XX_ETH_STATS_LEN; | 1036 | return ARRAY_SIZE(mv643xx_eth_stats); |
1025 | default: | 1037 | default: |
1026 | return -EOPNOTSUPP; | 1038 | return -EOPNOTSUPP; |
1027 | } | 1039 | } |