1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * Driver for the Intel Broxton PMC
4*4882a593Smuzhiyun *
5*4882a593Smuzhiyun * (C) Copyright 2014 - 2020 Intel Corporation
6*4882a593Smuzhiyun *
7*4882a593Smuzhiyun * This driver is based on Intel SCU IPC driver (intel_scu_ipc.c) by
8*4882a593Smuzhiyun * Sreedhara DS <sreedhara.ds@intel.com>
9*4882a593Smuzhiyun *
10*4882a593Smuzhiyun * The PMC (Power Management Controller) running on the ARC processor
11*4882a593Smuzhiyun * communicates with another entity running in the IA (Intel Architecture)
12*4882a593Smuzhiyun * core through an IPC (Intel Processor Communications) mechanism which in
13*4882a593Smuzhiyun * turn sends messages between the IA and the PMC.
14*4882a593Smuzhiyun */
15*4882a593Smuzhiyun
16*4882a593Smuzhiyun #include <linux/acpi.h>
17*4882a593Smuzhiyun #include <linux/delay.h>
18*4882a593Smuzhiyun #include <linux/errno.h>
19*4882a593Smuzhiyun #include <linux/interrupt.h>
20*4882a593Smuzhiyun #include <linux/io-64-nonatomic-lo-hi.h>
21*4882a593Smuzhiyun #include <linux/mfd/core.h>
22*4882a593Smuzhiyun #include <linux/mfd/intel_pmc_bxt.h>
23*4882a593Smuzhiyun #include <linux/module.h>
24*4882a593Smuzhiyun #include <linux/platform_device.h>
25*4882a593Smuzhiyun #include <linux/platform_data/itco_wdt.h>
26*4882a593Smuzhiyun
27*4882a593Smuzhiyun #include <asm/intel_scu_ipc.h>
28*4882a593Smuzhiyun
29*4882a593Smuzhiyun /* Residency with clock rate at 19.2MHz to usecs */
30*4882a593Smuzhiyun #define S0IX_RESIDENCY_IN_USECS(d, s) \
31*4882a593Smuzhiyun ({ \
32*4882a593Smuzhiyun u64 result = 10ull * ((d) + (s)); \
33*4882a593Smuzhiyun do_div(result, 192); \
34*4882a593Smuzhiyun result; \
35*4882a593Smuzhiyun })
36*4882a593Smuzhiyun
37*4882a593Smuzhiyun /* Resources exported from IFWI */
38*4882a593Smuzhiyun #define PLAT_RESOURCE_IPC_INDEX 0
39*4882a593Smuzhiyun #define PLAT_RESOURCE_IPC_SIZE 0x1000
40*4882a593Smuzhiyun #define PLAT_RESOURCE_GCR_OFFSET 0x1000
41*4882a593Smuzhiyun #define PLAT_RESOURCE_GCR_SIZE 0x1000
42*4882a593Smuzhiyun #define PLAT_RESOURCE_BIOS_DATA_INDEX 1
43*4882a593Smuzhiyun #define PLAT_RESOURCE_BIOS_IFACE_INDEX 2
44*4882a593Smuzhiyun #define PLAT_RESOURCE_TELEM_SSRAM_INDEX 3
45*4882a593Smuzhiyun #define PLAT_RESOURCE_ISP_DATA_INDEX 4
46*4882a593Smuzhiyun #define PLAT_RESOURCE_ISP_IFACE_INDEX 5
47*4882a593Smuzhiyun #define PLAT_RESOURCE_GTD_DATA_INDEX 6
48*4882a593Smuzhiyun #define PLAT_RESOURCE_GTD_IFACE_INDEX 7
49*4882a593Smuzhiyun #define PLAT_RESOURCE_ACPI_IO_INDEX 0
50*4882a593Smuzhiyun
51*4882a593Smuzhiyun /*
52*4882a593Smuzhiyun * BIOS does not create an ACPI device for each PMC function, but
53*4882a593Smuzhiyun * exports multiple resources from one ACPI device (IPC) for multiple
54*4882a593Smuzhiyun * functions. This driver is responsible for creating a child device and
55*4882a593Smuzhiyun * to export resources for those functions.
56*4882a593Smuzhiyun */
57*4882a593Smuzhiyun #define SMI_EN_OFFSET 0x0040
58*4882a593Smuzhiyun #define SMI_EN_SIZE 4
59*4882a593Smuzhiyun #define TCO_BASE_OFFSET 0x0060
60*4882a593Smuzhiyun #define TCO_REGS_SIZE 16
61*4882a593Smuzhiyun #define TELEM_SSRAM_SIZE 240
62*4882a593Smuzhiyun #define TELEM_PMC_SSRAM_OFFSET 0x1b00
63*4882a593Smuzhiyun #define TELEM_PUNIT_SSRAM_OFFSET 0x1a00
64*4882a593Smuzhiyun
65*4882a593Smuzhiyun /* Commands */
66*4882a593Smuzhiyun #define PMC_NORTHPEAK_CTRL 0xed
67*4882a593Smuzhiyun
is_gcr_valid(u32 offset)68*4882a593Smuzhiyun static inline bool is_gcr_valid(u32 offset)
69*4882a593Smuzhiyun {
70*4882a593Smuzhiyun return offset < PLAT_RESOURCE_GCR_SIZE - 8;
71*4882a593Smuzhiyun }
72*4882a593Smuzhiyun
73*4882a593Smuzhiyun /**
74*4882a593Smuzhiyun * intel_pmc_gcr_read64() - Read a 64-bit PMC GCR register
75*4882a593Smuzhiyun * @pmc: PMC device pointer
76*4882a593Smuzhiyun * @offset: offset of GCR register from GCR address base
77*4882a593Smuzhiyun * @data: data pointer for storing the register output
78*4882a593Smuzhiyun *
79*4882a593Smuzhiyun * Reads the 64-bit PMC GCR register at given offset.
80*4882a593Smuzhiyun *
81*4882a593Smuzhiyun * Return: Negative value on error or 0 on success.
82*4882a593Smuzhiyun */
intel_pmc_gcr_read64(struct intel_pmc_dev * pmc,u32 offset,u64 * data)83*4882a593Smuzhiyun int intel_pmc_gcr_read64(struct intel_pmc_dev *pmc, u32 offset, u64 *data)
84*4882a593Smuzhiyun {
85*4882a593Smuzhiyun if (!is_gcr_valid(offset))
86*4882a593Smuzhiyun return -EINVAL;
87*4882a593Smuzhiyun
88*4882a593Smuzhiyun spin_lock(&pmc->gcr_lock);
89*4882a593Smuzhiyun *data = readq(pmc->gcr_mem_base + offset);
90*4882a593Smuzhiyun spin_unlock(&pmc->gcr_lock);
91*4882a593Smuzhiyun
92*4882a593Smuzhiyun return 0;
93*4882a593Smuzhiyun }
94*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(intel_pmc_gcr_read64);
95*4882a593Smuzhiyun
96*4882a593Smuzhiyun /**
97*4882a593Smuzhiyun * intel_pmc_gcr_update() - Update PMC GCR register bits
98*4882a593Smuzhiyun * @pmc: PMC device pointer
99*4882a593Smuzhiyun * @offset: offset of GCR register from GCR address base
100*4882a593Smuzhiyun * @mask: bit mask for update operation
101*4882a593Smuzhiyun * @val: update value
102*4882a593Smuzhiyun *
103*4882a593Smuzhiyun * Updates the bits of given GCR register as specified by
104*4882a593Smuzhiyun * @mask and @val.
105*4882a593Smuzhiyun *
106*4882a593Smuzhiyun * Return: Negative value on error or 0 on success.
107*4882a593Smuzhiyun */
intel_pmc_gcr_update(struct intel_pmc_dev * pmc,u32 offset,u32 mask,u32 val)108*4882a593Smuzhiyun int intel_pmc_gcr_update(struct intel_pmc_dev *pmc, u32 offset, u32 mask, u32 val)
109*4882a593Smuzhiyun {
110*4882a593Smuzhiyun u32 new_val;
111*4882a593Smuzhiyun
112*4882a593Smuzhiyun if (!is_gcr_valid(offset))
113*4882a593Smuzhiyun return -EINVAL;
114*4882a593Smuzhiyun
115*4882a593Smuzhiyun spin_lock(&pmc->gcr_lock);
116*4882a593Smuzhiyun new_val = readl(pmc->gcr_mem_base + offset);
117*4882a593Smuzhiyun
118*4882a593Smuzhiyun new_val = (new_val & ~mask) | (val & mask);
119*4882a593Smuzhiyun writel(new_val, pmc->gcr_mem_base + offset);
120*4882a593Smuzhiyun
121*4882a593Smuzhiyun new_val = readl(pmc->gcr_mem_base + offset);
122*4882a593Smuzhiyun spin_unlock(&pmc->gcr_lock);
123*4882a593Smuzhiyun
124*4882a593Smuzhiyun /* Check whether the bit update is successful */
125*4882a593Smuzhiyun return (new_val & mask) != (val & mask) ? -EIO : 0;
126*4882a593Smuzhiyun }
127*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(intel_pmc_gcr_update);
128*4882a593Smuzhiyun
129*4882a593Smuzhiyun /**
130*4882a593Smuzhiyun * intel_pmc_s0ix_counter_read() - Read S0ix residency
131*4882a593Smuzhiyun * @pmc: PMC device pointer
132*4882a593Smuzhiyun * @data: Out param that contains current S0ix residency count.
133*4882a593Smuzhiyun *
134*4882a593Smuzhiyun * Writes to @data how many usecs the system has been in low-power S0ix
135*4882a593Smuzhiyun * state.
136*4882a593Smuzhiyun *
137*4882a593Smuzhiyun * Return: An error code or 0 on success.
138*4882a593Smuzhiyun */
intel_pmc_s0ix_counter_read(struct intel_pmc_dev * pmc,u64 * data)139*4882a593Smuzhiyun int intel_pmc_s0ix_counter_read(struct intel_pmc_dev *pmc, u64 *data)
140*4882a593Smuzhiyun {
141*4882a593Smuzhiyun u64 deep, shlw;
142*4882a593Smuzhiyun
143*4882a593Smuzhiyun spin_lock(&pmc->gcr_lock);
144*4882a593Smuzhiyun deep = readq(pmc->gcr_mem_base + PMC_GCR_TELEM_DEEP_S0IX_REG);
145*4882a593Smuzhiyun shlw = readq(pmc->gcr_mem_base + PMC_GCR_TELEM_SHLW_S0IX_REG);
146*4882a593Smuzhiyun spin_unlock(&pmc->gcr_lock);
147*4882a593Smuzhiyun
148*4882a593Smuzhiyun *data = S0IX_RESIDENCY_IN_USECS(deep, shlw);
149*4882a593Smuzhiyun return 0;
150*4882a593Smuzhiyun }
151*4882a593Smuzhiyun EXPORT_SYMBOL_GPL(intel_pmc_s0ix_counter_read);
152*4882a593Smuzhiyun
153*4882a593Smuzhiyun /**
154*4882a593Smuzhiyun * simplecmd_store() - Send a simple IPC command
155*4882a593Smuzhiyun * @dev: Device under the attribute is
156*4882a593Smuzhiyun * @attr: Attribute in question
157*4882a593Smuzhiyun * @buf: Buffer holding data to be stored to the attribute
158*4882a593Smuzhiyun * @count: Number of bytes in @buf
159*4882a593Smuzhiyun *
160*4882a593Smuzhiyun * Expects a string with two integers separated with space. These two
161*4882a593Smuzhiyun * values hold command and subcommand that is send to PMC.
162*4882a593Smuzhiyun *
163*4882a593Smuzhiyun * Return: Number number of bytes written (@count) or negative errno in
164*4882a593Smuzhiyun * case of error.
165*4882a593Smuzhiyun */
simplecmd_store(struct device * dev,struct device_attribute * attr,const char * buf,size_t count)166*4882a593Smuzhiyun static ssize_t simplecmd_store(struct device *dev, struct device_attribute *attr,
167*4882a593Smuzhiyun const char *buf, size_t count)
168*4882a593Smuzhiyun {
169*4882a593Smuzhiyun struct intel_pmc_dev *pmc = dev_get_drvdata(dev);
170*4882a593Smuzhiyun struct intel_scu_ipc_dev *scu = pmc->scu;
171*4882a593Smuzhiyun int subcmd;
172*4882a593Smuzhiyun int cmd;
173*4882a593Smuzhiyun int ret;
174*4882a593Smuzhiyun
175*4882a593Smuzhiyun ret = sscanf(buf, "%d %d", &cmd, &subcmd);
176*4882a593Smuzhiyun if (ret != 2) {
177*4882a593Smuzhiyun dev_err(dev, "Invalid values, expected: cmd subcmd\n");
178*4882a593Smuzhiyun return -EINVAL;
179*4882a593Smuzhiyun }
180*4882a593Smuzhiyun
181*4882a593Smuzhiyun ret = intel_scu_ipc_dev_simple_command(scu, cmd, subcmd);
182*4882a593Smuzhiyun if (ret)
183*4882a593Smuzhiyun return ret;
184*4882a593Smuzhiyun
185*4882a593Smuzhiyun return count;
186*4882a593Smuzhiyun }
187*4882a593Smuzhiyun static DEVICE_ATTR_WO(simplecmd);
188*4882a593Smuzhiyun
189*4882a593Smuzhiyun /**
190*4882a593Smuzhiyun * northpeak_store() - Enable or disable Northpeak
191*4882a593Smuzhiyun * @dev: Device under the attribute is
192*4882a593Smuzhiyun * @attr: Attribute in question
193*4882a593Smuzhiyun * @buf: Buffer holding data to be stored to the attribute
194*4882a593Smuzhiyun * @count: Number of bytes in @buf
195*4882a593Smuzhiyun *
196*4882a593Smuzhiyun * Expects an unsigned integer. Non-zero enables Northpeak and zero
197*4882a593Smuzhiyun * disables it.
198*4882a593Smuzhiyun *
199*4882a593Smuzhiyun * Return: Number number of bytes written (@count) or negative errno in
200*4882a593Smuzhiyun * case of error.
201*4882a593Smuzhiyun */
northpeak_store(struct device * dev,struct device_attribute * attr,const char * buf,size_t count)202*4882a593Smuzhiyun static ssize_t northpeak_store(struct device *dev, struct device_attribute *attr,
203*4882a593Smuzhiyun const char *buf, size_t count)
204*4882a593Smuzhiyun {
205*4882a593Smuzhiyun struct intel_pmc_dev *pmc = dev_get_drvdata(dev);
206*4882a593Smuzhiyun struct intel_scu_ipc_dev *scu = pmc->scu;
207*4882a593Smuzhiyun unsigned long val;
208*4882a593Smuzhiyun int subcmd;
209*4882a593Smuzhiyun int ret;
210*4882a593Smuzhiyun
211*4882a593Smuzhiyun ret = kstrtoul(buf, 0, &val);
212*4882a593Smuzhiyun if (ret)
213*4882a593Smuzhiyun return ret;
214*4882a593Smuzhiyun
215*4882a593Smuzhiyun /* Northpeak is enabled if subcmd == 1 and disabled if it is 0 */
216*4882a593Smuzhiyun if (val)
217*4882a593Smuzhiyun subcmd = 1;
218*4882a593Smuzhiyun else
219*4882a593Smuzhiyun subcmd = 0;
220*4882a593Smuzhiyun
221*4882a593Smuzhiyun ret = intel_scu_ipc_dev_simple_command(scu, PMC_NORTHPEAK_CTRL, subcmd);
222*4882a593Smuzhiyun if (ret)
223*4882a593Smuzhiyun return ret;
224*4882a593Smuzhiyun
225*4882a593Smuzhiyun return count;
226*4882a593Smuzhiyun }
227*4882a593Smuzhiyun static DEVICE_ATTR_WO(northpeak);
228*4882a593Smuzhiyun
229*4882a593Smuzhiyun static struct attribute *intel_pmc_attrs[] = {
230*4882a593Smuzhiyun &dev_attr_northpeak.attr,
231*4882a593Smuzhiyun &dev_attr_simplecmd.attr,
232*4882a593Smuzhiyun NULL
233*4882a593Smuzhiyun };
234*4882a593Smuzhiyun
235*4882a593Smuzhiyun static const struct attribute_group intel_pmc_group = {
236*4882a593Smuzhiyun .attrs = intel_pmc_attrs,
237*4882a593Smuzhiyun };
238*4882a593Smuzhiyun
239*4882a593Smuzhiyun static const struct attribute_group *intel_pmc_groups[] = {
240*4882a593Smuzhiyun &intel_pmc_group,
241*4882a593Smuzhiyun NULL
242*4882a593Smuzhiyun };
243*4882a593Smuzhiyun
244*4882a593Smuzhiyun static struct resource punit_res[6];
245*4882a593Smuzhiyun
246*4882a593Smuzhiyun static struct mfd_cell punit = {
247*4882a593Smuzhiyun .name = "intel_punit_ipc",
248*4882a593Smuzhiyun .resources = punit_res,
249*4882a593Smuzhiyun };
250*4882a593Smuzhiyun
251*4882a593Smuzhiyun static struct itco_wdt_platform_data tco_pdata = {
252*4882a593Smuzhiyun .name = "Apollo Lake SoC",
253*4882a593Smuzhiyun .version = 5,
254*4882a593Smuzhiyun .no_reboot_use_pmc = true,
255*4882a593Smuzhiyun };
256*4882a593Smuzhiyun
257*4882a593Smuzhiyun static struct resource tco_res[2];
258*4882a593Smuzhiyun
259*4882a593Smuzhiyun static const struct mfd_cell tco = {
260*4882a593Smuzhiyun .name = "iTCO_wdt",
261*4882a593Smuzhiyun .ignore_resource_conflicts = true,
262*4882a593Smuzhiyun .resources = tco_res,
263*4882a593Smuzhiyun .num_resources = ARRAY_SIZE(tco_res),
264*4882a593Smuzhiyun .platform_data = &tco_pdata,
265*4882a593Smuzhiyun .pdata_size = sizeof(tco_pdata),
266*4882a593Smuzhiyun };
267*4882a593Smuzhiyun
268*4882a593Smuzhiyun static const struct resource telem_res[] = {
269*4882a593Smuzhiyun DEFINE_RES_MEM(TELEM_PUNIT_SSRAM_OFFSET, TELEM_SSRAM_SIZE),
270*4882a593Smuzhiyun DEFINE_RES_MEM(TELEM_PMC_SSRAM_OFFSET, TELEM_SSRAM_SIZE),
271*4882a593Smuzhiyun };
272*4882a593Smuzhiyun
273*4882a593Smuzhiyun static const struct mfd_cell telem = {
274*4882a593Smuzhiyun .name = "intel_telemetry",
275*4882a593Smuzhiyun .resources = telem_res,
276*4882a593Smuzhiyun .num_resources = ARRAY_SIZE(telem_res),
277*4882a593Smuzhiyun };
278*4882a593Smuzhiyun
intel_pmc_get_tco_resources(struct platform_device * pdev)279*4882a593Smuzhiyun static int intel_pmc_get_tco_resources(struct platform_device *pdev)
280*4882a593Smuzhiyun {
281*4882a593Smuzhiyun struct resource *res;
282*4882a593Smuzhiyun
283*4882a593Smuzhiyun if (acpi_has_watchdog())
284*4882a593Smuzhiyun return 0;
285*4882a593Smuzhiyun
286*4882a593Smuzhiyun res = platform_get_resource(pdev, IORESOURCE_IO,
287*4882a593Smuzhiyun PLAT_RESOURCE_ACPI_IO_INDEX);
288*4882a593Smuzhiyun if (!res) {
289*4882a593Smuzhiyun dev_err(&pdev->dev, "Failed to get IO resource\n");
290*4882a593Smuzhiyun return -EINVAL;
291*4882a593Smuzhiyun }
292*4882a593Smuzhiyun
293*4882a593Smuzhiyun tco_res[0].flags = IORESOURCE_IO;
294*4882a593Smuzhiyun tco_res[0].start = res->start + TCO_BASE_OFFSET;
295*4882a593Smuzhiyun tco_res[0].end = tco_res[0].start + TCO_REGS_SIZE - 1;
296*4882a593Smuzhiyun tco_res[1].flags = IORESOURCE_IO;
297*4882a593Smuzhiyun tco_res[1].start = res->start + SMI_EN_OFFSET;
298*4882a593Smuzhiyun tco_res[1].end = tco_res[1].start + SMI_EN_SIZE - 1;
299*4882a593Smuzhiyun
300*4882a593Smuzhiyun return 0;
301*4882a593Smuzhiyun }
302*4882a593Smuzhiyun
intel_pmc_get_resources(struct platform_device * pdev,struct intel_pmc_dev * pmc,struct intel_scu_ipc_data * scu_data)303*4882a593Smuzhiyun static int intel_pmc_get_resources(struct platform_device *pdev,
304*4882a593Smuzhiyun struct intel_pmc_dev *pmc,
305*4882a593Smuzhiyun struct intel_scu_ipc_data *scu_data)
306*4882a593Smuzhiyun {
307*4882a593Smuzhiyun struct resource gcr_res;
308*4882a593Smuzhiyun size_t npunit_res = 0;
309*4882a593Smuzhiyun struct resource *res;
310*4882a593Smuzhiyun int ret;
311*4882a593Smuzhiyun
312*4882a593Smuzhiyun scu_data->irq = platform_get_irq_optional(pdev, 0);
313*4882a593Smuzhiyun
314*4882a593Smuzhiyun res = platform_get_resource(pdev, IORESOURCE_MEM,
315*4882a593Smuzhiyun PLAT_RESOURCE_IPC_INDEX);
316*4882a593Smuzhiyun if (!res) {
317*4882a593Smuzhiyun dev_err(&pdev->dev, "Failed to get IPC resource\n");
318*4882a593Smuzhiyun return -EINVAL;
319*4882a593Smuzhiyun }
320*4882a593Smuzhiyun
321*4882a593Smuzhiyun /* IPC registers */
322*4882a593Smuzhiyun scu_data->mem.flags = res->flags;
323*4882a593Smuzhiyun scu_data->mem.start = res->start;
324*4882a593Smuzhiyun scu_data->mem.end = res->start + PLAT_RESOURCE_IPC_SIZE - 1;
325*4882a593Smuzhiyun
326*4882a593Smuzhiyun /* GCR registers */
327*4882a593Smuzhiyun gcr_res.flags = res->flags;
328*4882a593Smuzhiyun gcr_res.start = res->start + PLAT_RESOURCE_GCR_OFFSET;
329*4882a593Smuzhiyun gcr_res.end = gcr_res.start + PLAT_RESOURCE_GCR_SIZE - 1;
330*4882a593Smuzhiyun
331*4882a593Smuzhiyun pmc->gcr_mem_base = devm_ioremap_resource(&pdev->dev, &gcr_res);
332*4882a593Smuzhiyun if (IS_ERR(pmc->gcr_mem_base))
333*4882a593Smuzhiyun return PTR_ERR(pmc->gcr_mem_base);
334*4882a593Smuzhiyun
335*4882a593Smuzhiyun /* Only register iTCO watchdog if there is no WDAT ACPI table */
336*4882a593Smuzhiyun ret = intel_pmc_get_tco_resources(pdev);
337*4882a593Smuzhiyun if (ret)
338*4882a593Smuzhiyun return ret;
339*4882a593Smuzhiyun
340*4882a593Smuzhiyun /* BIOS data register */
341*4882a593Smuzhiyun res = platform_get_resource(pdev, IORESOURCE_MEM,
342*4882a593Smuzhiyun PLAT_RESOURCE_BIOS_DATA_INDEX);
343*4882a593Smuzhiyun if (!res) {
344*4882a593Smuzhiyun dev_err(&pdev->dev, "Failed to get resource of P-unit BIOS data\n");
345*4882a593Smuzhiyun return -EINVAL;
346*4882a593Smuzhiyun }
347*4882a593Smuzhiyun punit_res[npunit_res++] = *res;
348*4882a593Smuzhiyun
349*4882a593Smuzhiyun /* BIOS interface register */
350*4882a593Smuzhiyun res = platform_get_resource(pdev, IORESOURCE_MEM,
351*4882a593Smuzhiyun PLAT_RESOURCE_BIOS_IFACE_INDEX);
352*4882a593Smuzhiyun if (!res) {
353*4882a593Smuzhiyun dev_err(&pdev->dev, "Failed to get resource of P-unit BIOS interface\n");
354*4882a593Smuzhiyun return -EINVAL;
355*4882a593Smuzhiyun }
356*4882a593Smuzhiyun punit_res[npunit_res++] = *res;
357*4882a593Smuzhiyun
358*4882a593Smuzhiyun /* ISP data register, optional */
359*4882a593Smuzhiyun res = platform_get_resource(pdev, IORESOURCE_MEM,
360*4882a593Smuzhiyun PLAT_RESOURCE_ISP_DATA_INDEX);
361*4882a593Smuzhiyun if (res)
362*4882a593Smuzhiyun punit_res[npunit_res++] = *res;
363*4882a593Smuzhiyun
364*4882a593Smuzhiyun /* ISP interface register, optional */
365*4882a593Smuzhiyun res = platform_get_resource(pdev, IORESOURCE_MEM,
366*4882a593Smuzhiyun PLAT_RESOURCE_ISP_IFACE_INDEX);
367*4882a593Smuzhiyun if (res)
368*4882a593Smuzhiyun punit_res[npunit_res++] = *res;
369*4882a593Smuzhiyun
370*4882a593Smuzhiyun /* GTD data register, optional */
371*4882a593Smuzhiyun res = platform_get_resource(pdev, IORESOURCE_MEM,
372*4882a593Smuzhiyun PLAT_RESOURCE_GTD_DATA_INDEX);
373*4882a593Smuzhiyun if (res)
374*4882a593Smuzhiyun punit_res[npunit_res++] = *res;
375*4882a593Smuzhiyun
376*4882a593Smuzhiyun /* GTD interface register, optional */
377*4882a593Smuzhiyun res = platform_get_resource(pdev, IORESOURCE_MEM,
378*4882a593Smuzhiyun PLAT_RESOURCE_GTD_IFACE_INDEX);
379*4882a593Smuzhiyun if (res)
380*4882a593Smuzhiyun punit_res[npunit_res++] = *res;
381*4882a593Smuzhiyun
382*4882a593Smuzhiyun punit.num_resources = npunit_res;
383*4882a593Smuzhiyun
384*4882a593Smuzhiyun /* Telemetry SSRAM is optional */
385*4882a593Smuzhiyun res = platform_get_resource(pdev, IORESOURCE_MEM,
386*4882a593Smuzhiyun PLAT_RESOURCE_TELEM_SSRAM_INDEX);
387*4882a593Smuzhiyun if (res)
388*4882a593Smuzhiyun pmc->telem_base = res;
389*4882a593Smuzhiyun
390*4882a593Smuzhiyun return 0;
391*4882a593Smuzhiyun }
392*4882a593Smuzhiyun
intel_pmc_create_devices(struct intel_pmc_dev * pmc)393*4882a593Smuzhiyun static int intel_pmc_create_devices(struct intel_pmc_dev *pmc)
394*4882a593Smuzhiyun {
395*4882a593Smuzhiyun int ret;
396*4882a593Smuzhiyun
397*4882a593Smuzhiyun if (!acpi_has_watchdog()) {
398*4882a593Smuzhiyun ret = devm_mfd_add_devices(pmc->dev, PLATFORM_DEVID_AUTO, &tco,
399*4882a593Smuzhiyun 1, NULL, 0, NULL);
400*4882a593Smuzhiyun if (ret)
401*4882a593Smuzhiyun return ret;
402*4882a593Smuzhiyun }
403*4882a593Smuzhiyun
404*4882a593Smuzhiyun ret = devm_mfd_add_devices(pmc->dev, PLATFORM_DEVID_AUTO, &punit, 1,
405*4882a593Smuzhiyun NULL, 0, NULL);
406*4882a593Smuzhiyun if (ret)
407*4882a593Smuzhiyun return ret;
408*4882a593Smuzhiyun
409*4882a593Smuzhiyun if (pmc->telem_base) {
410*4882a593Smuzhiyun ret = devm_mfd_add_devices(pmc->dev, PLATFORM_DEVID_AUTO,
411*4882a593Smuzhiyun &telem, 1, pmc->telem_base, 0, NULL);
412*4882a593Smuzhiyun }
413*4882a593Smuzhiyun
414*4882a593Smuzhiyun return ret;
415*4882a593Smuzhiyun }
416*4882a593Smuzhiyun
417*4882a593Smuzhiyun static const struct acpi_device_id intel_pmc_acpi_ids[] = {
418*4882a593Smuzhiyun { "INT34D2" },
419*4882a593Smuzhiyun { }
420*4882a593Smuzhiyun };
421*4882a593Smuzhiyun MODULE_DEVICE_TABLE(acpi, intel_pmc_acpi_ids);
422*4882a593Smuzhiyun
intel_pmc_probe(struct platform_device * pdev)423*4882a593Smuzhiyun static int intel_pmc_probe(struct platform_device *pdev)
424*4882a593Smuzhiyun {
425*4882a593Smuzhiyun struct intel_scu_ipc_data scu_data = {};
426*4882a593Smuzhiyun struct intel_pmc_dev *pmc;
427*4882a593Smuzhiyun int ret;
428*4882a593Smuzhiyun
429*4882a593Smuzhiyun pmc = devm_kzalloc(&pdev->dev, sizeof(*pmc), GFP_KERNEL);
430*4882a593Smuzhiyun if (!pmc)
431*4882a593Smuzhiyun return -ENOMEM;
432*4882a593Smuzhiyun
433*4882a593Smuzhiyun pmc->dev = &pdev->dev;
434*4882a593Smuzhiyun spin_lock_init(&pmc->gcr_lock);
435*4882a593Smuzhiyun
436*4882a593Smuzhiyun ret = intel_pmc_get_resources(pdev, pmc, &scu_data);
437*4882a593Smuzhiyun if (ret) {
438*4882a593Smuzhiyun dev_err(&pdev->dev, "Failed to request resources\n");
439*4882a593Smuzhiyun return ret;
440*4882a593Smuzhiyun }
441*4882a593Smuzhiyun
442*4882a593Smuzhiyun pmc->scu = devm_intel_scu_ipc_register(&pdev->dev, &scu_data);
443*4882a593Smuzhiyun if (IS_ERR(pmc->scu))
444*4882a593Smuzhiyun return PTR_ERR(pmc->scu);
445*4882a593Smuzhiyun
446*4882a593Smuzhiyun platform_set_drvdata(pdev, pmc);
447*4882a593Smuzhiyun
448*4882a593Smuzhiyun ret = intel_pmc_create_devices(pmc);
449*4882a593Smuzhiyun if (ret)
450*4882a593Smuzhiyun dev_err(&pdev->dev, "Failed to create PMC devices\n");
451*4882a593Smuzhiyun
452*4882a593Smuzhiyun return ret;
453*4882a593Smuzhiyun }
454*4882a593Smuzhiyun
455*4882a593Smuzhiyun static struct platform_driver intel_pmc_driver = {
456*4882a593Smuzhiyun .probe = intel_pmc_probe,
457*4882a593Smuzhiyun .driver = {
458*4882a593Smuzhiyun .name = "intel_pmc_bxt",
459*4882a593Smuzhiyun .acpi_match_table = intel_pmc_acpi_ids,
460*4882a593Smuzhiyun .dev_groups = intel_pmc_groups,
461*4882a593Smuzhiyun },
462*4882a593Smuzhiyun };
463*4882a593Smuzhiyun module_platform_driver(intel_pmc_driver);
464*4882a593Smuzhiyun
465*4882a593Smuzhiyun MODULE_AUTHOR("Mika Westerberg <mika.westerberg@linux.intel.com>");
466*4882a593Smuzhiyun MODULE_AUTHOR("Zha Qipeng <qipeng.zha@intel.com>");
467*4882a593Smuzhiyun MODULE_DESCRIPTION("Intel Broxton PMC driver");
468*4882a593Smuzhiyun MODULE_LICENSE("GPL v2");
469