xref: /linux/drivers/thermal/intel/intel_tcc.c (revision 17e548405a81665fd14cee960db7d093d1396400)
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(intel_tcc_cpu_ids);
128 	if (id)
129 		memcpy(&intel_tcc_temp_masks, (const void *)id->driver_data,
130 		       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, &low, &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, &low, &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, &low, &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, low, 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, &low, &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