| |
| /* |
| * acpi_lpit.c - LPIT table processing functions |
| * |
| * Copyright (C) 2017 Intel Corporation. All rights reserved. |
| * |
| * This program is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU General Public License version |
| * 2 as published by the Free Software Foundation. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| */ |
| |
| #include <linux/cpu.h> |
| #include <linux/acpi.h> |
| #include <asm/msr.h> |
| #include <asm/tsc.h> |
| |
| struct lpit_residency_info { |
| struct acpi_generic_address gaddr; |
| u64 frequency; |
| void __iomem *iomem_addr; |
| }; |
| |
| /* Storage for an memory mapped and FFH based entries */ |
| static struct lpit_residency_info residency_info_mem; |
| static struct lpit_residency_info residency_info_ffh; |
| |
| static int lpit_read_residency_counter_us(u64 *counter, bool io_mem) |
| { |
| int err; |
| |
| if (io_mem) { |
| u64 count = 0; |
| int error; |
| |
| error = acpi_os_read_iomem(residency_info_mem.iomem_addr, &count, |
| residency_info_mem.gaddr.bit_width); |
| if (error) |
| return error; |
| |
| *counter = div64_u64(count * 1000000ULL, residency_info_mem.frequency); |
| return 0; |
| } |
| |
| err = rdmsrl_safe(residency_info_ffh.gaddr.address, counter); |
| if (!err) { |
| u64 mask = GENMASK_ULL(residency_info_ffh.gaddr.bit_offset + |
| residency_info_ffh.gaddr. bit_width - 1, |
| residency_info_ffh.gaddr.bit_offset); |
| |
| *counter &= mask; |
| *counter >>= residency_info_ffh.gaddr.bit_offset; |
| *counter = div64_u64(*counter * 1000000ULL, residency_info_ffh.frequency); |
| return 0; |
| } |
| |
| return -ENODATA; |
| } |
| |
| static ssize_t low_power_idle_system_residency_us_show(struct device *dev, |
| struct device_attribute *attr, |
| char *buf) |
| { |
| u64 counter; |
| int ret; |
| |
| ret = lpit_read_residency_counter_us(&counter, true); |
| if (ret) |
| return ret; |
| |
| return sprintf(buf, "%llu\n", counter); |
| } |
| static DEVICE_ATTR_RO(low_power_idle_system_residency_us); |
| |
| static ssize_t low_power_idle_cpu_residency_us_show(struct device *dev, |
| struct device_attribute *attr, |
| char *buf) |
| { |
| u64 counter; |
| int ret; |
| |
| ret = lpit_read_residency_counter_us(&counter, false); |
| if (ret) |
| return ret; |
| |
| return sprintf(buf, "%llu\n", counter); |
| } |
| static DEVICE_ATTR_RO(low_power_idle_cpu_residency_us); |
| |
| int lpit_read_residency_count_address(u64 *address) |
| { |
| if (!residency_info_mem.gaddr.address) |
| return -EINVAL; |
| |
| *address = residency_info_mem.gaddr.address; |
| |
| return 0; |
| } |
| |
| static void lpit_update_residency(struct lpit_residency_info *info, |
| struct acpi_lpit_native *lpit_native) |
| { |
| info->frequency = lpit_native->counter_frequency ? |
| lpit_native->counter_frequency : tsc_khz * 1000; |
| if (!info->frequency) |
| info->frequency = 1; |
| |
| info->gaddr = lpit_native->residency_counter; |
| if (info->gaddr.space_id == ACPI_ADR_SPACE_SYSTEM_MEMORY) { |
| info->iomem_addr = ioremap_nocache(info->gaddr.address, |
| info->gaddr.bit_width / 8); |
| if (!info->iomem_addr) |
| return; |
| |
| /* Silently fail, if cpuidle attribute group is not present */ |
| sysfs_add_file_to_group(&cpu_subsys.dev_root->kobj, |
| &dev_attr_low_power_idle_system_residency_us.attr, |
| "cpuidle"); |
| } else if (info->gaddr.space_id == ACPI_ADR_SPACE_FIXED_HARDWARE) { |
| /* Silently fail, if cpuidle attribute group is not present */ |
| sysfs_add_file_to_group(&cpu_subsys.dev_root->kobj, |
| &dev_attr_low_power_idle_cpu_residency_us.attr, |
| "cpuidle"); |
| } |
| } |
| |
| static void lpit_process(u64 begin, u64 end) |
| { |
| while (begin + sizeof(struct acpi_lpit_native) < end) { |
| struct acpi_lpit_native *lpit_native = (struct acpi_lpit_native *)begin; |
| |
| if (!lpit_native->header.type && !lpit_native->header.flags) { |
| if (lpit_native->residency_counter.space_id == ACPI_ADR_SPACE_SYSTEM_MEMORY && |
| !residency_info_mem.gaddr.address) { |
| lpit_update_residency(&residency_info_mem, lpit_native); |
| } else if (lpit_native->residency_counter.space_id == ACPI_ADR_SPACE_FIXED_HARDWARE && |
| !residency_info_ffh.gaddr.address) { |
| lpit_update_residency(&residency_info_ffh, lpit_native); |
| } |
| } |
| begin += lpit_native->header.length; |
| } |
| } |
| |
| void acpi_init_lpit(void) |
| { |
| acpi_status status; |
| u64 lpit_begin; |
| struct acpi_table_lpit *lpit; |
| |
| status = acpi_get_table(ACPI_SIG_LPIT, 0, (struct acpi_table_header **)&lpit); |
| |
| if (ACPI_FAILURE(status)) |
| return; |
| |
| lpit_begin = (u64)lpit + sizeof(*lpit); |
| lpit_process(lpit_begin, lpit_begin + lpit->header.length); |
| } |