472 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			472 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0-only
 | |
| /*
 | |
|  *  (C) 2004-2009  Dominik Brodowski <linux@dominikbrodowski.de>
 | |
|  *  (C) 2011       Thomas Renninger <trenn@novell.com> Novell Inc.
 | |
|  */
 | |
| 
 | |
| #include <stdio.h>
 | |
| #include <errno.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <sys/types.h>
 | |
| #include <sys/stat.h>
 | |
| #include <fcntl.h>
 | |
| #include <unistd.h>
 | |
| 
 | |
| #include "helpers/sysfs.h"
 | |
| 
 | |
| unsigned int sysfs_read_file(const char *path, char *buf, size_t buflen)
 | |
| {
 | |
| 	int fd;
 | |
| 	ssize_t numread;
 | |
| 
 | |
| 	fd = open(path, O_RDONLY);
 | |
| 	if (fd == -1)
 | |
| 		return 0;
 | |
| 
 | |
| 	numread = read(fd, buf, buflen - 1);
 | |
| 	if (numread < 1) {
 | |
| 		close(fd);
 | |
| 		return 0;
 | |
| 	}
 | |
| 
 | |
| 	buf[numread] = '\0';
 | |
| 	close(fd);
 | |
| 
 | |
| 	return (unsigned int) numread;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Detect whether a CPU is online
 | |
|  *
 | |
|  * Returns:
 | |
|  *     1 -> if CPU is online
 | |
|  *     0 -> if CPU is offline
 | |
|  *     negative errno values in error case
 | |
|  */
 | |
| int sysfs_is_cpu_online(unsigned int cpu)
 | |
| {
 | |
| 	char path[SYSFS_PATH_MAX];
 | |
| 	int fd;
 | |
| 	ssize_t numread;
 | |
| 	unsigned long long value;
 | |
| 	char linebuf[MAX_LINE_LEN];
 | |
| 	char *endp;
 | |
| 	struct stat statbuf;
 | |
| 
 | |
| 	snprintf(path, sizeof(path), PATH_TO_CPU "cpu%u", cpu);
 | |
| 
 | |
| 	if (stat(path, &statbuf) != 0)
 | |
| 		return 0;
 | |
| 
 | |
| 	/*
 | |
| 	 * kernel without CONFIG_HOTPLUG_CPU
 | |
| 	 * -> cpuX directory exists, but not cpuX/online file
 | |
| 	 */
 | |
| 	snprintf(path, sizeof(path), PATH_TO_CPU "cpu%u/online", cpu);
 | |
| 	if (stat(path, &statbuf) != 0)
 | |
| 		return 1;
 | |
| 
 | |
| 	fd = open(path, O_RDONLY);
 | |
| 	if (fd == -1)
 | |
| 		return -errno;
 | |
| 
 | |
| 	numread = read(fd, linebuf, MAX_LINE_LEN - 1);
 | |
| 	if (numread < 1) {
 | |
| 		close(fd);
 | |
| 		return -EIO;
 | |
| 	}
 | |
| 	linebuf[numread] = '\0';
 | |
| 	close(fd);
 | |
| 
 | |
| 	value = strtoull(linebuf, &endp, 0);
 | |
| 	if (value > 1)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	return value;
 | |
| }
 | |
| 
 | |
| /* CPUidle idlestate specific /sys/devices/system/cpu/cpuX/cpuidle/ access */
 | |
| 
 | |
| 
 | |
| /* CPUidle idlestate specific /sys/devices/system/cpu/cpuX/cpuidle/ access */
 | |
| 
 | |
| /*
 | |
|  * helper function to check whether a file under "../cpuX/cpuidle/stateX/" dir
 | |
|  * exists.
 | |
|  * For example the functionality to disable c-states was introduced in later
 | |
|  * kernel versions, this function can be used to explicitly check for this
 | |
|  * feature.
 | |
|  *
 | |
|  * returns 1 if the file exists, 0 otherwise.
 | |
|  */
 | |
| unsigned int sysfs_idlestate_file_exists(unsigned int cpu,
 | |
| 					 unsigned int idlestate,
 | |
| 					 const char *fname)
 | |
| {
 | |
| 	char path[SYSFS_PATH_MAX];
 | |
| 	struct stat statbuf;
 | |
| 
 | |
| 
 | |
| 	snprintf(path, sizeof(path), PATH_TO_CPU "cpu%u/cpuidle/state%u/%s",
 | |
| 		 cpu, idlestate, fname);
 | |
| 	if (stat(path, &statbuf) != 0)
 | |
| 		return 0;
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * helper function to read file from /sys into given buffer
 | |
|  * fname is a relative path under "cpuX/cpuidle/stateX/" dir
 | |
|  * cstates starting with 0, C0 is not counted as cstate.
 | |
|  * This means if you want C1 info, pass 0 as idlestate param
 | |
|  */
 | |
| unsigned int sysfs_idlestate_read_file(unsigned int cpu, unsigned int idlestate,
 | |
| 			     const char *fname, char *buf, size_t buflen)
 | |
| {
 | |
| 	char path[SYSFS_PATH_MAX];
 | |
| 	int fd;
 | |
| 	ssize_t numread;
 | |
| 
 | |
| 	snprintf(path, sizeof(path), PATH_TO_CPU "cpu%u/cpuidle/state%u/%s",
 | |
| 		 cpu, idlestate, fname);
 | |
| 
 | |
| 	fd = open(path, O_RDONLY);
 | |
| 	if (fd == -1)
 | |
| 		return 0;
 | |
| 
 | |
| 	numread = read(fd, buf, buflen - 1);
 | |
| 	if (numread < 1) {
 | |
| 		close(fd);
 | |
| 		return 0;
 | |
| 	}
 | |
| 
 | |
| 	buf[numread] = '\0';
 | |
| 	close(fd);
 | |
| 
 | |
| 	return (unsigned int) numread;
 | |
| }
 | |
| 
 | |
| /* 
 | |
|  * helper function to write a new value to a /sys file
 | |
|  * fname is a relative path under "../cpuX/cpuidle/cstateY/" dir
 | |
|  *
 | |
|  * Returns the number of bytes written or 0 on error
 | |
|  */
 | |
| static
 | |
| unsigned int sysfs_idlestate_write_file(unsigned int cpu,
 | |
| 					unsigned int idlestate,
 | |
| 					const char *fname,
 | |
| 					const char *value, size_t len)
 | |
| {
 | |
| 	char path[SYSFS_PATH_MAX];
 | |
| 	int fd;
 | |
| 	ssize_t numwrite;
 | |
| 
 | |
| 	snprintf(path, sizeof(path), PATH_TO_CPU "cpu%u/cpuidle/state%u/%s",
 | |
| 		 cpu, idlestate, fname);
 | |
| 
 | |
| 	fd = open(path, O_WRONLY);
 | |
| 	if (fd == -1)
 | |
| 		return 0;
 | |
| 
 | |
| 	numwrite = write(fd, value, len);
 | |
| 	if (numwrite < 1) {
 | |
| 		close(fd);
 | |
| 		return 0;
 | |
| 	}
 | |
| 
 | |
| 	close(fd);
 | |
| 
 | |
| 	return (unsigned int) numwrite;
 | |
| }
 | |
| 
 | |
| /* read access to files which contain one numeric value */
 | |
| 
 | |
| enum idlestate_value {
 | |
| 	IDLESTATE_USAGE,
 | |
| 	IDLESTATE_POWER,
 | |
| 	IDLESTATE_LATENCY,
 | |
| 	IDLESTATE_TIME,
 | |
| 	IDLESTATE_DISABLE,
 | |
| 	MAX_IDLESTATE_VALUE_FILES
 | |
| };
 | |
| 
 | |
| static const char *idlestate_value_files[MAX_IDLESTATE_VALUE_FILES] = {
 | |
| 	[IDLESTATE_USAGE] = "usage",
 | |
| 	[IDLESTATE_POWER] = "power",
 | |
| 	[IDLESTATE_LATENCY] = "latency",
 | |
| 	[IDLESTATE_TIME]  = "time",
 | |
| 	[IDLESTATE_DISABLE]  = "disable",
 | |
| };
 | |
| 
 | |
| static unsigned long long sysfs_idlestate_get_one_value(unsigned int cpu,
 | |
| 						     unsigned int idlestate,
 | |
| 						     enum idlestate_value which)
 | |
| {
 | |
| 	unsigned long long value;
 | |
| 	unsigned int len;
 | |
| 	char linebuf[MAX_LINE_LEN];
 | |
| 	char *endp;
 | |
| 
 | |
| 	if (which >= MAX_IDLESTATE_VALUE_FILES)
 | |
| 		return 0;
 | |
| 
 | |
| 	len = sysfs_idlestate_read_file(cpu, idlestate,
 | |
| 					idlestate_value_files[which],
 | |
| 					linebuf, sizeof(linebuf));
 | |
| 	if (len == 0)
 | |
| 		return 0;
 | |
| 
 | |
| 	value = strtoull(linebuf, &endp, 0);
 | |
| 
 | |
| 	if (endp == linebuf || errno == ERANGE)
 | |
| 		return 0;
 | |
| 
 | |
| 	return value;
 | |
| }
 | |
| 
 | |
| /* read access to files which contain one string */
 | |
| 
 | |
| enum idlestate_string {
 | |
| 	IDLESTATE_DESC,
 | |
| 	IDLESTATE_NAME,
 | |
| 	MAX_IDLESTATE_STRING_FILES
 | |
| };
 | |
| 
 | |
| static const char *idlestate_string_files[MAX_IDLESTATE_STRING_FILES] = {
 | |
| 	[IDLESTATE_DESC] = "desc",
 | |
| 	[IDLESTATE_NAME] = "name",
 | |
| };
 | |
| 
 | |
| 
 | |
| static char *sysfs_idlestate_get_one_string(unsigned int cpu,
 | |
| 					unsigned int idlestate,
 | |
| 					enum idlestate_string which)
 | |
| {
 | |
| 	char linebuf[MAX_LINE_LEN];
 | |
| 	char *result;
 | |
| 	unsigned int len;
 | |
| 
 | |
| 	if (which >= MAX_IDLESTATE_STRING_FILES)
 | |
| 		return NULL;
 | |
| 
 | |
| 	len = sysfs_idlestate_read_file(cpu, idlestate,
 | |
| 					idlestate_string_files[which],
 | |
| 					linebuf, sizeof(linebuf));
 | |
| 	if (len == 0)
 | |
| 		return NULL;
 | |
| 
 | |
| 	result = strdup(linebuf);
 | |
| 	if (result == NULL)
 | |
| 		return NULL;
 | |
| 
 | |
| 	if (result[strlen(result) - 1] == '\n')
 | |
| 		result[strlen(result) - 1] = '\0';
 | |
| 
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Returns:
 | |
|  *    1  if disabled
 | |
|  *    0  if enabled
 | |
|  *    -1 if idlestate is not available
 | |
|  *    -2 if disabling is not supported by the kernel
 | |
|  */
 | |
| int sysfs_is_idlestate_disabled(unsigned int cpu,
 | |
| 				unsigned int idlestate)
 | |
| {
 | |
| 	if (sysfs_get_idlestate_count(cpu) <= idlestate)
 | |
| 		return -1;
 | |
| 
 | |
| 	if (!sysfs_idlestate_file_exists(cpu, idlestate,
 | |
| 				 idlestate_value_files[IDLESTATE_DISABLE]))
 | |
| 		return -2;
 | |
| 	return sysfs_idlestate_get_one_value(cpu, idlestate, IDLESTATE_DISABLE);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Pass 1 as last argument to disable or 0 to enable the state
 | |
|  * Returns:
 | |
|  *    0  on success
 | |
|  *    negative values on error, for example:
 | |
|  *      -1 if idlestate is not available
 | |
|  *      -2 if disabling is not supported by the kernel
 | |
|  *      -3 No write access to disable/enable C-states
 | |
|  */
 | |
| int sysfs_idlestate_disable(unsigned int cpu,
 | |
| 			    unsigned int idlestate,
 | |
| 			    unsigned int disable)
 | |
| {
 | |
| 	char value[SYSFS_PATH_MAX];
 | |
| 	int bytes_written;
 | |
| 
 | |
| 	if (sysfs_get_idlestate_count(cpu) <= idlestate)
 | |
| 		return -1;
 | |
| 
 | |
| 	if (!sysfs_idlestate_file_exists(cpu, idlestate,
 | |
| 				 idlestate_value_files[IDLESTATE_DISABLE]))
 | |
| 		return -2;
 | |
| 
 | |
| 	snprintf(value, SYSFS_PATH_MAX, "%u", disable);
 | |
| 
 | |
| 	bytes_written = sysfs_idlestate_write_file(cpu, idlestate, "disable",
 | |
| 						   value, sizeof(disable));
 | |
| 	if (bytes_written)
 | |
| 		return 0;
 | |
| 	return -3;
 | |
| }
 | |
| 
 | |
| unsigned long sysfs_get_idlestate_latency(unsigned int cpu,
 | |
| 					  unsigned int idlestate)
 | |
| {
 | |
| 	return sysfs_idlestate_get_one_value(cpu, idlestate, IDLESTATE_LATENCY);
 | |
| }
 | |
| 
 | |
| unsigned long sysfs_get_idlestate_usage(unsigned int cpu,
 | |
| 					unsigned int idlestate)
 | |
| {
 | |
| 	return sysfs_idlestate_get_one_value(cpu, idlestate, IDLESTATE_USAGE);
 | |
| }
 | |
| 
 | |
| unsigned long long sysfs_get_idlestate_time(unsigned int cpu,
 | |
| 					unsigned int idlestate)
 | |
| {
 | |
| 	return sysfs_idlestate_get_one_value(cpu, idlestate, IDLESTATE_TIME);
 | |
| }
 | |
| 
 | |
| char *sysfs_get_idlestate_name(unsigned int cpu, unsigned int idlestate)
 | |
| {
 | |
| 	return sysfs_idlestate_get_one_string(cpu, idlestate, IDLESTATE_NAME);
 | |
| }
 | |
| 
 | |
| char *sysfs_get_idlestate_desc(unsigned int cpu, unsigned int idlestate)
 | |
| {
 | |
| 	return sysfs_idlestate_get_one_string(cpu, idlestate, IDLESTATE_DESC);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Returns number of supported C-states of CPU core cpu
 | |
|  * Negativ in error case
 | |
|  * Zero if cpuidle does not export any C-states
 | |
|  */
 | |
| unsigned int sysfs_get_idlestate_count(unsigned int cpu)
 | |
| {
 | |
| 	char file[SYSFS_PATH_MAX];
 | |
| 	struct stat statbuf;
 | |
| 	int idlestates = 1;
 | |
| 
 | |
| 
 | |
| 	snprintf(file, SYSFS_PATH_MAX, PATH_TO_CPU "cpuidle");
 | |
| 	if (stat(file, &statbuf) != 0 || !S_ISDIR(statbuf.st_mode))
 | |
| 		return 0;
 | |
| 
 | |
| 	snprintf(file, SYSFS_PATH_MAX, PATH_TO_CPU "cpu%u/cpuidle/state0", cpu);
 | |
| 	if (stat(file, &statbuf) != 0 || !S_ISDIR(statbuf.st_mode))
 | |
| 		return 0;
 | |
| 
 | |
| 	while (stat(file, &statbuf) == 0 && S_ISDIR(statbuf.st_mode)) {
 | |
| 		snprintf(file, SYSFS_PATH_MAX, PATH_TO_CPU
 | |
| 			 "cpu%u/cpuidle/state%d", cpu, idlestates);
 | |
| 		idlestates++;
 | |
| 	}
 | |
| 	idlestates--;
 | |
| 	return idlestates;
 | |
| }
 | |
| 
 | |
| /* CPUidle general /sys/devices/system/cpu/cpuidle/ sysfs access ********/
 | |
| 
 | |
| /*
 | |
|  * helper function to read file from /sys into given buffer
 | |
|  * fname is a relative path under "cpu/cpuidle/" dir
 | |
|  */
 | |
| static unsigned int sysfs_cpuidle_read_file(const char *fname, char *buf,
 | |
| 					    size_t buflen)
 | |
| {
 | |
| 	char path[SYSFS_PATH_MAX];
 | |
| 
 | |
| 	snprintf(path, sizeof(path), PATH_TO_CPU "cpuidle/%s", fname);
 | |
| 
 | |
| 	return sysfs_read_file(path, buf, buflen);
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /* read access to files which contain one string */
 | |
| 
 | |
| enum cpuidle_string {
 | |
| 	CPUIDLE_GOVERNOR,
 | |
| 	CPUIDLE_GOVERNOR_RO,
 | |
| 	CPUIDLE_DRIVER,
 | |
| 	MAX_CPUIDLE_STRING_FILES
 | |
| };
 | |
| 
 | |
| static const char *cpuidle_string_files[MAX_CPUIDLE_STRING_FILES] = {
 | |
| 	[CPUIDLE_GOVERNOR]	= "current_governor",
 | |
| 	[CPUIDLE_GOVERNOR_RO]	= "current_governor_ro",
 | |
| 	[CPUIDLE_DRIVER]	= "current_driver",
 | |
| };
 | |
| 
 | |
| 
 | |
| static char *sysfs_cpuidle_get_one_string(enum cpuidle_string which)
 | |
| {
 | |
| 	char linebuf[MAX_LINE_LEN];
 | |
| 	char *result;
 | |
| 	unsigned int len;
 | |
| 
 | |
| 	if (which >= MAX_CPUIDLE_STRING_FILES)
 | |
| 		return NULL;
 | |
| 
 | |
| 	len = sysfs_cpuidle_read_file(cpuidle_string_files[which],
 | |
| 				linebuf, sizeof(linebuf));
 | |
| 	if (len == 0)
 | |
| 		return NULL;
 | |
| 
 | |
| 	result = strdup(linebuf);
 | |
| 	if (result == NULL)
 | |
| 		return NULL;
 | |
| 
 | |
| 	if (result[strlen(result) - 1] == '\n')
 | |
| 		result[strlen(result) - 1] = '\0';
 | |
| 
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| char *sysfs_get_cpuidle_governor(void)
 | |
| {
 | |
| 	char *tmp = sysfs_cpuidle_get_one_string(CPUIDLE_GOVERNOR_RO);
 | |
| 	if (!tmp)
 | |
| 		return sysfs_cpuidle_get_one_string(CPUIDLE_GOVERNOR);
 | |
| 	else
 | |
| 		return tmp;
 | |
| }
 | |
| 
 | |
| char *sysfs_get_cpuidle_driver(void)
 | |
| {
 | |
| 	return sysfs_cpuidle_get_one_string(CPUIDLE_DRIVER);
 | |
| }
 | |
| /* CPUidle idlestate specific /sys/devices/system/cpu/cpuX/cpuidle/ access */
 | |
| 
 | |
| /*
 | |
|  * Get sched_mc or sched_smt settings
 | |
|  * Pass "mc" or "smt" as argument
 | |
|  *
 | |
|  * Returns negative value on failure
 | |
|  */
 | |
| int sysfs_get_sched(const char *smt_mc)
 | |
| {
 | |
| 	return -ENODEV;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Get sched_mc or sched_smt settings
 | |
|  * Pass "mc" or "smt" as argument
 | |
|  *
 | |
|  * Returns negative value on failure
 | |
|  */
 | |
| int sysfs_set_sched(const char *smt_mc, int val)
 | |
| {
 | |
| 	return -ENODEV;
 | |
| }
 |