xref: /openbmc/linux/drivers/acpi/acpi_lpit.c (revision d1ac288b)
11802d0beSThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only
2eeb2d80dSSrinivas Pandruvada 
3eeb2d80dSSrinivas Pandruvada /*
4eeb2d80dSSrinivas Pandruvada  * acpi_lpit.c - LPIT table processing functions
5eeb2d80dSSrinivas Pandruvada  *
6eeb2d80dSSrinivas Pandruvada  * Copyright (C) 2017 Intel Corporation. All rights reserved.
7eeb2d80dSSrinivas Pandruvada  */
8eeb2d80dSSrinivas Pandruvada 
9eeb2d80dSSrinivas Pandruvada #include <linux/cpu.h>
10eeb2d80dSSrinivas Pandruvada #include <linux/acpi.h>
11eeb2d80dSSrinivas Pandruvada #include <asm/msr.h>
12eeb2d80dSSrinivas Pandruvada #include <asm/tsc.h>
13446c85afSAmmar Faizi #include "internal.h"
14eeb2d80dSSrinivas Pandruvada 
15eeb2d80dSSrinivas Pandruvada struct lpit_residency_info {
16eeb2d80dSSrinivas Pandruvada 	struct acpi_generic_address gaddr;
17eeb2d80dSSrinivas Pandruvada 	u64 frequency;
18eeb2d80dSSrinivas Pandruvada 	void __iomem *iomem_addr;
19eeb2d80dSSrinivas Pandruvada };
20eeb2d80dSSrinivas Pandruvada 
21eeb2d80dSSrinivas Pandruvada /* Storage for an memory mapped and FFH based entries */
22eeb2d80dSSrinivas Pandruvada static struct lpit_residency_info residency_info_mem;
23eeb2d80dSSrinivas Pandruvada static struct lpit_residency_info residency_info_ffh;
24eeb2d80dSSrinivas Pandruvada 
lpit_read_residency_counter_us(u64 * counter,bool io_mem)25eeb2d80dSSrinivas Pandruvada static int lpit_read_residency_counter_us(u64 *counter, bool io_mem)
26eeb2d80dSSrinivas Pandruvada {
27eeb2d80dSSrinivas Pandruvada 	int err;
28eeb2d80dSSrinivas Pandruvada 
29eeb2d80dSSrinivas Pandruvada 	if (io_mem) {
30eeb2d80dSSrinivas Pandruvada 		u64 count = 0;
31eeb2d80dSSrinivas Pandruvada 		int error;
32eeb2d80dSSrinivas Pandruvada 
33eeb2d80dSSrinivas Pandruvada 		error = acpi_os_read_iomem(residency_info_mem.iomem_addr, &count,
34eeb2d80dSSrinivas Pandruvada 					   residency_info_mem.gaddr.bit_width);
35eeb2d80dSSrinivas Pandruvada 		if (error)
36eeb2d80dSSrinivas Pandruvada 			return error;
37eeb2d80dSSrinivas Pandruvada 
38eeb2d80dSSrinivas Pandruvada 		*counter = div64_u64(count * 1000000ULL, residency_info_mem.frequency);
39eeb2d80dSSrinivas Pandruvada 		return 0;
40eeb2d80dSSrinivas Pandruvada 	}
41eeb2d80dSSrinivas Pandruvada 
42eeb2d80dSSrinivas Pandruvada 	err = rdmsrl_safe(residency_info_ffh.gaddr.address, counter);
43eeb2d80dSSrinivas Pandruvada 	if (!err) {
44eeb2d80dSSrinivas Pandruvada 		u64 mask = GENMASK_ULL(residency_info_ffh.gaddr.bit_offset +
45eeb2d80dSSrinivas Pandruvada 				       residency_info_ffh.gaddr. bit_width - 1,
46eeb2d80dSSrinivas Pandruvada 				       residency_info_ffh.gaddr.bit_offset);
47eeb2d80dSSrinivas Pandruvada 
48eeb2d80dSSrinivas Pandruvada 		*counter &= mask;
49eeb2d80dSSrinivas Pandruvada 		*counter >>= residency_info_ffh.gaddr.bit_offset;
50eeb2d80dSSrinivas Pandruvada 		*counter = div64_u64(*counter * 1000000ULL, residency_info_ffh.frequency);
51eeb2d80dSSrinivas Pandruvada 		return 0;
52eeb2d80dSSrinivas Pandruvada 	}
53eeb2d80dSSrinivas Pandruvada 
54eeb2d80dSSrinivas Pandruvada 	return -ENODATA;
55eeb2d80dSSrinivas Pandruvada }
56eeb2d80dSSrinivas Pandruvada 
low_power_idle_system_residency_us_show(struct device * dev,struct device_attribute * attr,char * buf)57eeb2d80dSSrinivas Pandruvada static ssize_t low_power_idle_system_residency_us_show(struct device *dev,
58eeb2d80dSSrinivas Pandruvada 						       struct device_attribute *attr,
59eeb2d80dSSrinivas Pandruvada 						       char *buf)
60eeb2d80dSSrinivas Pandruvada {
61eeb2d80dSSrinivas Pandruvada 	u64 counter;
62eeb2d80dSSrinivas Pandruvada 	int ret;
63eeb2d80dSSrinivas Pandruvada 
64eeb2d80dSSrinivas Pandruvada 	ret = lpit_read_residency_counter_us(&counter, true);
65eeb2d80dSSrinivas Pandruvada 	if (ret)
66eeb2d80dSSrinivas Pandruvada 		return ret;
67eeb2d80dSSrinivas Pandruvada 
68eeb2d80dSSrinivas Pandruvada 	return sprintf(buf, "%llu\n", counter);
69eeb2d80dSSrinivas Pandruvada }
70eeb2d80dSSrinivas Pandruvada static DEVICE_ATTR_RO(low_power_idle_system_residency_us);
71eeb2d80dSSrinivas Pandruvada 
low_power_idle_cpu_residency_us_show(struct device * dev,struct device_attribute * attr,char * buf)72eeb2d80dSSrinivas Pandruvada static ssize_t low_power_idle_cpu_residency_us_show(struct device *dev,
73eeb2d80dSSrinivas Pandruvada 						    struct device_attribute *attr,
74eeb2d80dSSrinivas Pandruvada 						    char *buf)
75eeb2d80dSSrinivas Pandruvada {
76eeb2d80dSSrinivas Pandruvada 	u64 counter;
77eeb2d80dSSrinivas Pandruvada 	int ret;
78eeb2d80dSSrinivas Pandruvada 
79eeb2d80dSSrinivas Pandruvada 	ret = lpit_read_residency_counter_us(&counter, false);
80eeb2d80dSSrinivas Pandruvada 	if (ret)
81eeb2d80dSSrinivas Pandruvada 		return ret;
82eeb2d80dSSrinivas Pandruvada 
83eeb2d80dSSrinivas Pandruvada 	return sprintf(buf, "%llu\n", counter);
84eeb2d80dSSrinivas Pandruvada }
85eeb2d80dSSrinivas Pandruvada static DEVICE_ATTR_RO(low_power_idle_cpu_residency_us);
86eeb2d80dSSrinivas Pandruvada 
lpit_read_residency_count_address(u64 * address)87eeb2d80dSSrinivas Pandruvada int lpit_read_residency_count_address(u64 *address)
88eeb2d80dSSrinivas Pandruvada {
89eeb2d80dSSrinivas Pandruvada 	if (!residency_info_mem.gaddr.address)
90eeb2d80dSSrinivas Pandruvada 		return -EINVAL;
91eeb2d80dSSrinivas Pandruvada 
92eeb2d80dSSrinivas Pandruvada 	*address = residency_info_mem.gaddr.address;
93eeb2d80dSSrinivas Pandruvada 
94eeb2d80dSSrinivas Pandruvada 	return 0;
95eeb2d80dSSrinivas Pandruvada }
969383bbadSSrinivas Pandruvada EXPORT_SYMBOL_GPL(lpit_read_residency_count_address);
97eeb2d80dSSrinivas Pandruvada 
lpit_update_residency(struct lpit_residency_info * info,struct acpi_lpit_native * lpit_native)98eeb2d80dSSrinivas Pandruvada static void lpit_update_residency(struct lpit_residency_info *info,
99eeb2d80dSSrinivas Pandruvada 				 struct acpi_lpit_native *lpit_native)
100eeb2d80dSSrinivas Pandruvada {
101af48ab8bSGreg Kroah-Hartman 	struct device *dev_root = bus_get_dev_root(&cpu_subsys);
102af48ab8bSGreg Kroah-Hartman 
103af48ab8bSGreg Kroah-Hartman 	/* Silently fail, if cpuidle attribute group is not present */
104af48ab8bSGreg Kroah-Hartman 	if (!dev_root)
105af48ab8bSGreg Kroah-Hartman 		return;
106af48ab8bSGreg Kroah-Hartman 
107eeb2d80dSSrinivas Pandruvada 	info->frequency = lpit_native->counter_frequency ?
108*d1ac288bSNikita Kiryushin 				lpit_native->counter_frequency : mul_u32_u32(tsc_khz, 1000U);
109eeb2d80dSSrinivas Pandruvada 	if (!info->frequency)
110eeb2d80dSSrinivas Pandruvada 		info->frequency = 1;
111eeb2d80dSSrinivas Pandruvada 
112eeb2d80dSSrinivas Pandruvada 	info->gaddr = lpit_native->residency_counter;
113eeb2d80dSSrinivas Pandruvada 	if (info->gaddr.space_id == ACPI_ADR_SPACE_SYSTEM_MEMORY) {
1144bdc0d67SChristoph Hellwig 		info->iomem_addr = ioremap(info->gaddr.address,
115eeb2d80dSSrinivas Pandruvada 						   info->gaddr.bit_width / 8);
116eeb2d80dSSrinivas Pandruvada 		if (!info->iomem_addr)
117af48ab8bSGreg Kroah-Hartman 			goto exit;
118eeb2d80dSSrinivas Pandruvada 
119af48ab8bSGreg Kroah-Hartman 		sysfs_add_file_to_group(&dev_root->kobj,
120eeb2d80dSSrinivas Pandruvada 					&dev_attr_low_power_idle_system_residency_us.attr,
121eeb2d80dSSrinivas Pandruvada 					"cpuidle");
122eeb2d80dSSrinivas Pandruvada 	} else if (info->gaddr.space_id == ACPI_ADR_SPACE_FIXED_HARDWARE) {
123af48ab8bSGreg Kroah-Hartman 		sysfs_add_file_to_group(&dev_root->kobj,
124eeb2d80dSSrinivas Pandruvada 					&dev_attr_low_power_idle_cpu_residency_us.attr,
125eeb2d80dSSrinivas Pandruvada 					"cpuidle");
126eeb2d80dSSrinivas Pandruvada 	}
127af48ab8bSGreg Kroah-Hartman exit:
128af48ab8bSGreg Kroah-Hartman 	put_device(dev_root);
129eeb2d80dSSrinivas Pandruvada }
130eeb2d80dSSrinivas Pandruvada 
lpit_process(u64 begin,u64 end)131eeb2d80dSSrinivas Pandruvada static void lpit_process(u64 begin, u64 end)
132eeb2d80dSSrinivas Pandruvada {
13332865e3eSLenny Szubowicz 	while (begin + sizeof(struct acpi_lpit_native) <= end) {
134eeb2d80dSSrinivas Pandruvada 		struct acpi_lpit_native *lpit_native = (struct acpi_lpit_native *)begin;
135eeb2d80dSSrinivas Pandruvada 
136eeb2d80dSSrinivas Pandruvada 		if (!lpit_native->header.type && !lpit_native->header.flags) {
137eeb2d80dSSrinivas Pandruvada 			if (lpit_native->residency_counter.space_id == ACPI_ADR_SPACE_SYSTEM_MEMORY &&
138eeb2d80dSSrinivas Pandruvada 			    !residency_info_mem.gaddr.address) {
139eeb2d80dSSrinivas Pandruvada 				lpit_update_residency(&residency_info_mem, lpit_native);
140eeb2d80dSSrinivas Pandruvada 			} else if (lpit_native->residency_counter.space_id == ACPI_ADR_SPACE_FIXED_HARDWARE &&
141eeb2d80dSSrinivas Pandruvada 				   !residency_info_ffh.gaddr.address) {
142eeb2d80dSSrinivas Pandruvada 				lpit_update_residency(&residency_info_ffh, lpit_native);
143eeb2d80dSSrinivas Pandruvada 			}
144eeb2d80dSSrinivas Pandruvada 		}
145eeb2d80dSSrinivas Pandruvada 		begin += lpit_native->header.length;
146eeb2d80dSSrinivas Pandruvada 	}
147eeb2d80dSSrinivas Pandruvada }
148eeb2d80dSSrinivas Pandruvada 
acpi_init_lpit(void)149eeb2d80dSSrinivas Pandruvada void acpi_init_lpit(void)
150eeb2d80dSSrinivas Pandruvada {
151eeb2d80dSSrinivas Pandruvada 	acpi_status status;
152eeb2d80dSSrinivas Pandruvada 	struct acpi_table_lpit *lpit;
153eeb2d80dSSrinivas Pandruvada 
154eeb2d80dSSrinivas Pandruvada 	status = acpi_get_table(ACPI_SIG_LPIT, 0, (struct acpi_table_header **)&lpit);
155eeb2d80dSSrinivas Pandruvada 	if (ACPI_FAILURE(status))
156eeb2d80dSSrinivas Pandruvada 		return;
157eeb2d80dSSrinivas Pandruvada 
15832865e3eSLenny Szubowicz 	lpit_process((u64)lpit + sizeof(*lpit),
15932865e3eSLenny Szubowicz 		     (u64)lpit + lpit->header.length);
160f8690227SHanjun Guo 
161f8690227SHanjun Guo 	acpi_put_table((struct acpi_table_header *)lpit);
162eeb2d80dSSrinivas Pandruvada }
163