Path: blob/master/drivers/cpufreq/amd_freq_sensitivity.c
26278 views
// SPDX-License-Identifier: GPL-2.0-only1/*2* amd_freq_sensitivity.c: AMD frequency sensitivity feedback powersave bias3* for the ondemand governor.4*5* Copyright (C) 2013 Advanced Micro Devices, Inc.6*7* Author: Jacob Shin <[email protected]>8*/910#include <linux/kernel.h>11#include <linux/module.h>12#include <linux/types.h>13#include <linux/pci.h>14#include <linux/percpu-defs.h>15#include <linux/init.h>16#include <linux/mod_devicetable.h>1718#include <asm/msr.h>19#include <asm/cpufeature.h>20#include <asm/cpu_device_id.h>2122#include "cpufreq_ondemand.h"2324#define MSR_AMD64_FREQ_SENSITIVITY_ACTUAL 0xc001008025#define MSR_AMD64_FREQ_SENSITIVITY_REFERENCE 0xc001008126#define CLASS_CODE_SHIFT 5627#define POWERSAVE_BIAS_MAX 100028#define POWERSAVE_BIAS_DEF 4002930struct cpu_data_t {31u64 actual;32u64 reference;33unsigned int freq_prev;34};3536static DEFINE_PER_CPU(struct cpu_data_t, cpu_data);3738static unsigned int amd_powersave_bias_target(struct cpufreq_policy *policy,39unsigned int freq_next,40unsigned int relation)41{42int sensitivity;43long d_actual, d_reference;44struct msr actual, reference;45struct cpu_data_t *data = &per_cpu(cpu_data, policy->cpu);46struct policy_dbs_info *policy_dbs = policy->governor_data;47struct dbs_data *od_data = policy_dbs->dbs_data;48struct od_dbs_tuners *od_tuners = od_data->tuners;4950if (!policy->freq_table)51return freq_next;5253rdmsr_on_cpu(policy->cpu, MSR_AMD64_FREQ_SENSITIVITY_ACTUAL,54&actual.l, &actual.h);55rdmsr_on_cpu(policy->cpu, MSR_AMD64_FREQ_SENSITIVITY_REFERENCE,56&reference.l, &reference.h);57actual.h &= 0x00ffffff;58reference.h &= 0x00ffffff;5960/* counter wrapped around, so stay on current frequency */61if (actual.q < data->actual || reference.q < data->reference) {62freq_next = policy->cur;63goto out;64}6566d_actual = actual.q - data->actual;67d_reference = reference.q - data->reference;6869/* divide by 0, so stay on current frequency as well */70if (d_reference == 0) {71freq_next = policy->cur;72goto out;73}7475sensitivity = POWERSAVE_BIAS_MAX -76(POWERSAVE_BIAS_MAX * (d_reference - d_actual) / d_reference);7778clamp(sensitivity, 0, POWERSAVE_BIAS_MAX);7980/* this workload is not CPU bound, so choose a lower freq */81if (sensitivity < od_tuners->powersave_bias) {82if (data->freq_prev == policy->cur)83freq_next = policy->cur;8485if (freq_next > policy->cur)86freq_next = policy->cur;87else if (freq_next < policy->cur)88freq_next = policy->min;89else {90unsigned int index;9192index = cpufreq_table_find_index_h(policy,93policy->cur - 1,94relation & CPUFREQ_RELATION_E);95freq_next = policy->freq_table[index].frequency;96}9798data->freq_prev = freq_next;99} else100data->freq_prev = 0;101102out:103data->actual = actual.q;104data->reference = reference.q;105return freq_next;106}107108static int __init amd_freq_sensitivity_init(void)109{110u64 val;111struct pci_dev *pcidev;112unsigned int pci_vendor;113114if (boot_cpu_data.x86_vendor == X86_VENDOR_AMD)115pci_vendor = PCI_VENDOR_ID_AMD;116else if (boot_cpu_data.x86_vendor == X86_VENDOR_HYGON)117pci_vendor = PCI_VENDOR_ID_HYGON;118else119return -ENODEV;120121pcidev = pci_get_device(pci_vendor,122PCI_DEVICE_ID_AMD_KERNCZ_SMBUS, NULL);123124if (!pcidev) {125if (!boot_cpu_has(X86_FEATURE_PROC_FEEDBACK))126return -ENODEV;127} else {128pci_dev_put(pcidev);129}130131if (rdmsrq_safe(MSR_AMD64_FREQ_SENSITIVITY_ACTUAL, &val))132return -ENODEV;133134if (!(val >> CLASS_CODE_SHIFT))135return -ENODEV;136137od_register_powersave_bias_handler(amd_powersave_bias_target,138POWERSAVE_BIAS_DEF);139return 0;140}141late_initcall(amd_freq_sensitivity_init);142143static void __exit amd_freq_sensitivity_exit(void)144{145od_unregister_powersave_bias_handler();146}147module_exit(amd_freq_sensitivity_exit);148149static const struct x86_cpu_id __maybe_unused amd_freq_sensitivity_ids[] = {150X86_MATCH_FEATURE(X86_FEATURE_PROC_FEEDBACK, NULL),151{}152};153MODULE_DEVICE_TABLE(x86cpu, amd_freq_sensitivity_ids);154155MODULE_AUTHOR("Jacob Shin <[email protected]>");156MODULE_DESCRIPTION("AMD frequency sensitivity feedback powersave bias for "157"the ondemand governor.");158MODULE_LICENSE("GPL");159160161