Blame cputree.c

Packit Service 2212bb
/* 
Packit Service 2212bb
 * Copyright (C) 2006, Intel Corporation
Packit Service 2212bb
 * Copyright (C) 2012, Neil Horman <nhorman@tuxdriver.com> 
Packit Service 2212bb
 * 
Packit Service 2212bb
 * This file is part of irqbalance
Packit Service 2212bb
 *
Packit Service 2212bb
 * This program file is free software; you can redistribute it and/or modify it
Packit Service 2212bb
 * under the terms of the GNU General Public License as published by the
Packit Service 2212bb
 * Free Software Foundation; version 2 of the License.
Packit Service 2212bb
 * 
Packit Service 2212bb
 * This program is distributed in the hope that it will be useful, but WITHOUT
Packit Service 2212bb
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
Packit Service 2212bb
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
Packit Service 2212bb
 * for more details.
Packit Service 2212bb
 * 
Packit Service 2212bb
 * You should have received a copy of the GNU General Public License
Packit Service 2212bb
 * along with this program in a file named COPYING; if not, write to the 
Packit Service 2212bb
 * Free Software Foundation, Inc., 
Packit Service 2212bb
 * 51 Franklin Street, Fifth Floor, 
Packit Service 2212bb
 * Boston, MA 02110-1301 USA
Packit Service 2212bb
 */
Packit Service 2212bb
Packit Service 2212bb
/*
Packit Service 2212bb
 * This file contains the code to construct and manipulate a hierarchy of processors,
Packit Service 2212bb
 * cache domains and processor cores.
Packit Service 2212bb
 */
Packit Service 2212bb
Packit Service 2212bb
#include "config.h"
Packit Service 2212bb
#include <stdio.h>
Packit Service 2212bb
#include <stdlib.h>
Packit Service 2212bb
#include <string.h>
Packit Service 2212bb
#include <unistd.h>
Packit Service 2212bb
#include <sys/types.h>
Packit Service 2212bb
#include <sys/stat.h>
Packit Service 2212bb
#include <dirent.h>
Packit Service 2212bb
Packit Service 2212bb
#include <glib.h>
Packit Service 2212bb
Packit Service 2212bb
#include "irqbalance.h"
Packit Service 2212bb
Packit Service 2212bb
extern char *banned_cpumask_from_ui;
Packit Service 2212bb
Packit Service 2212bb
GList *cpus;
Packit Service 2212bb
GList *cache_domains;
Packit Service 2212bb
GList *packages;
Packit Service 2212bb
Packit Service 2212bb
int package_count;
Packit Service 2212bb
int cache_domain_count;
Packit Service 2212bb
int core_count;
Packit Service 2212bb
Packit Service 2212bb
/* Users want to be able to keep interrupts away from some cpus; store these in a cpumask_t */
Packit Service 2212bb
cpumask_t banned_cpus;
Packit Service 2212bb
Packit Service 2212bb
cpumask_t cpu_possible_map;
Packit Service 2212bb
Packit Service 2212bb
/* 
Packit Service 2212bb
   it's convenient to have the complement of banned_cpus available so that 
Packit Service 2212bb
   the AND operator can be used to mask out unwanted cpus
Packit Service 2212bb
*/
Packit Service 2212bb
cpumask_t unbanned_cpus;
Packit Service 2212bb
Packit Service 2212bb
/*
Packit Service 2212bb
 * By default do not place IRQs on CPUs the kernel keeps isolated or
Packit Service 2212bb
 * nohz_full, as specified through the boot commandline. Users can
Packit Service 2212bb
 * override this with the IRQBALANCE_BANNED_CPUS environment variable.
Packit Service 2212bb
 */
