1 // SPDX-License-Identifier: GPL-2.0-only
3 * FUJITSU Extended Socket Network Device driver
4 * Copyright (c) 2015 FUJITSU LIMITED
7 /* ethtool support for fjes */
9 #include <linux/vmalloc.h>
10 #include <linux/netdevice.h>
11 #include <linux/ethtool.h>
12 #include <linux/platform_device.h>
17 char stat_string
[ETH_GSTRING_LEN
];
22 #define FJES_STAT(name, stat) { \
23 .stat_string = name, \
24 .sizeof_stat = sizeof_field(struct fjes_adapter, stat), \
25 .stat_offset = offsetof(struct fjes_adapter, stat) \
28 static const struct fjes_stats fjes_gstrings_stats
[] = {
29 FJES_STAT("rx_packets", stats64
.rx_packets
),
30 FJES_STAT("tx_packets", stats64
.tx_packets
),
31 FJES_STAT("rx_bytes", stats64
.rx_bytes
),
32 FJES_STAT("tx_bytes", stats64
.rx_bytes
),
33 FJES_STAT("rx_dropped", stats64
.rx_dropped
),
34 FJES_STAT("tx_dropped", stats64
.tx_dropped
),
37 #define FJES_EP_STATS_LEN 14
38 #define FJES_STATS_LEN \
39 (ARRAY_SIZE(fjes_gstrings_stats) + \
40 ((&((struct fjes_adapter *)netdev_priv(netdev))->hw)->max_epid - 1) * \
43 static void fjes_get_ethtool_stats(struct net_device
*netdev
,
44 struct ethtool_stats
*stats
, u64
*data
)
46 struct fjes_adapter
*adapter
= netdev_priv(netdev
);
47 struct fjes_hw
*hw
= &adapter
->hw
;
52 for (i
= 0; i
< ARRAY_SIZE(fjes_gstrings_stats
); i
++) {
53 p
= (char *)adapter
+ fjes_gstrings_stats
[i
].stat_offset
;
54 data
[i
] = (fjes_gstrings_stats
[i
].sizeof_stat
== sizeof(u64
))
55 ? *(u64
*)p
: *(u32
*)p
;
57 for (epidx
= 0; epidx
< hw
->max_epid
; epidx
++) {
58 if (epidx
== hw
->my_epid
)
60 data
[i
++] = hw
->ep_shm_info
[epidx
].ep_stats
62 data
[i
++] = hw
->ep_shm_info
[epidx
].ep_stats
63 .com_unregist_buf_exec
;
64 data
[i
++] = hw
->ep_shm_info
[epidx
].ep_stats
.send_intr_rx
;
65 data
[i
++] = hw
->ep_shm_info
[epidx
].ep_stats
.send_intr_unshare
;
66 data
[i
++] = hw
->ep_shm_info
[epidx
].ep_stats
67 .send_intr_zoneupdate
;
68 data
[i
++] = hw
->ep_shm_info
[epidx
].ep_stats
.recv_intr_rx
;
69 data
[i
++] = hw
->ep_shm_info
[epidx
].ep_stats
.recv_intr_unshare
;
70 data
[i
++] = hw
->ep_shm_info
[epidx
].ep_stats
.recv_intr_stop
;
71 data
[i
++] = hw
->ep_shm_info
[epidx
].ep_stats
72 .recv_intr_zoneupdate
;
73 data
[i
++] = hw
->ep_shm_info
[epidx
].ep_stats
.tx_buffer_full
;
74 data
[i
++] = hw
->ep_shm_info
[epidx
].ep_stats
75 .tx_dropped_not_shared
;
76 data
[i
++] = hw
->ep_shm_info
[epidx
].ep_stats
77 .tx_dropped_ver_mismatch
;
78 data
[i
++] = hw
->ep_shm_info
[epidx
].ep_stats
79 .tx_dropped_buf_size_mismatch
;
80 data
[i
++] = hw
->ep_shm_info
[epidx
].ep_stats
81 .tx_dropped_vlanid_mismatch
;
85 static void fjes_get_strings(struct net_device
*netdev
,
86 u32 stringset
, u8
*data
)
88 struct fjes_adapter
*adapter
= netdev_priv(netdev
);
89 struct fjes_hw
*hw
= &adapter
->hw
;
92 if (stringset
!= ETH_SS_STATS
)
95 for (i
= 0; i
< ARRAY_SIZE(fjes_gstrings_stats
); i
++)
96 ethtool_puts(&data
, fjes_gstrings_stats
[i
].stat_string
);
98 for (i
= 0; i
< hw
->max_epid
; i
++) {
101 ethtool_sprintf(&data
, "ep%u_com_regist_buf_exec", i
);
102 ethtool_sprintf(&data
, "ep%u_com_unregist_buf_exec", i
);
103 ethtool_sprintf(&data
, "ep%u_send_intr_rx", i
);
104 ethtool_sprintf(&data
, "ep%u_send_intr_unshare", i
);
105 ethtool_sprintf(&data
, "ep%u_send_intr_zoneupdate", i
);
106 ethtool_sprintf(&data
, "ep%u_recv_intr_rx", i
);
107 ethtool_sprintf(&data
, "ep%u_recv_intr_unshare", i
);
108 ethtool_sprintf(&data
, "ep%u_recv_intr_stop", i
);
109 ethtool_sprintf(&data
, "ep%u_recv_intr_zoneupdate", i
);
110 ethtool_sprintf(&data
, "ep%u_tx_buffer_full", i
);
111 ethtool_sprintf(&data
, "ep%u_tx_dropped_not_shared", i
);
112 ethtool_sprintf(&data
, "ep%u_tx_dropped_ver_mismatch", i
);
113 ethtool_sprintf(&data
, "ep%u_tx_dropped_buf_size_mismatch", i
);
114 ethtool_sprintf(&data
, "ep%u_tx_dropped_vlanid_mismatch", i
);
118 static int fjes_get_sset_count(struct net_device
*netdev
, int sset
)
122 return FJES_STATS_LEN
;
128 static void fjes_get_drvinfo(struct net_device
*netdev
,
129 struct ethtool_drvinfo
*drvinfo
)
131 struct fjes_adapter
*adapter
= netdev_priv(netdev
);
132 struct platform_device
*plat_dev
;
134 plat_dev
= adapter
->plat_dev
;
136 strscpy(drvinfo
->driver
, fjes_driver_name
, sizeof(drvinfo
->driver
));
137 strscpy(drvinfo
->version
, fjes_driver_version
,
138 sizeof(drvinfo
->version
));
140 strscpy(drvinfo
->fw_version
, "none", sizeof(drvinfo
->fw_version
));
141 snprintf(drvinfo
->bus_info
, sizeof(drvinfo
->bus_info
),
142 "platform:%s", plat_dev
->name
);
145 static int fjes_get_link_ksettings(struct net_device
*netdev
,
146 struct ethtool_link_ksettings
*ecmd
)
148 ethtool_link_ksettings_zero_link_mode(ecmd
, supported
);
149 ethtool_link_ksettings_zero_link_mode(ecmd
, advertising
);
150 ecmd
->base
.duplex
= DUPLEX_FULL
;
151 ecmd
->base
.autoneg
= AUTONEG_DISABLE
;
152 ecmd
->base
.port
= PORT_NONE
;
153 ecmd
->base
.speed
= 20000; /* 20Gb/s */
158 static int fjes_get_regs_len(struct net_device
*netdev
)
160 #define FJES_REGS_LEN 37
161 return FJES_REGS_LEN
* sizeof(u32
);
164 static void fjes_get_regs(struct net_device
*netdev
,
165 struct ethtool_regs
*regs
, void *p
)
167 struct fjes_adapter
*adapter
= netdev_priv(netdev
);
168 struct fjes_hw
*hw
= &adapter
->hw
;
171 memset(p
, 0, FJES_REGS_LEN
* sizeof(u32
));
175 /* Information registers */
176 regs_buff
[0] = rd32(XSCT_OWNER_EPID
);
177 regs_buff
[1] = rd32(XSCT_MAX_EP
);
179 /* Device Control registers */
180 regs_buff
[4] = rd32(XSCT_DCTL
);
182 /* Command Control registers */
183 regs_buff
[8] = rd32(XSCT_CR
);
184 regs_buff
[9] = rd32(XSCT_CS
);
185 regs_buff
[10] = rd32(XSCT_SHSTSAL
);
186 regs_buff
[11] = rd32(XSCT_SHSTSAH
);
188 regs_buff
[13] = rd32(XSCT_REQBL
);
189 regs_buff
[14] = rd32(XSCT_REQBAL
);
190 regs_buff
[15] = rd32(XSCT_REQBAH
);
192 regs_buff
[17] = rd32(XSCT_RESPBL
);
193 regs_buff
[18] = rd32(XSCT_RESPBAL
);
194 regs_buff
[19] = rd32(XSCT_RESPBAH
);
196 /* Interrupt Control registers */
197 regs_buff
[32] = rd32(XSCT_IS
);
198 regs_buff
[33] = rd32(XSCT_IMS
);
199 regs_buff
[34] = rd32(XSCT_IMC
);
200 regs_buff
[35] = rd32(XSCT_IG
);
201 regs_buff
[36] = rd32(XSCT_ICTL
);
204 static int fjes_set_dump(struct net_device
*netdev
, struct ethtool_dump
*dump
)
206 struct fjes_adapter
*adapter
= netdev_priv(netdev
);
207 struct fjes_hw
*hw
= &adapter
->hw
;
214 hw
->debug_mode
= dump
->flag
;
216 /* enable debug mode */
217 mutex_lock(&hw
->hw_info
.lock
);
218 ret
= fjes_hw_start_debug(hw
);
219 mutex_unlock(&hw
->hw_info
.lock
);
227 /* disable debug mode */
228 mutex_lock(&hw
->hw_info
.lock
);
229 ret
= fjes_hw_stop_debug(hw
);
230 mutex_unlock(&hw
->hw_info
.lock
);
236 static int fjes_get_dump_flag(struct net_device
*netdev
,
237 struct ethtool_dump
*dump
)
239 struct fjes_adapter
*adapter
= netdev_priv(netdev
);
240 struct fjes_hw
*hw
= &adapter
->hw
;
242 dump
->len
= hw
->hw_info
.trace_size
;
244 dump
->flag
= hw
->debug_mode
;
249 static int fjes_get_dump_data(struct net_device
*netdev
,
250 struct ethtool_dump
*dump
, void *buf
)
252 struct fjes_adapter
*adapter
= netdev_priv(netdev
);
253 struct fjes_hw
*hw
= &adapter
->hw
;
256 if (hw
->hw_info
.trace
)
257 memcpy(buf
, hw
->hw_info
.trace
, hw
->hw_info
.trace_size
);
264 static const struct ethtool_ops fjes_ethtool_ops
= {
265 .get_drvinfo
= fjes_get_drvinfo
,
266 .get_ethtool_stats
= fjes_get_ethtool_stats
,
267 .get_strings
= fjes_get_strings
,
268 .get_sset_count
= fjes_get_sset_count
,
269 .get_regs
= fjes_get_regs
,
270 .get_regs_len
= fjes_get_regs_len
,
271 .set_dump
= fjes_set_dump
,
272 .get_dump_flag
= fjes_get_dump_flag
,
273 .get_dump_data
= fjes_get_dump_data
,
274 .get_link_ksettings
= fjes_get_link_ksettings
,
277 void fjes_set_ethtool_ops(struct net_device
*netdev
)
279 netdev
->ethtool_ops
= &fjes_ethtool_ops
;