Add a quirk for Turbo Boost.
It was observed that at least Sverre Froyen's ThinkPad T500 reports values that do not match readings from the IA32_PERF_STATUS register. This only applied to the P0-state. Thus, for now, skip the status check if Turbo Boost has been detected and the requested state is P0. This needs to be revisited once Turbo Boost actually works in NetBSD. It is unclear whether this is a BIOS flaw or not; these values may well be what we get from IA32_PERF_STATUS once the CPU actually uses the +133.33 MHz boost.
This commit is contained in:
parent
4362bda118
commit
503e356fbc
@ -1,4 +1,4 @@
|
|||||||
/* $NetBSD: acpi_cpu_md.c,v 1.23 2010/08/21 04:36:29 jruoho Exp $ */
|
/* $NetBSD: acpi_cpu_md.c,v 1.24 2010/08/21 05:10:43 jruoho Exp $ */
|
||||||
|
|
||||||
/*-
|
/*-
|
||||||
* Copyright (c) 2010 Jukka Ruohonen <jruohonen@iki.fi>
|
* Copyright (c) 2010 Jukka Ruohonen <jruohonen@iki.fi>
|
||||||
@ -27,7 +27,7 @@
|
|||||||
* SUCH DAMAGE.
|
* SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
#include <sys/cdefs.h>
|
#include <sys/cdefs.h>
|
||||||
__KERNEL_RCSID(0, "$NetBSD: acpi_cpu_md.c,v 1.23 2010/08/21 04:36:29 jruoho Exp $");
|
__KERNEL_RCSID(0, "$NetBSD: acpi_cpu_md.c,v 1.24 2010/08/21 05:10:43 jruoho Exp $");
|
||||||
|
|
||||||
#include <sys/param.h>
|
#include <sys/param.h>
|
||||||
#include <sys/bus.h>
|
#include <sys/bus.h>
|
||||||
@ -156,8 +156,8 @@ acpicpu_md_quirks(void)
|
|||||||
if ((regs[2] & __BIT(0)) != 0) /* ECX.06[0] */
|
if ((regs[2] & __BIT(0)) != 0) /* ECX.06[0] */
|
||||||
val |= ACPICPU_FLAG_P_HW;
|
val |= ACPICPU_FLAG_P_HW;
|
||||||
|
|
||||||
if ((regs[0] & __BIT(1)) != 0)
|
if ((regs[0] & __BIT(1)) != 0) /* EAX.06[1] */
|
||||||
val |= ACPICPU_FLAG_P_TURBO; /* EAX.06[1] */
|
val |= ACPICPU_FLAG_P_TURBO;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -166,7 +166,7 @@ acpicpu_md_quirks(void)
|
|||||||
* at constant rate. Depending on the CPU, this may
|
* at constant rate. Depending on the CPU, this may
|
||||||
* affect P- and T-state changes, but especially
|
* affect P- and T-state changes, but especially
|
||||||
* relevant are C-states; with variant TSC, states
|
* relevant are C-states; with variant TSC, states
|
||||||
* larger than C1 will completely stop the timer.
|
* larger than C1 may completely stop the counter.
|
||||||
*/
|
*/
|
||||||
x86_cpuid(0x80000000, regs);
|
x86_cpuid(0x80000000, regs);
|
||||||
|
|
||||||
@ -415,6 +415,30 @@ acpicpu_md_pstate_pss(struct acpicpu_softc *sc)
|
|||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* When the state is P0 and Turbo Boost has been
|
||||||
|
* detected, we need to skip the status check as
|
||||||
|
* BIOS may not report right comparison values for
|
||||||
|
* the IA32_PERF_STATUS MSR. Note that this is a
|
||||||
|
* BIOS issue, and that for instance AMD documents
|
||||||
|
* clearly state that vendors should never expose
|
||||||
|
* "turbo" in the ACPI objects (namely, in _PSS).
|
||||||
|
*
|
||||||
|
* For discussion, see:
|
||||||
|
*
|
||||||
|
* Intel Corporation: Intel Turbo Boost Technology
|
||||||
|
* in Intel Core(tm) Microarchitectures (Nehalem)
|
||||||
|
* Based Processors. White Paper, November 2008.
|
||||||
|
*/
|
||||||
|
if (cpu_vendor != CPUVENDOR_INTEL)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if ((sc->sc_flags & ACPICPU_FLAG_P_TURBO) == 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (sc->sc_pstate[1].ps_freq + 1 == sc->sc_pstate[0].ps_freq)
|
||||||
|
sc->sc_pstate[0].ps_flags |= ACPICPU_FLAG_P_TURBO;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -471,7 +495,7 @@ acpicpu_md_pstate_set(struct acpicpu_pstate *ps)
|
|||||||
msr.msr_type = ps->ps_control_addr;
|
msr.msr_type = ps->ps_control_addr;
|
||||||
msr.msr_value = ps->ps_control;
|
msr.msr_value = ps->ps_control;
|
||||||
|
|
||||||
if (ps->ps_control_mask != 0) {
|
if (__predict_true(ps->ps_control_mask != 0)) {
|
||||||
msr.msr_mask = ps->ps_control_mask;
|
msr.msr_mask = ps->ps_control_mask;
|
||||||
msr.msr_read = true;
|
msr.msr_read = true;
|
||||||
}
|
}
|
||||||
@ -479,7 +503,10 @@ acpicpu_md_pstate_set(struct acpicpu_pstate *ps)
|
|||||||
xc = xc_broadcast(0, (xcfunc_t)x86_msr_xcall, &msr, NULL);
|
xc = xc_broadcast(0, (xcfunc_t)x86_msr_xcall, &msr, NULL);
|
||||||
xc_wait(xc);
|
xc_wait(xc);
|
||||||
|
|
||||||
if (ps->ps_status_addr == 0)
|
if (__predict_false(ps->ps_status_addr == 0))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if ((ps->ps_flags & ACPICPU_FLAG_P_TURBO) != 0)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
xc = xc_broadcast(0, (xcfunc_t)acpicpu_md_pstate_status, ps, &rv);
|
xc = xc_broadcast(0, (xcfunc_t)acpicpu_md_pstate_status, ps, &rv);
|
||||||
@ -499,7 +526,7 @@ acpicpu_md_pstate_status(void *arg1, void *arg2)
|
|||||||
|
|
||||||
val = rdmsr(ps->ps_status_addr);
|
val = rdmsr(ps->ps_status_addr);
|
||||||
|
|
||||||
if (ps->ps_status_mask != 0)
|
if (__predict_true(ps->ps_status_mask != 0))
|
||||||
val = val & ps->ps_status_mask;
|
val = val & ps->ps_status_mask;
|
||||||
|
|
||||||
if (val == ps->ps_status)
|
if (val == ps->ps_status)
|
||||||
|
Loading…
Reference in New Issue
Block a user