Packit Service 2212bb
static void setup_banned_cpus(void)
Packit Service 2212bb
{
Packit Service 2212bb
	FILE *file;
Packit Service 2212bb
	char *line = NULL;
Packit Service 2212bb
	size_t size = 0;
Packit Service 2212bb
	char buffer[4096];
Packit Service 2212bb
	cpumask_t nohz_full;
Packit Service 2212bb
	cpumask_t isolated_cpus;
Packit Service 2212bb
Packit Service 2212bb
	cpus_clear(isolated_cpus);
Packit Service 2212bb
	cpus_clear(nohz_full);
Packit Service 2212bb
Packit Service 2212bb
	/* A manually specified cpumask overrides auto-detection. */
Packit Service 2212bb
	if (banned_cpumask_from_ui != NULL) {
Packit Service 2212bb
		cpulist_parse(banned_cpumask_from_ui,
Packit Service 2212bb
			strlen(banned_cpumask_from_ui), banned_cpus);
Packit Service 2212bb
		goto out;
Packit Service 2212bb
	}
Packit Service 2212bb
	if (getenv("IRQBALANCE_BANNED_CPUS"))  {
Packit Service 2212bb
		cpumask_parse_user(getenv("IRQBALANCE_BANNED_CPUS"), strlen(getenv("IRQBALANCE_BANNED_CPUS")), banned_cpus);
Packit Service 2212bb
		goto out;
Packit Service 2212bb
	}
Packit Service 2212bb
	file = fopen("/sys/devices/system/cpu/isolated", "r");
Packit Service 2212bb
	if (file) {
Packit Service 2212bb
		if (getline(&line, &size, file) > 0) {
Packit Service 2212bb
			if (strlen(line) && line[0] != '\n')
Packit Service 2212bb
				cpulist_parse(line, strlen(line), isolated_cpus);
Packit Service 2212bb
			free(line);
Packit Service 2212bb
			line = NULL;
Packit Service 2212bb
			size = 0;
Packit Service 2212bb
		}
Packit Service 2212bb
		fclose(file);
Packit Service 2212bb
	}
Packit Service 2212bb
Packit Service 2212bb
	file = fopen("/sys/devices/system/cpu/nohz_full", "r");
Packit Service 2212bb
	if (file) {
Packit Service 2212bb
		if (getline(&line, &size, file) > 0) {
Packit Service 2212bb
			if (strlen(line) && line[0] != '\n')
Packit Service 2212bb
				cpulist_parse(line, strlen(line), nohz_full);
Packit Service 2212bb
			free(line);
Packit Service 2212bb
			line = NULL;
Packit Service 2212bb
			size = 0;
Packit Service 2212bb
		}
Packit Service 2212bb
		fclose(file);
Packit Service 2212bb
	}
Packit Service 2212bb
Packit Service 2212bb
	cpus_or(banned_cpus, nohz_full, isolated_cpus);
Packit Service 2212bb
Packit Service 2212bb
out:
Packit Service 2212bb
	cpumask_scnprintf(buffer, 4096, isolated_cpus);
Packit Service 2212bb
	log(TO_CONSOLE, LOG_INFO, "Isolated CPUs: %s\n", buffer);
Packit Service 2212bb
	cpumask_scnprintf(buffer, 4096, nohz_full);
Packit Service 2212bb
	log(TO_CONSOLE, LOG_INFO, "Adaptive-ticks CPUs: %s\n", buffer);
Packit Service 2212bb
	cpumask_scnprintf(buffer, 4096, banned_cpus);
Packit Service 2212bb
	log(TO_CONSOLE, LOG_INFO, "Banned CPUs: %s\n", buffer);
Packit Service 2212bb
}
Packit Service 2212bb
Packit Service 2212bb
static void add_numa_node_to_topo_obj(struct topo_obj *obj, int nodeid)
Packit Service 2212bb
{
Packit Service 2212bb
	GList *entry;
Packit Service 2212bb
	struct topo_obj *node;
Packit Service 2212bb
	struct topo_obj *cand_node;
Packit Service 2212bb
Packit Service 2212bb
	node = get_numa_node(nodeid);
Packit Service 2212bb
	if (!node || node->number == -1)
Packit Service 2212bb
		return;
Packit Service 2212bb
Packit Service 2212bb
	entry = g_list_first(obj->numa_nodes);
Packit Service 2212bb
	while (entry) {
Packit Service 2212bb
		cand_node = entry->data;
Packit Service 2212bb
		if (cand_node == node)
Packit Service 2212bb
			break;
Packit Service 2212bb
		entry = g_list_next(entry);
Packit Service 2212bb
	}
Packit Service 2212bb
Packit Service 2212bb
	if (!entry)
Packit Service 2212bb
		obj->numa_nodes = g_list_append(obj->numa_nodes, node);
Packit Service 2212bb
}
Packit Service 2212bb
Packit Service 2212bb
static struct topo_obj* add_cache_domain_to_package(struct topo_obj *cache,
Packit Service 2212bb
						    int packageid,
Packit Service 2212bb
						    cpumask_t package_mask,
Packit Service 2212bb
						    int nodeid)
