mirror of
				https://github.com/torvalds/linux.git
				synced 2025-10-31 08:38:45 +02:00 
			
		
		
		
	 f13a48bd79
			
		
	
	
		f13a48bd79
		
	
	
	
	
		
			
			Move slow_work's debugging proc file to debugfs. Signed-off-by: David Howells <dhowells@redhat.com> Requested-and-acked-by: Ingo Molnar <mingo@elte.hu> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
		
			
				
	
	
		
			227 lines
		
	
	
	
		
			5.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			227 lines
		
	
	
	
		
			5.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* Slow work debugging
 | |
|  *
 | |
|  * Copyright (C) 2009 Red Hat, Inc. All Rights Reserved.
 | |
|  * Written by David Howells (dhowells@redhat.com)
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or
 | |
|  * modify it under the terms of the GNU General Public Licence
 | |
|  * as published by the Free Software Foundation; either version
 | |
|  * 2 of the Licence, or (at your option) any later version.
 | |
|  */
 | |
| 
 | |
| #include <linux/module.h>
 | |
| #include <linux/slow-work.h>
 | |
| #include <linux/fs.h>
 | |
| #include <linux/time.h>
 | |
| #include <linux/seq_file.h>
 | |
| #include "slow-work.h"
 | |
| 
 | |
| #define ITERATOR_SHIFT		(BITS_PER_LONG - 4)
 | |
| #define ITERATOR_SELECTOR	(0xfUL << ITERATOR_SHIFT)
 | |
| #define ITERATOR_COUNTER	(~ITERATOR_SELECTOR)
 | |
| 
 | |
| void slow_work_new_thread_desc(struct slow_work *work, struct seq_file *m)
 | |
