3 * acpi_lpit.c - LPIT table processing functions
5 * Copyright (C) 2017 Intel Corporation. All rights reserved.
7 * This program is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU General Public License version
9 * 2 as published by the Free Software Foundation.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
17 #include <linux/cpu.h>
18 #include <linux/acpi.h>
22 struct lpit_residency_info
{
23 struct acpi_generic_address gaddr
;
25 void __iomem
*iomem_addr
;
28 /* Storage for an memory mapped and FFH based entries */
29 static struct lpit_residency_info residency_info_mem
;
30 static struct lpit_residency_info residency_info_ffh
;
32 static int lpit_read_residency_counter_us(u64
*counter
, bool io_mem
)
40 error
= acpi_os_read_iomem(residency_info_mem
.iomem_addr
, &count
,
41 residency_info_mem
.gaddr
.bit_width
);
45 *counter
= div64_u64(count
* 1000000ULL, residency_info_mem
.frequency
);
49 err
= rdmsrl_safe(residency_info_ffh
.gaddr
.address
, counter
);
51 u64 mask
= GENMASK_ULL(residency_info_ffh
.gaddr
.bit_offset
+
52 residency_info_ffh
.gaddr
. bit_width
- 1,
53 residency_info_ffh
.gaddr
.bit_offset
);
56 *counter
>>= residency_info_ffh
.gaddr
.bit_offset
;
57 *counter
= div64_u64(*counter
* 1000000ULL, residency_info_ffh
.frequency
);
64 static ssize_t
low_power_idle_system_residency_us_show(struct device
*dev
,
65 struct device_attribute
*attr
,
71 ret
= lpit_read_residency_counter_us(&counter
, true);
75 return sprintf(buf
, "%llu\n", counter
);
77 static DEVICE_ATTR_RO(low_power_idle_system_residency_us
);
79 static ssize_t
low_power_idle_cpu_residency_us_show(struct device
*dev
,
80 struct device_attribute
*attr
,
86 ret
= lpit_read_residency_counter_us(&counter
, false);
90 return sprintf(buf
, "%llu\n", counter
);
92 static DEVICE_ATTR_RO(low_power_idle_cpu_residency_us
);
94 int lpit_read_residency_count_address(u64
*address
)
96 if (!residency_info_mem
.gaddr
.address
)
99 *address
= residency_info_mem
.gaddr
.address
;
104 static void lpit_update_residency(struct lpit_residency_info
*info
,
105 struct acpi_lpit_native
*lpit_native
)
107 info
->frequency
= lpit_native
->counter_frequency
?
108 lpit_native
->counter_frequency
: tsc_khz
* 1000;
109 if (!info
->frequency
)
112 info
->gaddr
= lpit_native
->residency_counter
;
113 if (info
->gaddr
.space_id
== ACPI_ADR_SPACE_SYSTEM_MEMORY
) {
114 info
->iomem_addr
= ioremap_nocache(info
->gaddr
.address
,
115 info
->gaddr
.bit_width
/ 8);
116 if (!info
->iomem_addr
)
119 /* Silently fail, if cpuidle attribute group is not present */
120 sysfs_add_file_to_group(&cpu_subsys
.dev_root
->kobj
,
121 &dev_attr_low_power_idle_system_residency_us
.attr
,
123 } else if (info
->gaddr
.space_id
== ACPI_ADR_SPACE_FIXED_HARDWARE
) {
124 /* Silently fail, if cpuidle attribute group is not present */
125 sysfs_add_file_to_group(&cpu_subsys
.dev_root
->kobj
,
126 &dev_attr_low_power_idle_cpu_residency_us
.attr
,
131 static void lpit_process(u64 begin
, u64 end
)
133 while (begin
+ sizeof(struct acpi_lpit_native
) < end
) {
134 struct acpi_lpit_native
*lpit_native
= (struct acpi_lpit_native
*)begin
;
136 if (!lpit_native
->header
.type
&& !lpit_native
->header
.flags
) {
137 if (lpit_native
->residency_counter
.space_id
== ACPI_ADR_SPACE_SYSTEM_MEMORY
&&
138 !residency_info_mem
.gaddr
.address
) {
139 lpit_update_residency(&residency_info_mem
, lpit_native
);
140 } else if (lpit_native
->residency_counter
.space_id
== ACPI_ADR_SPACE_FIXED_HARDWARE
&&
141 !residency_info_ffh
.gaddr
.address
) {
142 lpit_update_residency(&residency_info_ffh
, lpit_native
);
145 begin
+= lpit_native
->header
.length
;
149 void acpi_init_lpit(void)
153 struct acpi_table_lpit
*lpit
;
155 status
= acpi_get_table(ACPI_SIG_LPIT
, 0, (struct acpi_table_header
**)&lpit
);
157 if (ACPI_FAILURE(status
))
160 lpit_begin
= (u64
)lpit
+ sizeof(*lpit
);
161 lpit_process(lpit_begin
, lpit_begin
+ lpit
->header
.length
);