Packit Service 2212bb
{
Packit Service 2212bb
	GList *entry;
Packit Service 2212bb
	struct topo_obj *package;
Packit Service 2212bb
	struct topo_obj *lcache; 
Packit Service 2212bb
Packit Service 2212bb
	entry = g_list_first(packages);
Packit Service 2212bb
Packit Service 2212bb
	while (entry) {
Packit Service 2212bb
		package = entry->data;
Packit Service 2212bb
		if (cpus_equal(package_mask, package->mask)) {
Packit Service 2212bb
			if (packageid != package->number)
Packit Service 2212bb
				log(TO_ALL, LOG_WARNING, "package_mask with different physical_package_id found!\n");
Packit Service 2212bb
			break;
Packit Service 2212bb
		}
Packit Service 2212bb
		entry = g_list_next(entry);
Packit Service 2212bb
	}
Packit Service 2212bb
Packit Service 2212bb
	if (!entry) {
Packit Service 2212bb
		package = calloc(sizeof(struct topo_obj), 1);
Packit Service 2212bb
		if (!package)
Packit Service 2212bb
			return NULL;
Packit Service 2212bb
		package->mask = package_mask;
Packit Service 2212bb
		package->obj_type = OBJ_TYPE_PACKAGE;
Packit Service 2212bb
		package->obj_type_list = &packages;
Packit Service 2212bb
		package->number = packageid;
Packit Service 2212bb
		packages = g_list_append(packages, package);
Packit Service 2212bb
		package_count++;
Packit Service 2212bb
	}
Packit Service 2212bb
Packit Service 2212bb
	entry = g_list_first(package->children);
Packit Service 2212bb
	while (entry) {
Packit Service 2212bb
		lcache = entry->data;
Packit Service 2212bb
		if (lcache == cache)
Packit Service 2212bb
			break;
Packit Service 2212bb
		entry = g_list_next(entry);
Packit Service 2212bb
	}
Packit Service 2212bb
Packit Service 2212bb
	if (!entry) {
Packit Service 2212bb
		package->children = g_list_append(package->children, cache);
Packit Service 2212bb
		cache->parent = package;
Packit Service 2212bb
	}
Packit Service 2212bb
Packit Service 2212bb
	if (nodeid > -1)
Packit Service 2212bb
		add_numa_node_to_topo_obj(package, nodeid);
Packit Service 2212bb
Packit Service 2212bb
	return package;
Packit Service 2212bb
}
Packit Service 2212bb
static struct topo_obj* add_cpu_to_cache_domain(struct topo_obj *cpu,
Packit Service 2212bb
						    cpumask_t cache_mask,
Packit Service 2212bb
						    int nodeid)
Packit Service 2212bb
{
Packit Service 2212bb
	GList *entry;
Packit Service 2212bb
	struct topo_obj *cache;
Packit Service 2212bb
	struct topo_obj *lcpu;
Packit Service 2212bb
Packit Service 2212bb
	entry = g_list_first(cache_domains);
Packit Service 2212bb
Packit Service 2212bb
	while (entry) {
Packit Service 2212bb
		cache = entry->data;
Packit Service 2212bb
		if (cpus_equal(cache_mask, cache->mask))
Packit Service 2212bb
			break;
Packit Service 2212bb
		entry = g_list_next(entry);
Packit Service 2212bb
	}
Packit Service 2212bb
Packit Service 2212bb
	if (!entry) {
Packit Service 2212bb
		cache = calloc(sizeof(struct topo_obj), 1);
Packit Service 2212bb
		if (!cache)
Packit Service 2212bb
			return NULL;
Packit Service 2212bb
		cache->obj_type = OBJ_TYPE_CACHE;
Packit Service 2212bb
		cache->mask = cache_mask;
Packit Service 2212bb
		cache->number = cache_domain_count;
Packit Service 2212bb
		cache->obj_type_list = &cache_domains;
Packit Service 2212bb
		cache_domains = g_list_append(cache_domains, cache);
Packit Service 2212bb
		cache_domain_count++;
Packit Service 2212bb
	}
Packit Service 2212bb
Packit Service 2212bb
	entry = g_list_first(cache->children);
Packit Service 2212bb
	while (entry) {
Packit Service 2212bb
		lcpu = entry->data;
Packit Service 2212bb
		if (lcpu == cpu)
Packit Service 2212bb
			break;
Packit Service 2212bb
		entry = g_list_next(entry);
Packit Service 2212bb
	}
Packit Service 2212bb
Packit Service 2212bb
	if (!entry) {
Packit Service 2212bb
		cache->children = g_list_append(cache->children, cpu);
Packit Service 2212bb
		cpu->parent = (struct topo_obj *)cache;
Packit Service 2212bb
	}
Packit Service 2212bb
Packit Service 2212bb
	if (nodeid > -1)
Packit Service 2212bb
		add_numa_node_to_topo_obj(cache, nodeid);
Packit Service 2212bb
Packit Service 2212bb
	return cache;
Packit Service 2212bb
}
Packit Service 2212bb
 
