| 1 | // SPDX-License-Identifier: GPL-2.0-only | 
|---|
| 2 | /* | 
|---|
| 3 | * intel_tcc.c - Library for Intel TCC (thermal control circuitry) MSR access | 
|---|
| 4 | * Copyright (c) 2022, Intel Corporation. | 
|---|
| 5 | */ | 
|---|
| 6 |  | 
|---|
| 7 | #include <linux/errno.h> | 
|---|
| 8 | #include <linux/intel_tcc.h> | 
|---|
| 9 | #include <asm/cpu_device_id.h> | 
|---|
| 10 | #include <asm/intel-family.h> | 
|---|
| 11 | #include <asm/msr.h> | 
|---|
| 12 |  | 
|---|
| 13 | /** | 
|---|
| 14 | * struct temp_masks - Bitmasks for temperature readings | 
|---|
| 15 | * @tcc_offset:			TCC offset in MSR_TEMPERATURE_TARGET | 
|---|
| 16 | * @digital_readout:		Digital readout in MSR_IA32_THERM_STATUS | 
|---|
| 17 | * @pkg_digital_readout:	Digital readout in MSR_IA32_PACKAGE_THERM_STATUS | 
|---|
| 18 | * | 
|---|
| 19 | * Bitmasks to extract the fields of the MSR_TEMPERATURE and IA32_[PACKAGE]_ | 
|---|
| 20 | * THERM_STATUS registers for different processor models. | 
|---|
| 21 | * | 
|---|
| 22 | * The bitmask of TjMax is not included in this structure. It is always 0xff. | 
|---|
| 23 | */ | 
|---|
| 24 | struct temp_masks { | 
|---|
| 25 | u32 tcc_offset; | 
|---|
| 26 | u32 digital_readout; | 
|---|
| 27 | u32 pkg_digital_readout; | 
|---|
| 28 | }; | 
|---|
| 29 |  | 
|---|
| 30 | #define TCC_MODEL_TEMP_MASKS(model, _tcc_offset, _digital_readout,	\ | 
|---|
| 31 | _pkg_digital_readout)			\ | 
|---|
| 32 | static const struct temp_masks temp_##model __initconst = {	\ | 
|---|
| 33 | .tcc_offset = _tcc_offset,				\ | 
|---|
| 34 | .digital_readout = _digital_readout,			\ | 
|---|
| 35 | .pkg_digital_readout = _pkg_digital_readout		\ | 
|---|
| 36 | } | 
|---|
| 37 |  | 
|---|
| 38 | TCC_MODEL_TEMP_MASKS(nehalem, 0, 0x7f, 0x7f); | 
|---|
| 39 | TCC_MODEL_TEMP_MASKS(haswell_x, 0xf, 0x7f, 0x7f); | 
|---|
| 40 | TCC_MODEL_TEMP_MASKS(broadwell, 0x3f, 0x7f, 0x7f); | 
|---|
| 41 | TCC_MODEL_TEMP_MASKS(goldmont, 0x7f, 0x7f, 0x7f); | 
|---|
| 42 | TCC_MODEL_TEMP_MASKS(tigerlake, 0x3f, 0xff, 0xff); | 
|---|
| 43 | TCC_MODEL_TEMP_MASKS(sapphirerapids, 0x3f, 0x7f, 0xff); | 
|---|
| 44 |  | 
|---|
| 45 | /* Use these masks for processors not included in @tcc_cpu_ids. */ | 
|---|
| 46 | static struct temp_masks intel_tcc_temp_masks __ro_after_init = { | 
|---|
| 47 | .tcc_offset = 0x7f, | 
|---|
| 48 | .digital_readout = 0xff, | 
|---|
| 49 | .pkg_digital_readout = 0xff, | 
|---|
| 50 | }; | 
|---|
| 51 |  | 
|---|
| 52 | static const struct x86_cpu_id intel_tcc_cpu_ids[] __initconst = { | 
|---|
| 53 | X86_MATCH_VFM(INTEL_CORE_YONAH,			&temp_nehalem), | 
|---|
| 54 | X86_MATCH_VFM(INTEL_CORE2_MEROM,		&temp_nehalem), | 
|---|
| 55 | X86_MATCH_VFM(INTEL_CORE2_MEROM_L,		&temp_nehalem), | 
|---|
| 56 | X86_MATCH_VFM(INTEL_CORE2_PENRYN,		&temp_nehalem), | 
|---|
| 57 | X86_MATCH_VFM(INTEL_CORE2_DUNNINGTON,		&temp_nehalem), | 
|---|
| 58 | X86_MATCH_VFM(INTEL_NEHALEM,			&temp_nehalem), | 
|---|
| 59 | X86_MATCH_VFM(INTEL_NEHALEM_G,			&temp_nehalem), | 
|---|
| 60 | X86_MATCH_VFM(INTEL_NEHALEM_EP,			&temp_nehalem), | 
|---|
| 61 | X86_MATCH_VFM(INTEL_NEHALEM_EX,			&temp_nehalem), | 
|---|
| 62 | X86_MATCH_VFM(INTEL_WESTMERE,			&temp_nehalem), | 
|---|
| 63 | X86_MATCH_VFM(INTEL_WESTMERE_EP,		&temp_nehalem), | 
|---|
| 64 | X86_MATCH_VFM(INTEL_WESTMERE_EX,		&temp_nehalem), | 
|---|
| 65 | X86_MATCH_VFM(INTEL_SANDYBRIDGE,		&temp_nehalem), | 
|---|
| 66 | X86_MATCH_VFM(INTEL_SANDYBRIDGE_X,		&temp_nehalem), | 
|---|
| 67 | X86_MATCH_VFM(INTEL_IVYBRIDGE,			&temp_nehalem), | 
|---|
| 68 | X86_MATCH_VFM(INTEL_IVYBRIDGE_X,		&temp_haswell_x), | 
|---|
| 69 | X86_MATCH_VFM(INTEL_HASWELL,			&temp_nehalem), | 
|---|
| 70 | X86_MATCH_VFM(INTEL_HASWELL_X,			&temp_haswell_x), | 
|---|
| 71 | X86_MATCH_VFM(INTEL_HASWELL_L,			&temp_nehalem), | 
|---|
| 72 | X86_MATCH_VFM(INTEL_HASWELL_G,			&temp_nehalem), | 
|---|
| 73 | X86_MATCH_VFM(INTEL_BROADWELL,			&temp_broadwell), | 
|---|
| 74 | X86_MATCH_VFM(INTEL_BROADWELL_G,		&temp_broadwell), | 
|---|
| 75 | X86_MATCH_VFM(INTEL_BROADWELL_X,		&temp_haswell_x), | 
|---|
| 76 | X86_MATCH_VFM(INTEL_BROADWELL_D,		&temp_haswell_x), | 
|---|
| 77 | X86_MATCH_VFM(INTEL_SKYLAKE_L,			&temp_broadwell), | 
|---|
| 78 | X86_MATCH_VFM(INTEL_SKYLAKE,			&temp_broadwell), | 
|---|
| 79 | X86_MATCH_VFM(INTEL_SKYLAKE_X,			&temp_haswell_x), | 
|---|
| 80 | X86_MATCH_VFM(INTEL_KABYLAKE_L,			&temp_broadwell), | 
|---|
| 81 | X86_MATCH_VFM(INTEL_KABYLAKE,			&temp_broadwell), | 
|---|
| 82 | X86_MATCH_VFM(INTEL_COMETLAKE,			&temp_broadwell), | 
|---|
| 83 | X86_MATCH_VFM(INTEL_COMETLAKE_L,		&temp_broadwell), | 
|---|
| 84 | X86_MATCH_VFM(INTEL_CANNONLAKE_L,		&temp_broadwell), | 
|---|
| 85 | X86_MATCH_VFM(INTEL_ICELAKE_X,			&temp_broadwell), | 
|---|
| 86 | X86_MATCH_VFM(INTEL_ICELAKE_D,			&temp_broadwell), | 
|---|
| 87 | X86_MATCH_VFM(INTEL_ICELAKE,			&temp_broadwell), | 
|---|
| 88 | X86_MATCH_VFM(INTEL_ICELAKE_L,			&temp_broadwell), | 
|---|
| 89 | X86_MATCH_VFM(INTEL_ICELAKE_NNPI,		&temp_broadwell), | 
|---|
| 90 | X86_MATCH_VFM(INTEL_ROCKETLAKE,			&temp_broadwell), | 
|---|
| 91 | X86_MATCH_VFM(INTEL_TIGERLAKE_L,		&temp_tigerlake), | 
|---|
| 92 | X86_MATCH_VFM(INTEL_TIGERLAKE,			&temp_tigerlake), | 
|---|
| 93 | X86_MATCH_VFM(INTEL_SAPPHIRERAPIDS_X,		&temp_sapphirerapids), | 
|---|
| 94 | X86_MATCH_VFM(INTEL_EMERALDRAPIDS_X,		&temp_sapphirerapids), | 
|---|
| 95 | X86_MATCH_VFM(INTEL_LAKEFIELD,			&temp_broadwell), | 
|---|
| 96 | X86_MATCH_VFM(INTEL_ALDERLAKE,			&temp_tigerlake), | 
|---|
| 97 | X86_MATCH_VFM(INTEL_ALDERLAKE_L,		&temp_tigerlake), | 
|---|
| 98 | X86_MATCH_VFM(INTEL_RAPTORLAKE,			&temp_tigerlake), | 
|---|
| 99 | X86_MATCH_VFM(INTEL_RAPTORLAKE_P,		&temp_tigerlake), | 
|---|
| 100 | X86_MATCH_VFM(INTEL_RAPTORLAKE_S,		&temp_tigerlake), | 
|---|
| 101 | X86_MATCH_VFM(INTEL_ATOM_BONNELL,		&temp_nehalem), | 
|---|
| 102 | X86_MATCH_VFM(INTEL_ATOM_BONNELL_MID,		&temp_nehalem), | 
|---|
| 103 | X86_MATCH_VFM(INTEL_ATOM_SALTWELL,		&temp_nehalem), | 
|---|
| 104 | X86_MATCH_VFM(INTEL_ATOM_SALTWELL_MID,		&temp_nehalem), | 
|---|
| 105 | X86_MATCH_VFM(INTEL_ATOM_SILVERMONT,		&temp_broadwell), | 
|---|
| 106 | X86_MATCH_VFM(INTEL_ATOM_SILVERMONT_D,		&temp_broadwell), | 
|---|
| 107 | X86_MATCH_VFM(INTEL_ATOM_SILVERMONT_MID,	&temp_broadwell), | 
|---|
| 108 | X86_MATCH_VFM(INTEL_ATOM_AIRMONT,		&temp_broadwell), | 
|---|
| 109 | X86_MATCH_VFM(INTEL_ATOM_SILVERMONT_MID2,	&temp_broadwell), | 
|---|
| 110 | X86_MATCH_VFM(INTEL_ATOM_AIRMONT_NP,		&temp_broadwell), | 
|---|
| 111 | X86_MATCH_VFM(INTEL_ATOM_GOLDMONT,		&temp_goldmont), | 
|---|
| 112 | X86_MATCH_VFM(INTEL_ATOM_GOLDMONT_D,		&temp_goldmont), | 
|---|
| 113 | X86_MATCH_VFM(INTEL_ATOM_GOLDMONT_PLUS,		&temp_goldmont), | 
|---|
| 114 | X86_MATCH_VFM(INTEL_ATOM_TREMONT_D,		&temp_broadwell), | 
|---|
| 115 | X86_MATCH_VFM(INTEL_ATOM_TREMONT,		&temp_broadwell), | 
|---|
| 116 | X86_MATCH_VFM(INTEL_ATOM_TREMONT_L,		&temp_broadwell), | 
|---|
| 117 | X86_MATCH_VFM(INTEL_ATOM_GRACEMONT,		&temp_tigerlake), | 
|---|
| 118 | X86_MATCH_VFM(INTEL_XEON_PHI_KNL,		&temp_broadwell), | 
|---|
| 119 | X86_MATCH_VFM(INTEL_XEON_PHI_KNM,		&temp_broadwell), | 
|---|
| 120 | {} | 
|---|
| 121 | }; | 
|---|
| 122 |  | 
|---|
| 123 | static int __init intel_tcc_init(void) | 
|---|
| 124 | { | 
|---|
| 125 | const struct x86_cpu_id *id; | 
|---|
| 126 |  | 
|---|
| 127 | id = x86_match_cpu(match: intel_tcc_cpu_ids); | 
|---|
| 128 | if (id) | 
|---|
| 129 | memcpy(to: &intel_tcc_temp_masks, from: (const void *)id->driver_data, | 
|---|
| 130 | len: sizeof(intel_tcc_temp_masks)); | 
|---|
| 131 |  | 
|---|
| 132 | return 0; | 
|---|
| 133 | } | 
|---|
| 134 | /* | 
|---|
| 135 | * Use subsys_initcall to ensure temperature bitmasks are initialized before | 
|---|
| 136 | * the drivers that use this library. | 
|---|
| 137 | */ | 
|---|
| 138 | subsys_initcall(intel_tcc_init); | 
|---|
| 139 |  | 
|---|
| 140 | /** | 
|---|
| 141 | * intel_tcc_get_offset_mask() - Returns the bitmask to read TCC offset | 
|---|
| 142 | * | 
|---|
| 143 | * Get the model-specific bitmask to extract TCC_OFFSET from the MSR | 
|---|
| 144 | * TEMPERATURE_TARGET register. If the mask is 0, it means the processor does | 
|---|
| 145 | * not support TCC offset. | 
|---|
| 146 | * | 
|---|
| 147 | * Return: The model-specific bitmask for TCC offset. | 
|---|
| 148 | */ | 
|---|
| 149 | u32 intel_tcc_get_offset_mask(void) | 
|---|
| 150 | { | 
|---|
| 151 | return intel_tcc_temp_masks.tcc_offset; | 
|---|
| 152 | } | 
|---|
| 153 | EXPORT_SYMBOL_NS(intel_tcc_get_offset_mask, "INTEL_TCC"); | 
|---|
| 154 |  | 
|---|
| 155 | /** | 
|---|
| 156 | * get_temp_mask() - Returns the model-specific bitmask for temperature | 
|---|
| 157 | * | 
|---|
| 158 | * @pkg: true: Package Thermal Sensor. false: Core Thermal Sensor. | 
|---|
| 159 | * | 
|---|
| 160 | * Get the model-specific bitmask to extract the temperature reading from the | 
|---|
| 161 | * MSR_IA32_[PACKAGE]_THERM_STATUS register. | 
|---|
| 162 | * | 
|---|
| 163 | * Callers must check if the thermal status registers are supported. | 
|---|
| 164 | * | 
|---|
| 165 | * Return: The model-specific bitmask for temperature reading | 
|---|
| 166 | */ | 
|---|
| 167 | static u32 get_temp_mask(bool pkg) | 
|---|
| 168 | { | 
|---|
| 169 | return pkg ? intel_tcc_temp_masks.pkg_digital_readout : | 
|---|
| 170 | intel_tcc_temp_masks.digital_readout; | 
|---|
| 171 | } | 
|---|
| 172 |  | 
|---|
| 173 | /** | 
|---|
| 174 | * intel_tcc_get_tjmax() - returns the default TCC activation Temperature | 
|---|
| 175 | * @cpu: cpu that the MSR should be run on, nagative value means any cpu. | 
|---|
| 176 | * | 
|---|
| 177 | * Get the TjMax value, which is the default thermal throttling or TCC | 
|---|
| 178 | * activation temperature in degrees C. | 
|---|
| 179 | * | 
|---|
| 180 | * Return: Tjmax value in degrees C on success, negative error code otherwise. | 
|---|
| 181 | */ | 
|---|
| 182 | int intel_tcc_get_tjmax(int cpu) | 
|---|
| 183 | { | 
|---|
| 184 | u32 low, high; | 
|---|
| 185 | int val, err; | 
|---|
| 186 |  | 
|---|
| 187 | if (cpu < 0) | 
|---|
| 188 | err = rdmsr_safe(MSR_IA32_TEMPERATURE_TARGET, &low, &high); | 
|---|
| 189 | else | 
|---|
| 190 | err = rdmsr_safe_on_cpu(cpu, MSR_IA32_TEMPERATURE_TARGET, l: &low, h: &high); | 
|---|
| 191 | if (err) | 
|---|
| 192 | return err; | 
|---|
| 193 |  | 
|---|
| 194 | val = (low >> 16) & 0xff; | 
|---|
| 195 |  | 
|---|
| 196 | return val ? val : -ENODATA; | 
|---|
| 197 | } | 
|---|
| 198 | EXPORT_SYMBOL_NS_GPL(intel_tcc_get_tjmax, "INTEL_TCC"); | 
|---|
| 199 |  | 
|---|
| 200 | /** | 
|---|
| 201 | * intel_tcc_get_offset() - returns the TCC Offset value to Tjmax | 
|---|
| 202 | * @cpu: cpu that the MSR should be run on, nagative value means any cpu. | 
|---|
| 203 | * | 
|---|
| 204 | * Get the TCC offset value to Tjmax. The effective thermal throttling or TCC | 
|---|
| 205 | * activation temperature equals "Tjmax" - "TCC Offset", in degrees C. | 
|---|
| 206 | * | 
|---|
| 207 | * Return: Tcc offset value in degrees C on success, negative error code otherwise. | 
|---|
| 208 | */ | 
|---|
| 209 | int intel_tcc_get_offset(int cpu) | 
|---|
| 210 | { | 
|---|
| 211 | u32 low, high; | 
|---|
| 212 | int err; | 
|---|
| 213 |  | 
|---|
| 214 | if (cpu < 0) | 
|---|
| 215 | err = rdmsr_safe(MSR_IA32_TEMPERATURE_TARGET, &low, &high); | 
|---|
| 216 | else | 
|---|
| 217 | err = rdmsr_safe_on_cpu(cpu, MSR_IA32_TEMPERATURE_TARGET, l: &low, h: &high); | 
|---|
| 218 | if (err) | 
|---|
| 219 | return err; | 
|---|
| 220 |  | 
|---|
| 221 | return (low >> 24) & intel_tcc_temp_masks.tcc_offset; | 
|---|
| 222 | } | 
|---|
| 223 | EXPORT_SYMBOL_NS_GPL(intel_tcc_get_offset, "INTEL_TCC"); | 
|---|
| 224 |  | 
|---|
| 225 | /** | 
|---|
| 226 | * intel_tcc_set_offset() - set the TCC offset value to Tjmax | 
|---|
| 227 | * @cpu: cpu that the MSR should be run on, nagative value means any cpu. | 
|---|
| 228 | * @offset: TCC offset value in degree C | 
|---|
| 229 | * | 
|---|
| 230 | * Set the TCC Offset value to Tjmax. The effective thermal throttling or TCC | 
|---|
| 231 | * activation temperature equals "Tjmax" - "TCC Offset", in degree C. | 
|---|
| 232 | * | 
|---|
| 233 | * Return: On success returns 0, negative error code otherwise. | 
|---|
| 234 | */ | 
|---|
| 235 |  | 
|---|
| 236 | int intel_tcc_set_offset(int cpu, int offset) | 
|---|
| 237 | { | 
|---|
| 238 | u32 low, high; | 
|---|
| 239 | int err; | 
|---|
| 240 |  | 
|---|
| 241 | if (!intel_tcc_temp_masks.tcc_offset) | 
|---|
| 242 | return -ENODEV; | 
|---|
| 243 |  | 
|---|
| 244 | if (offset < 0 || offset > intel_tcc_temp_masks.tcc_offset) | 
|---|
| 245 | return -EINVAL; | 
|---|
| 246 |  | 
|---|
| 247 | if (cpu < 0) | 
|---|
| 248 | err = rdmsr_safe(MSR_IA32_TEMPERATURE_TARGET, &low, &high); | 
|---|
| 249 | else | 
|---|
| 250 | err = rdmsr_safe_on_cpu(cpu, MSR_IA32_TEMPERATURE_TARGET, l: &low, h: &high); | 
|---|
| 251 | if (err) | 
|---|
| 252 | return err; | 
|---|
| 253 |  | 
|---|
| 254 | /* MSR Locked */ | 
|---|
| 255 | if (low & BIT(31)) | 
|---|
| 256 | return -EPERM; | 
|---|
| 257 |  | 
|---|
| 258 | low &= ~(intel_tcc_temp_masks.tcc_offset << 24); | 
|---|
| 259 | low |= offset << 24; | 
|---|
| 260 |  | 
|---|
| 261 | if (cpu < 0) | 
|---|
| 262 | return wrmsr_safe(MSR_IA32_TEMPERATURE_TARGET, low, high); | 
|---|
| 263 | else | 
|---|
| 264 | return wrmsr_safe_on_cpu(cpu, MSR_IA32_TEMPERATURE_TARGET, l: low, h: high); | 
|---|
| 265 | } | 
|---|
| 266 | EXPORT_SYMBOL_NS_GPL(intel_tcc_set_offset, "INTEL_TCC"); | 
|---|
| 267 |  | 
|---|
| 268 | /** | 
|---|
| 269 | * intel_tcc_get_temp() - returns the current temperature | 
|---|
| 270 | * @cpu: cpu that the MSR should be run on, nagative value means any cpu. | 
|---|
| 271 | * @temp: pointer to the memory for saving cpu temperature. | 
|---|
| 272 | * @pkg: true: Package Thermal Sensor. false: Core Thermal Sensor. | 
|---|
| 273 | * | 
|---|
| 274 | * Get the current temperature returned by the CPU core/package level | 
|---|
| 275 | * thermal sensor, in degrees C. | 
|---|
| 276 | * | 
|---|
| 277 | * Return: 0 on success, negative error code otherwise. | 
|---|
| 278 | */ | 
|---|
| 279 | int intel_tcc_get_temp(int cpu, int *temp, bool pkg) | 
|---|
| 280 | { | 
|---|
| 281 | u32 msr = pkg ? MSR_IA32_PACKAGE_THERM_STATUS : MSR_IA32_THERM_STATUS; | 
|---|
| 282 | u32 low, high, mask; | 
|---|
| 283 | int tjmax, err; | 
|---|
| 284 |  | 
|---|
| 285 | tjmax = intel_tcc_get_tjmax(cpu); | 
|---|
| 286 | if (tjmax < 0) | 
|---|
| 287 | return tjmax; | 
|---|
| 288 |  | 
|---|
| 289 | if (cpu < 0) | 
|---|
| 290 | err = rdmsr_safe(msr, &low, &high); | 
|---|
| 291 | else | 
|---|
| 292 | err = rdmsr_safe_on_cpu(cpu, msr_no: msr, l: &low, h: &high); | 
|---|
| 293 | if (err) | 
|---|
| 294 | return err; | 
|---|
| 295 |  | 
|---|
| 296 | /* Temperature is beyond the valid thermal sensor range */ | 
|---|
| 297 | if (!(low & BIT(31))) | 
|---|
| 298 | return -ENODATA; | 
|---|
| 299 |  | 
|---|
| 300 | mask = get_temp_mask(pkg); | 
|---|
| 301 |  | 
|---|
| 302 | *temp = tjmax - ((low >> 16) & mask); | 
|---|
| 303 |  | 
|---|
| 304 | return 0; | 
|---|
| 305 | } | 
|---|
| 306 | EXPORT_SYMBOL_NS_GPL(intel_tcc_get_temp, "INTEL_TCC"); | 
|---|
| 307 |  | 
|---|