Linux Audio

Check our new training course

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