]> Git Repo - linux.git/blob - drivers/thermal/intel/intel_tcc.c
crypto: akcipher - Drop sign/verify operations
[linux.git] / drivers / thermal / intel / intel_tcc.c
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_AIRMONT_MID,           &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);
This page took 0.049416 seconds and 4 git commands to generate.