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