| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306 |
- // SPDX-License-Identifier: GPL-2.0-only
- /*
- * intel_tcc.c - Library for Intel TCC (thermal control circuitry) MSR access
- * Copyright (c) 2022, Intel Corporation.
- */
- #include <linux/errno.h>
- #include <linux/intel_tcc.h>
- #include <asm/cpu_device_id.h>
- #include <asm/intel-family.h>
- #include <asm/msr.h>
- /**
- * struct temp_masks - Bitmasks for temperature readings
- * @tcc_offset: TCC offset in MSR_TEMPERATURE_TARGET
- * @digital_readout: Digital readout in MSR_IA32_THERM_STATUS
- * @pkg_digital_readout: Digital readout in MSR_IA32_PACKAGE_THERM_STATUS
- *
- * Bitmasks to extract the fields of the MSR_TEMPERATURE and IA32_[PACKAGE]_
- * THERM_STATUS registers for different processor models.
- *
- * The bitmask of TjMax is not included in this structure. It is always 0xff.
- */
- struct temp_masks {
- u32 tcc_offset;
- u32 digital_readout;
- u32 pkg_digital_readout;
- };
- #define TCC_MODEL_TEMP_MASKS(model, _tcc_offset, _digital_readout, \
- _pkg_digital_readout) \
- static const struct temp_masks temp_##model __initconst = { \
- .tcc_offset = _tcc_offset, \
- .digital_readout = _digital_readout, \
- .pkg_digital_readout = _pkg_digital_readout \
- }
- TCC_MODEL_TEMP_MASKS(nehalem, 0, 0x7f, 0x7f);
- TCC_MODEL_TEMP_MASKS(haswell_x, 0xf, 0x7f, 0x7f);
- TCC_MODEL_TEMP_MASKS(broadwell, 0x3f, 0x7f, 0x7f);
- TCC_MODEL_TEMP_MASKS(goldmont, 0x7f, 0x7f, 0x7f);
- TCC_MODEL_TEMP_MASKS(tigerlake, 0x3f, 0xff, 0xff);
- TCC_MODEL_TEMP_MASKS(sapphirerapids, 0x3f, 0x7f, 0xff);
- /* Use these masks for processors not included in @tcc_cpu_ids. */
- static struct temp_masks intel_tcc_temp_masks __ro_after_init = {
- .tcc_offset = 0x7f,
- .digital_readout = 0xff,
- .pkg_digital_readout = 0xff,
- };
- static const struct x86_cpu_id intel_tcc_cpu_ids[] __initconst = {
- X86_MATCH_VFM(INTEL_CORE_YONAH, &temp_nehalem),
- X86_MATCH_VFM(INTEL_CORE2_MEROM, &temp_nehalem),
- X86_MATCH_VFM(INTEL_CORE2_MEROM_L, &temp_nehalem),
- X86_MATCH_VFM(INTEL_CORE2_PENRYN, &temp_nehalem),
- X86_MATCH_VFM(INTEL_CORE2_DUNNINGTON, &temp_nehalem),
- X86_MATCH_VFM(INTEL_NEHALEM, &temp_nehalem),
- X86_MATCH_VFM(INTEL_NEHALEM_G, &temp_nehalem),
- X86_MATCH_VFM(INTEL_NEHALEM_EP, &temp_nehalem),
- X86_MATCH_VFM(INTEL_NEHALEM_EX, &temp_nehalem),
- X86_MATCH_VFM(INTEL_WESTMERE, &temp_nehalem),
- X86_MATCH_VFM(INTEL_WESTMERE_EP, &temp_nehalem),
- X86_MATCH_VFM(INTEL_WESTMERE_EX, &temp_nehalem),
- X86_MATCH_VFM(INTEL_SANDYBRIDGE, &temp_nehalem),
- X86_MATCH_VFM(INTEL_SANDYBRIDGE_X, &temp_nehalem),
- X86_MATCH_VFM(INTEL_IVYBRIDGE, &temp_nehalem),
- X86_MATCH_VFM(INTEL_IVYBRIDGE_X, &temp_haswell_x),
- X86_MATCH_VFM(INTEL_HASWELL, &temp_nehalem),
- X86_MATCH_VFM(INTEL_HASWELL_X, &temp_haswell_x),
- X86_MATCH_VFM(INTEL_HASWELL_L, &temp_nehalem),
- X86_MATCH_VFM(INTEL_HASWELL_G, &temp_nehalem),
- X86_MATCH_VFM(INTEL_BROADWELL, &temp_broadwell),
- X86_MATCH_VFM(INTEL_BROADWELL_G, &temp_broadwell),
- X86_MATCH_VFM(INTEL_BROADWELL_X, &temp_haswell_x),
- X86_MATCH_VFM(INTEL_BROADWELL_D, &temp_haswell_x),
- X86_MATCH_VFM(INTEL_SKYLAKE_L, &temp_broadwell),
- X86_MATCH_VFM(INTEL_SKYLAKE, &temp_broadwell),
- X86_MATCH_VFM(INTEL_SKYLAKE_X, &temp_haswell_x),
- X86_MATCH_VFM(INTEL_KABYLAKE_L, &temp_broadwell),
- X86_MATCH_VFM(INTEL_KABYLAKE, &temp_broadwell),
- X86_MATCH_VFM(INTEL_COMETLAKE, &temp_broadwell),
- X86_MATCH_VFM(INTEL_COMETLAKE_L, &temp_broadwell),
- X86_MATCH_VFM(INTEL_CANNONLAKE_L, &temp_broadwell),
- X86_MATCH_VFM(INTEL_ICELAKE_X, &temp_broadwell),
- X86_MATCH_VFM(INTEL_ICELAKE_D, &temp_broadwell),
- X86_MATCH_VFM(INTEL_ICELAKE, &temp_broadwell),
- X86_MATCH_VFM(INTEL_ICELAKE_L, &temp_broadwell),
- X86_MATCH_VFM(INTEL_ICELAKE_NNPI, &temp_broadwell),
- X86_MATCH_VFM(INTEL_ROCKETLAKE, &temp_broadwell),
- X86_MATCH_VFM(INTEL_TIGERLAKE_L, &temp_tigerlake),
- X86_MATCH_VFM(INTEL_TIGERLAKE, &temp_tigerlake),
- X86_MATCH_VFM(INTEL_SAPPHIRERAPIDS_X, &temp_sapphirerapids),
- X86_MATCH_VFM(INTEL_EMERALDRAPIDS_X, &temp_sapphirerapids),
- X86_MATCH_VFM(INTEL_LAKEFIELD, &temp_broadwell),
- X86_MATCH_VFM(INTEL_ALDERLAKE, &temp_tigerlake),
- X86_MATCH_VFM(INTEL_ALDERLAKE_L, &temp_tigerlake),
- X86_MATCH_VFM(INTEL_RAPTORLAKE, &temp_tigerlake),
- X86_MATCH_VFM(INTEL_RAPTORLAKE_P, &temp_tigerlake),
- X86_MATCH_VFM(INTEL_RAPTORLAKE_S, &temp_tigerlake),
- X86_MATCH_VFM(INTEL_ATOM_BONNELL, &temp_nehalem),
- X86_MATCH_VFM(INTEL_ATOM_BONNELL_MID, &temp_nehalem),
- X86_MATCH_VFM(INTEL_ATOM_SALTWELL, &temp_nehalem),
- X86_MATCH_VFM(INTEL_ATOM_SALTWELL_MID, &temp_nehalem),
- X86_MATCH_VFM(INTEL_ATOM_SILVERMONT, &temp_broadwell),
- X86_MATCH_VFM(INTEL_ATOM_SILVERMONT_D, &temp_broadwell),
- X86_MATCH_VFM(INTEL_ATOM_SILVERMONT_MID, &temp_broadwell),
- X86_MATCH_VFM(INTEL_ATOM_AIRMONT, &temp_broadwell),
- X86_MATCH_VFM(INTEL_ATOM_SILVERMONT_MID2, &temp_broadwell),
- X86_MATCH_VFM(INTEL_ATOM_AIRMONT_NP, &temp_broadwell),
- X86_MATCH_VFM(INTEL_ATOM_GOLDMONT, &temp_goldmont),
- X86_MATCH_VFM(INTEL_ATOM_GOLDMONT_D, &temp_goldmont),
- X86_MATCH_VFM(INTEL_ATOM_GOLDMONT_PLUS, &temp_goldmont),
- X86_MATCH_VFM(INTEL_ATOM_TREMONT_D, &temp_broadwell),
- X86_MATCH_VFM(INTEL_ATOM_TREMONT, &temp_broadwell),
- X86_MATCH_VFM(INTEL_ATOM_TREMONT_L, &temp_broadwell),
- X86_MATCH_VFM(INTEL_ATOM_GRACEMONT, &temp_tigerlake),
- X86_MATCH_VFM(INTEL_XEON_PHI_KNL, &temp_broadwell),
- X86_MATCH_VFM(INTEL_XEON_PHI_KNM, &temp_broadwell),
- {}
- };
- static int __init intel_tcc_init(void)
- {
- const struct x86_cpu_id *id;
- id = x86_match_cpu(intel_tcc_cpu_ids);
- if (id)
- memcpy(&intel_tcc_temp_masks, (const void *)id->driver_data,
- sizeof(intel_tcc_temp_masks));
- return 0;
- }
- /*
- * Use subsys_initcall to ensure temperature bitmasks are initialized before
- * the drivers that use this library.
- */
- subsys_initcall(intel_tcc_init);
- /**
- * intel_tcc_get_offset_mask() - Returns the bitmask to read TCC offset
- *
- * Get the model-specific bitmask to extract TCC_OFFSET from the MSR
- * TEMPERATURE_TARGET register. If the mask is 0, it means the processor does
- * not support TCC offset.
- *
- * Return: The model-specific bitmask for TCC offset.
- */
- u32 intel_tcc_get_offset_mask(void)
- {
- return intel_tcc_temp_masks.tcc_offset;
- }
- EXPORT_SYMBOL_NS(intel_tcc_get_offset_mask, "INTEL_TCC");
- /**
- * get_temp_mask() - Returns the model-specific bitmask for temperature
- *
- * @pkg: true: Package Thermal Sensor. false: Core Thermal Sensor.
- *
- * Get the model-specific bitmask to extract the temperature reading from the
- * MSR_IA32_[PACKAGE]_THERM_STATUS register.
- *
- * Callers must check if the thermal status registers are supported.
- *
- * Return: The model-specific bitmask for temperature reading
- */
- static u32 get_temp_mask(bool pkg)
- {
- return pkg ? intel_tcc_temp_masks.pkg_digital_readout :
- intel_tcc_temp_masks.digital_readout;
- }
- /**
- * intel_tcc_get_tjmax() - returns the default TCC activation Temperature
- * @cpu: cpu that the MSR should be run on, negative value means any cpu.
- *
- * Get the TjMax value, which is the default thermal throttling or TCC
- * activation temperature in degrees C.
- *
- * Return: Tjmax value in degrees C on success, negative error code otherwise.
- */
- int intel_tcc_get_tjmax(int cpu)
- {
- u32 low, high;
- int val, err;
- if (cpu < 0)
- err = rdmsr_safe(MSR_IA32_TEMPERATURE_TARGET, &low, &high);
- else
- err = rdmsr_safe_on_cpu(cpu, MSR_IA32_TEMPERATURE_TARGET, &low, &high);
- if (err)
- return err;
- val = (low >> 16) & 0xff;
- return val ? val : -ENODATA;
- }
- EXPORT_SYMBOL_NS_GPL(intel_tcc_get_tjmax, "INTEL_TCC");
- /**
- * intel_tcc_get_offset() - returns the TCC Offset value to Tjmax
- * @cpu: cpu that the MSR should be run on, negative value means any cpu.
- *
- * Get the TCC offset value to Tjmax. The effective thermal throttling or TCC
- * activation temperature equals "Tjmax" - "TCC Offset", in degrees C.
- *
- * Return: Tcc offset value in degrees C on success, negative error code otherwise.
- */
- int intel_tcc_get_offset(int cpu)
- {
- u32 low, high;
- int err;
- if (cpu < 0)
- err = rdmsr_safe(MSR_IA32_TEMPERATURE_TARGET, &low, &high);
- else
- err = rdmsr_safe_on_cpu(cpu, MSR_IA32_TEMPERATURE_TARGET, &low, &high);
- if (err)
- return err;
- return (low >> 24) & intel_tcc_temp_masks.tcc_offset;
- }
- EXPORT_SYMBOL_NS_GPL(intel_tcc_get_offset, "INTEL_TCC");
- /**
- * intel_tcc_set_offset() - set the TCC offset value to Tjmax
- * @cpu: cpu that the MSR should be run on, negative value means any cpu.
- * @offset: TCC offset value in degree C
- *
- * Set the TCC Offset value to Tjmax. The effective thermal throttling or TCC
- * activation temperature equals "Tjmax" - "TCC Offset", in degree C.
- *
- * Return: On success returns 0, negative error code otherwise.
- */
- int intel_tcc_set_offset(int cpu, int offset)
- {
- u32 low, high;
- int err;
- if (!intel_tcc_temp_masks.tcc_offset)
- return -ENODEV;
- if (offset < 0 || offset > intel_tcc_temp_masks.tcc_offset)
- return -EINVAL;
- if (cpu < 0)
- err = rdmsr_safe(MSR_IA32_TEMPERATURE_TARGET, &low, &high);
- else
- err = rdmsr_safe_on_cpu(cpu, MSR_IA32_TEMPERATURE_TARGET, &low, &high);
- if (err)
- return err;
- /* MSR Locked */
- if (low & BIT(31))
- return -EPERM;
- low &= ~(intel_tcc_temp_masks.tcc_offset << 24);
- low |= offset << 24;
- if (cpu < 0)
- return wrmsr_safe(MSR_IA32_TEMPERATURE_TARGET, low, high);
- else
- return wrmsr_safe_on_cpu(cpu, MSR_IA32_TEMPERATURE_TARGET, low, high);
- }
- EXPORT_SYMBOL_NS_GPL(intel_tcc_set_offset, "INTEL_TCC");
- /**
- * intel_tcc_get_temp() - returns the current temperature
- * @cpu: cpu that the MSR should be run on, negative value means any cpu.
- * @temp: pointer to the memory for saving cpu temperature.
- * @pkg: true: Package Thermal Sensor. false: Core Thermal Sensor.
- *
- * Get the current temperature returned by the CPU core/package level
- * thermal sensor, in degrees C.
- *
- * Return: 0 on success, negative error code otherwise.
- */
- int intel_tcc_get_temp(int cpu, int *temp, bool pkg)
- {
- u32 msr = pkg ? MSR_IA32_PACKAGE_THERM_STATUS : MSR_IA32_THERM_STATUS;
- u32 low, high, mask;
- int tjmax, err;
- tjmax = intel_tcc_get_tjmax(cpu);
- if (tjmax < 0)
- return tjmax;
- if (cpu < 0)
- err = rdmsr_safe(msr, &low, &high);
- else
- err = rdmsr_safe_on_cpu(cpu, msr, &low, &high);
- if (err)
- return err;
- /* Temperature is beyond the valid thermal sensor range */
- if (!(low & BIT(31)))
- return -ENODATA;
- mask = get_temp_mask(pkg);
- *temp = tjmax - ((low >> 16) & mask);
- return 0;
- }
- EXPORT_SYMBOL_NS_GPL(intel_tcc_get_temp, "INTEL_TCC");
|