2019-07-10 21:44:30 +08:00
// SPDX-License-Identifier: GPL-2.0-only
/*
* Intel Running Average Power Limit ( RAPL ) Driver via MSR interface
* Copyright ( c ) 2019 , Intel Corporation .
*/
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/list.h>
# include <linux/types.h>
# include <linux/device.h>
# include <linux/slab.h>
# include <linux/log2.h>
# include <linux/bitmap.h>
# include <linux/delay.h>
# include <linux/sysfs.h>
# include <linux/cpu.h>
# include <linux/powercap.h>
# include <linux/suspend.h>
# include <linux/intel_rapl.h>
# include <linux/processor.h>
2019-07-10 21:44:34 +08:00
# include <linux/platform_device.h>
2019-07-10 21:44:30 +08:00
# include <asm/cpu_device_id.h>
# include <asm/intel-family.h>
/* Local defines */
# define MSR_PLATFORM_POWER_LIMIT 0x0000065C
2020-07-16 23:14:55 +05:30
# define MSR_VR_CURRENT_CONFIG 0x00000601
2019-07-10 21:44:30 +08:00
/* private data for RAPL MSR Interface */
2020-10-27 07:23:55 +00:00
static struct rapl_if_priv * rapl_msr_priv ;
static struct rapl_if_priv rapl_msr_priv_intel = {
2023-04-19 10:44:17 +08:00
. type = RAPL_IF_MSR ,
2019-07-10 21:44:30 +08:00
. reg_unit = MSR_RAPL_POWER_UNIT ,
. regs [ RAPL_DOMAIN_PACKAGE ] = {
MSR_PKG_POWER_LIMIT , MSR_PKG_ENERGY_STATUS , MSR_PKG_PERF_STATUS , 0 , MSR_PKG_POWER_INFO } ,
. regs [ RAPL_DOMAIN_PP0 ] = {
MSR_PP0_POWER_LIMIT , MSR_PP0_ENERGY_STATUS , 0 , MSR_PP0_POLICY , 0 } ,
. regs [ RAPL_DOMAIN_PP1 ] = {
MSR_PP1_POWER_LIMIT , MSR_PP1_ENERGY_STATUS , 0 , MSR_PP1_POLICY , 0 } ,
. regs [ RAPL_DOMAIN_DRAM ] = {
MSR_DRAM_POWER_LIMIT , MSR_DRAM_ENERGY_STATUS , MSR_DRAM_PERF_STATUS , 0 , MSR_DRAM_POWER_INFO } ,
. regs [ RAPL_DOMAIN_PLATFORM ] = {
MSR_PLATFORM_POWER_LIMIT , MSR_PLATFORM_ENERGY_STATUS , 0 , 0 , 0 } ,
2023-04-19 10:44:12 +08:00
. limits [ RAPL_DOMAIN_PACKAGE ] = BIT ( POWER_LIMIT2 ) ,
. limits [ RAPL_DOMAIN_PLATFORM ] = BIT ( POWER_LIMIT2 ) ,
2019-07-10 21:44:30 +08:00
} ;
2020-10-27 07:23:56 +00:00
static struct rapl_if_priv rapl_msr_priv_amd = {
2023-04-19 10:44:17 +08:00
. type = RAPL_IF_MSR ,
2020-10-27 07:23:56 +00:00
. reg_unit = MSR_AMD_RAPL_POWER_UNIT ,
. regs [ RAPL_DOMAIN_PACKAGE ] = {
0 , MSR_AMD_PKG_ENERGY_STATUS , 0 , 0 , 0 } ,
. regs [ RAPL_DOMAIN_PP0 ] = {
0 , MSR_AMD_CORE_ENERGY_STATUS , 0 , 0 , 0 } ,
} ;
2019-07-10 21:44:30 +08:00
/* Handles CPU hotplug on multi-socket systems.
* If a CPU goes online as the first CPU of the physical package
* we add the RAPL package to the system . Similarly , when the last
* CPU of the package is removed , we remove the RAPL package and its
* associated domains . Cooling devices are handled accordingly at
* per - domain level .
*/
static int rapl_cpu_online ( unsigned int cpu )
{
struct rapl_package * rp ;
powercap: intel_rapl: Make cpu optional for rapl_package
MSR RAPL Interface always removes a rapl_package when all the CPUs in
that rapl_package are offlined. This is because it relies on an online
CPU to access the MSR.
But for RAPL Interface using MMIO registers, when all the cpus within
the rapl_package are offlined,
1. the register can still be accessed
2. monitoring and setting the Power Pimits for the rapl_package is still
meaningful because of uncore power.
This means that, a valid rapl_package doesn't rely on one or more cpus
being onlined.
For this sense, make cpu optional for rapl_package. A rapl_package can
be registered either using a CPU id to represent the physical
package/die, or using the physical package id directly.
Note that, the thermal throttling interrupt is not disabled via
MSR_IA32_PACKAGE_THERM_INTERRUPT for such rapl_package at the moment.
If it is still needed in the future, this can be achieved by selecting
an onlined CPU using the physical package id.
Note that, processor_thermal_rapl, the current MMIO RAPL Interface
driver, can also be converted to register using a package id instead.
But this is not done right now because processor_thermal_rapl driver
works on single-package systems only, and offlining the only package
will not happen. So keep the previous logic.
Signed-off-by: Zhang Rui <rui.zhang@intel.com>
Tested-by: Wang Wendy <wendy.wang@intel.com>
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
2023-04-19 10:44:16 +08:00
rp = rapl_find_package_domain ( cpu , rapl_msr_priv , true ) ;
2019-07-10 21:44:30 +08:00
if ( ! rp ) {
powercap: intel_rapl: Make cpu optional for rapl_package
MSR RAPL Interface always removes a rapl_package when all the CPUs in
that rapl_package are offlined. This is because it relies on an online
CPU to access the MSR.
But for RAPL Interface using MMIO registers, when all the cpus within
the rapl_package are offlined,
1. the register can still be accessed
2. monitoring and setting the Power Pimits for the rapl_package is still
meaningful because of uncore power.
This means that, a valid rapl_package doesn't rely on one or more cpus
being onlined.
For this sense, make cpu optional for rapl_package. A rapl_package can
be registered either using a CPU id to represent the physical
package/die, or using the physical package id directly.
Note that, the thermal throttling interrupt is not disabled via
MSR_IA32_PACKAGE_THERM_INTERRUPT for such rapl_package at the moment.
If it is still needed in the future, this can be achieved by selecting
an onlined CPU using the physical package id.
Note that, processor_thermal_rapl, the current MMIO RAPL Interface
driver, can also be converted to register using a package id instead.
But this is not done right now because processor_thermal_rapl driver
works on single-package systems only, and offlining the only package
will not happen. So keep the previous logic.
Signed-off-by: Zhang Rui <rui.zhang@intel.com>
Tested-by: Wang Wendy <wendy.wang@intel.com>
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
2023-04-19 10:44:16 +08:00
rp = rapl_add_package ( cpu , rapl_msr_priv , true ) ;
2019-07-10 21:44:30 +08:00
if ( IS_ERR ( rp ) )
return PTR_ERR ( rp ) ;
}
cpumask_set_cpu ( cpu , & rp - > cpumask ) ;
return 0 ;
}
static int rapl_cpu_down_prep ( unsigned int cpu )
{
struct rapl_package * rp ;
int lead_cpu ;
powercap: intel_rapl: Make cpu optional for rapl_package
MSR RAPL Interface always removes a rapl_package when all the CPUs in
that rapl_package are offlined. This is because it relies on an online
CPU to access the MSR.
But for RAPL Interface using MMIO registers, when all the cpus within
the rapl_package are offlined,
1. the register can still be accessed
2. monitoring and setting the Power Pimits for the rapl_package is still
meaningful because of uncore power.
This means that, a valid rapl_package doesn't rely on one or more cpus
being onlined.
For this sense, make cpu optional for rapl_package. A rapl_package can
be registered either using a CPU id to represent the physical
package/die, or using the physical package id directly.
Note that, the thermal throttling interrupt is not disabled via
MSR_IA32_PACKAGE_THERM_INTERRUPT for such rapl_package at the moment.
If it is still needed in the future, this can be achieved by selecting
an onlined CPU using the physical package id.
Note that, processor_thermal_rapl, the current MMIO RAPL Interface
driver, can also be converted to register using a package id instead.
But this is not done right now because processor_thermal_rapl driver
works on single-package systems only, and offlining the only package
will not happen. So keep the previous logic.
Signed-off-by: Zhang Rui <rui.zhang@intel.com>
Tested-by: Wang Wendy <wendy.wang@intel.com>
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
2023-04-19 10:44:16 +08:00
rp = rapl_find_package_domain ( cpu , rapl_msr_priv , true ) ;
2019-07-10 21:44:30 +08:00
if ( ! rp )
return 0 ;
cpumask_clear_cpu ( cpu , & rp - > cpumask ) ;
lead_cpu = cpumask_first ( & rp - > cpumask ) ;
if ( lead_cpu > = nr_cpu_ids )
rapl_remove_package ( rp ) ;
else if ( rp - > lead_cpu = = cpu )
rp - > lead_cpu = lead_cpu ;
return 0 ;
}
static int rapl_msr_read_raw ( int cpu , struct reg_action * ra )
{
2019-07-10 21:44:31 +08:00
u32 msr = ( u32 ) ra - > reg ;
if ( rdmsrl_safe_on_cpu ( cpu , msr , & ra - > value ) ) {
pr_debug ( " failed to read msr 0x%x on cpu %d \n " , msr , cpu ) ;
2019-07-10 21:44:30 +08:00
return - EIO ;
}
ra - > value & = ra - > mask ;
return 0 ;
}
static void rapl_msr_update_func ( void * info )
{
struct reg_action * ra = info ;
2019-07-10 21:44:31 +08:00
u32 msr = ( u32 ) ra - > reg ;
2019-07-10 21:44:30 +08:00
u64 val ;
2019-07-10 21:44:31 +08:00
ra - > err = rdmsrl_safe ( msr , & val ) ;
2019-07-10 21:44:30 +08:00
if ( ra - > err )
return ;
val & = ~ ra - > mask ;
val | = ra - > value ;
2019-07-10 21:44:31 +08:00
ra - > err = wrmsrl_safe ( msr , val ) ;
2019-07-10 21:44:30 +08:00
}
static int rapl_msr_write_raw ( int cpu , struct reg_action * ra )
{
int ret ;
ret = smp_call_function_single ( cpu , rapl_msr_update_func , ra , 1 ) ;
if ( WARN_ON_ONCE ( ret ) )
return ret ;
return ra - > err ;
}
2020-07-16 23:14:55 +05:30
/* List of verified CPUs. */
static const struct x86_cpu_id pl4_support_ids [ ] = {
2023-06-08 08:00:06 +05:30
X86_MATCH_INTEL_FAM6_MODEL ( TIGERLAKE_L , NULL ) ,
X86_MATCH_INTEL_FAM6_MODEL ( ALDERLAKE , NULL ) ,
X86_MATCH_INTEL_FAM6_MODEL ( ALDERLAKE_L , NULL ) ,
X86_MATCH_INTEL_FAM6_MODEL ( ALDERLAKE_N , NULL ) ,
X86_MATCH_INTEL_FAM6_MODEL ( RAPTORLAKE , NULL ) ,
X86_MATCH_INTEL_FAM6_MODEL ( RAPTORLAKE_P , NULL ) ,
X86_MATCH_INTEL_FAM6_MODEL ( METEORLAKE , NULL ) ,
X86_MATCH_INTEL_FAM6_MODEL ( METEORLAKE_L , NULL ) ,
2020-07-16 23:14:55 +05:30
{ }
} ;
2019-07-10 21:44:34 +08:00
static int rapl_msr_probe ( struct platform_device * pdev )
2019-07-10 21:44:30 +08:00
{
2020-07-16 23:14:55 +05:30
const struct x86_cpu_id * id = x86_match_cpu ( pl4_support_ids ) ;
2019-07-10 21:44:30 +08:00
int ret ;
2020-10-27 07:23:56 +00:00
switch ( boot_cpu_data . x86_vendor ) {
case X86_VENDOR_INTEL :
rapl_msr_priv = & rapl_msr_priv_intel ;
break ;
2021-03-02 10:01:08 +08:00
case X86_VENDOR_HYGON :
2020-10-27 07:23:56 +00:00
case X86_VENDOR_AMD :
rapl_msr_priv = & rapl_msr_priv_amd ;
break ;
default :
pr_err ( " intel-rapl does not support CPU vendor %d \n " , boot_cpu_data . x86_vendor ) ;
return - ENODEV ;
}
2020-10-27 07:23:55 +00:00
rapl_msr_priv - > read_raw = rapl_msr_read_raw ;
rapl_msr_priv - > write_raw = rapl_msr_write_raw ;
2019-07-10 21:44:30 +08:00
2020-07-16 23:14:55 +05:30
if ( id ) {
2023-04-19 10:44:12 +08:00
rapl_msr_priv - > limits [ RAPL_DOMAIN_PACKAGE ] | = BIT ( POWER_LIMIT4 ) ;
2020-10-27 07:23:55 +00:00
rapl_msr_priv - > regs [ RAPL_DOMAIN_PACKAGE ] [ RAPL_DOMAIN_REG_PL4 ] =
2020-07-16 23:14:55 +05:30
MSR_VR_CURRENT_CONFIG ;
pr_info ( " PL4 support detected. \n " ) ;
}
2020-10-27 07:23:55 +00:00
rapl_msr_priv - > control_type = powercap_register_control_type ( NULL , " intel-rapl " , NULL ) ;
if ( IS_ERR ( rapl_msr_priv - > control_type ) ) {
2019-07-10 21:44:30 +08:00
pr_debug ( " failed to register powercap control_type. \n " ) ;
2020-10-27 07:23:55 +00:00
return PTR_ERR ( rapl_msr_priv - > control_type ) ;
2019-07-10 21:44:30 +08:00
}
ret = cpuhp_setup_state ( CPUHP_AP_ONLINE_DYN , " powercap/rapl:online " ,
rapl_cpu_online , rapl_cpu_down_prep ) ;
if ( ret < 0 )
goto out ;
2020-10-27 07:23:55 +00:00
rapl_msr_priv - > pcap_rapl_online = ret ;
2019-07-10 21:44:30 +08:00
return 0 ;
out :
if ( ret )
2020-10-27 07:23:55 +00:00
powercap_unregister_control_type ( rapl_msr_priv - > control_type ) ;
2019-07-10 21:44:30 +08:00
return ret ;
}
2019-07-10 21:44:34 +08:00
static int rapl_msr_remove ( struct platform_device * pdev )
2019-07-10 21:44:30 +08:00
{
2020-10-27 07:23:55 +00:00
cpuhp_remove_state ( rapl_msr_priv - > pcap_rapl_online ) ;
powercap_unregister_control_type ( rapl_msr_priv - > control_type ) ;
2019-07-10 21:44:34 +08:00
return 0 ;
2019-07-10 21:44:30 +08:00
}
2019-07-10 21:44:34 +08:00
static const struct platform_device_id rapl_msr_ids [ ] = {
{ . name = " intel_rapl_msr " , } ,
{ }
} ;
MODULE_DEVICE_TABLE ( platform , rapl_msr_ids ) ;
static struct platform_driver intel_rapl_msr_driver = {
. probe = rapl_msr_probe ,
. remove = rapl_msr_remove ,
. id_table = rapl_msr_ids ,
. driver = {
. name = " intel_rapl_msr " ,
} ,
} ;
module_platform_driver ( intel_rapl_msr_driver ) ;
2019-07-10 21:44:30 +08:00
MODULE_DESCRIPTION ( " Driver for Intel RAPL (Running Average Power Limit) control via MSR interface " ) ;
MODULE_AUTHOR ( " Zhang Rui <rui.zhang@intel.com> " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;