| {
 | |
| 	seq_puts(m, "Slow-work: New thread");
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Render the time mark field on a work item into a 5-char time with units plus
 | |
|  * a space
 | |
|  */
 | |
| static void slow_work_print_mark(struct seq_file *m, struct slow_work *work)
 | |
| {
 | |
| 	struct timespec now, diff;
 | |
| 
 | |
| 	now = CURRENT_TIME;
 | |
| 	diff = timespec_sub(now, work->mark);
 | |
| 
 | |
| 	if (diff.tv_sec < 0)
 | |
| 		seq_puts(m, "  -ve ");
 | |
| 	else if (diff.tv_sec == 0 && diff.tv_nsec < 1000)
 | |
| 		seq_printf(m, "%3luns ", diff.tv_nsec);
 | |
| 	else if (diff.tv_sec == 0 && diff.tv_nsec < 1000000)
 | |
| 		seq_printf(m, "%3luus ", diff.tv_nsec / 1000);
 | |
| 	else if (diff.tv_sec == 0 && diff.tv_nsec < 1000000000)
 | |
| 		seq_printf(m, "%3lums ", diff.tv_nsec / 1000000);
 | |
| 	else if (diff.tv_sec <= 1)
 | |
| 		seq_puts(m, "   1s ");
 | |
| 	else if (diff.tv_sec < 60)
 | |
| 		seq_printf(m, "%4lus ", diff.tv_sec);
 | |
| 	else if (diff.tv_sec < 60 * 60)
 | |
| 		seq_printf(m, "%4lum ", diff.tv_sec / 60);
 | |
| 	else if (diff.tv_sec < 60 * 60 * 24)
 | |
| 		seq_printf(m, "%4luh ", diff.tv_sec / 3600);
 | |
| 	else
 | |
| 		seq_puts(m, "exces ");
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Describe a slow work item for debugfs
 | |
|  */
 | |
| static int slow_work_runqueue_show(struct seq_file *m, void *v)
 | |
| {
 | |
| 	struct slow_work *work;
 | |
| 	struct list_head *p = v;
 | |
| 	unsigned long id;
 | |
| 
 | |
| 	switch ((unsigned long) v) {
 | |
| 	case 1:
 | |
| 		seq_puts(m, "THR PID   ITEM ADDR        FL MARK  DESC\n");
 | |
| 		return 0;
 | |
| 	case 2:
 | |
| 		seq_puts(m, "=== ===== ================ == ===== ==========\n");
 | |
| 		return 0;
 | |
| 
 | |
| 	case 3 ... 3 + SLOW_WORK_THREAD_LIMIT - 1:
 | |
| 		id = (unsigned long) v - 3;
 | |
| 
 | |
| 		read_lock(&slow_work_execs_lock);
 | |
| 		work = slow_work_execs[id];
 | |
| 		if (work) {
 | |
| 			smp_read_barrier_depends();
 | |
| 
 | |
| 			seq_printf(m, "%3lu %5d %16p %2lx ",
 | |
| 				   id, slow_work_pids[id], work, work->flags);
 | |
| 			slow_work_print_mark(m, work);
 | |
| 
 | |
| 			if (work->ops->desc)
 | |
| 				work->ops->desc(work, m);
 | |
| 			seq_putc(m, '\n');
 | |
| 		}
 | |
| 		read_unlock(&slow_work_execs_lock);
 | |
| 		return 0;
 | |
| 
 | |
| 	default:
 | |
| 		work = list_entry(p, struct slow_work, link);
 | |
| 		seq_printf(m, "%3s     - %16p %2lx ",
 | |
| 			   work->flags & SLOW_WORK_VERY_SLOW ? "vsq" : "sq",
 | |
| 			   work, work->flags);
 | |
| 		slow_work_print_mark(m, work);
 | |
| 
 | |
| 		if (work->ops->desc)
 | |
| 			work->ops->desc(work, m);
 | |
| 		seq_putc(m, '\n');
 | |
| 		return 0;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * map the iterator to a work item
 | |
|  */
 | |
| static void *slow_work_runqueue_index(struct seq_file *m, loff_t *_pos)
 | |
| {
 | |
| 	struct list_head *p;
 | |
| 	unsigned long count, id;
 | |
| 
 | |
| 	switch (*_pos >> ITERATOR_SHIFT) {
 | |
| 	case 0x0:
 | |
| 		if (*_pos == 0)
 | |
| 			*_pos = 1;
 | |
| 		if (*_pos < 3)
 | |
| 			return (void *)(unsigned long) *_pos;
 | |
| 		if (*_pos < 3 + SLOW_WORK_THREAD_LIMIT)
 | |
| 			for (id = *_pos - 3;
 | |
| 			     id < SLOW_WORK_THREAD_LIMIT;
 | |
| 			     id++, (*_pos)++)
 | |
| 				if (slow_work_execs[id])
 | |
| 					return (void *)(unsigned long) *_pos;
 | |
| 		*_pos = 0x1UL << ITERATOR_SHIFT;
 | |
| 
 | |
| 	case 0x1:
 | |
| 		count = *_pos & ITERATOR_COUNTER;
 | |
| 		list_for_each(p, &slow_work_queue) {
 | |
| 			if (count == 0)
 | |
| 				return p;
 | |
| 			count--;
 | |
| 		}
 | |
| 		*_pos = 0x2UL << ITERATOR_SHIFT;
 | |
| 
 | |
| 	case 0x2:
 | |
| 		count = *_pos & ITERATOR_COUNTER;
 | |
| 		list_for_each(p, &vslow_work_queue) {
 | |
| 			if (count == 0)
 | |
| 				return p;
 | |
| 			count--;
 | |
| 		}
 | |
| 		*_pos = 0x3UL << ITERATOR_SHIFT;
 | |
| 
 | |
| 	default:
 | |
| 		return NULL;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * set up the iterator to start reading from the first line
 | |
|  */
 | |
| static void *slow_work_runqueue_start(struct seq_file *m, loff_t *_pos)
 | |
| {
 | |
| 	spin_lock_irq(&slow_work_queue_lock);
 | |
| 	return slow_work_runqueue_index(m, _pos);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * move to the next line
 | |
|  */
 | |
| static void *slow_work_runqueue_next(struct seq_file *m, void *v, loff_t *_pos)
 | |
| {
 | |
| 	struct list_head *p = v;
 | |
| 	unsigned long selector = *_pos >> ITERATOR_SHIFT;
 | |
| 
 | |
| 	(*_pos)++;
 | |
| 	switch (selector) {
 | |
| 	case 0x0:
 | |
| 		return slow_work_runqueue_index(m, _pos);
 | |
| 
 | |
| 	case 0x1:
 | |
| 		if (*_pos >> ITERATOR_SHIFT == 0x1) {
 | |
| 			p = p->next;
 | |
| 			if (p != &slow_work_queue)
 | |
| 				return p;
 | |
| 		}
 | |
| 		*_pos = 0x2UL << ITERATOR_SHIFT;
 | |
| 		p = &vslow_work_queue;
 | |
| 
 | |
| 	case 0x2:
 | |
| 		if (*_pos >> ITERATOR_SHIFT == 0x2) {
 | |
| 			p = p->next;
 | |
| 			if (p != &vslow_work_queue)
 | |
| 				return p;
 | |
| 		}
 | |
| 		*_pos = 0x3UL << ITERATOR_SHIFT;
 | |
| 
 | |
| 	default:
 | |
| 		return NULL;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * clean up after reading
 | |
|  */
 | |
| static void slow_work_runqueue_stop(struct seq_file *m, void *v)
 | |
| {
 | |
| 	spin_unlock_irq(&slow_work_queue_lock);
 | |
| }
 | |
| 
 | |
| static const struct seq_operations slow_work_runqueue_ops = {
 | |
| 	.start		= slow_work_runqueue_start,
 | |
| 	.stop		= slow_work_runqueue_stop,
 | |
| 	.next		= slow_work_runqueue_next,
 | |
| 	.show		= slow_work_runqueue_show,
 | |
| };
 | |
| 
 | |
| /*
 | |
|  * open "/sys/kernel/debug/slow_work/runqueue" to list queue contents
 | |
|  */
 | |
| static int slow_work_runqueue_open(struct inode *inode, struct file *file)
 | |
| {
 | |
| 	return seq_open(file, &slow_work_runqueue_ops);
 | |
| }
 | |
| 
 | |
| const struct file_operations slow_work_runqueue_fops = {
 | |
| 	.owner		= THIS_MODULE,
 | |
| 	.open		= slow_work_runqueue_open,
 | |
| 	.read		= seq_read,
 | |
| 	.llseek		= seq_lseek,
 | |
| 	.release	= seq_release,
 | |
| };
 |