diff options
Diffstat (limited to 'fs/proc/stat.c')
| -rw-r--r-- | fs/proc/stat.c | 198 | 
1 files changed, 117 insertions, 81 deletions
diff --git a/fs/proc/stat.c b/fs/proc/stat.c index e15a19c93ba..bf2d03f8fd3 100644 --- a/fs/proc/stat.c +++ b/fs/proc/stat.c @@ -9,7 +9,8 @@  #include <linux/slab.h>  #include <linux/time.h>  #include <linux/irqnr.h> -#include <asm/cputime.h> +#include <linux/cputime.h> +#include <linux/tick.h>  #ifndef arch_irq_stat_cpu  #define arch_irq_stat_cpu(cpu) 0 @@ -17,40 +18,93 @@  #ifndef arch_irq_stat  #define arch_irq_stat() 0  #endif -#ifndef arch_idle_time -#define arch_idle_time(cpu) 0 + +#ifdef arch_idle_time + +static cputime64_t get_idle_time(int cpu) +{ +	cputime64_t idle; + +	idle = kcpustat_cpu(cpu).cpustat[CPUTIME_IDLE]; +	if (cpu_online(cpu) && !nr_iowait_cpu(cpu)) +		idle += arch_idle_time(cpu); +	return idle; +} + +static cputime64_t get_iowait_time(int cpu) +{ +	cputime64_t iowait; + +	iowait = kcpustat_cpu(cpu).cpustat[CPUTIME_IOWAIT]; +	if (cpu_online(cpu) && nr_iowait_cpu(cpu)) +		iowait += arch_idle_time(cpu); +	return iowait; +} + +#else + +static u64 get_idle_time(int cpu) +{ +	u64 idle, idle_time = -1ULL; + +	if (cpu_online(cpu)) +		idle_time = get_cpu_idle_time_us(cpu, NULL); + +	if (idle_time == -1ULL) +		/* !NO_HZ or cpu offline so we can rely on cpustat.idle */ +		idle = kcpustat_cpu(cpu).cpustat[CPUTIME_IDLE]; +	else +		idle = usecs_to_cputime64(idle_time); + +	return idle; +} + +static u64 get_iowait_time(int cpu) +{ +	u64 iowait, iowait_time = -1ULL; + +	if (cpu_online(cpu)) +		iowait_time = get_cpu_iowait_time_us(cpu, NULL); + +	if (iowait_time == -1ULL) +		/* !NO_HZ or cpu offline so we can rely on cpustat.iowait */ +		iowait = kcpustat_cpu(cpu).cpustat[CPUTIME_IOWAIT]; +	else +		iowait = usecs_to_cputime64(iowait_time); + +	return iowait; +} +  #endif  static int show_stat(struct seq_file *p, void *v)  {  	int i, j;  	unsigned long jif; -	cputime64_t user, nice, system, idle, iowait, irq, softirq, steal; -	cputime64_t guest, guest_nice; +	u64 user, nice, system, idle, iowait, irq, softirq, steal; +	u64 guest, guest_nice;  	u64 sum = 0;  	u64 sum_softirq = 0;  	unsigned int per_softirq_sums[NR_SOFTIRQS] = {0};  	struct timespec boottime;  	user = nice = system = idle = iowait = -		irq = softirq = steal = cputime64_zero; -	guest = guest_nice = cputime64_zero; +		irq = softirq = steal = 0; +	guest = guest_nice = 0;  	getboottime(&boottime);  	jif = boottime.tv_sec;  	for_each_possible_cpu(i) { -		user = cputime64_add(user, kstat_cpu(i).cpustat.user); -		nice = cputime64_add(nice, kstat_cpu(i).cpustat.nice); -		system = cputime64_add(system, kstat_cpu(i).cpustat.system); -		idle = cputime64_add(idle, kstat_cpu(i).cpustat.idle); -		idle = cputime64_add(idle, arch_idle_time(i)); -		iowait = cputime64_add(iowait, kstat_cpu(i).cpustat.iowait); -		irq = cputime64_add(irq, kstat_cpu(i).cpustat.irq); -		softirq = cputime64_add(softirq, kstat_cpu(i).cpustat.softirq); -		steal = cputime64_add(steal, kstat_cpu(i).cpustat.steal); -		guest = cputime64_add(guest, kstat_cpu(i).cpustat.guest); -		guest_nice = cputime64_add(guest_nice, -			kstat_cpu(i).cpustat.guest_nice); +		user += kcpustat_cpu(i).cpustat[CPUTIME_USER]; +		nice += kcpustat_cpu(i).cpustat[CPUTIME_NICE]; +		system += kcpustat_cpu(i).cpustat[CPUTIME_SYSTEM]; +		idle += get_idle_time(i); +		iowait += get_iowait_time(i); +		irq += kcpustat_cpu(i).cpustat[CPUTIME_IRQ]; +		softirq += kcpustat_cpu(i).cpustat[CPUTIME_SOFTIRQ]; +		steal += kcpustat_cpu(i).cpustat[CPUTIME_STEAL]; +		guest += kcpustat_cpu(i).cpustat[CPUTIME_GUEST]; +		guest_nice += kcpustat_cpu(i).cpustat[CPUTIME_GUEST_NICE];  		sum += kstat_cpu_irqs_sum(i);  		sum += arch_irq_stat_cpu(i); @@ -63,52 +117,49 @@ static int show_stat(struct seq_file *p, void *v)  	}  	sum += arch_irq_stat(); -	seq_printf(p, "cpu  %llu %llu %llu %llu %llu %llu %llu %llu %llu " -		"%llu\n", -		(unsigned long long)cputime64_to_clock_t(user), -		(unsigned long long)cputime64_to_clock_t(nice), -		(unsigned long long)cputime64_to_clock_t(system), -		(unsigned long long)cputime64_to_clock_t(idle), -		(unsigned long long)cputime64_to_clock_t(iowait), -		(unsigned long long)cputime64_to_clock_t(irq), -		(unsigned long long)cputime64_to_clock_t(softirq), -		(unsigned long long)cputime64_to_clock_t(steal), -		(unsigned long long)cputime64_to_clock_t(guest), -		(unsigned long long)cputime64_to_clock_t(guest_nice)); -	for_each_online_cpu(i) { +	seq_puts(p, "cpu "); +	seq_put_decimal_ull(p, ' ', cputime64_to_clock_t(user)); +	seq_put_decimal_ull(p, ' ', cputime64_to_clock_t(nice)); +	seq_put_decimal_ull(p, ' ', cputime64_to_clock_t(system)); +	seq_put_decimal_ull(p, ' ', cputime64_to_clock_t(idle)); +	seq_put_decimal_ull(p, ' ', cputime64_to_clock_t(iowait)); +	seq_put_decimal_ull(p, ' ', cputime64_to_clock_t(irq)); +	seq_put_decimal_ull(p, ' ', cputime64_to_clock_t(softirq)); +	seq_put_decimal_ull(p, ' ', cputime64_to_clock_t(steal)); +	seq_put_decimal_ull(p, ' ', cputime64_to_clock_t(guest)); +	seq_put_decimal_ull(p, ' ', cputime64_to_clock_t(guest_nice)); +	seq_putc(p, '\n'); +	for_each_online_cpu(i) {  		/* Copy values here to work around gcc-2.95.3, gcc-2.96 */ -		user = kstat_cpu(i).cpustat.user; -		nice = kstat_cpu(i).cpustat.nice; -		system = kstat_cpu(i).cpustat.system; -		idle = kstat_cpu(i).cpustat.idle; -		idle = cputime64_add(idle, arch_idle_time(i)); -		iowait = kstat_cpu(i).cpustat.iowait; -		irq = kstat_cpu(i).cpustat.irq; -		softirq = kstat_cpu(i).cpustat.softirq; -		steal = kstat_cpu(i).cpustat.steal; -		guest = kstat_cpu(i).cpustat.guest; -		guest_nice = kstat_cpu(i).cpustat.guest_nice; -		seq_printf(p, -			"cpu%d %llu %llu %llu %llu %llu %llu %llu %llu %llu " -			"%llu\n", -			i, -			(unsigned long long)cputime64_to_clock_t(user), -			(unsigned long long)cputime64_to_clock_t(nice), -			(unsigned long long)cputime64_to_clock_t(system), -			(unsigned long long)cputime64_to_clock_t(idle), -			(unsigned long long)cputime64_to_clock_t(iowait), -			(unsigned long long)cputime64_to_clock_t(irq), -			(unsigned long long)cputime64_to_clock_t(softirq), -			(unsigned long long)cputime64_to_clock_t(steal), -			(unsigned long long)cputime64_to_clock_t(guest), -			(unsigned long long)cputime64_to_clock_t(guest_nice)); +		user = kcpustat_cpu(i).cpustat[CPUTIME_USER]; +		nice = kcpustat_cpu(i).cpustat[CPUTIME_NICE]; +		system = kcpustat_cpu(i).cpustat[CPUTIME_SYSTEM]; +		idle = get_idle_time(i); +		iowait = get_iowait_time(i); +		irq = kcpustat_cpu(i).cpustat[CPUTIME_IRQ]; +		softirq = kcpustat_cpu(i).cpustat[CPUTIME_SOFTIRQ]; +		steal = kcpustat_cpu(i).cpustat[CPUTIME_STEAL]; +		guest = kcpustat_cpu(i).cpustat[CPUTIME_GUEST]; +		guest_nice = kcpustat_cpu(i).cpustat[CPUTIME_GUEST_NICE]; +		seq_printf(p, "cpu%d", i); +		seq_put_decimal_ull(p, ' ', cputime64_to_clock_t(user)); +		seq_put_decimal_ull(p, ' ', cputime64_to_clock_t(nice)); +		seq_put_decimal_ull(p, ' ', cputime64_to_clock_t(system)); +		seq_put_decimal_ull(p, ' ', cputime64_to_clock_t(idle)); +		seq_put_decimal_ull(p, ' ', cputime64_to_clock_t(iowait)); +		seq_put_decimal_ull(p, ' ', cputime64_to_clock_t(irq)); +		seq_put_decimal_ull(p, ' ', cputime64_to_clock_t(softirq)); +		seq_put_decimal_ull(p, ' ', cputime64_to_clock_t(steal)); +		seq_put_decimal_ull(p, ' ', cputime64_to_clock_t(guest)); +		seq_put_decimal_ull(p, ' ', cputime64_to_clock_t(guest_nice)); +		seq_putc(p, '\n');  	}  	seq_printf(p, "intr %llu", (unsigned long long)sum);  	/* sum again ? it could be updated? */  	for_each_irq_nr(j) -		seq_printf(p, " %u", kstat_irqs(j)); +		seq_put_decimal_ull(p, ' ', kstat_irqs(j));  	seq_printf(p,  		"\nctxt %llu\n" @@ -125,34 +176,19 @@ static int show_stat(struct seq_file *p, void *v)  	seq_printf(p, "softirq %llu", (unsigned long long)sum_softirq);  	for (i = 0; i < NR_SOFTIRQS; i++) -		seq_printf(p, " %u", per_softirq_sums[i]); -	seq_printf(p, "\n"); +		seq_put_decimal_ull(p, ' ', per_softirq_sums[i]); +	seq_putc(p, '\n');  	return 0;  }  static int stat_open(struct inode *inode, struct file *file)  { -	unsigned size = 4096 * (1 + num_possible_cpus() / 32); -	char *buf; -	struct seq_file *m; -	int res; - -	/* don't ask for more than the kmalloc() max size, currently 128 KB */ -	if (size > 128 * 1024) -		size = 128 * 1024; -	buf = kmalloc(size, GFP_KERNEL); -	if (!buf) -		return -ENOMEM; - -	res = single_open(file, show_stat, NULL); -	if (!res) { -		m = file->private_data; -		m->buf = buf; -		m->size = size; -	} else -		kfree(buf); -	return res; +	size_t size = 1024 + 128 * num_online_cpus(); + +	/* minimum size to display an interrupt count : 2 bytes */ +	size += 2 * nr_irqs; +	return single_open_size(file, show_stat, NULL, size);  }  static const struct file_operations proc_stat_operations = { @@ -167,4 +203,4 @@ static int __init proc_stat_init(void)  	proc_create("stat", 0, NULL, &proc_stat_operations);  	return 0;  } -module_init(proc_stat_init); +fs_initcall(proc_stat_init);  | 