Packit Service 2212bb
static void do_one_cpu(char *path)
Packit Service 2212bb
{
Packit Service 2212bb
	struct topo_obj *cpu;
Packit Service 2212bb
	FILE *file;
Packit Service 2212bb
	char new_path[PATH_MAX];
Packit Service 2212bb
	cpumask_t cache_mask, package_mask;
Packit Service 2212bb
	struct topo_obj *cache;
Packit Service 2212bb
	DIR *dir;
Packit Service 2212bb
	struct dirent *entry;
Packit Service 2212bb
	int nodeid;
Packit Service 2212bb
	int packageid = 0;
Packit Service 2212bb
	unsigned int max_cache_index, cache_index, cache_stat;
Packit Service 2212bb
Packit Service 2212bb
	/* skip offline cpus */
Packit Service 2212bb
	snprintf(new_path, PATH_MAX, "%s/online", path);
Packit Service 2212bb
	file = fopen(new_path, "r");
Packit Service 2212bb
	if (file) {
Packit Service 2212bb
		char *line = NULL;
Packit Service 2212bb
		size_t size = 0;
Packit Service 2212bb
		if (getline(&line, &size, file)==0)
Packit Service 2212bb
			return;
Packit Service 2212bb
		fclose(file);
Packit Service 2212bb
		if (line && line[0]=='0') {
Packit Service 2212bb
			free(line);
Packit Service 2212bb
			return;
Packit Service 2212bb
		}
Packit Service 2212bb
		free(line);
Packit Service 2212bb
	}
Packit Service 2212bb
Packit Service 2212bb
	cpu = calloc(sizeof(struct topo_obj), 1);
Packit Service 2212bb
	if (!cpu)
Packit Service 2212bb
		return;
Packit Service 2212bb
Packit Service 2212bb
	cpu->obj_type = OBJ_TYPE_CPU;
Packit Service 2212bb
Packit Service 2212bb
	cpu->number = strtoul(&path[27], NULL, 10);
Packit Service 2212bb
Packit Service 2212bb
	cpu_set(cpu->number, cpu_possible_map);
Packit Service 2212bb
	
Packit Service 2212bb
	cpu_set(cpu->number, cpu->mask);
Packit Service 2212bb
Packit Service 2212bb
	/*
Packit Service 2212bb
 	 * Default the cache_domain mask to be equal to the cpu
Packit Service 2212bb
 	 */
Packit Service 2212bb
	cpus_clear(cache_mask);
Packit Service 2212bb
	cpu_set(cpu->number, cache_mask);
Packit Service 2212bb
Packit Service 2212bb
	/* if the cpu is on the banned list, just don't add it */
Packit Service 2212bb
	if (cpus_intersects(cpu->mask, banned_cpus)) {
Packit Service 2212bb
		free(cpu);
Packit Service 2212bb
		/* even though we don't use the cpu we do need to count it */
Packit Service 2212bb
		core_count++;
Packit Service 2212bb
		return;
Packit Service 2212bb
	}
Packit Service 2212bb
Packit Service 2212bb
Packit Service 2212bb
	/* try to read the package mask; if it doesn't exist assume solitary */
Packit Service 2212bb
	snprintf(new_path, PATH_MAX, "%s/topology/core_siblings", path);
Packit Service 2212bb
	file = fopen(new_path, "r");
Packit Service 2212bb
	cpu_set(cpu->number, package_mask);
Packit Service 2212bb
	if (file) {
Packit Service 2212bb
		char *line = NULL;
Packit Service 2212bb
		size_t size = 0;
Packit Service 2212bb
		if (getline(&line, &size, file)) 
Packit Service 2212bb
			cpumask_parse_user(line, strlen(line), package_mask);
Packit Service 2212bb
		fclose(file);
Packit Service 2212bb
		free(line);
Packit Service 2212bb
	}
Packit Service 2212bb
	/* try to read the package id */
Packit Service 2212bb
	snprintf(new_path, PATH_MAX, "%s/topology/physical_package_id", path);
Packit Service 2212bb
	file = fopen(new_path, "r");
Packit Service 2212bb
	if (file) {
Packit Service 2212bb
		char *line = NULL;
Packit Service 2212bb
		size_t size = 0;
Packit Service 2212bb
		if (getline(&line, &size, file))
Packit Service 2212bb
			packageid = strtoul(line, NULL, 10);
Packit Service 2212bb
		fclose(file);
Packit Service 2212bb
		free(line);
Packit Service 2212bb
	}
Packit Service 2212bb
Packit Service 2212bb
	/* try to read the cache mask; if it doesn't exist assume solitary */
Packit Service 2212bb
	/* We want the deepest cache level available */
Packit Service 2212bb
	cpu_set(cpu->number, cache_mask);
Packit Service 2212bb
	max_cache_index = 0;
Packit Service 2212bb
	cache_index = 1;
Packit Service 2212bb
	do {
Packit Service 2212bb
		struct stat sb;
Packit Service 2212bb
		snprintf(new_path, PATH_MAX, "%s/cache/index%d/shared_cpu_map", path, cache_index);
Packit Service 2212bb
		cache_stat = stat(new_path, &sb);
Packit Service 2212bb
		if (!cache_stat) {
Packit Service 2212bb
			max_cache_index = cache_index;
Packit Service 2212bb
			if (max_cache_index == deepest_cache)
Packit Service 2212bb
				break;
Packit Service 2212bb
			cache_index ++;
Packit Service 2212bb
		}
Packit Service 2212bb
	} while(!cache_stat);
Packit Service 2212bb
Packit Service 2212bb
	if (max_cache_index > 0) {
Packit Service 2212bb
		snprintf(new_path, PATH_MAX, "%s/cache/index%d/shared_cpu_map", path, max_cache_index);
Packit Service 2212bb
		file = fopen(new_path, "r");
Packit Service 2212bb
		if (file) {
Packit Service 2212bb
			char *line = NULL;
Packit Service 2212bb
			size_t size = 0;
Packit Service 2212bb
			if (getline(&line, &size, file))
Packit Service 2212bb
				cpumask_parse_user(line, strlen(line), cache_mask);
Packit Service 2212bb
			fclose(file);
Packit Service 2212bb
			free(line);
Packit Service 2212bb
		}
Packit Service 2212bb
	}
Packit Service 2212bb
Packit Service 2212bb
	nodeid=-1;
Packit Service 2212bb
	if (numa_avail) {
Packit Service 2212bb
		struct topo_obj *node;
Packit Service 2212bb
Packit Service 2212bb
		dir = opendir(path);
Packit Service 2212bb
		do {
Packit Service 2212bb
			entry = readdir(dir);
Packit Service 2212bb
			if (!entry)
Packit Service 2212bb
				break;
Packit Service adcb48
			if (strstr(entry->d_name, "node")) {
Packit Service adcb48
				nodeid = strtoul(&entry->d_name[4], NULL, 10);
Packit Service adcb48
				break;
Packit Service 2212bb
			}
Packit Service 2212bb
		} while (entry);
Packit Service 2212bb
		closedir(dir);
Packit Service 2212bb
Packit Service 2212bb
		/*
Packit Service 2212bb
		 * In case of multiple NUMA nodes within a CPU package,
Packit Service 2212bb
		 * we override package_mask with node mask.
Packit Service 2212bb
		 */
Packit Service 2212bb
		node = get_numa_node(nodeid);
Packit Service 2212bb
		if (node && (cpus_weight(package_mask) > cpus_weight(node->mask)))
Packit Service 2212bb
			cpus_and(package_mask, package_mask, node->mask);
Packit Service 2212bb
	}
Packit Service 2212bb
Packit Service 2212bb
	/*
Packit Service 2212bb
	   blank out the banned cpus from the various masks so that interrupts
Packit Service 2212bb
	   will never be told to go there
Packit Service 2212bb
	 */
Packit Service 2212bb
	cpus_and(cache_mask, cache_mask, unbanned_cpus);
Packit Service 2212bb
	cpus_and(package_mask, package_mask, unbanned_cpus);
Packit Service 2212bb
Packit Service 2212bb
	cache = add_cpu_to_cache_domain(cpu, cache_mask, nodeid);
Packit Service 2212bb
	add_cache_domain_to_package(cache, packageid, package_mask,
Packit Service 2212bb
	    nodeid);
Packit Service 2212bb
Packit Service 2212bb
	cpu->obj_type_list = &cpus;
Packit Service 2212bb
	cpus = g_list_append(cpus, cpu);
Packit Service 2212bb
	core_count++;
Packit Service 2212bb
}
Packit Service 2212bb
Packit Service 2212bb
static void dump_irq(struct irq_info *info, void *data)
Packit Service 2212bb
{
Packit Service 2212bb
	int spaces = (long int)data;
Packit Service 2212bb
	int i;
Packit Service 2212bb
	char * indent = malloc (sizeof(char) * (spaces + 1));
Packit Service 2212bb
Packit Service 2212bb
	for ( i = 0; i < spaces; i++ )
Packit Service 2212bb
		indent[i] = log_indent[0];
Packit Service 2212bb
Packit Service 2212bb
	indent[i] = '\0';
Packit Service 2212bb
	log(TO_CONSOLE, LOG_INFO, "%sInterrupt %i node_num is %d (%s/%lu:%lu) \n", indent,
Packit Service 2212bb
	    info->irq, irq_numa_node(info)->number, classes[info->class], info->load, (info->irq_count - info->last_irq_count));
Packit Service 2212bb
	free(indent);
Packit Service 2212bb
}
Packit Service 2212bb
Packit Service 2212bb
static void dump_numa_node_num(struct topo_obj *p, void *data __attribute__((unused)))
Packit Service 2212bb
{
Packit Service 2212bb
	log(TO_CONSOLE, LOG_INFO, "%d ", p->number);
Packit Service 2212bb
}
Packit Service 2212bb
Packit Service 2212bb
static void dump_balance_obj(struct topo_obj *d, void *data __attribute__((unused)))
Packit Service 2212bb
{
Packit Service 2212bb
	struct topo_obj *c = (struct topo_obj *)d;
Packit Service 2212bb
	log(TO_CONSOLE, LOG_INFO, "%s%s%s%sCPU number %i  numa_node is ",
Packit Service 2212bb
	    log_indent, log_indent, log_indent, log_indent, c->number);
Packit Service 2212bb
	for_each_object(cpu_numa_node(c), dump_numa_node_num, NULL);
Packit Service 2212bb
	log(TO_CONSOLE, LOG_INFO, "(load %lu)\n", (unsigned long)c->load);
Packit Service 2212bb
	if (c->interrupts)
Packit Service 2212bb
		for_each_irq(c->interrupts, dump_irq, (void *)18);
Packit Service 2212bb
}
Packit Service 2212bb
Packit Service 2212bb
static void dump_cache_domain(struct topo_obj *d, void *data)
Packit Service 2212bb
{
Packit Service 2212bb
	char *buffer = data;
Packit Service 2212bb
	cpumask_scnprintf(buffer, 4095, d->mask);
Packit Service 2212bb
	log(TO_CONSOLE, LOG_INFO, "%s%sCache domain %i:  numa_node is ",
Packit Service 2212bb
	    log_indent, log_indent, d->number);
Packit Service 2212bb
	for_each_object(d->numa_nodes, dump_numa_node_num, NULL);
Packit Service 2212bb
	log(TO_CONSOLE, LOG_INFO, "cpu mask is %s  (load %lu) \n", buffer,
Packit Service 2212bb
	    (unsigned long)d->load);
Packit Service 2212bb
	if (d->children)
Packit Service 2212bb
		for_each_object(d->children, dump_balance_obj, NULL);
Packit Service 2212bb
	if (g_list_length(d->interrupts) > 0)
Packit Service 2212bb
		for_each_irq(d->interrupts, dump_irq, (void *)10);
Packit Service 2212bb
}
Packit Service 2212bb
Packit Service 2212bb
static void dump_package(struct topo_obj *d, void *data)
Packit Service 2212bb
{
Packit Service 2212bb
	char *buffer = data;
Packit Service 2212bb
	cpumask_scnprintf(buffer, 4096, d->mask);
Packit Service 2212bb
	log(TO_CONSOLE, LOG_INFO, "Package %i:  numa_node ", d->number);
Packit Service 2212bb
	for_each_object(d->numa_nodes, dump_numa_node_num, NULL);
Packit Service 2212bb
	log(TO_CONSOLE, LOG_INFO, "cpu mask is %s (load %lu)\n",
Packit Service 2212bb
	    buffer, (unsigned long)d->load);
Packit Service 2212bb
	if (d->children)
Packit Service 2212bb
		for_each_object(d->children, dump_cache_domain, buffer);
Packit Service 2212bb
	if (g_list_length(d->interrupts) > 0)
Packit Service 2212bb
		for_each_irq(d->interrupts, dump_irq, (void *)2);
Packit Service 2212bb
}
Packit Service 2212bb
Packit Service 2212bb
void dump_tree(void)
Packit Service 2212bb
{
Packit Service 2212bb
	char buffer[4096];
Packit Service 2212bb
	for_each_object(packages, dump_package, buffer);
Packit Service 2212bb
}
Packit Service 2212bb
Packit Service 2212bb
static void clear_irq_stats(struct irq_info *info, void *data __attribute__((unused)))
Packit Service 2212bb
{
Packit Service 2212bb
	info->load = 0;
Packit Service 2212bb
}
Packit Service 2212bb
Packit Service 2212bb
static void clear_obj_stats(struct topo_obj *d, void *data __attribute__((unused)))
Packit Service 2212bb
{
Packit Service 2212bb
	for_each_object(d->children, clear_obj_stats, NULL);
Packit Service 2212bb
	for_each_irq(d->interrupts, clear_irq_stats, NULL);
Packit Service 2212bb
}
Packit Service 2212bb
Packit Service 2212bb
/*
Packit Service 2212bb
 * this function removes previous state from the cpu tree, such as
Packit Service 2212bb
 * which level does how much work and the actual lists of interrupts 
Packit Service 2212bb
 * assigned to each component
Packit Service 2212bb
 */
