cachepc-linux

Fork of AMDESE/linux with modifications for CachePC side-channel attack
git clone https://git.sinitax.com/sinitax/cachepc-linux
Log | Files | Refs | README | LICENSE | sfeed.txt

cpu.c (7502B)


      1// SPDX-License-Identifier: GPL-2.0-only
      2// Copyright (c) 2021 Intel Corporation
      3
      4#include <linux/auxiliary_bus.h>
      5#include <linux/module.h>
      6#include <linux/peci.h>
      7#include <linux/peci-cpu.h>
      8#include <linux/slab.h>
      9
     10#include "internal.h"
     11
     12/**
     13 * peci_temp_read() - read the maximum die temperature from PECI target device
     14 * @device: PECI device to which request is going to be sent
     15 * @temp_raw: where to store the read temperature
     16 *
     17 * It uses GetTemp PECI command.
     18 *
     19 * Return: 0 if succeeded, other values in case errors.
     20 */
     21int peci_temp_read(struct peci_device *device, s16 *temp_raw)
     22{
     23	struct peci_request *req;
     24
     25	req = peci_xfer_get_temp(device);
     26	if (IS_ERR(req))
     27		return PTR_ERR(req);
     28
     29	*temp_raw = peci_request_temp_read(req);
     30
     31	peci_request_free(req);
     32
     33	return 0;
     34}
     35EXPORT_SYMBOL_NS_GPL(peci_temp_read, PECI_CPU);
     36
     37/**
     38 * peci_pcs_read() - read PCS register
     39 * @device: PECI device to which request is going to be sent
     40 * @index: PCS index
     41 * @param: PCS parameter
     42 * @data: where to store the read data
     43 *
     44 * It uses RdPkgConfig PECI command.
     45 *
     46 * Return: 0 if succeeded, other values in case errors.
     47 */
     48int peci_pcs_read(struct peci_device *device, u8 index, u16 param, u32 *data)
     49{
     50	struct peci_request *req;
     51	int ret;
     52
     53	req = peci_xfer_pkg_cfg_readl(device, index, param);
     54	if (IS_ERR(req))
     55		return PTR_ERR(req);
     56
     57	ret = peci_request_status(req);
     58	if (ret)
     59		goto out_req_free;
     60
     61	*data = peci_request_data_readl(req);
     62out_req_free:
     63	peci_request_free(req);
     64
     65	return ret;
     66}
     67EXPORT_SYMBOL_NS_GPL(peci_pcs_read, PECI_CPU);
     68
     69/**
     70 * peci_pci_local_read() - read 32-bit memory location using raw address
     71 * @device: PECI device to which request is going to be sent
     72 * @bus: bus
     73 * @dev: device
     74 * @func: function
     75 * @reg: register
     76 * @data: where to store the read data
     77 *
     78 * It uses RdPCIConfigLocal PECI command.
     79 *
     80 * Return: 0 if succeeded, other values in case errors.
     81 */
     82int peci_pci_local_read(struct peci_device *device, u8 bus, u8 dev, u8 func,
     83			u16 reg, u32 *data)
     84{
     85	struct peci_request *req;
     86	int ret;
     87
     88	req = peci_xfer_pci_cfg_local_readl(device, bus, dev, func, reg);
     89	if (IS_ERR(req))
     90		return PTR_ERR(req);
     91
     92	ret = peci_request_status(req);
     93	if (ret)
     94		goto out_req_free;
     95
     96	*data = peci_request_data_readl(req);
     97out_req_free:
     98	peci_request_free(req);
     99
    100	return ret;
    101}
    102EXPORT_SYMBOL_NS_GPL(peci_pci_local_read, PECI_CPU);
    103
    104/**
    105 * peci_ep_pci_local_read() - read 32-bit memory location using raw address
    106 * @device: PECI device to which request is going to be sent
    107 * @seg: PCI segment
    108 * @bus: bus
    109 * @dev: device
    110 * @func: function
    111 * @reg: register
    112 * @data: where to store the read data
    113 *
    114 * Like &peci_pci_local_read, but it uses RdEndpointConfig PECI command.
    115 *
    116 * Return: 0 if succeeded, other values in case errors.
    117 */
    118int peci_ep_pci_local_read(struct peci_device *device, u8 seg,
    119			   u8 bus, u8 dev, u8 func, u16 reg, u32 *data)
    120{
    121	struct peci_request *req;
    122	int ret;
    123
    124	req = peci_xfer_ep_pci_cfg_local_readl(device, seg, bus, dev, func, reg);
    125	if (IS_ERR(req))
    126		return PTR_ERR(req);
    127
    128	ret = peci_request_status(req);
    129	if (ret)
    130		goto out_req_free;
    131
    132	*data = peci_request_data_readl(req);
    133out_req_free:
    134	peci_request_free(req);
    135
    136	return ret;
    137}
    138EXPORT_SYMBOL_NS_GPL(peci_ep_pci_local_read, PECI_CPU);
    139
    140/**
    141 * peci_mmio_read() - read 32-bit memory location using 64-bit bar offset address
    142 * @device: PECI device to which request is going to be sent
    143 * @bar: PCI bar
    144 * @seg: PCI segment
    145 * @bus: bus
    146 * @dev: device
    147 * @func: function
    148 * @address: 64-bit MMIO address
    149 * @data: where to store the read data
    150 *
    151 * It uses RdEndpointConfig PECI command.
    152 *
    153 * Return: 0 if succeeded, other values in case errors.
    154 */
    155int peci_mmio_read(struct peci_device *device, u8 bar, u8 seg,
    156		   u8 bus, u8 dev, u8 func, u64 address, u32 *data)
    157{
    158	struct peci_request *req;
    159	int ret;
    160
    161	req = peci_xfer_ep_mmio64_readl(device, bar, seg, bus, dev, func, address);
    162	if (IS_ERR(req))
    163		return PTR_ERR(req);
    164
    165	ret = peci_request_status(req);
    166	if (ret)
    167		goto out_req_free;
    168
    169	*data = peci_request_data_readl(req);
    170out_req_free:
    171	peci_request_free(req);
    172
    173	return ret;
    174}
    175EXPORT_SYMBOL_NS_GPL(peci_mmio_read, PECI_CPU);
    176
    177static const char * const peci_adev_types[] = {
    178	"cputemp",
    179	"dimmtemp",
    180};
    181
    182struct peci_cpu {
    183	struct peci_device *device;
    184	const struct peci_device_id *id;
    185};
    186
    187static void adev_release(struct device *dev)
    188{
    189	struct auxiliary_device *adev = to_auxiliary_dev(dev);
    190
    191	auxiliary_device_uninit(adev);
    192
    193	kfree(adev->name);
    194	kfree(adev);
    195}
    196
    197static struct auxiliary_device *adev_alloc(struct peci_cpu *priv, int idx)
    198{
    199	struct peci_controller *controller = to_peci_controller(priv->device->dev.parent);
    200	struct auxiliary_device *adev;
    201	const char *name;
    202	int ret;
    203
    204	adev = kzalloc(sizeof(*adev), GFP_KERNEL);
    205	if (!adev)
    206		return ERR_PTR(-ENOMEM);
    207
    208	name = kasprintf(GFP_KERNEL, "%s.%s", peci_adev_types[idx], (const char *)priv->id->data);
    209	if (!name) {
    210		ret = -ENOMEM;
    211		goto free_adev;
    212	}
    213
    214	adev->name = name;
    215	adev->dev.parent = &priv->device->dev;
    216	adev->dev.release = adev_release;
    217	adev->id = (controller->id << 16) | (priv->device->addr);
    218
    219	ret = auxiliary_device_init(adev);
    220	if (ret)
    221		goto free_name;
    222
    223	return adev;
    224
    225free_name:
    226	kfree(name);
    227free_adev:
    228	kfree(adev);
    229	return ERR_PTR(ret);
    230}
    231
    232static void unregister_adev(void *_adev)
    233{
    234	struct auxiliary_device *adev = _adev;
    235
    236	auxiliary_device_delete(adev);
    237}
    238
    239static int devm_adev_add(struct device *dev, int idx)
    240{
    241	struct peci_cpu *priv = dev_get_drvdata(dev);
    242	struct auxiliary_device *adev;
    243	int ret;
    244
    245	adev = adev_alloc(priv, idx);
    246	if (IS_ERR(adev))
    247		return PTR_ERR(adev);
    248
    249	ret = auxiliary_device_add(adev);
    250	if (ret) {
    251		auxiliary_device_uninit(adev);
    252		return ret;
    253	}
    254
    255	ret = devm_add_action_or_reset(&priv->device->dev, unregister_adev, adev);
    256	if (ret)
    257		return ret;
    258
    259	return 0;
    260}
    261
    262static void peci_cpu_add_adevices(struct peci_cpu *priv)
    263{
    264	struct device *dev = &priv->device->dev;
    265	int ret, i;
    266
    267	for (i = 0; i < ARRAY_SIZE(peci_adev_types); i++) {
    268		ret = devm_adev_add(dev, i);
    269		if (ret) {
    270			dev_warn(dev, "Failed to register PECI auxiliary: %s, ret = %d\n",
    271				 peci_adev_types[i], ret);
    272			continue;
    273		}
    274	}
    275}
    276
    277static int
    278peci_cpu_probe(struct peci_device *device, const struct peci_device_id *id)
    279{
    280	struct device *dev = &device->dev;
    281	struct peci_cpu *priv;
    282
    283	priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
    284	if (!priv)
    285		return -ENOMEM;
    286
    287	dev_set_drvdata(dev, priv);
    288	priv->device = device;
    289	priv->id = id;
    290
    291	peci_cpu_add_adevices(priv);
    292
    293	return 0;
    294}
    295
    296static const struct peci_device_id peci_cpu_device_ids[] = {
    297	{ /* Haswell Xeon */
    298		.family	= 6,
    299		.model	= INTEL_FAM6_HASWELL_X,
    300		.data	= "hsx",
    301	},
    302	{ /* Broadwell Xeon */
    303		.family	= 6,
    304		.model	= INTEL_FAM6_BROADWELL_X,
    305		.data	= "bdx",
    306	},
    307	{ /* Broadwell Xeon D */
    308		.family	= 6,
    309		.model	= INTEL_FAM6_BROADWELL_D,
    310		.data	= "bdxd",
    311	},
    312	{ /* Skylake Xeon */
    313		.family	= 6,
    314		.model	= INTEL_FAM6_SKYLAKE_X,
    315		.data	= "skx",
    316	},
    317	{ /* Icelake Xeon */
    318		.family	= 6,
    319		.model	= INTEL_FAM6_ICELAKE_X,
    320		.data	= "icx",
    321	},
    322	{ /* Icelake Xeon D */
    323		.family	= 6,
    324		.model	= INTEL_FAM6_ICELAKE_D,
    325		.data	= "icxd",
    326	},
    327	{ }
    328};
    329MODULE_DEVICE_TABLE(peci, peci_cpu_device_ids);
    330
    331static struct peci_driver peci_cpu_driver = {
    332	.probe		= peci_cpu_probe,
    333	.id_table	= peci_cpu_device_ids,
    334	.driver		= {
    335		.name		= "peci-cpu",
    336	},
    337};
    338module_peci_driver(peci_cpu_driver);
    339
    340MODULE_AUTHOR("Iwona Winiarska <iwona.winiarska@intel.com>");
    341MODULE_DESCRIPTION("PECI CPU driver");
    342MODULE_LICENSE("GPL");
    343MODULE_IMPORT_NS(PECI);