Merge tag 'trace-printf-v6.13' of git://git.kernel.org/pub/scm/linux/kernel/git/trace...
[drm/drm-misc.git] / drivers / peci / device.c
blob37ca7dd61807c7db5d0ccf684883b05b64d97cb1
1 // SPDX-License-Identifier: GPL-2.0-only
2 // Copyright (c) 2018-2021 Intel Corporation
4 #include <linux/bitfield.h>
5 #include <linux/peci.h>
6 #include <linux/peci-cpu.h>
7 #include <linux/slab.h>
9 #include "internal.h"
12 * PECI device can be removed using sysfs, but the removal can also happen as
13 * a result of controller being removed.
14 * Mutex is used to protect PECI device from being double-deleted.
16 static DEFINE_MUTEX(peci_device_del_lock);
18 #define REVISION_NUM_MASK GENMASK(15, 8)
19 static int peci_get_revision(struct peci_device *device, u8 *revision)
21 struct peci_request *req;
22 u64 dib;
24 req = peci_xfer_get_dib(device);
25 if (IS_ERR(req))
26 return PTR_ERR(req);
29 * PECI device may be in a state where it is unable to return a proper
30 * DIB, in which case it returns 0 as DIB value.
31 * Let's treat this as an error to avoid carrying on with the detection
32 * using invalid revision.
34 dib = peci_request_dib_read(req);
35 if (dib == 0) {
36 peci_request_free(req);
37 return -EIO;
40 *revision = FIELD_GET(REVISION_NUM_MASK, dib);
42 peci_request_free(req);
44 return 0;
47 static int peci_get_cpu_id(struct peci_device *device, u32 *cpu_id)
49 struct peci_request *req;
50 int ret;
52 req = peci_xfer_pkg_cfg_readl(device, PECI_PCS_PKG_ID, PECI_PKG_ID_CPU_ID);
53 if (IS_ERR(req))
54 return PTR_ERR(req);
56 ret = peci_request_status(req);
57 if (ret)
58 goto out_req_free;
60 *cpu_id = peci_request_data_readl(req);
61 out_req_free:
62 peci_request_free(req);
64 return ret;
67 static unsigned int peci_x86_cpu_family(unsigned int sig)
69 unsigned int x86;
71 x86 = (sig >> 8) & 0xf;
73 if (x86 == 0xf)
74 x86 += (sig >> 20) & 0xff;
76 return x86;
79 static unsigned int peci_x86_cpu_model(unsigned int sig)
81 unsigned int fam, model;
83 fam = peci_x86_cpu_family(sig);
85 model = (sig >> 4) & 0xf;
87 if (fam >= 0x6)
88 model += ((sig >> 16) & 0xf) << 4;
90 return model;
93 static int peci_device_info_init(struct peci_device *device)
95 u8 revision;
96 u32 cpu_id;
97 int ret;
99 ret = peci_get_cpu_id(device, &cpu_id);
100 if (ret)
101 return ret;
103 device->info.x86_vfm = IFM(peci_x86_cpu_family(cpu_id), peci_x86_cpu_model(cpu_id));
105 ret = peci_get_revision(device, &revision);
106 if (ret)
107 return ret;
108 device->info.peci_revision = revision;
110 device->info.socket_id = device->addr - PECI_BASE_ADDR;
112 return 0;
115 static int peci_detect(struct peci_controller *controller, u8 addr)
118 * PECI Ping is a command encoded by tx_len = 0, rx_len = 0.
119 * We expect correct Write FCS if the device at the target address
120 * is able to respond.
122 struct peci_request req = { 0 };
123 int ret;
125 mutex_lock(&controller->bus_lock);
126 ret = controller->ops->xfer(controller, addr, &req);
127 mutex_unlock(&controller->bus_lock);
129 return ret;
132 static bool peci_addr_valid(u8 addr)
134 return addr >= PECI_BASE_ADDR && addr < PECI_BASE_ADDR + PECI_DEVICE_NUM_MAX;
137 static int peci_dev_exists(struct device *dev, void *data)
139 struct peci_device *device = to_peci_device(dev);
140 u8 *addr = data;
142 if (device->addr == *addr)
143 return -EBUSY;
145 return 0;
148 int peci_device_create(struct peci_controller *controller, u8 addr)
150 struct peci_device *device;
151 int ret;
153 if (!peci_addr_valid(addr))
154 return -EINVAL;
156 /* Check if we have already detected this device before. */
157 ret = device_for_each_child(&controller->dev, &addr, peci_dev_exists);
158 if (ret)
159 return 0;
161 ret = peci_detect(controller, addr);
162 if (ret) {
164 * Device not present or host state doesn't allow successful
165 * detection at this time.
167 if (ret == -EIO || ret == -ETIMEDOUT)
168 return 0;
170 return ret;
173 device = kzalloc(sizeof(*device), GFP_KERNEL);
174 if (!device)
175 return -ENOMEM;
177 device_initialize(&device->dev);
179 device->addr = addr;
180 device->dev.parent = &controller->dev;
181 device->dev.bus = &peci_bus_type;
182 device->dev.type = &peci_device_type;
184 ret = peci_device_info_init(device);
185 if (ret)
186 goto err_put;
188 ret = dev_set_name(&device->dev, "%d-%02x", controller->id, device->addr);
189 if (ret)
190 goto err_put;
192 ret = device_add(&device->dev);
193 if (ret)
194 goto err_put;
196 return 0;
198 err_put:
199 put_device(&device->dev);
201 return ret;
204 void peci_device_destroy(struct peci_device *device)
206 mutex_lock(&peci_device_del_lock);
207 if (!device->deleted) {
208 device_unregister(&device->dev);
209 device->deleted = true;
211 mutex_unlock(&peci_device_del_lock);
214 int __peci_driver_register(struct peci_driver *driver, struct module *owner,
215 const char *mod_name)
217 driver->driver.bus = &peci_bus_type;
218 driver->driver.owner = owner;
219 driver->driver.mod_name = mod_name;
221 if (!driver->probe) {
222 pr_err("peci: trying to register driver without probe callback\n");
223 return -EINVAL;
226 if (!driver->id_table) {
227 pr_err("peci: trying to register driver without device id table\n");
228 return -EINVAL;
231 return driver_register(&driver->driver);
233 EXPORT_SYMBOL_NS_GPL(__peci_driver_register, PECI);
235 void peci_driver_unregister(struct peci_driver *driver)
237 driver_unregister(&driver->driver);
239 EXPORT_SYMBOL_NS_GPL(peci_driver_unregister, PECI);
241 static void peci_device_release(struct device *dev)
243 struct peci_device *device = to_peci_device(dev);
245 kfree(device);
248 const struct device_type peci_device_type = {
249 .groups = peci_device_groups,
250 .release = peci_device_release,