Packit Service 2212bb
void clear_work_stats(void)
Packit Service 2212bb
{
Packit Service 2212bb
	for_each_object(numa_nodes, clear_obj_stats, NULL);
Packit Service 2212bb
}
Packit Service 2212bb
Packit Service 2212bb
Packit Service 2212bb
void parse_cpu_tree(void)
Packit Service 2212bb
{
Packit Service 2212bb
	DIR *dir;
Packit Service 2212bb
	struct dirent *entry;
Packit Service 2212bb
Packit Service 2212bb
	setup_banned_cpus();
Packit Service 2212bb
Packit Service 2212bb
	cpus_complement(unbanned_cpus, banned_cpus);
Packit Service 2212bb
Packit Service 2212bb
	dir = opendir("/sys/devices/system/cpu");
Packit Service 2212bb
	if (!dir)
Packit Service 2212bb
		return;
Packit Service 2212bb
	do {
Packit Service 2212bb
		int num;
Packit Service 2212bb
		char pad;
Packit Service 2212bb
		entry = readdir(dir);
Packit Service 2212bb
		/*
Packit Service 2212bb
		 * We only want to count real cpus, not cpufreq and
Packit Service 2212bb
		 * cpuidle
Packit Service 2212bb
		 */
Packit Service 2212bb
		if (entry &&
Packit Service 2212bb
		    sscanf(entry->d_name, "cpu%d%c", &num, &pad) == 1 &&
Packit Service 2212bb
		    !strchr(entry->d_name, ' ')) {
Packit Service 2212bb
			char new_path[PATH_MAX];
Packit Service 2212bb
			sprintf(new_path, "/sys/devices/system/cpu/%s", entry->d_name);
Packit Service 2212bb
			do_one_cpu(new_path);
Packit Service 2212bb
		}
Packit Service 2212bb
	} while (entry);
Packit Service 2212bb
	closedir(dir);
Packit Service 2212bb
	for_each_object(packages, connect_cpu_mem_topo, NULL);
Packit Service 2212bb
Packit Service 2212bb
	if (debug_mode)
Packit Service 2212bb
		dump_tree();
Packit Service 2212bb
Packit Service 2212bb
}
Packit Service 2212bb
Packit Service 2212bb
Packit Service 2212bb
/*
Packit Service 2212bb
 * This function frees all memory related to a cpu tree so that a new tree
Packit Service 2212bb
 * can be read
Packit Service 2212bb
 */
