diff options
Diffstat (limited to 'tools/power/x86/turbostat/turbostat.c')
| -rw-r--r-- | tools/power/x86/turbostat/turbostat.c | 677 | 
1 files changed, 639 insertions, 38 deletions
diff --git a/tools/power/x86/turbostat/turbostat.c b/tools/power/x86/turbostat/turbostat.c index ea095abbe97..ce6d46038f7 100644 --- a/tools/power/x86/turbostat/turbostat.c +++ b/tools/power/x86/turbostat/turbostat.c @@ -20,6 +20,7 @@   */  #define _GNU_SOURCE +#include <asm/msr.h>  #include <stdio.h>  #include <unistd.h>  #include <sys/types.h> @@ -35,28 +36,18 @@  #include <ctype.h>  #include <sched.h> -#define MSR_NEHALEM_PLATFORM_INFO	0xCE -#define MSR_NEHALEM_TURBO_RATIO_LIMIT	0x1AD -#define MSR_IVT_TURBO_RATIO_LIMIT	0x1AE -#define MSR_APERF	0xE8 -#define MSR_MPERF	0xE7 -#define MSR_PKG_C2_RESIDENCY	0x60D	/* SNB only */ -#define MSR_PKG_C3_RESIDENCY	0x3F8 -#define MSR_PKG_C6_RESIDENCY	0x3F9 -#define MSR_PKG_C7_RESIDENCY	0x3FA	/* SNB only */ -#define MSR_CORE_C3_RESIDENCY	0x3FC -#define MSR_CORE_C6_RESIDENCY	0x3FD -#define MSR_CORE_C7_RESIDENCY	0x3FE	/* SNB only */ -  char *proc_stat = "/proc/stat";  unsigned int interval_sec = 5;	/* set with -i interval_sec */  unsigned int verbose;		/* set with -v */ +unsigned int rapl_verbose;	/* set with -R */ +unsigned int thermal_verbose;	/* set with -T */  unsigned int summary_only;	/* set with -s */  unsigned int skip_c0;  unsigned int skip_c1;  unsigned int do_nhm_cstates;  unsigned int do_snb_cstates;  unsigned int has_aperf; +unsigned int has_epb;  unsigned int units = 1000000000;	/* Ghz etc */  unsigned int genuine_intel;  unsigned int has_invariant_tsc; @@ -74,6 +65,23 @@ unsigned int show_cpu;  unsigned int show_pkg_only;  unsigned int show_core_only;  char *output_buffer, *outp; +unsigned int do_rapl; +unsigned int do_dts; +unsigned int do_ptm; +unsigned int tcc_activation_temp; +unsigned int tcc_activation_temp_override; +double rapl_power_units, rapl_energy_units, rapl_time_units; +double rapl_joule_counter_range; + +#define RAPL_PKG	(1 << 0) +#define RAPL_CORES	(1 << 1) +#define RAPL_GFX	(1 << 2) +#define RAPL_DRAM	(1 << 3) +#define RAPL_PKG_PERF_STATUS	(1 << 4) +#define RAPL_DRAM_PERF_STATUS	(1 << 5) +#define	TJMAX_DEFAULT	100 + +#define MAX(a, b) ((a) > (b) ? (a) : (b))  int aperf_mperf_unstable;  int backwards_count; @@ -101,6 +109,7 @@ struct core_data {  	unsigned long long c3;  	unsigned long long c6;  	unsigned long long c7; +	unsigned int core_temp_c;  	unsigned int core_id;  } *core_even, *core_odd; @@ -110,6 +119,14 @@ struct pkg_data {  	unsigned long long pc6;  	unsigned long long pc7;  	unsigned int package_id; +	unsigned int energy_pkg;	/* MSR_PKG_ENERGY_STATUS */ +	unsigned int energy_dram;	/* MSR_DRAM_ENERGY_STATUS */ +	unsigned int energy_cores;	/* MSR_PP0_ENERGY_STATUS */ +	unsigned int energy_gfx;	/* MSR_PP1_ENERGY_STATUS */ +	unsigned int rapl_pkg_perf_status;	/* MSR_PKG_PERF_STATUS */ +	unsigned int rapl_dram_perf_status;	/* MSR_DRAM_PERF_STATUS */ +	unsigned int pkg_temp_c; +  } *package_even, *package_odd;  #define ODD_COUNTERS thread_odd, core_odd, package_odd @@ -247,6 +264,12 @@ void print_header(void)  		outp += sprintf(outp, "    %%c6");  	if (do_snb_cstates)  		outp += sprintf(outp, "    %%c7"); + +	if (do_dts) +		outp += sprintf(outp, " CTMP"); +	if (do_ptm) +		outp += sprintf(outp, " PTMP"); +  	if (do_snb_cstates)  		outp += sprintf(outp, "   %%pc2");  	if (do_nhm_cstates) @@ -256,6 +279,19 @@ void print_header(void)  	if (do_snb_cstates)  		outp += sprintf(outp, "   %%pc7"); +	if (do_rapl & RAPL_PKG) +		outp += sprintf(outp, "  Pkg_W"); +	if (do_rapl & RAPL_CORES) +		outp += sprintf(outp, "  Cor_W"); +	if (do_rapl & RAPL_GFX) +		outp += sprintf(outp, " GFX_W"); +	if (do_rapl & RAPL_DRAM) +		outp += sprintf(outp, " RAM_W"); +	if (do_rapl & RAPL_PKG_PERF_STATUS) +		outp += sprintf(outp, " PKG_%%"); +	if (do_rapl & RAPL_DRAM_PERF_STATUS) +		outp += sprintf(outp, " RAM_%%"); +  	outp += sprintf(outp, "\n");  } @@ -285,6 +321,7 @@ int dump_counters(struct thread_data *t, struct core_data *c,  		fprintf(stderr, "c3: %016llX\n", c->c3);  		fprintf(stderr, "c6: %016llX\n", c->c6);  		fprintf(stderr, "c7: %016llX\n", c->c7); +		fprintf(stderr, "DTS: %dC\n", c->core_temp_c);  	}  	if (p) { @@ -293,6 +330,13 @@ int dump_counters(struct thread_data *t, struct core_data *c,  		fprintf(stderr, "pc3: %016llX\n", p->pc3);  		fprintf(stderr, "pc6: %016llX\n", p->pc6);  		fprintf(stderr, "pc7: %016llX\n", p->pc7); +		fprintf(stderr, "Joules PKG: %0X\n", p->energy_pkg); +		fprintf(stderr, "Joules COR: %0X\n", p->energy_cores); +		fprintf(stderr, "Joules GFX: %0X\n", p->energy_gfx); +		fprintf(stderr, "Joules RAM: %0X\n", p->energy_dram); +		fprintf(stderr, "Throttle PKG: %0X\n", p->rapl_pkg_perf_status); +		fprintf(stderr, "Throttle RAM: %0X\n", p->rapl_dram_perf_status); +		fprintf(stderr, "PTM: %dC\n", p->pkg_temp_c);  	}  	return 0;  } @@ -302,14 +346,21 @@ int dump_counters(struct thread_data *t, struct core_data *c,   * package: "pk" 2 columns %2d   * core: "cor" 3 columns %3d   * CPU: "CPU" 3 columns %3d + * Pkg_W: %6.2 + * Cor_W: %6.2 + * GFX_W: %5.2 + * RAM_W: %5.2   * GHz: "GHz" 3 columns %3.2   * TSC: "TSC" 3 columns %3.2   * percentage " %pc3" %6.2 + * Perf Status percentage: %5.2 + * "CTMP" 4 columns %4d   */  int format_counters(struct thread_data *t, struct core_data *c,  	struct pkg_data *p)  {  	double interval_float; +	char *fmt5, *fmt6;  	 /* if showing only 1st thread in core and this isn't one, bail out */  	if (show_core_only && !(t->flags & CPU_IS_FIRST_THREAD_IN_CORE)) @@ -349,7 +400,6 @@ int format_counters(struct thread_data *t, struct core_data *c,  		if (show_cpu)  			outp += sprintf(outp, " %3d", t->cpu_id);  	} -  	/* %c0 */  	if (do_nhm_cstates) {  		if (show_pkg || show_core || show_cpu) @@ -414,10 +464,16 @@ int format_counters(struct thread_data *t, struct core_data *c,  	if (do_snb_cstates)  		outp += sprintf(outp, " %6.2f", 100.0 * c->c7/t->tsc); +	if (do_dts) +		outp += sprintf(outp, " %4d", c->core_temp_c); +  	/* print per-package data only for 1st core in package */  	if (!(t->flags & CPU_IS_FIRST_CORE_IN_PACKAGE))  		goto done; +	if (do_ptm) +		outp += sprintf(outp, " %4d", p->pkg_temp_c); +  	if (do_snb_cstates)  		outp += sprintf(outp, " %6.2f", 100.0 * p->pc2/t->tsc);  	if (do_nhm_cstates) @@ -426,6 +482,32 @@ int format_counters(struct thread_data *t, struct core_data *c,  		outp += sprintf(outp, " %6.2f", 100.0 * p->pc6/t->tsc);  	if (do_snb_cstates)  		outp += sprintf(outp, " %6.2f", 100.0 * p->pc7/t->tsc); + +	/* + 	 * If measurement interval exceeds minimum RAPL Joule Counter range, + 	 * indicate that results are suspect by printing "**" in fraction place. + 	 */ +	if (interval_float < rapl_joule_counter_range) { +		fmt5 = " %5.2f"; +		fmt6 = " %6.2f"; +	} else { +		fmt5 = " %3.0f**"; +		fmt6 = " %4.0f**"; +	} + +	if (do_rapl & RAPL_PKG) +		outp += sprintf(outp, fmt6, p->energy_pkg * rapl_energy_units / interval_float); +	if (do_rapl & RAPL_CORES) +		outp += sprintf(outp, fmt6, p->energy_cores * rapl_energy_units / interval_float); +	if (do_rapl & RAPL_GFX) +		outp += sprintf(outp, fmt5, p->energy_gfx * rapl_energy_units / interval_float);  +	if (do_rapl & RAPL_DRAM) +		outp += sprintf(outp, fmt5, p->energy_dram * rapl_energy_units / interval_float); +	if (do_rapl & RAPL_PKG_PERF_STATUS ) +		outp += sprintf(outp, fmt5, 100.0 * p->rapl_pkg_perf_status * rapl_time_units / interval_float); +	if (do_rapl & RAPL_DRAM_PERF_STATUS ) +		outp += sprintf(outp, fmt5, 100.0 * p->rapl_dram_perf_status * rapl_time_units / interval_float); +  done:  	outp += sprintf(outp, "\n"); @@ -435,6 +517,7 @@ done:  void flush_stdout()  {  	fputs(output_buffer, stdout); +	fflush(stdout);  	outp = output_buffer;  }  void flush_stderr() @@ -461,6 +544,13 @@ void format_all_counters(struct thread_data *t, struct core_data *c, struct pkg_  	for_all_cpus(format_counters, t, c, p);  } +#define DELTA_WRAP32(new, old)			\ +	if (new > old) {			\ +		old = new - old;		\ +	} else {				\ +		old = 0x100000000 + new - old;	\ +	} +  void  delta_package(struct pkg_data *new, struct pkg_data *old)  { @@ -468,6 +558,14 @@ delta_package(struct pkg_data *new, struct pkg_data *old)  	old->pc3 = new->pc3 - old->pc3;  	old->pc6 = new->pc6 - old->pc6;  	old->pc7 = new->pc7 - old->pc7; +	old->pkg_temp_c = new->pkg_temp_c; + +	DELTA_WRAP32(new->energy_pkg, old->energy_pkg); +	DELTA_WRAP32(new->energy_cores, old->energy_cores); +	DELTA_WRAP32(new->energy_gfx, old->energy_gfx); +	DELTA_WRAP32(new->energy_dram, old->energy_dram); +	DELTA_WRAP32(new->rapl_pkg_perf_status, old->rapl_pkg_perf_status); +	DELTA_WRAP32(new->rapl_dram_perf_status, old->rapl_dram_perf_status);  }  void @@ -476,6 +574,7 @@ delta_core(struct core_data *new, struct core_data *old)  	old->c3 = new->c3 - old->c3;  	old->c6 = new->c6 - old->c6;  	old->c7 = new->c7 - old->c7; +	old->core_temp_c = new->core_temp_c;  }  /* @@ -582,11 +681,20 @@ void clear_counters(struct thread_data *t, struct core_data *c, struct pkg_data  	c->c3 = 0;  	c->c6 = 0;  	c->c7 = 0; +	c->core_temp_c = 0;  	p->pc2 = 0;  	p->pc3 = 0;  	p->pc6 = 0;  	p->pc7 = 0; + +	p->energy_pkg = 0; +	p->energy_dram = 0; +	p->energy_cores = 0; +	p->energy_gfx = 0; +	p->rapl_pkg_perf_status = 0; +	p->rapl_dram_perf_status = 0; +	p->pkg_temp_c = 0;  }  int sum_counters(struct thread_data *t, struct core_data *c,  	struct pkg_data *p) @@ -607,6 +715,8 @@ int sum_counters(struct thread_data *t, struct core_data *c,  	average.cores.c6 += c->c6;  	average.cores.c7 += c->c7; +	average.cores.core_temp_c = MAX(average.cores.core_temp_c, c->core_temp_c); +  	/* sum per-pkg values only for 1st core in pkg */  	if (!(t->flags & CPU_IS_FIRST_CORE_IN_PACKAGE))  		return 0; @@ -616,6 +726,15 @@ int sum_counters(struct thread_data *t, struct core_data *c,  	average.packages.pc6 += p->pc6;  	average.packages.pc7 += p->pc7; +	average.packages.energy_pkg += p->energy_pkg; +	average.packages.energy_dram += p->energy_dram; +	average.packages.energy_cores += p->energy_cores; +	average.packages.energy_gfx += p->energy_gfx; + +	average.packages.pkg_temp_c = MAX(average.packages.pkg_temp_c, p->pkg_temp_c); + +	average.packages.rapl_pkg_perf_status += p->rapl_pkg_perf_status; +	average.packages.rapl_dram_perf_status += p->rapl_dram_perf_status;  	return 0;  }  /* @@ -667,23 +786,26 @@ static unsigned long long rdtsc(void)  int get_counters(struct thread_data *t, struct core_data *c, struct pkg_data *p)  {  	int cpu = t->cpu_id; +	unsigned long long msr; -	if (cpu_migrate(cpu)) +	if (cpu_migrate(cpu)) { +		fprintf(stderr, "Could not migrate to CPU %d\n", cpu);  		return -1; +	}  	t->tsc = rdtsc();	/* we are running on local CPU of interest */  	if (has_aperf) { -		if (get_msr(cpu, MSR_APERF, &t->aperf)) +		if (get_msr(cpu, MSR_IA32_APERF, &t->aperf))  			return -3; -		if (get_msr(cpu, MSR_MPERF, &t->mperf)) +		if (get_msr(cpu, MSR_IA32_MPERF, &t->mperf))  			return -4;  	}  	if (extra_delta_offset32) { -		if (get_msr(cpu, extra_delta_offset32, &t->extra_delta32)) +		if (get_msr(cpu, extra_delta_offset32, &msr))  			return -5; -		t->extra_delta32 &= 0xFFFFFFFF; +		t->extra_delta32 = msr & 0xFFFFFFFF;  	}  	if (extra_delta_offset64) @@ -691,9 +813,9 @@ int get_counters(struct thread_data *t, struct core_data *c, struct pkg_data *p)  			return -5;  	if (extra_msr_offset32) { -		if (get_msr(cpu, extra_msr_offset32, &t->extra_msr32)) +		if (get_msr(cpu, extra_msr_offset32, &msr))  			return -5; -		t->extra_msr32 &= 0xFFFFFFFF; +		t->extra_msr32 = msr & 0xFFFFFFFF;  	}  	if (extra_msr_offset64) @@ -715,6 +837,13 @@ int get_counters(struct thread_data *t, struct core_data *c, struct pkg_data *p)  		if (get_msr(cpu, MSR_CORE_C7_RESIDENCY, &c->c7))  			return -8; +	if (do_dts) { +		if (get_msr(cpu, MSR_IA32_THERM_STATUS, &msr)) +			return -9; +		c->core_temp_c = tcc_activation_temp - ((msr >> 16) & 0x7F); +	} + +  	/* collect package counters only for 1st core in package */  	if (!(t->flags & CPU_IS_FIRST_CORE_IN_PACKAGE))  		return 0; @@ -731,6 +860,41 @@ int get_counters(struct thread_data *t, struct core_data *c, struct pkg_data *p)  		if (get_msr(cpu, MSR_PKG_C7_RESIDENCY, &p->pc7))  			return -12;  	} +	if (do_rapl & RAPL_PKG) { +		if (get_msr(cpu, MSR_PKG_ENERGY_STATUS, &msr)) +			return -13; +		p->energy_pkg = msr & 0xFFFFFFFF; +	} +	if (do_rapl & RAPL_CORES) { +		if (get_msr(cpu, MSR_PP0_ENERGY_STATUS, &msr)) +			return -14; +		p->energy_cores = msr & 0xFFFFFFFF; +	} +	if (do_rapl & RAPL_DRAM) { +		if (get_msr(cpu, MSR_DRAM_ENERGY_STATUS, &msr)) +			return -15; +		p->energy_dram = msr & 0xFFFFFFFF; +	} +	if (do_rapl & RAPL_GFX) { +		if (get_msr(cpu, MSR_PP1_ENERGY_STATUS, &msr)) +			return -16; +		p->energy_gfx = msr & 0xFFFFFFFF; +	} +	if (do_rapl & RAPL_PKG_PERF_STATUS) { +		if (get_msr(cpu, MSR_PKG_PERF_STATUS, &msr)) +			return -16; +		p->rapl_pkg_perf_status = msr & 0xFFFFFFFF; +	} +	if (do_rapl & RAPL_DRAM_PERF_STATUS) { +		if (get_msr(cpu, MSR_DRAM_PERF_STATUS, &msr)) +			return -16; +		p->rapl_dram_perf_status = msr & 0xFFFFFFFF; +	} +	if (do_ptm) { +		if (get_msr(cpu, MSR_IA32_PACKAGE_THERM_STATUS, &msr)) +			return -17; +		p->pkg_temp_c = tcc_activation_temp - ((msr >> 16) & 0x7F); +	}  	return 0;  } @@ -742,10 +906,10 @@ void print_verbose_header(void)  	if (!do_nehalem_platform_info)  		return; -	get_msr(0, MSR_NEHALEM_PLATFORM_INFO, &msr); +	get_msr(0, MSR_NHM_PLATFORM_INFO, &msr); -	if (verbose > 1) -		fprintf(stderr, "MSR_NEHALEM_PLATFORM_INFO: 0x%llx\n", msr); +	if (verbose) +		fprintf(stderr, "cpu0: MSR_NHM_PLATFORM_INFO: 0x%08llx\n", msr);  	ratio = (msr >> 40) & 0xFF;  	fprintf(stderr, "%d * %.0f = %.0f MHz max efficiency\n", @@ -760,8 +924,8 @@ void print_verbose_header(void)  	get_msr(0, MSR_IVT_TURBO_RATIO_LIMIT, &msr); -	if (verbose > 1) -		fprintf(stderr, "MSR_IVT_TURBO_RATIO_LIMIT: 0x%llx\n", msr); +	if (verbose) +		fprintf(stderr, "cpu0: MSR_IVT_TURBO_RATIO_LIMIT: 0x%08llx\n", msr);  	ratio = (msr >> 56) & 0xFF;  	if (ratio) @@ -804,14 +968,56 @@ void print_verbose_header(void)  			ratio, bclk, ratio * bclk);  print_nhm_turbo_ratio_limits: +	get_msr(0, MSR_NHM_SNB_PKG_CST_CFG_CTL, &msr); + +#define SNB_C1_AUTO_UNDEMOTE              (1UL << 27) +#define SNB_C3_AUTO_UNDEMOTE              (1UL << 28) + +	fprintf(stderr, "cpu0: MSR_NHM_SNB_PKG_CST_CFG_CTL: 0x%08llx", msr); + +	fprintf(stderr, " (%s%s%s%s%slocked: pkg-cstate-limit=%d: ", +		(msr & SNB_C3_AUTO_UNDEMOTE) ? "UNdemote-C3, " : "", +		(msr & SNB_C1_AUTO_UNDEMOTE) ? "UNdemote-C1, " : "", +		(msr & NHM_C3_AUTO_DEMOTE) ? "demote-C3, " : "", +		(msr & NHM_C1_AUTO_DEMOTE) ? "demote-C1, " : "", +		(msr & (1 << 15)) ? "" : "UN", +		(unsigned int)msr & 7); + + +	switch(msr & 0x7) { +	case 0: +		fprintf(stderr, "pc0"); +		break; +	case 1: +		fprintf(stderr, do_snb_cstates ? "pc2" : "pc0"); +		break; +	case 2: +		fprintf(stderr, do_snb_cstates ? "pc6-noret" : "pc3"); +		break; +	case 3: +		fprintf(stderr, "pc6"); +		break; +	case 4: +		fprintf(stderr, "pc7"); +		break; +	case 5: +		fprintf(stderr, do_snb_cstates ? "pc7s" : "invalid"); +		break; +	case 7: +		fprintf(stderr, "unlimited"); +		break; +	default: +		fprintf(stderr, "invalid"); +	} +	fprintf(stderr, ")\n");  	if (!do_nehalem_turbo_ratio_limit)  		return; -	get_msr(0, MSR_NEHALEM_TURBO_RATIO_LIMIT, &msr); +	get_msr(0, MSR_NHM_TURBO_RATIO_LIMIT, &msr); -	if (verbose > 1) -		fprintf(stderr, "MSR_NEHALEM_TURBO_RATIO_LIMIT: 0x%llx\n", msr); +	if (verbose) +		fprintf(stderr, "cpu0: MSR_NHM_TURBO_RATIO_LIMIT: 0x%08llx\n", msr);  	ratio = (msr >> 56) & 0xFF;  	if (ratio) @@ -1100,15 +1306,22 @@ int mark_cpu_present(int cpu)  void turbostat_loop()  {  	int retval; +	int restarted = 0;  restart: +	restarted++; +  	retval = for_all_cpus(get_counters, EVEN_COUNTERS);  	if (retval < -1) {  		exit(retval);  	} else if (retval == -1) { +		if (restarted > 1) { +			exit(retval); +		}  		re_initialize();  		goto restart;  	} +	restarted = 0;  	gettimeofday(&tv_even, (struct timezone *)NULL);  	while (1) { @@ -1207,6 +1420,299 @@ int has_ivt_turbo_ratio_limit(unsigned int family, unsigned int model)  	}  } +/* + * print_epb() + * Decode the ENERGY_PERF_BIAS MSR + */ +int print_epb(struct thread_data *t, struct core_data *c, struct pkg_data *p) +{ +	unsigned long long msr; +	char *epb_string; +	int cpu; + +	if (!has_epb) +		return 0; + +	cpu = t->cpu_id; + +	/* EPB is per-package */ +	if (!(t->flags & CPU_IS_FIRST_THREAD_IN_CORE) || !(t->flags & CPU_IS_FIRST_CORE_IN_PACKAGE)) +		return 0; + +	if (cpu_migrate(cpu)) { +		fprintf(stderr, "Could not migrate to CPU %d\n", cpu); +		return -1; +	} + +	if (get_msr(cpu, MSR_IA32_ENERGY_PERF_BIAS, &msr)) +		return 0; + +	switch (msr & 0x7) { +	case ENERGY_PERF_BIAS_PERFORMANCE: +		epb_string = "performance"; +		break; +	case ENERGY_PERF_BIAS_NORMAL: +		epb_string = "balanced"; +		break; +	case ENERGY_PERF_BIAS_POWERSAVE: +		epb_string = "powersave"; +		break; +	default: +		epb_string = "custom"; +		break; +	} +	fprintf(stderr, "cpu%d: MSR_IA32_ENERGY_PERF_BIAS: 0x%08llx (%s)\n", cpu, msr, epb_string); + +	return 0; +} + +#define	RAPL_POWER_GRANULARITY	0x7FFF	/* 15 bit power granularity */ +#define	RAPL_TIME_GRANULARITY	0x3F /* 6 bit time granularity */ + +/* + * rapl_probe() + * + * sets do_rapl + */ +void rapl_probe(unsigned int family, unsigned int model) +{ +	unsigned long long msr; +	double tdp; + +	if (!genuine_intel) +		return; + +	if (family != 6) +		return; + +	switch (model) { +	case 0x2A: +	case 0x3A: +		do_rapl = RAPL_PKG | RAPL_CORES | RAPL_GFX; +		break; +	case 0x2D: +	case 0x3E: +		do_rapl = RAPL_PKG | RAPL_CORES | RAPL_DRAM | RAPL_PKG_PERF_STATUS | RAPL_DRAM_PERF_STATUS; +		break; +	default: +		return; +	} + +	/* units on package 0, verify later other packages match */ +	if (get_msr(0, MSR_RAPL_POWER_UNIT, &msr)) +		return; + +	rapl_power_units = 1.0 / (1 << (msr & 0xF)); +	rapl_energy_units = 1.0 / (1 << (msr >> 8 & 0x1F)); +	rapl_time_units = 1.0 / (1 << (msr >> 16 & 0xF)); + +	/* get TDP to determine energy counter range */ +	if (get_msr(0, MSR_PKG_POWER_INFO, &msr)) +		return; + +	tdp = ((msr >> 0) & RAPL_POWER_GRANULARITY) * rapl_power_units; + +	rapl_joule_counter_range = 0xFFFFFFFF * rapl_energy_units / tdp; + +	if (verbose) +		fprintf(stderr, "RAPL: %.0f sec. Joule Counter Range\n", rapl_joule_counter_range); + +	return; +} + +int print_thermal(struct thread_data *t, struct core_data *c, struct pkg_data *p) +{ +	unsigned long long msr; +	unsigned int dts; +	int cpu; + +	if (!(do_dts || do_ptm)) +		return 0; + +	cpu = t->cpu_id; + +	/* DTS is per-core, no need to print for each thread */ +	if (!(t->flags & CPU_IS_FIRST_THREAD_IN_CORE))  +		return 0; + +	if (cpu_migrate(cpu)) { +		fprintf(stderr, "Could not migrate to CPU %d\n", cpu); +		return -1; +	} + +	if (do_ptm && (t->flags & CPU_IS_FIRST_CORE_IN_PACKAGE)) { +		if (get_msr(cpu, MSR_IA32_PACKAGE_THERM_STATUS, &msr)) +			return 0; + +		dts = (msr >> 16) & 0x7F; +		fprintf(stderr, "cpu%d: MSR_IA32_PACKAGE_THERM_STATUS: 0x%08llx (%d C)\n", +			cpu, msr, tcc_activation_temp - dts); + +#ifdef	THERM_DEBUG +		if (get_msr(cpu, MSR_IA32_PACKAGE_THERM_INTERRUPT, &msr)) +			return 0; + +		dts = (msr >> 16) & 0x7F; +		dts2 = (msr >> 8) & 0x7F; +		fprintf(stderr, "cpu%d: MSR_IA32_PACKAGE_THERM_INTERRUPT: 0x%08llx (%d C, %d C)\n", +			cpu, msr, tcc_activation_temp - dts, tcc_activation_temp - dts2); +#endif +	} + + +	if (do_dts) { +		unsigned int resolution; + +		if (get_msr(cpu, MSR_IA32_THERM_STATUS, &msr)) +			return 0; + +		dts = (msr >> 16) & 0x7F; +		resolution = (msr >> 27) & 0xF; +		fprintf(stderr, "cpu%d: MSR_IA32_THERM_STATUS: 0x%08llx (%d C +/- %d)\n", +			cpu, msr, tcc_activation_temp - dts, resolution); + +#ifdef THERM_DEBUG +		if (get_msr(cpu, MSR_IA32_THERM_INTERRUPT, &msr)) +			return 0; + +		dts = (msr >> 16) & 0x7F; +		dts2 = (msr >> 8) & 0x7F; +		fprintf(stderr, "cpu%d: MSR_IA32_THERM_INTERRUPT: 0x%08llx (%d C, %d C)\n", +			cpu, msr, tcc_activation_temp - dts, tcc_activation_temp - dts2); +#endif +	} + +	return 0; +} +	 +void print_power_limit_msr(int cpu, unsigned long long msr, char *label) +{ +	fprintf(stderr, "cpu%d: %s: %sabled (%f Watts, %f sec, clamp %sabled)\n", +		cpu, label, +		((msr >> 15) & 1) ? "EN" : "DIS", +		((msr >> 0) & 0x7FFF) * rapl_power_units, +		(1.0 + (((msr >> 22) & 0x3)/4.0)) * (1 << ((msr >> 17) & 0x1F)) * rapl_time_units, +		(((msr >> 16) & 1) ? "EN" : "DIS")); + +	return; +} + +int print_rapl(struct thread_data *t, struct core_data *c, struct pkg_data *p) +{ +	unsigned long long msr; +	int cpu; +	double local_rapl_power_units, local_rapl_energy_units, local_rapl_time_units; + +	if (!do_rapl) +		return 0; + +	/* RAPL counters are per package, so print only for 1st thread/package */ +	if (!(t->flags & CPU_IS_FIRST_THREAD_IN_CORE) || !(t->flags & CPU_IS_FIRST_CORE_IN_PACKAGE)) +		return 0; + +	cpu = t->cpu_id; +	if (cpu_migrate(cpu)) { +		fprintf(stderr, "Could not migrate to CPU %d\n", cpu); +		return -1; +	} + +	if (get_msr(cpu, MSR_RAPL_POWER_UNIT, &msr)) +		return -1; + +	local_rapl_power_units = 1.0 / (1 << (msr & 0xF)); +	local_rapl_energy_units = 1.0 / (1 << (msr >> 8 & 0x1F)); +	local_rapl_time_units = 1.0 / (1 << (msr >> 16 & 0xF)); + +	if (local_rapl_power_units != rapl_power_units) +		fprintf(stderr, "cpu%d, ERROR: Power units mis-match\n", cpu); +	if (local_rapl_energy_units != rapl_energy_units) +		fprintf(stderr, "cpu%d, ERROR: Energy units mis-match\n", cpu); +	if (local_rapl_time_units != rapl_time_units) +		fprintf(stderr, "cpu%d, ERROR: Time units mis-match\n", cpu); + +	if (verbose) { +		fprintf(stderr, "cpu%d: MSR_RAPL_POWER_UNIT: 0x%08llx " +			"(%f Watts, %f Joules, %f sec.)\n", cpu, msr, +			local_rapl_power_units, local_rapl_energy_units, local_rapl_time_units); +	} +	if (do_rapl & RAPL_PKG) { +		if (get_msr(cpu, MSR_PKG_POWER_INFO, &msr)) +                	return -5; + + +		fprintf(stderr, "cpu%d: MSR_PKG_POWER_INFO: 0x%08llx (%.0f W TDP, RAPL %.0f - %.0f W, %f sec.)\n", +			cpu, msr, +			((msr >>  0) & RAPL_POWER_GRANULARITY) * rapl_power_units, +			((msr >> 16) & RAPL_POWER_GRANULARITY) * rapl_power_units, +			((msr >> 32) & RAPL_POWER_GRANULARITY) * rapl_power_units, +			((msr >> 48) & RAPL_TIME_GRANULARITY) * rapl_time_units); + +		if (get_msr(cpu, MSR_PKG_POWER_LIMIT, &msr)) +			return -9; + +		fprintf(stderr, "cpu%d: MSR_PKG_POWER_LIMIT: 0x%08llx (%slocked)\n", +			cpu, msr, (msr >> 63) & 1 ? "": "UN"); + +		print_power_limit_msr(cpu, msr, "PKG Limit #1"); +		fprintf(stderr, "cpu%d: PKG Limit #2: %sabled (%f Watts, %f* sec, clamp %sabled)\n", +			cpu, +			((msr >> 47) & 1) ? "EN" : "DIS", +			((msr >> 32) & 0x7FFF) * rapl_power_units, +			(1.0 + (((msr >> 54) & 0x3)/4.0)) * (1 << ((msr >> 49) & 0x1F)) * rapl_time_units, +			((msr >> 48) & 1) ? "EN" : "DIS"); +	} + +	if (do_rapl & RAPL_DRAM) { +		if (get_msr(cpu, MSR_DRAM_POWER_INFO, &msr)) +                	return -6; + + +		fprintf(stderr, "cpu%d: MSR_DRAM_POWER_INFO,: 0x%08llx (%.0f W TDP, RAPL %.0f - %.0f W, %f sec.)\n", +			cpu, msr, +			((msr >>  0) & RAPL_POWER_GRANULARITY) * rapl_power_units, +			((msr >> 16) & RAPL_POWER_GRANULARITY) * rapl_power_units, +			((msr >> 32) & RAPL_POWER_GRANULARITY) * rapl_power_units, +			((msr >> 48) & RAPL_TIME_GRANULARITY) * rapl_time_units); + + +		if (get_msr(cpu, MSR_DRAM_POWER_LIMIT, &msr)) +			return -9; +		fprintf(stderr, "cpu%d: MSR_DRAM_POWER_LIMIT: 0x%08llx (%slocked)\n", +				cpu, msr, (msr >> 31) & 1 ? "": "UN"); + +		print_power_limit_msr(cpu, msr, "DRAM Limit"); +	} +	if (do_rapl & RAPL_CORES) { +		if (verbose) { +			if (get_msr(cpu, MSR_PP0_POLICY, &msr)) +				return -7; + +			fprintf(stderr, "cpu%d: MSR_PP0_POLICY: %lld\n", cpu, msr & 0xF); + +			if (get_msr(cpu, MSR_PP0_POWER_LIMIT, &msr)) +				return -9; +			fprintf(stderr, "cpu%d: MSR_PP0_POWER_LIMIT: 0x%08llx (%slocked)\n", +					cpu, msr, (msr >> 31) & 1 ? "": "UN"); +			print_power_limit_msr(cpu, msr, "Cores Limit"); +		} +	} +	if (do_rapl & RAPL_GFX) { +		if (verbose) { +			if (get_msr(cpu, MSR_PP1_POLICY, &msr)) +				return -8; + +			fprintf(stderr, "cpu%d: MSR_PP1_POLICY: %lld\n", cpu, msr & 0xF); + +			if (get_msr(cpu, MSR_PP1_POWER_LIMIT, &msr)) +				return -9; +			fprintf(stderr, "cpu%d: MSR_PP1_POWER_LIMIT: 0x%08llx (%slocked)\n", +					cpu, msr, (msr >> 31) & 1 ? "": "UN"); +			print_power_limit_msr(cpu, msr, "GFX Limit"); +		} +	} +	return 0; +} +  int is_snb(unsigned int family, unsigned int model)  { @@ -1231,6 +1737,72 @@ double discover_bclk(unsigned int family, unsigned int model)  		return 133.33;  } +/* + * MSR_IA32_TEMPERATURE_TARGET indicates the temperature where + * the Thermal Control Circuit (TCC) activates. + * This is usually equal to tjMax. + * + * Older processors do not have this MSR, so there we guess, + * but also allow cmdline over-ride with -T. + * + * Several MSR temperature values are in units of degrees-C + * below this value, including the Digital Thermal Sensor (DTS), + * Package Thermal Management Sensor (PTM), and thermal event thresholds. + */ +int set_temperature_target(struct thread_data *t, struct core_data *c, struct pkg_data *p) +{ +	unsigned long long msr; +	unsigned int target_c_local; +	int cpu; + +	/* tcc_activation_temp is used only for dts or ptm */ +	if (!(do_dts || do_ptm)) +		return 0; + +	/* this is a per-package concept */ +	if (!(t->flags & CPU_IS_FIRST_THREAD_IN_CORE) || !(t->flags & CPU_IS_FIRST_CORE_IN_PACKAGE)) +		return 0; + +	cpu = t->cpu_id; +	if (cpu_migrate(cpu)) { +		fprintf(stderr, "Could not migrate to CPU %d\n", cpu); +		return -1; +	} + +	if (tcc_activation_temp_override != 0) { +		tcc_activation_temp = tcc_activation_temp_override; +		fprintf(stderr, "cpu%d: Using cmdline TCC Target (%d C)\n", +			cpu, tcc_activation_temp); +		return 0; +	} + +	/* Temperature Target MSR is Nehalem and newer only */ +	if (!do_nehalem_platform_info) +		goto guess; + +	if (get_msr(0, MSR_IA32_TEMPERATURE_TARGET, &msr)) +		goto guess; + +	target_c_local = (msr >> 16) & 0x7F; + +	if (verbose) +		fprintf(stderr, "cpu%d: MSR_IA32_TEMPERATURE_TARGET: 0x%08llx (%d C)\n", +			cpu, msr, target_c_local); + +	if (target_c_local < 85 || target_c_local > 120) +		goto guess; + +	tcc_activation_temp = target_c_local; + +	return 0; + +guess: +	tcc_activation_temp = TJMAX_DEFAULT; +	fprintf(stderr, "cpu%d: Guessing tjMax %d C, Please use -T to specify\n", +		cpu, tcc_activation_temp); + +	return 0; +}  void check_cpuid()  {  	unsigned int eax, ebx, ecx, edx, max_level; @@ -1244,7 +1816,7 @@ void check_cpuid()  		genuine_intel = 1;  	if (verbose) -		fprintf(stderr, "%.4s%.4s%.4s ", +		fprintf(stderr, "CPUID(0): %.4s%.4s%.4s ",  			(char *)&ebx, (char *)&edx, (char *)&ecx);  	asm("cpuid" : "=a" (fms), "=c" (ecx), "=d" (edx) : "a" (1) : "ebx"); @@ -1295,10 +1867,19 @@ void check_cpuid()  	asm("cpuid" : "=a" (eax), "=b" (ebx), "=c" (ecx), "=d" (edx) : "a" (0x6));  	has_aperf = ecx & (1 << 0); -	if (!has_aperf) { -		fprintf(stderr, "No APERF MSR\n"); -		exit(1); -	} +	do_dts = eax & (1 << 0); +	do_ptm = eax & (1 << 6); +	has_epb = ecx & (1 << 3); + +	if (verbose) +		fprintf(stderr, "CPUID(6): %s%s%s%s\n", +			has_aperf ? "APERF" : "No APERF!", +			do_dts ? ", DTS" : "", +			do_ptm ? ", PTM": "", +			has_epb ? ", EPB": ""); + +	if (!has_aperf) +		exit(-1);  	do_nehalem_platform_info = genuine_intel && has_invariant_tsc;  	do_nhm_cstates = genuine_intel;	/* all Intel w/ non-stop TSC have NHM counters */ @@ -1307,12 +1888,15 @@ void check_cpuid()  	do_nehalem_turbo_ratio_limit = has_nehalem_turbo_ratio_limit(family, model);  	do_ivt_turbo_ratio_limit = has_ivt_turbo_ratio_limit(family, model); +	rapl_probe(family, model); + +	return;  }  void usage()  { -	fprintf(stderr, "%s: [-v][-p|-P|-S][-c MSR# | -s]][-C MSR#][-m MSR#][-M MSR#][-i interval_sec | command ...]\n", +	fprintf(stderr, "%s: [-v][-R][-T][-p|-P|-S][-c MSR# | -s]][-C MSR#][-m MSR#][-M MSR#][-i interval_sec | command ...]\n",  		progname);  	exit(1);  } @@ -1548,6 +2132,17 @@ void turbostat_init()  	if (verbose)  		print_verbose_header(); + +	if (verbose) +		for_all_cpus(print_epb, ODD_COUNTERS); + +	if (verbose) +		for_all_cpus(print_rapl, ODD_COUNTERS); + +	for_all_cpus(set_temperature_target, ODD_COUNTERS); + +	if (verbose) +		for_all_cpus(print_thermal, ODD_COUNTERS);  }  int fork_it(char **argv) @@ -1604,7 +2199,7 @@ void cmdline(int argc, char **argv)  	progname = argv[0]; -	while ((opt = getopt(argc, argv, "+pPSvi:sc:sC:m:M:")) != -1) { +	while ((opt = getopt(argc, argv, "+pPSvi:sc:sC:m:M:RT:")) != -1) {  		switch (opt) {  		case 'p':  			show_core_only++; @@ -1636,6 +2231,12 @@ void cmdline(int argc, char **argv)  		case 'M':  			sscanf(optarg, "%x", &extra_msr_offset64);  			break; +		case 'R': +			rapl_verbose++; +			break; +		case 'T': +			tcc_activation_temp_override = atoi(optarg); +			break;  		default:  			usage();  		} @@ -1646,8 +2247,8 @@ int main(int argc, char **argv)  {  	cmdline(argc, argv); -	if (verbose > 1) -		fprintf(stderr, "turbostat v2.1 October 6, 2012" +	if (verbose) +		fprintf(stderr, "turbostat v3.0 November 23, 2012"  			" - Len Brown <lenb@kernel.org>\n");  	turbostat_init();  |