3 * linux/drivers/s390/net/qeth_fs.c
5 * Linux on zSeries OSA Express and HiperSockets support
6 * This file contains code related to procfs.
8 * Copyright 2000,2003 IBM Corporation
10 * Author(s): Thomas Spatzier <tspat@de.ibm.com>
13 #include <linux/module.h>
14 #include <linux/init.h>
15 #include <linux/proc_fs.h>
16 #include <linux/seq_file.h>
17 #include <linux/list.h>
18 #include <linux/rwsem.h>
24 /***** /proc/qeth *****/
25 #define QETH_PROCFILE_NAME "qeth"
26 static struct proc_dir_entry
*qeth_procfile
;
29 qeth_procfile_seq_match(struct device
*dev
, void *data
)
35 qeth_procfile_seq_start(struct seq_file
*s
, loff_t
*offset
)
37 struct device
*dev
= NULL
;
41 return SEQ_START_TOKEN
;
43 dev
= driver_find_device(&qeth_ccwgroup_driver
.driver
, dev
,
44 NULL
, qeth_procfile_seq_match
);
53 qeth_procfile_seq_stop(struct seq_file
*s
, void* it
)
58 qeth_procfile_seq_next(struct seq_file
*s
, void *it
, loff_t
*offset
)
60 struct device
*prev
, *next
;
62 if (it
== SEQ_START_TOKEN
)
65 prev
= (struct device
*) it
;
66 next
= driver_find_device(&qeth_ccwgroup_driver
.driver
,
67 prev
, NULL
, qeth_procfile_seq_match
);
72 static inline const char *
73 qeth_get_router_str(struct qeth_card
*card
, int ipv
)
75 enum qeth_routing_types routing_type
= NO_ROUTER
;
78 routing_type
= card
->options
.route4
.type
;
80 #ifdef CONFIG_QETH_IPV6
81 routing_type
= card
->options
.route6
.type
;
84 #endif /* CONFIG_QETH_IPV6 */
87 switch (routing_type
){
90 case SECONDARY_ROUTER
:
92 case MULTICAST_ROUTER
:
93 if (card
->info
.broadcast_capable
== QETH_BROADCAST_WITHOUT_ECHO
)
96 case PRIMARY_CONNECTOR
:
97 if (card
->info
.broadcast_capable
== QETH_BROADCAST_WITHOUT_ECHO
)
100 case SECONDARY_CONNECTOR
:
101 if (card
->info
.broadcast_capable
== QETH_BROADCAST_WITHOUT_ECHO
)
104 default: /* NO_ROUTER */
110 qeth_procfile_seq_show(struct seq_file
*s
, void *it
)
112 struct device
*device
;
113 struct qeth_card
*card
;
114 char tmp
[12]; /* for qeth_get_prioq_str */
116 if (it
== SEQ_START_TOKEN
){
117 seq_printf(s
, "devices CHPID interface "
118 "cardtype port chksum prio-q'ing rtr4 "
120 seq_printf(s
, "-------------------------- ----- ---------- "
121 "-------------- ---- ------ ---------- ---- "
122 "---- ----- -----\n");
124 device
= (struct device
*) it
;
125 card
= device
->driver_data
;
126 seq_printf(s
, "%s/%s/%s x%02X %-10s %-14s %-4i ",
131 QETH_CARD_IFNAME(card
),
132 qeth_get_cardname_short(card
),
134 if (card
->lan_online
)
135 seq_printf(s
, "%-6s %-10s %-4s %-4s %-5s %-5i\n",
136 qeth_get_checksum_str(card
),
137 qeth_get_prioq_str(card
, tmp
),
138 qeth_get_router_str(card
, 4),
139 qeth_get_router_str(card
, 6),
140 qeth_get_bufsize_str(card
),
141 card
->qdio
.in_buf_pool
.buf_count
);
143 seq_printf(s
, " +++ LAN OFFLINE +++\n");
149 static const struct seq_operations qeth_procfile_seq_ops
= {
150 .start
= qeth_procfile_seq_start
,
151 .stop
= qeth_procfile_seq_stop
,
152 .next
= qeth_procfile_seq_next
,
153 .show
= qeth_procfile_seq_show
,
157 qeth_procfile_open(struct inode
*inode
, struct file
*file
)
159 return seq_open(file
, &qeth_procfile_seq_ops
);
162 static const struct file_operations qeth_procfile_fops
= {
163 .owner
= THIS_MODULE
,
164 .open
= qeth_procfile_open
,
167 .release
= seq_release
,
170 /***** /proc/qeth_perf *****/
171 #define QETH_PERF_PROCFILE_NAME "qeth_perf"
172 static struct proc_dir_entry
*qeth_perf_procfile
;
175 qeth_perf_procfile_seq_show(struct seq_file
*s
, void *it
)
177 struct device
*device
;
178 struct qeth_card
*card
;
181 if (it
== SEQ_START_TOKEN
)
184 device
= (struct device
*) it
;
185 card
= device
->driver_data
;
186 seq_printf(s
, "For card with devnos %s/%s/%s (%s):\n",
190 QETH_CARD_IFNAME(card
)
192 if (!card
->options
.performance_stats
)
193 seq_printf(s
, "Performance statistics are deactivated.\n");
194 seq_printf(s
, " Skb's/buffers received : %lu/%u\n"
195 " Skb's/buffers sent : %lu/%u\n\n",
196 card
->stats
.rx_packets
-
197 card
->perf_stats
.initial_rx_packets
,
198 card
->perf_stats
.bufs_rec
,
199 card
->stats
.tx_packets
-
200 card
->perf_stats
.initial_tx_packets
,
201 card
->perf_stats
.bufs_sent
203 seq_printf(s
, " Skb's/buffers sent without packing : %lu/%u\n"
204 " Skb's/buffers sent with packing : %u/%u\n\n",
205 card
->stats
.tx_packets
- card
->perf_stats
.initial_tx_packets
206 - card
->perf_stats
.skbs_sent_pack
,
207 card
->perf_stats
.bufs_sent
- card
->perf_stats
.bufs_sent_pack
,
208 card
->perf_stats
.skbs_sent_pack
,
209 card
->perf_stats
.bufs_sent_pack
211 seq_printf(s
, " Skbs sent in SG mode : %u\n"
212 " Skb fragments sent in SG mode : %u\n\n",
213 card
->perf_stats
.sg_skbs_sent
,
214 card
->perf_stats
.sg_frags_sent
);
215 seq_printf(s
, " Skbs received in SG mode : %u\n"
216 " Skb fragments received in SG mode : %u\n"
217 " Page allocations for rx SG mode : %u\n\n",
218 card
->perf_stats
.sg_skbs_rx
,
219 card
->perf_stats
.sg_frags_rx
,
220 card
->perf_stats
.sg_alloc_page_rx
);
221 seq_printf(s
, " large_send tx (in Kbytes) : %u\n"
222 " large_send count : %u\n\n",
223 card
->perf_stats
.large_send_bytes
>> 10,
224 card
->perf_stats
.large_send_cnt
);
225 seq_printf(s
, " Packing state changes no pkg.->packing : %u/%u\n"
226 " Watermarks L/H : %i/%i\n"
227 " Current buffer usage (outbound q's) : "
229 card
->perf_stats
.sc_dp_p
, card
->perf_stats
.sc_p_dp
,
230 QETH_LOW_WATERMARK_PACK
, QETH_HIGH_WATERMARK_PACK
,
231 atomic_read(&card
->qdio
.out_qs
[0]->used_buffers
),
232 (card
->qdio
.no_out_queues
> 1)?
233 atomic_read(&card
->qdio
.out_qs
[1]->used_buffers
)
235 (card
->qdio
.no_out_queues
> 2)?
236 atomic_read(&card
->qdio
.out_qs
[2]->used_buffers
)
238 (card
->qdio
.no_out_queues
> 3)?
239 atomic_read(&card
->qdio
.out_qs
[3]->used_buffers
)
242 seq_printf(s
, " Inbound handler time (in us) : %u\n"
243 " Inbound handler count : %u\n"
244 " Inbound do_QDIO time (in us) : %u\n"
245 " Inbound do_QDIO count : %u\n\n"
246 " Outbound handler time (in us) : %u\n"
247 " Outbound handler count : %u\n\n"
248 " Outbound time (in us, incl QDIO) : %u\n"
249 " Outbound count : %u\n"
250 " Outbound do_QDIO time (in us) : %u\n"
251 " Outbound do_QDIO count : %u\n\n",
252 card
->perf_stats
.inbound_time
,
253 card
->perf_stats
.inbound_cnt
,
254 card
->perf_stats
.inbound_do_qdio_time
,
255 card
->perf_stats
.inbound_do_qdio_cnt
,
256 card
->perf_stats
.outbound_handler_time
,
257 card
->perf_stats
.outbound_handler_cnt
,
258 card
->perf_stats
.outbound_time
,
259 card
->perf_stats
.outbound_cnt
,
260 card
->perf_stats
.outbound_do_qdio_time
,
261 card
->perf_stats
.outbound_do_qdio_cnt
267 static const struct seq_operations qeth_perf_procfile_seq_ops
= {
268 .start
= qeth_procfile_seq_start
,
269 .stop
= qeth_procfile_seq_stop
,
270 .next
= qeth_procfile_seq_next
,
271 .show
= qeth_perf_procfile_seq_show
,
275 qeth_perf_procfile_open(struct inode
*inode
, struct file
*file
)
277 return seq_open(file
, &qeth_perf_procfile_seq_ops
);
280 static const struct file_operations qeth_perf_procfile_fops
= {
281 .owner
= THIS_MODULE
,
282 .open
= qeth_perf_procfile_open
,
285 .release
= seq_release
,
289 qeth_create_procfs_entries(void)
291 qeth_procfile
= create_proc_entry(QETH_PROCFILE_NAME
,
292 S_IFREG
| 0444, NULL
);
294 qeth_procfile
->proc_fops
= &qeth_procfile_fops
;
296 qeth_perf_procfile
= create_proc_entry(QETH_PERF_PROCFILE_NAME
,
297 S_IFREG
| 0444, NULL
);
298 if (qeth_perf_procfile
)
299 qeth_perf_procfile
->proc_fops
= &qeth_perf_procfile_fops
;
309 qeth_remove_procfs_entries(void)
312 remove_proc_entry(QETH_PROCFILE_NAME
, NULL
);
313 if (qeth_perf_procfile
)
314 remove_proc_entry(QETH_PERF_PROCFILE_NAME
, NULL
);