Packit Service 2212bb
void clear_cpu_tree(void)
Packit Service 2212bb
{
Packit Service 2212bb
	GList *item;
Packit Service 2212bb
	struct topo_obj *cpu;
Packit Service 2212bb
	struct topo_obj *cache_domain;
Packit Service 2212bb
	struct topo_obj *package;
Packit Service 2212bb
Packit Service 2212bb
	while (packages) {
Packit Service 2212bb
		item = g_list_first(packages);
Packit Service 2212bb
		package = item->data;
Packit Service 2212bb
		g_list_free(package->children);
Packit Service 2212bb
		g_list_free(package->interrupts);
Packit Service 2212bb
		g_list_free(package->numa_nodes);
Packit Service 2212bb
		free(package);
Packit Service 2212bb
		packages = g_list_delete_link(packages, item);
Packit Service 2212bb
	}
Packit Service 2212bb
	package_count = 0;
Packit Service 2212bb
Packit Service 2212bb
	while (cache_domains) {
Packit Service 2212bb
		item = g_list_first(cache_domains);
Packit Service 2212bb
		cache_domain = item->data;
Packit Service 2212bb
		g_list_free(cache_domain->children);
Packit Service 2212bb
		g_list_free(cache_domain->interrupts);
Packit Service 2212bb
		g_list_free(cache_domain->numa_nodes);
Packit Service 2212bb
		free(cache_domain);
Packit Service 2212bb
		cache_domains = g_list_delete_link(cache_domains, item);
Packit Service 2212bb
	}
Packit Service 2212bb
	cache_domain_count = 0;
Packit Service 2212bb
Packit Service 2212bb
Packit Service 2212bb
	while (cpus) {
Packit Service 2212bb
		item = g_list_first(cpus);
Packit Service 2212bb
		cpu = item->data;
Packit Service 2212bb
		g_list_free(cpu->interrupts);
Packit Service 2212bb
		free(cpu);
Packit Service 2212bb
		cpus = g_list_delete_link(cpus, item);
Packit Service 2212bb
	}
Packit Service 2212bb
	core_count = 0;
Packit Service 2212bb
Packit Service 2212bb
}
Packit Service 2212bb
Packit Service 2212bb
static gint compare_cpus(gconstpointer a, gconstpointer b)
Packit Service 2212bb
{
Packit Service 2212bb
	const struct topo_obj *ai = a;
Packit Service 2212bb
	const struct topo_obj *bi = b;
Packit Service 2212bb
Packit Service 2212bb
	return ai->number - bi->number;	
Packit Service 2212bb
}
Packit Service 2212bb
Packit Service 2212bb
struct topo_obj *find_cpu_core(int cpunr)
Packit Service 2212bb
{
Packit Service 2212bb
	GList *entry;
Packit Service 2212bb
	struct topo_obj find;
Packit Service 2212bb
Packit Service 2212bb
	find.number = cpunr;
Packit Service 2212bb
	entry = g_list_find_custom(cpus, &find, compare_cpus);
Packit Service 2212bb
Packit Service 2212bb
	return entry ? entry->data : NULL;
Packit Service 2212bb
}	
Packit Service 2212bb
Packit Service 2212bb
int get_cpu_count(void)
Packit Service 2212bb
{
Packit Service 2212bb
	return g_list_length(cpus);
Packit Service 2212bb
}
Packit Service 2212bb