1 #define pr_fmt(fmt) "drbd debugfs: " fmt
2 #include <linux/kernel.h>
3 #include <linux/module.h>
4 #include <linux/debugfs.h>
5 #include <linux/seq_file.h>
6 #include <linux/stat.h>
7 #include <linux/jiffies.h>
8 #include <linux/list.h>
12 #include "drbd_debugfs.h"
15 /**********************************************************************
16 * Whenever you change the file format, remember to bump the version. *
17 **********************************************************************/
19 static struct dentry
*drbd_debugfs_root
;
20 static struct dentry
*drbd_debugfs_version
;
21 static struct dentry
*drbd_debugfs_resources
;
22 static struct dentry
*drbd_debugfs_minors
;
24 static void seq_print_age_or_dash(struct seq_file
*m
, bool valid
, unsigned long dt
)
27 seq_printf(m
, "\t%d", jiffies_to_msecs(dt
));
32 static void __seq_print_rq_state_bit(struct seq_file
*m
,
33 bool is_set
, char *sep
, const char *set_name
, const char *unset_name
)
35 if (is_set
&& set_name
) {
37 seq_puts(m
, set_name
);
39 } else if (!is_set
&& unset_name
) {
41 seq_puts(m
, unset_name
);
46 static void seq_print_rq_state_bit(struct seq_file
*m
,
47 bool is_set
, char *sep
, const char *set_name
)
49 __seq_print_rq_state_bit(m
, is_set
, sep
, set_name
, NULL
);
52 /* pretty print enum drbd_req_state_bits req->rq_state */
53 static void seq_print_request_state(struct seq_file
*m
, struct drbd_request
*req
)
55 unsigned int s
= req
->rq_state
;
57 seq_printf(m
, "\t0x%08x", s
);
58 seq_printf(m
, "\tmaster: %s", req
->master_bio
? "pending" : "completed");
60 /* RQ_WRITE ignored, already reported */
61 seq_puts(m
, "\tlocal:");
62 seq_print_rq_state_bit(m
, s
& RQ_IN_ACT_LOG
, &sep
, "in-AL");
63 seq_print_rq_state_bit(m
, s
& RQ_POSTPONED
, &sep
, "postponed");
64 seq_print_rq_state_bit(m
, s
& RQ_COMPLETION_SUSP
, &sep
, "suspended");
66 seq_print_rq_state_bit(m
, s
& RQ_LOCAL_PENDING
, &sep
, "pending");
67 seq_print_rq_state_bit(m
, s
& RQ_LOCAL_COMPLETED
, &sep
, "completed");
68 seq_print_rq_state_bit(m
, s
& RQ_LOCAL_ABORTED
, &sep
, "aborted");
69 seq_print_rq_state_bit(m
, s
& RQ_LOCAL_OK
, &sep
, "ok");
73 /* for_each_connection ... */
74 seq_printf(m
, "\tnet:");
76 seq_print_rq_state_bit(m
, s
& RQ_NET_PENDING
, &sep
, "pending");
77 seq_print_rq_state_bit(m
, s
& RQ_NET_QUEUED
, &sep
, "queued");
78 seq_print_rq_state_bit(m
, s
& RQ_NET_SENT
, &sep
, "sent");
79 seq_print_rq_state_bit(m
, s
& RQ_NET_DONE
, &sep
, "done");
80 seq_print_rq_state_bit(m
, s
& RQ_NET_SIS
, &sep
, "sis");
81 seq_print_rq_state_bit(m
, s
& RQ_NET_OK
, &sep
, "ok");
87 seq_print_rq_state_bit(m
, s
& RQ_EXP_RECEIVE_ACK
, &sep
, "B");
88 seq_print_rq_state_bit(m
, s
& RQ_EXP_WRITE_ACK
, &sep
, "C");
89 seq_print_rq_state_bit(m
, s
& RQ_EXP_BARR_ACK
, &sep
, "barr");
95 static void seq_print_one_request(struct seq_file
*m
, struct drbd_request
*req
, unsigned long now
)
97 /* change anything here, fixup header below! */
98 unsigned int s
= req
->rq_state
;
100 #define RQ_HDR_1 "epoch\tsector\tsize\trw"
101 seq_printf(m
, "0x%x\t%llu\t%u\t%s",
103 (unsigned long long)req
->i
.sector
, req
->i
.size
>> 9,
104 (s
& RQ_WRITE
) ? "W" : "R");
106 #define RQ_HDR_2 "\tstart\tin AL\tsubmit"
107 seq_printf(m
, "\t%d", jiffies_to_msecs(now
- req
->start_jif
));
108 seq_print_age_or_dash(m
, s
& RQ_IN_ACT_LOG
, now
- req
->in_actlog_jif
);
109 seq_print_age_or_dash(m
, s
& RQ_LOCAL_PENDING
, now
- req
->pre_submit_jif
);
111 #define RQ_HDR_3 "\tsent\tacked\tdone"
112 seq_print_age_or_dash(m
, s
& RQ_NET_SENT
, now
- req
->pre_send_jif
);
113 seq_print_age_or_dash(m
, (s
& RQ_NET_SENT
) && !(s
& RQ_NET_PENDING
), now
- req
->acked_jif
);
114 seq_print_age_or_dash(m
, s
& RQ_NET_DONE
, now
- req
->net_done_jif
);
116 #define RQ_HDR_4 "\tstate\n"
117 seq_print_request_state(m
, req
);
119 #define RQ_HDR RQ_HDR_1 RQ_HDR_2 RQ_HDR_3 RQ_HDR_4
121 static void seq_print_minor_vnr_req(struct seq_file
*m
, struct drbd_request
*req
, unsigned long now
)
123 seq_printf(m
, "%u\t%u\t", req
->device
->minor
, req
->device
->vnr
);
124 seq_print_one_request(m
, req
, now
);
127 static void seq_print_resource_pending_meta_io(struct seq_file
*m
, struct drbd_resource
*resource
, unsigned long now
)
129 struct drbd_device
*device
;
132 seq_puts(m
, "minor\tvnr\tstart\tsubmit\tintent\n");
134 idr_for_each_entry(&resource
->devices
, device
, i
) {
135 struct drbd_md_io tmp
;
136 /* In theory this is racy,
137 * in the sense that there could have been a
138 * drbd_md_put_buffer(); drbd_md_get_buffer();
139 * between accessing these members here. */
141 if (atomic_read(&tmp
.in_use
)) {
142 seq_printf(m
, "%u\t%u\t%d\t",
143 device
->minor
, device
->vnr
,
144 jiffies_to_msecs(now
- tmp
.start_jif
));
145 if (time_before(tmp
.submit_jif
, tmp
.start_jif
))
148 seq_printf(m
, "%d\t", jiffies_to_msecs(now
- tmp
.submit_jif
));
149 seq_printf(m
, "%s\n", tmp
.current_use
);
155 static void seq_print_waiting_for_AL(struct seq_file
*m
, struct drbd_resource
*resource
, unsigned long now
)
157 struct drbd_device
*device
;
160 seq_puts(m
, "minor\tvnr\tage\t#waiting\n");
162 idr_for_each_entry(&resource
->devices
, device
, i
) {
164 struct drbd_request
*req
;
165 int n
= atomic_read(&device
->ap_actlog_cnt
);
167 spin_lock_irq(&device
->resource
->req_lock
);
168 req
= list_first_entry_or_null(&device
->pending_master_completion
[1],
169 struct drbd_request
, req_pending_master_completion
);
170 /* if the oldest request does not wait for the activity log
171 * it is not interesting for us here */
172 if (req
&& !(req
->rq_state
& RQ_IN_ACT_LOG
))
173 jif
= req
->start_jif
;
176 spin_unlock_irq(&device
->resource
->req_lock
);
179 seq_printf(m
, "%u\t%u\t", device
->minor
, device
->vnr
);
181 seq_printf(m
, "%u\t", jiffies_to_msecs(now
- jif
));
184 seq_printf(m
, "%u\n", n
);
190 static void seq_print_device_bitmap_io(struct seq_file
*m
, struct drbd_device
*device
, unsigned long now
)
192 struct drbd_bm_aio_ctx
*ctx
;
193 unsigned long start_jif
;
194 unsigned int in_flight
;
196 spin_lock_irq(&device
->resource
->req_lock
);
197 ctx
= list_first_entry_or_null(&device
->pending_bitmap_io
, struct drbd_bm_aio_ctx
, list
);
198 if (ctx
&& ctx
->done
)
201 start_jif
= ctx
->start_jif
;
202 in_flight
= atomic_read(&ctx
->in_flight
);
205 spin_unlock_irq(&device
->resource
->req_lock
);
207 seq_printf(m
, "%u\t%u\t%c\t%u\t%u\n",
208 device
->minor
, device
->vnr
,
209 (flags
& BM_AIO_READ
) ? 'R' : 'W',
210 jiffies_to_msecs(now
- start_jif
),
215 static void seq_print_resource_pending_bitmap_io(struct seq_file
*m
, struct drbd_resource
*resource
, unsigned long now
)
217 struct drbd_device
*device
;
220 seq_puts(m
, "minor\tvnr\trw\tage\t#in-flight\n");
222 idr_for_each_entry(&resource
->devices
, device
, i
) {
223 seq_print_device_bitmap_io(m
, device
, now
);
228 /* pretty print enum peer_req->flags */
229 static void seq_print_peer_request_flags(struct seq_file
*m
, struct drbd_peer_request
*peer_req
)
231 unsigned long f
= peer_req
->flags
;
234 __seq_print_rq_state_bit(m
, f
& EE_SUBMITTED
, &sep
, "submitted", "preparing");
235 __seq_print_rq_state_bit(m
, f
& EE_APPLICATION
, &sep
, "application", "internal");
236 seq_print_rq_state_bit(m
, f
& EE_CALL_AL_COMPLETE_IO
, &sep
, "in-AL");
237 seq_print_rq_state_bit(m
, f
& EE_SEND_WRITE_ACK
, &sep
, "C");
238 seq_print_rq_state_bit(m
, f
& EE_MAY_SET_IN_SYNC
, &sep
, "set-in-sync");
240 if (f
& EE_IS_TRIM
) {
243 if (f
& EE_IS_TRIM_USE_ZEROOUT
)
244 seq_puts(m
, "zero-out");
251 static void seq_print_peer_request(struct seq_file
*m
,
252 struct drbd_device
*device
, struct list_head
*lh
,
255 bool reported_preparing
= false;
256 struct drbd_peer_request
*peer_req
;
257 list_for_each_entry(peer_req
, lh
, w
.list
) {
258 if (reported_preparing
&& !(peer_req
->flags
& EE_SUBMITTED
))
262 seq_printf(m
, "%u\t%u\t", device
->minor
, device
->vnr
);
264 seq_printf(m
, "%llu\t%u\t%c\t%u\t",
265 (unsigned long long)peer_req
->i
.sector
, peer_req
->i
.size
>> 9,
266 (peer_req
->flags
& EE_WRITE
) ? 'W' : 'R',
267 jiffies_to_msecs(now
- peer_req
->submit_jif
));
268 seq_print_peer_request_flags(m
, peer_req
);
269 if (peer_req
->flags
& EE_SUBMITTED
)
272 reported_preparing
= true;
276 static void seq_print_device_peer_requests(struct seq_file
*m
,
277 struct drbd_device
*device
, unsigned long now
)
279 seq_puts(m
, "minor\tvnr\tsector\tsize\trw\tage\tflags\n");
280 spin_lock_irq(&device
->resource
->req_lock
);
281 seq_print_peer_request(m
, device
, &device
->active_ee
, now
);
282 seq_print_peer_request(m
, device
, &device
->read_ee
, now
);
283 seq_print_peer_request(m
, device
, &device
->sync_ee
, now
);
284 spin_unlock_irq(&device
->resource
->req_lock
);
285 if (test_bit(FLUSH_PENDING
, &device
->flags
)) {
286 seq_printf(m
, "%u\t%u\t-\t-\tF\t%u\tflush\n",
287 device
->minor
, device
->vnr
,
288 jiffies_to_msecs(now
- device
->flush_jif
));
292 static void seq_print_resource_pending_peer_requests(struct seq_file
*m
,
293 struct drbd_resource
*resource
, unsigned long now
)
295 struct drbd_device
*device
;
299 idr_for_each_entry(&resource
->devices
, device
, i
) {
300 seq_print_device_peer_requests(m
, device
, now
);
305 static void seq_print_resource_transfer_log_summary(struct seq_file
*m
,
306 struct drbd_resource
*resource
,
307 struct drbd_connection
*connection
,
310 struct drbd_request
*req
;
311 unsigned int count
= 0;
312 unsigned int show_state
= 0;
314 seq_puts(m
, "n\tdevice\tvnr\t" RQ_HDR
);
315 spin_lock_irq(&resource
->req_lock
);
316 list_for_each_entry(req
, &connection
->transfer_log
, tl_requests
) {
317 unsigned int tmp
= 0;
321 /* don't disable irq "forever" */
322 if (!(count
& 0x1ff)) {
323 struct drbd_request
*req_next
;
324 kref_get(&req
->kref
);
325 spin_unlock_irq(&resource
->req_lock
);
327 spin_lock_irq(&resource
->req_lock
);
328 req_next
= list_next_entry(req
, tl_requests
);
329 if (kref_put(&req
->kref
, drbd_req_destroy
))
331 if (&req
->tl_requests
== &connection
->transfer_log
)
337 /* This is meant to summarize timing issues, to be able to tell
338 * local disk problems from network problems.
339 * Skip requests, if we have shown an even older request with
340 * similar aspects already. */
341 if (req
->master_bio
== NULL
)
343 if ((s
& RQ_LOCAL_MASK
) && (s
& RQ_LOCAL_PENDING
))
345 if (s
& RQ_NET_MASK
) {
346 if (!(s
& RQ_NET_SENT
))
348 if (s
& RQ_NET_PENDING
)
350 if (!(s
& RQ_NET_DONE
))
353 if ((tmp
& show_state
) == tmp
)
356 seq_printf(m
, "%u\t", count
);
357 seq_print_minor_vnr_req(m
, req
, now
);
358 if (show_state
== 0x1f)
361 spin_unlock_irq(&resource
->req_lock
);
364 /* TODO: transfer_log and friends should be moved to resource */
365 static int in_flight_summary_show(struct seq_file
*m
, void *pos
)
367 struct drbd_resource
*resource
= m
->private;
368 struct drbd_connection
*connection
;
369 unsigned long jif
= jiffies
;
371 connection
= first_connection(resource
);
372 /* This does not happen, actually.
373 * But be robust and prepare for future code changes. */
374 if (!connection
|| !kref_get_unless_zero(&connection
->kref
))
377 /* BUMP me if you change the file format/content/presentation */
378 seq_printf(m
, "v: %u\n\n", 0);
380 seq_puts(m
, "oldest bitmap IO\n");
381 seq_print_resource_pending_bitmap_io(m
, resource
, jif
);
384 seq_puts(m
, "meta data IO\n");
385 seq_print_resource_pending_meta_io(m
, resource
, jif
);
388 seq_puts(m
, "socket buffer stats\n");
389 /* for each connection ... once we have more than one */
391 if (connection
->data
.socket
) {
392 /* open coded SIOCINQ, the "relevant" part */
393 struct tcp_sock
*tp
= tcp_sk(connection
->data
.socket
->sk
);
394 int answ
= tp
->rcv_nxt
- tp
->copied_seq
;
395 seq_printf(m
, "unread receive buffer: %u Byte\n", answ
);
396 /* open coded SIOCOUTQ, the "relevant" part */
397 answ
= tp
->write_seq
- tp
->snd_una
;
398 seq_printf(m
, "unacked send buffer: %u Byte\n", answ
);
403 seq_puts(m
, "oldest peer requests\n");
404 seq_print_resource_pending_peer_requests(m
, resource
, jif
);
407 seq_puts(m
, "application requests waiting for activity log\n");
408 seq_print_waiting_for_AL(m
, resource
, jif
);
411 seq_puts(m
, "oldest application requests\n");
412 seq_print_resource_transfer_log_summary(m
, resource
, connection
, jif
);
417 seq_printf(m
, "generated in %d ms\n", jiffies_to_msecs(jif
));
418 kref_put(&connection
->kref
, drbd_destroy_connection
);
422 /* make sure at *open* time that the respective object won't go away. */
423 static int drbd_single_open(struct file
*file
, int (*show
)(struct seq_file
*, void *),
424 void *data
, struct kref
*kref
,
425 void (*release
)(struct kref
*))
427 struct dentry
*parent
;
430 /* Are we still linked,
431 * or has debugfs_remove() already been called? */
432 parent
= file
->f_path
.dentry
->d_parent
;
433 /* not sure if this can happen: */
434 if (!parent
|| d_really_is_negative(parent
))
436 /* serialize with d_delete() */
437 inode_lock(d_inode(parent
));
438 /* Make sure the object is still alive */
439 if (simple_positive(file
->f_path
.dentry
)
440 && kref_get_unless_zero(kref
))
442 inode_unlock(d_inode(parent
));
444 ret
= single_open(file
, show
, data
);
446 kref_put(kref
, release
);
452 static int in_flight_summary_open(struct inode
*inode
, struct file
*file
)
454 struct drbd_resource
*resource
= inode
->i_private
;
455 return drbd_single_open(file
, in_flight_summary_show
, resource
,
456 &resource
->kref
, drbd_destroy_resource
);
459 static int in_flight_summary_release(struct inode
*inode
, struct file
*file
)
461 struct drbd_resource
*resource
= inode
->i_private
;
462 kref_put(&resource
->kref
, drbd_destroy_resource
);
463 return single_release(inode
, file
);
466 static const struct file_operations in_flight_summary_fops
= {
467 .owner
= THIS_MODULE
,
468 .open
= in_flight_summary_open
,
471 .release
= in_flight_summary_release
,
474 void drbd_debugfs_resource_add(struct drbd_resource
*resource
)
476 struct dentry
*dentry
;
477 if (!drbd_debugfs_resources
)
480 dentry
= debugfs_create_dir(resource
->name
, drbd_debugfs_resources
);
481 if (IS_ERR_OR_NULL(dentry
))
483 resource
->debugfs_res
= dentry
;
485 dentry
= debugfs_create_dir("volumes", resource
->debugfs_res
);
486 if (IS_ERR_OR_NULL(dentry
))
488 resource
->debugfs_res_volumes
= dentry
;
490 dentry
= debugfs_create_dir("connections", resource
->debugfs_res
);
491 if (IS_ERR_OR_NULL(dentry
))
493 resource
->debugfs_res_connections
= dentry
;
495 dentry
= debugfs_create_file("in_flight_summary", S_IRUSR
|S_IRGRP
,
496 resource
->debugfs_res
, resource
,
497 &in_flight_summary_fops
);
498 if (IS_ERR_OR_NULL(dentry
))
500 resource
->debugfs_res_in_flight_summary
= dentry
;
504 drbd_debugfs_resource_cleanup(resource
);
505 drbd_err(resource
, "failed to create debugfs dentry\n");
508 static void drbd_debugfs_remove(struct dentry
**dp
)
514 void drbd_debugfs_resource_cleanup(struct drbd_resource
*resource
)
516 /* it is ok to call debugfs_remove(NULL) */
517 drbd_debugfs_remove(&resource
->debugfs_res_in_flight_summary
);
518 drbd_debugfs_remove(&resource
->debugfs_res_connections
);
519 drbd_debugfs_remove(&resource
->debugfs_res_volumes
);
520 drbd_debugfs_remove(&resource
->debugfs_res
);
523 static void seq_print_one_timing_detail(struct seq_file
*m
,
524 const struct drbd_thread_timing_details
*tdp
,
527 struct drbd_thread_timing_details td
;
529 * use temporary assignment to get at consistent data. */
532 } while (td
.cb_nr
!= tdp
->cb_nr
);
535 seq_printf(m
, "%u\t%d\t%s:%u\t%ps\n",
537 jiffies_to_msecs(now
- td
.start_jif
),
538 td
.caller_fn
, td
.line
,
542 static void seq_print_timing_details(struct seq_file
*m
,
544 unsigned int cb_nr
, struct drbd_thread_timing_details
*tdp
, unsigned long now
)
546 unsigned int start_idx
;
549 seq_printf(m
, "%s\n", title
);
550 /* If not much is going on, this will result in natural ordering.
551 * If it is very busy, we will possibly skip events, or even see wrap
552 * arounds, which could only be avoided with locking.
554 start_idx
= cb_nr
% DRBD_THREAD_DETAILS_HIST
;
555 for (i
= start_idx
; i
< DRBD_THREAD_DETAILS_HIST
; i
++)
556 seq_print_one_timing_detail(m
, tdp
+i
, now
);
557 for (i
= 0; i
< start_idx
; i
++)
558 seq_print_one_timing_detail(m
, tdp
+i
, now
);
561 static int callback_history_show(struct seq_file
*m
, void *ignored
)
563 struct drbd_connection
*connection
= m
->private;
564 unsigned long jif
= jiffies
;
566 /* BUMP me if you change the file format/content/presentation */
567 seq_printf(m
, "v: %u\n\n", 0);
569 seq_puts(m
, "n\tage\tcallsite\tfn\n");
570 seq_print_timing_details(m
, "worker", connection
->w_cb_nr
, connection
->w_timing_details
, jif
);
571 seq_print_timing_details(m
, "receiver", connection
->r_cb_nr
, connection
->r_timing_details
, jif
);
575 static int callback_history_open(struct inode
*inode
, struct file
*file
)
577 struct drbd_connection
*connection
= inode
->i_private
;
578 return drbd_single_open(file
, callback_history_show
, connection
,
579 &connection
->kref
, drbd_destroy_connection
);
582 static int callback_history_release(struct inode
*inode
, struct file
*file
)
584 struct drbd_connection
*connection
= inode
->i_private
;
585 kref_put(&connection
->kref
, drbd_destroy_connection
);
586 return single_release(inode
, file
);
589 static const struct file_operations connection_callback_history_fops
= {
590 .owner
= THIS_MODULE
,
591 .open
= callback_history_open
,
594 .release
= callback_history_release
,
597 static int connection_oldest_requests_show(struct seq_file
*m
, void *ignored
)
599 struct drbd_connection
*connection
= m
->private;
600 unsigned long now
= jiffies
;
601 struct drbd_request
*r1
, *r2
;
603 /* BUMP me if you change the file format/content/presentation */
604 seq_printf(m
, "v: %u\n\n", 0);
606 spin_lock_irq(&connection
->resource
->req_lock
);
607 r1
= connection
->req_next
;
609 seq_print_minor_vnr_req(m
, r1
, now
);
610 r2
= connection
->req_ack_pending
;
611 if (r2
&& r2
!= r1
) {
613 seq_print_minor_vnr_req(m
, r1
, now
);
615 r2
= connection
->req_not_net_done
;
617 seq_print_minor_vnr_req(m
, r2
, now
);
618 spin_unlock_irq(&connection
->resource
->req_lock
);
622 static int connection_oldest_requests_open(struct inode
*inode
, struct file
*file
)
624 struct drbd_connection
*connection
= inode
->i_private
;
625 return drbd_single_open(file
, connection_oldest_requests_show
, connection
,
626 &connection
->kref
, drbd_destroy_connection
);
629 static int connection_oldest_requests_release(struct inode
*inode
, struct file
*file
)
631 struct drbd_connection
*connection
= inode
->i_private
;
632 kref_put(&connection
->kref
, drbd_destroy_connection
);
633 return single_release(inode
, file
);
636 static const struct file_operations connection_oldest_requests_fops
= {
637 .owner
= THIS_MODULE
,
638 .open
= connection_oldest_requests_open
,
641 .release
= connection_oldest_requests_release
,
644 void drbd_debugfs_connection_add(struct drbd_connection
*connection
)
646 struct dentry
*conns_dir
= connection
->resource
->debugfs_res_connections
;
647 struct dentry
*dentry
;
651 /* Once we enable mutliple peers,
652 * these connections will have descriptive names.
653 * For now, it is just the one connection to the (only) "peer". */
654 dentry
= debugfs_create_dir("peer", conns_dir
);
655 if (IS_ERR_OR_NULL(dentry
))
657 connection
->debugfs_conn
= dentry
;
659 dentry
= debugfs_create_file("callback_history", S_IRUSR
|S_IRGRP
,
660 connection
->debugfs_conn
, connection
,
661 &connection_callback_history_fops
);
662 if (IS_ERR_OR_NULL(dentry
))
664 connection
->debugfs_conn_callback_history
= dentry
;
666 dentry
= debugfs_create_file("oldest_requests", S_IRUSR
|S_IRGRP
,
667 connection
->debugfs_conn
, connection
,
668 &connection_oldest_requests_fops
);
669 if (IS_ERR_OR_NULL(dentry
))
671 connection
->debugfs_conn_oldest_requests
= dentry
;
675 drbd_debugfs_connection_cleanup(connection
);
676 drbd_err(connection
, "failed to create debugfs dentry\n");
679 void drbd_debugfs_connection_cleanup(struct drbd_connection
*connection
)
681 drbd_debugfs_remove(&connection
->debugfs_conn_callback_history
);
682 drbd_debugfs_remove(&connection
->debugfs_conn_oldest_requests
);
683 drbd_debugfs_remove(&connection
->debugfs_conn
);
686 static void resync_dump_detail(struct seq_file
*m
, struct lc_element
*e
)
688 struct bm_extent
*bme
= lc_entry(e
, struct bm_extent
, lce
);
690 seq_printf(m
, "%5d %s %s %s", bme
->rs_left
,
691 test_bit(BME_NO_WRITES
, &bme
->flags
) ? "NO_WRITES" : "---------",
692 test_bit(BME_LOCKED
, &bme
->flags
) ? "LOCKED" : "------",
693 test_bit(BME_PRIORITY
, &bme
->flags
) ? "PRIORITY" : "--------"
697 static int device_resync_extents_show(struct seq_file
*m
, void *ignored
)
699 struct drbd_device
*device
= m
->private;
701 /* BUMP me if you change the file format/content/presentation */
702 seq_printf(m
, "v: %u\n\n", 0);
704 if (get_ldev_if_state(device
, D_FAILED
)) {
705 lc_seq_printf_stats(m
, device
->resync
);
706 lc_seq_dump_details(m
, device
->resync
, "rs_left flags", resync_dump_detail
);
712 static int device_act_log_extents_show(struct seq_file
*m
, void *ignored
)
714 struct drbd_device
*device
= m
->private;
716 /* BUMP me if you change the file format/content/presentation */
717 seq_printf(m
, "v: %u\n\n", 0);
719 if (get_ldev_if_state(device
, D_FAILED
)) {
720 lc_seq_printf_stats(m
, device
->act_log
);
721 lc_seq_dump_details(m
, device
->act_log
, "", NULL
);
727 static int device_oldest_requests_show(struct seq_file
*m
, void *ignored
)
729 struct drbd_device
*device
= m
->private;
730 struct drbd_resource
*resource
= device
->resource
;
731 unsigned long now
= jiffies
;
732 struct drbd_request
*r1
, *r2
;
735 /* BUMP me if you change the file format/content/presentation */
736 seq_printf(m
, "v: %u\n\n", 0);
739 spin_lock_irq(&resource
->req_lock
);
740 /* WRITE, then READ */
741 for (i
= 1; i
>= 0; --i
) {
742 r1
= list_first_entry_or_null(&device
->pending_master_completion
[i
],
743 struct drbd_request
, req_pending_master_completion
);
744 r2
= list_first_entry_or_null(&device
->pending_completion
[i
],
745 struct drbd_request
, req_pending_local
);
747 seq_print_one_request(m
, r1
, now
);
749 seq_print_one_request(m
, r2
, now
);
751 spin_unlock_irq(&resource
->req_lock
);
755 static int device_data_gen_id_show(struct seq_file
*m
, void *ignored
)
757 struct drbd_device
*device
= m
->private;
759 enum drbd_uuid_index idx
;
761 if (!get_ldev_if_state(device
, D_FAILED
))
764 md
= &device
->ldev
->md
;
765 spin_lock_irq(&md
->uuid_lock
);
766 for (idx
= UI_CURRENT
; idx
<= UI_HISTORY_END
; idx
++) {
767 seq_printf(m
, "0x%016llX\n", md
->uuid
[idx
]);
769 spin_unlock_irq(&md
->uuid_lock
);
774 static int device_ed_gen_id_show(struct seq_file
*m
, void *ignored
)
776 struct drbd_device
*device
= m
->private;
777 seq_printf(m
, "0x%016llX\n", (unsigned long long)device
->ed_uuid
);
781 #define drbd_debugfs_device_attr(name) \
782 static int device_ ## name ## _open(struct inode *inode, struct file *file) \
784 struct drbd_device *device = inode->i_private; \
785 return drbd_single_open(file, device_ ## name ## _show, device, \
786 &device->kref, drbd_destroy_device); \
788 static int device_ ## name ## _release(struct inode *inode, struct file *file) \
790 struct drbd_device *device = inode->i_private; \
791 kref_put(&device->kref, drbd_destroy_device); \
792 return single_release(inode, file); \
794 static const struct file_operations device_ ## name ## _fops = { \
795 .owner = THIS_MODULE, \
796 .open = device_ ## name ## _open, \
798 .llseek = seq_lseek, \
799 .release = device_ ## name ## _release, \
802 drbd_debugfs_device_attr(oldest_requests
)
803 drbd_debugfs_device_attr(act_log_extents
)
804 drbd_debugfs_device_attr(resync_extents
)
805 drbd_debugfs_device_attr(data_gen_id
)
806 drbd_debugfs_device_attr(ed_gen_id
)
808 void drbd_debugfs_device_add(struct drbd_device
*device
)
810 struct dentry
*vols_dir
= device
->resource
->debugfs_res_volumes
;
811 char minor_buf
[8]; /* MINORMASK, MINORBITS == 20; */
812 char vnr_buf
[8]; /* volume number vnr is even 16 bit only; */
813 char *slink_name
= NULL
;
815 struct dentry
*dentry
;
816 if (!vols_dir
|| !drbd_debugfs_minors
)
819 snprintf(vnr_buf
, sizeof(vnr_buf
), "%u", device
->vnr
);
820 dentry
= debugfs_create_dir(vnr_buf
, vols_dir
);
821 if (IS_ERR_OR_NULL(dentry
))
823 device
->debugfs_vol
= dentry
;
825 snprintf(minor_buf
, sizeof(minor_buf
), "%u", device
->minor
);
826 slink_name
= kasprintf(GFP_KERNEL
, "../resources/%s/volumes/%u",
827 device
->resource
->name
, device
->vnr
);
830 dentry
= debugfs_create_symlink(minor_buf
, drbd_debugfs_minors
, slink_name
);
833 if (IS_ERR_OR_NULL(dentry
))
835 device
->debugfs_minor
= dentry
;
837 #define DCF(name) do { \
838 dentry = debugfs_create_file(#name, S_IRUSR|S_IRGRP, \
839 device->debugfs_vol, device, \
840 &device_ ## name ## _fops); \
841 if (IS_ERR_OR_NULL(dentry)) \
843 device->debugfs_vol_ ## name = dentry; \
846 DCF(oldest_requests
);
847 DCF(act_log_extents
);
855 drbd_debugfs_device_cleanup(device
);
856 drbd_err(device
, "failed to create debugfs entries\n");
859 void drbd_debugfs_device_cleanup(struct drbd_device
*device
)
861 drbd_debugfs_remove(&device
->debugfs_minor
);
862 drbd_debugfs_remove(&device
->debugfs_vol_oldest_requests
);
863 drbd_debugfs_remove(&device
->debugfs_vol_act_log_extents
);
864 drbd_debugfs_remove(&device
->debugfs_vol_resync_extents
);
865 drbd_debugfs_remove(&device
->debugfs_vol_data_gen_id
);
866 drbd_debugfs_remove(&device
->debugfs_vol_ed_gen_id
);
867 drbd_debugfs_remove(&device
->debugfs_vol
);
870 void drbd_debugfs_peer_device_add(struct drbd_peer_device
*peer_device
)
872 struct dentry
*conn_dir
= peer_device
->connection
->debugfs_conn
;
873 struct dentry
*dentry
;
879 snprintf(vnr_buf
, sizeof(vnr_buf
), "%u", peer_device
->device
->vnr
);
880 dentry
= debugfs_create_dir(vnr_buf
, conn_dir
);
881 if (IS_ERR_OR_NULL(dentry
))
883 peer_device
->debugfs_peer_dev
= dentry
;
887 drbd_debugfs_peer_device_cleanup(peer_device
);
888 drbd_err(peer_device
, "failed to create debugfs entries\n");
891 void drbd_debugfs_peer_device_cleanup(struct drbd_peer_device
*peer_device
)
893 drbd_debugfs_remove(&peer_device
->debugfs_peer_dev
);
896 static int drbd_version_show(struct seq_file
*m
, void *ignored
)
898 seq_printf(m
, "# %s\n", drbd_buildtag());
899 seq_printf(m
, "VERSION=%s\n", REL_VERSION
);
900 seq_printf(m
, "API_VERSION=%u\n", API_VERSION
);
901 seq_printf(m
, "PRO_VERSION_MIN=%u\n", PRO_VERSION_MIN
);
902 seq_printf(m
, "PRO_VERSION_MAX=%u\n", PRO_VERSION_MAX
);
906 static int drbd_version_open(struct inode
*inode
, struct file
*file
)
908 return single_open(file
, drbd_version_show
, NULL
);
911 static struct file_operations drbd_version_fops
= {
912 .owner
= THIS_MODULE
,
913 .open
= drbd_version_open
,
916 .release
= single_release
,
919 /* not __exit, may be indirectly called
920 * from the module-load-failure path as well. */
921 void drbd_debugfs_cleanup(void)
923 drbd_debugfs_remove(&drbd_debugfs_resources
);
924 drbd_debugfs_remove(&drbd_debugfs_minors
);
925 drbd_debugfs_remove(&drbd_debugfs_version
);
926 drbd_debugfs_remove(&drbd_debugfs_root
);
929 int __init
drbd_debugfs_init(void)
931 struct dentry
*dentry
;
933 dentry
= debugfs_create_dir("drbd", NULL
);
934 if (IS_ERR_OR_NULL(dentry
))
936 drbd_debugfs_root
= dentry
;
938 dentry
= debugfs_create_file("version", 0444, drbd_debugfs_root
, NULL
, &drbd_version_fops
);
939 if (IS_ERR_OR_NULL(dentry
))
941 drbd_debugfs_version
= dentry
;
943 dentry
= debugfs_create_dir("resources", drbd_debugfs_root
);
944 if (IS_ERR_OR_NULL(dentry
))
946 drbd_debugfs_resources
= dentry
;
948 dentry
= debugfs_create_dir("minors", drbd_debugfs_root
);
949 if (IS_ERR_OR_NULL(dentry
))
951 drbd_debugfs_minors
= dentry
;
955 drbd_debugfs_cleanup();
957 return PTR_ERR(dentry
);