cregit-Linux how code gets into the kernel

Release 4.16 drivers/acpi/acpi_lpit.c

Directory: drivers/acpi
/*
 * 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; }

Contributors

PersonTokensPropCommitsCommitProp
Srinivas Pandruvada154100.00%1100.00%
Total154100.00%1100.00%


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); }

Contributors

PersonTokensPropCommitsCommitProp
Srinivas Pandruvada53100.00%1100.00%
Total53100.00%1100.00%

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); }

Contributors

PersonTokensPropCommitsCommitProp
Srinivas Pandruvada53100.00%1100.00%
Total53100.00%1100.00%

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; }

Contributors

PersonTokensPropCommitsCommitProp
Srinivas Pandruvada34100.00%1100.00%
Total34100.00%1100.00%

EXPORT_SYMBOL_GPL(lpit_read_residency_count_address);
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"); } }

Contributors

PersonTokensPropCommitsCommitProp
Srinivas Pandruvada143100.00%1100.00%
Total143100.00%1100.00%


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; } }

Contributors

PersonTokensPropCommitsCommitProp
Srinivas Pandruvada119100.00%1100.00%
Total119100.00%1100.00%


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); }

Contributors

PersonTokensPropCommitsCommitProp
Srinivas Pandruvada70100.00%1100.00%
Total70100.00%1100.00%


Overall Contributors

PersonTokensPropCommitsCommitProp
Srinivas Pandruvada684100.00%2100.00%
Total684100.00%2100.00%
Directory: drivers/acpi
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.