300 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			300 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0-only
 | |
| /*
 | |
|  *  (C) 2016 SUSE Software Solutions GmbH
 | |
|  *           Thomas Renninger <trenn@suse.de>
 | |
|  */
 | |
| 
 | |
| #include <sys/types.h>
 | |
| #include <sys/stat.h>
 | |
| #include <unistd.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <fcntl.h>
 | |
| #include <stdio.h>
 | |
| #include <dirent.h>
 | |
| 
 | |
| #include "powercap.h"
 | |
| 
 | |
| static 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;
 | |
| }
 | |
| 
 | |
| static int sysfs_get_enabled(char *path, int *mode)
 | |
| {
 | |
| 	int fd;
 | |
| 	char yes_no;
 | |
| 	int ret = 0;
 | |
| 
 | |
| 	*mode = 0;
 | |
| 
 | |
| 	fd = open(path, O_RDONLY);
 | |
| 	if (fd == -1) {
 | |
| 		ret = -1;
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	if (read(fd, &yes_no, 1) != 1) {
 | |
| 		ret = -1;
 | |
| 		goto out_close;
 | |
| 	}
 | |
| 
 | |
| 	if (yes_no == '1') {
 | |
| 		*mode = 1;
 | |
| 		goto out_close;
 | |
| 	} else if (yes_no == '0') {
 | |
| 		goto out_close;
 | |
| 	} else {
 | |
| 		ret = -1;
 | |
| 		goto out_close;
 | |
| 	}
 | |
| out_close:
 | |
| 	close(fd);
 | |
| out:
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| int powercap_get_enabled(int *mode)
 | |
| {
 | |
| 	char path[SYSFS_PATH_MAX] = PATH_TO_POWERCAP "/intel-rapl/enabled";
 | |
| 
 | |
| 	return sysfs_get_enabled(path, mode);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Hardcoded, because rapl is the only powercap implementation
 | |
| - * this needs to get more generic if more powercap implementations
 | |
|  * should show up
 | |
|  */
 | |
| int powercap_get_driver(char *driver, int buflen)
 | |
| {
 | |
| 	char file[SYSFS_PATH_MAX] = PATH_TO_RAPL;
 | |
| 
 | |
| 	struct stat statbuf;
 | |
| 
 | |
| 	if (stat(file, &statbuf) != 0 || !S_ISDIR(statbuf.st_mode)) {
 | |
| 		driver = "";
 | |
| 		return -1;
 | |
| 	} else if (buflen > 10) {
 | |
| 		strcpy(driver, "intel-rapl");
 | |
| 		return 0;
 | |
| 	} else
 | |
| 		return -1;
 | |
| }
 | |
| 
 | |
| enum powercap_get64 {
 | |
| 	GET_ENERGY_UJ,
 | |
| 	GET_MAX_ENERGY_RANGE_UJ,
 | |
| 	GET_POWER_UW,
 | |
| 	GET_MAX_POWER_RANGE_UW,
 | |
| 	MAX_GET_64_FILES
 | |
| };
 | |
| 
 | |
| static const char *powercap_get64_files[MAX_GET_64_FILES] = {
 | |
| 	[GET_POWER_UW] = "power_uw",
 | |
| 	[GET_MAX_POWER_RANGE_UW] = "max_power_range_uw",
 | |
| 	[GET_ENERGY_UJ] = "energy_uj",
 | |
| 	[GET_MAX_ENERGY_RANGE_UJ] = "max_energy_range_uj",
 | |
| };
 | |
| 
 | |
| static int sysfs_powercap_get64_val(struct powercap_zone *zone,
 | |
| 				      enum powercap_get64 which,
 | |
| 				      uint64_t *val)
 | |
| {
 | |
| 	char file[SYSFS_PATH_MAX] = PATH_TO_POWERCAP "/";
 | |
| 	int ret;
 | |
| 	char buf[MAX_LINE_LEN];
 | |
| 
 | |
| 	strcat(file, zone->sys_name);
 | |
| 	strcat(file, "/");
 | |
| 	strcat(file, powercap_get64_files[which]);
 | |
| 
 | |
| 	ret = sysfs_read_file(file, buf, MAX_LINE_LEN);
 | |
| 	if (ret < 0)
 | |
| 		return ret;
 | |
| 	if (ret == 0)
 | |
| 		return -1;
 | |
| 
 | |
| 	*val = strtoll(buf, NULL, 10);
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| int powercap_get_max_energy_range_uj(struct powercap_zone *zone, uint64_t *val)
 | |
| {
 | |
| 	return sysfs_powercap_get64_val(zone, GET_MAX_ENERGY_RANGE_UJ, val);
 | |
| }
 | |
| 
 | |
| int powercap_get_energy_uj(struct powercap_zone *zone, uint64_t *val)
 | |
| {
 | |
| 	return sysfs_powercap_get64_val(zone, GET_ENERGY_UJ, val);
 | |
| }
 | |
| 
 | |
| int powercap_get_max_power_range_uw(struct powercap_zone *zone, uint64_t *val)
 | |
| {
 | |
| 	return sysfs_powercap_get64_val(zone, GET_MAX_POWER_RANGE_UW, val);
 | |
| }
 | |
| 
 | |
| int powercap_get_power_uw(struct powercap_zone *zone, uint64_t *val)
 | |
| {
 | |
| 	return sysfs_powercap_get64_val(zone, GET_POWER_UW, val);
 | |
| }
 | |
| 
 | |
| int powercap_zone_get_enabled(struct powercap_zone *zone, int *mode)
 | |
| {
 | |
| 	char path[SYSFS_PATH_MAX] = PATH_TO_POWERCAP;
 | |
| 
 | |
| 	if ((strlen(PATH_TO_POWERCAP) + strlen(zone->sys_name)) +
 | |
| 	    strlen("/enabled") + 1 >= SYSFS_PATH_MAX)
 | |
| 		return -1;
 | |
| 
 | |
| 	strcat(path, "/");
 | |
| 	strcat(path, zone->sys_name);
 | |
| 	strcat(path, "/enabled");
 | |
| 
 | |
| 	return sysfs_get_enabled(path, mode);
 | |
| }
 | |
| 
 | |
| int powercap_zone_set_enabled(struct powercap_zone *zone, int mode)
 | |
| {
 | |
| 	/* To be done if needed */
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| int powercap_read_zone(struct powercap_zone *zone)
 | |
| {
 | |
| 	struct dirent *dent;
 | |
| 	DIR *zone_dir;
 | |
| 	char sysfs_dir[SYSFS_PATH_MAX] = PATH_TO_POWERCAP;
 | |
| 	struct powercap_zone *child_zone;
 | |
| 	char file[SYSFS_PATH_MAX] = PATH_TO_POWERCAP;
 | |
| 	int i, ret = 0;
 | |
| 	uint64_t val = 0;
 | |
| 
 | |
| 	strcat(sysfs_dir, "/");
 | |
| 	strcat(sysfs_dir, zone->sys_name);
 | |
| 
 | |
| 	zone_dir = opendir(sysfs_dir);
 | |
| 	if (zone_dir == NULL)
 | |
| 		return -1;
 | |
| 
 | |
| 	strcat(file, "/");
 | |
| 	strcat(file, zone->sys_name);
 | |
| 	strcat(file, "/name");
 | |
| 	sysfs_read_file(file, zone->name, MAX_LINE_LEN);
 | |
| 	if (zone->parent)
 | |
| 		zone->tree_depth = zone->parent->tree_depth + 1;
 | |
| 	ret = powercap_get_energy_uj(zone, &val);
 | |
| 	if (ret == 0)
 | |
| 		zone->has_energy_uj = 1;
 | |
| 	ret = powercap_get_power_uw(zone, &val);
 | |
| 	if (ret == 0)
 | |
| 		zone->has_power_uw = 1;
 | |
| 
 | |
| 	while ((dent = readdir(zone_dir)) != NULL) {
 | |
| 		struct stat st;
 | |
| 
 | |
| 		if (strcmp(dent->d_name, ".") == 0 || strcmp(dent->d_name, "..") == 0)
 | |
| 			continue;
 | |
| 
 | |
| 		if (stat(dent->d_name, &st) != 0 || !S_ISDIR(st.st_mode))
 | |
| 			if (fstatat(dirfd(zone_dir), dent->d_name, &st, 0) < 0)
 | |
| 				continue;
 | |
| 
 | |
| 		if (strncmp(dent->d_name, "intel-rapl:", 11) != 0)
 | |
| 			continue;
 | |
| 
 | |
| 		child_zone = calloc(1, sizeof(struct powercap_zone));
 | |
| 		if (child_zone == NULL)
 | |
| 			return -1;
 | |
| 		for (i = 0; i < POWERCAP_MAX_CHILD_ZONES; i++) {
 | |
| 			if (zone->children[i] == NULL) {
 | |
| 				zone->children[i] = child_zone;
 | |
| 				break;
 | |
| 			}
 | |
| 			if (i == POWERCAP_MAX_CHILD_ZONES - 1) {
 | |
| 				free(child_zone);
 | |
| 				fprintf(stderr, "Reached POWERCAP_MAX_CHILD_ZONES %d\n",
 | |
| 				       POWERCAP_MAX_CHILD_ZONES);
 | |
| 				return -1;
 | |
| 			}
 | |
| 		}
 | |
| 		strcpy(child_zone->sys_name, zone->sys_name);
 | |
| 		strcat(child_zone->sys_name, "/");
 | |
| 		strcat(child_zone->sys_name, dent->d_name);
 | |
| 		child_zone->parent = zone;
 | |
| 		if (zone->tree_depth >= POWERCAP_MAX_TREE_DEPTH) {
 | |
| 			fprintf(stderr, "Maximum zone hierarchy depth[%d] reached\n",
 | |
| 				POWERCAP_MAX_TREE_DEPTH);
 | |
| 			ret = -1;
 | |
| 			break;
 | |
| 		}
 | |
| 		powercap_read_zone(child_zone);
 | |
| 	}
 | |
| 	closedir(zone_dir);
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| struct powercap_zone *powercap_init_zones(void)
 | |
| {
 | |
| 	int enabled;
 | |
| 	struct powercap_zone *root_zone;
 | |
| 	int ret;
 | |
| 	char file[SYSFS_PATH_MAX] = PATH_TO_RAPL "/enabled";
 | |
| 
 | |
| 	ret = sysfs_get_enabled(file, &enabled);
 | |
| 
 | |
| 	if (ret)
 | |
| 		return NULL;
 | |
| 
 | |
| 	if (!enabled)
 | |
| 		return NULL;
 | |
| 
 | |
| 	root_zone = calloc(1, sizeof(struct powercap_zone));
 | |
| 	if (!root_zone)
 | |
| 		return NULL;
 | |
| 
 | |
| 	strcpy(root_zone->sys_name, "intel-rapl/intel-rapl:0");
 | |
| 
 | |
| 	powercap_read_zone(root_zone);
 | |
| 
 | |
| 	return root_zone;
 | |
| }
 | |
| 
 | |
| /* Call function *f on the passed zone and all its children */
 | |
| 
 | |
| int powercap_walk_zones(struct powercap_zone *zone,
 | |
| 			int (*f)(struct powercap_zone *zone))
 | |
| {
 | |
| 	int i, ret;
 | |
| 
 | |
| 	if (!zone)
 | |
| 		return -1;
 | |
| 
 | |
| 	ret = f(zone);
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 
 | |
| 	for (i = 0; i < POWERCAP_MAX_CHILD_ZONES; i++) {
 | |
| 		if (zone->children[i] != NULL)
 | |
| 			powercap_walk_zones(zone->children[i], f);
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 |