mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 02:30:34 +02:00 
			
		
		
		
	Clean up the existing export namespace code along the same lines of
commit 33def8498f ("treewide: Convert macro and uses of __section(foo)
to __section("foo")") and for the same reason, it is not desired for the
namespace argument to be a macro expansion itself.
Scripted using
  git grep -l -e MODULE_IMPORT_NS -e EXPORT_SYMBOL_NS | while read file;
  do
    awk -i inplace '
      /^#define EXPORT_SYMBOL_NS/ {
        gsub(/__stringify\(ns\)/, "ns");
        print;
        next;
      }
      /^#define MODULE_IMPORT_NS/ {
        gsub(/__stringify\(ns\)/, "ns");
        print;
        next;
      }
      /MODULE_IMPORT_NS/ {
        $0 = gensub(/MODULE_IMPORT_NS\(([^)]*)\)/, "MODULE_IMPORT_NS(\"\\1\")", "g");
      }
      /EXPORT_SYMBOL_NS/ {
        if ($0 ~ /(EXPORT_SYMBOL_NS[^(]*)\(([^,]+),/) {
  	if ($0 !~ /(EXPORT_SYMBOL_NS[^(]*)\(([^,]+), ([^)]+)\)/ &&
  	    $0 !~ /(EXPORT_SYMBOL_NS[^(]*)\(\)/ &&
  	    $0 !~ /^my/) {
  	  getline line;
  	  gsub(/[[:space:]]*\\$/, "");
  	  gsub(/[[:space:]]/, "", line);
  	  $0 = $0 " " line;
  	}
  	$0 = gensub(/(EXPORT_SYMBOL_NS[^(]*)\(([^,]+), ([^)]+)\)/,
  		    "\\1(\\2, \"\\3\")", "g");
        }
      }
      { print }' $file;
  done
Requested-by: Masahiro Yamada <masahiroy@kernel.org>
Signed-off-by: Peter Zijlstra (Intel) <peterz@infradead.org>
Link: https://mail.google.com/mail/u/2/#inbox/FMfcgzQXKWgMmjdFwwdsfgxzKpVHWPlc
Acked-by: Greg KH <gregkh@linuxfoundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
		
	
			
		
			
				
	
	
		
			306 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			306 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
// 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_AIRMONT_MID,		&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, nagative 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, nagative 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, nagative 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, nagative 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");
 |