thermal: intel: hfi: Notify user space for HFI events
When the hardware issues an HFI event, relay a notification to user space. This allows user space to respond by reading performance and efficiency of each CPU and take appropriate action. For example, when the performance and efficiency of a CPU is 0, user space can either offline the CPU or inject idle. Also, if user space notices a downward trend in performance, it may proactively adjust power limits to avoid future situations in which performance drops to 0. To avoid excessive notifications, the rate is limited by one HZ per event. To limit the netlink message size, send parameters for up to 16 CPUs in a single message. If there are more than 16 CPUs, issue as many messages as needed to notify the status of all CPUs. In the HFI specification, both performance and efficiency capabilities are defined in the [0, 255] range. The existing implementations of HFI hardware do not scale the maximum values to 255. Since userspace cares about capability values that are either 0 or show a downward/upward trend, this fact does not matter much. Relative changes in capabilities are enough. To comply with the thermal netlink ABI, scale both performance and efficiency capabilities to the [0, 1023] interval. Reviewed-by: Len Brown <len.brown@intel.com> Signed-off-by: Srinivas Pandruvada <srinivas.pandruvada@linux.intel.com> Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
This commit is contained in:
parent
e4b1eb24ce
commit
bd30cdfd9b
@ -104,6 +104,7 @@ config INTEL_HFI_THERMAL
|
|||||||
bool "Intel Hardware Feedback Interface"
|
bool "Intel Hardware Feedback Interface"
|
||||||
depends on CPU_SUP_INTEL
|
depends on CPU_SUP_INTEL
|
||||||
depends on X86_THERMAL_VECTOR
|
depends on X86_THERMAL_VECTOR
|
||||||
|
select THERMAL_NETLINK
|
||||||
help
|
help
|
||||||
Select this option to enable the Hardware Feedback Interface. If
|
Select this option to enable the Hardware Feedback Interface. If
|
||||||
selected, hardware provides guidance to the operating system on
|
selected, hardware provides guidance to the operating system on
|
||||||
|
@ -40,6 +40,7 @@
|
|||||||
|
|
||||||
#include <asm/msr.h>
|
#include <asm/msr.h>
|
||||||
|
|
||||||
|
#include "../thermal_core.h"
|
||||||
#include "intel_hfi.h"
|
#include "intel_hfi.h"
|
||||||
|
|
||||||
#define THERM_STATUS_CLEAR_PKG_MASK (BIT(1) | BIT(3) | BIT(5) | BIT(7) | \
|
#define THERM_STATUS_CLEAR_PKG_MASK (BIT(1) | BIT(3) | BIT(5) | BIT(7) | \
|
||||||
@ -163,6 +164,78 @@ static DEFINE_MUTEX(hfi_instance_lock);
|
|||||||
|
|
||||||
static struct workqueue_struct *hfi_updates_wq;
|
static struct workqueue_struct *hfi_updates_wq;
|
||||||
#define HFI_UPDATE_INTERVAL HZ
|
#define HFI_UPDATE_INTERVAL HZ
|
||||||
|
#define HFI_MAX_THERM_NOTIFY_COUNT 16
|
||||||
|
|
||||||
|
static void get_hfi_caps(struct hfi_instance *hfi_instance,
|
||||||
|
struct thermal_genl_cpu_caps *cpu_caps)
|
||||||
|
{
|
||||||
|
int cpu, i = 0;
|
||||||
|
|
||||||
|
raw_spin_lock_irq(&hfi_instance->table_lock);
|
||||||
|
for_each_cpu(cpu, hfi_instance->cpus) {
|
||||||
|
struct hfi_cpu_data *caps;
|
||||||
|
s16 index;
|
||||||
|
|
||||||
|
index = per_cpu(hfi_cpu_info, cpu).index;
|
||||||
|
caps = hfi_instance->data + index * hfi_features.cpu_stride;
|
||||||
|
cpu_caps[i].cpu = cpu;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Scale performance and energy efficiency to
|
||||||
|
* the [0, 1023] interval that thermal netlink uses.
|
||||||
|
*/
|
||||||
|
cpu_caps[i].performance = caps->perf_cap << 2;
|
||||||
|
cpu_caps[i].efficiency = caps->ee_cap << 2;
|
||||||
|
|
||||||
|
++i;
|
||||||
|
}
|
||||||
|
raw_spin_unlock_irq(&hfi_instance->table_lock);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Call update_capabilities() when there are changes in the HFI table.
|
||||||
|
*/
|
||||||
|
static void update_capabilities(struct hfi_instance *hfi_instance)
|
||||||
|
{
|
||||||
|
struct thermal_genl_cpu_caps *cpu_caps;
|
||||||
|
int i = 0, cpu_count;
|
||||||
|
|
||||||
|
/* CPUs may come online/offline while processing an HFI update. */
|
||||||
|
mutex_lock(&hfi_instance_lock);
|
||||||
|
|
||||||
|
cpu_count = cpumask_weight(hfi_instance->cpus);
|
||||||
|
|
||||||
|
/* No CPUs to report in this hfi_instance. */
|
||||||
|
if (!cpu_count)
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
cpu_caps = kcalloc(cpu_count, sizeof(*cpu_caps), GFP_KERNEL);
|
||||||
|
if (!cpu_caps)
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
get_hfi_caps(hfi_instance, cpu_caps);
|
||||||
|
|
||||||
|
if (cpu_count < HFI_MAX_THERM_NOTIFY_COUNT)
|
||||||
|
goto last_cmd;
|
||||||
|
|
||||||
|
/* Process complete chunks of HFI_MAX_THERM_NOTIFY_COUNT capabilities. */
|
||||||
|
for (i = 0;
|
||||||
|
(i + HFI_MAX_THERM_NOTIFY_COUNT) <= cpu_count;
|
||||||
|
i += HFI_MAX_THERM_NOTIFY_COUNT)
|
||||||
|
thermal_genl_cpu_capability_event(HFI_MAX_THERM_NOTIFY_COUNT,
|
||||||
|
&cpu_caps[i]);
|
||||||
|
|
||||||
|
cpu_count = cpu_count - i;
|
||||||
|
|
||||||
|
last_cmd:
|
||||||
|
/* Process the remaining capabilities if any. */
|
||||||
|
if (cpu_count)
|
||||||
|
thermal_genl_cpu_capability_event(cpu_count, &cpu_caps[i]);
|
||||||
|
|
||||||
|
kfree(cpu_caps);
|
||||||
|
out:
|
||||||
|
mutex_unlock(&hfi_instance_lock);
|
||||||
|
}
|
||||||
|
|
||||||
static void hfi_update_work_fn(struct work_struct *work)
|
static void hfi_update_work_fn(struct work_struct *work)
|
||||||
{
|
{
|
||||||
@ -173,7 +246,7 @@ static void hfi_update_work_fn(struct work_struct *work)
|
|||||||
if (!hfi_instance)
|
if (!hfi_instance)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
/* TODO: Consume update here. */
|
update_capabilities(hfi_instance);
|
||||||
}
|
}
|
||||||
|
|
||||||
void intel_hfi_process_event(__u64 pkg_therm_status_msr_val)
|
void intel_hfi_process_event(__u64 pkg_therm_status_msr_val)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user