Commit 7cde4910 authored by Michael Holzheu's avatar Michael Holzheu Committed by Martin Schwidefsky

s390/numa: make core to node mapping data dynamic

The core to node mapping data consumes about 2 KB bss data. To save memory
for the non-NUMA case, make the data dynamic. In addition change the
"core_to_node" array from "int" to "s32" which saves 1 KB also for the
NUMA case.
Suggested-by: default avatarHeiko Carstens <heiko.carstens@de.ibm.com>
Signed-off-by: default avatarMichael Holzheu <holzheu@linux.vnet.ibm.com>
Signed-off-by: default avatarMartin Schwidefsky <schwidefsky@de.ibm.com>
parent 854508c0
...@@ -23,6 +23,7 @@ ...@@ -23,6 +23,7 @@
#include <linux/memblock.h> #include <linux/memblock.h>
#include <linux/node.h> #include <linux/node.h>
#include <linux/memory.h> #include <linux/memory.h>
#include <linux/slab.h>
#include <asm/smp.h> #include <asm/smp.h>
#include <asm/topology.h> #include <asm/topology.h>
#include "numa_mode.h" #include "numa_mode.h"
...@@ -56,26 +57,24 @@ static unsigned long emu_size; ...@@ -56,26 +57,24 @@ static unsigned long emu_size;
* Node to core pinning information updates are protected by * Node to core pinning information updates are protected by
* "sched_domains_mutex". * "sched_domains_mutex".
*/ */
/* Pinned core to node mapping */ static struct {
static int cores_to_node_id[CONFIG_NR_CPUS]; s32 to_node_id[CONFIG_NR_CPUS]; /* Pinned core to node mapping */
/* Total number of pinned cores */ int total; /* Total number of pinned cores */
static int cores_total; int per_node_target; /* Cores per node without extra cores */
/* Number of cores per node without extra cores */ int per_node[MAX_NUMNODES]; /* Number of cores pinned to node */
static int cores_per_node_target; } *emu_cores;
/* Number of cores pinned to node */
static int cores_per_node[MAX_NUMNODES];
/* /*
* Pin a core to a node * Pin a core to a node
*/ */
static void pin_core_to_node(int core_id, int node_id) static void pin_core_to_node(int core_id, int node_id)
{ {
if (cores_to_node_id[core_id] == NODE_ID_FREE) { if (emu_cores->to_node_id[core_id] == NODE_ID_FREE) {
cores_per_node[node_id]++; emu_cores->per_node[node_id]++;
cores_to_node_id[core_id] = node_id; emu_cores->to_node_id[core_id] = node_id;
cores_total++; emu_cores->total++;
} else { } else {
WARN_ON(cores_to_node_id[core_id] != node_id); WARN_ON(emu_cores->to_node_id[core_id] != node_id);
} }
} }
...@@ -84,7 +83,7 @@ static void pin_core_to_node(int core_id, int node_id) ...@@ -84,7 +83,7 @@ static void pin_core_to_node(int core_id, int node_id)
*/ */
static int cores_pinned(struct toptree *node) static int cores_pinned(struct toptree *node)
{ {
return cores_per_node[node->id]; return emu_cores->per_node[node->id];
} }
/* /*
...@@ -92,7 +91,7 @@ static int cores_pinned(struct toptree *node) ...@@ -92,7 +91,7 @@ static int cores_pinned(struct toptree *node)
*/ */
static int core_pinned_to_node_id(struct toptree *core) static int core_pinned_to_node_id(struct toptree *core)
{ {
return cores_to_node_id[core->id]; return emu_cores->to_node_id[core->id];
} }
/* /*
...@@ -174,14 +173,15 @@ static void toptree_unify_tree(struct toptree *tree) ...@@ -174,14 +173,15 @@ static void toptree_unify_tree(struct toptree *tree)
/* /*
* Find the best/nearest node for a given core and ensure that no node * Find the best/nearest node for a given core and ensure that no node
* gets more than "cores_per_node_target + extra" cores. * gets more than "emu_cores->per_node_target + extra" cores.
*/ */
static struct toptree *node_for_core(struct toptree *numa, struct toptree *core, static struct toptree *node_for_core(struct toptree *numa, struct toptree *core,
int extra) int extra)
{ {
struct toptree *node, *node_best = NULL; struct toptree *node, *node_best = NULL;
int dist_cur, dist_best; int dist_cur, dist_best, cores_target;
cores_target = emu_cores->per_node_target + extra;
dist_best = DIST_MAX; dist_best = DIST_MAX;
node_best = NULL; node_best = NULL;
toptree_for_each(node, numa, NODE) { toptree_for_each(node, numa, NODE) {
...@@ -191,7 +191,7 @@ static struct toptree *node_for_core(struct toptree *numa, struct toptree *core, ...@@ -191,7 +191,7 @@ static struct toptree *node_for_core(struct toptree *numa, struct toptree *core,
break; break;
} }
/* Skip nodes that already have enough cores */ /* Skip nodes that already have enough cores */
if (cores_pinned(node) >= cores_per_node_target + extra) if (cores_pinned(node) >= cores_target)
continue; continue;
dist_cur = dist_node_to_core(node, core); dist_cur = dist_node_to_core(node, core);
if (dist_cur < dist_best) { if (dist_cur < dist_best) {
...@@ -225,11 +225,11 @@ static void toptree_to_numa_single(struct toptree *numa, struct toptree *phys, ...@@ -225,11 +225,11 @@ static void toptree_to_numa_single(struct toptree *numa, struct toptree *phys,
static void move_level_to_numa_node(struct toptree *node, struct toptree *phys, static void move_level_to_numa_node(struct toptree *node, struct toptree *phys,
enum toptree_level level, bool perfect) enum toptree_level level, bool perfect)
{ {
int cores_free, cores_target = emu_cores->per_node_target;
struct toptree *cur, *tmp; struct toptree *cur, *tmp;
int cores_free;
toptree_for_each_safe(cur, tmp, phys, level) { toptree_for_each_safe(cur, tmp, phys, level) {
cores_free = cores_per_node_target - toptree_count(node, CORE); cores_free = cores_target - toptree_count(node, CORE);
if (perfect) { if (perfect) {
if (cores_free == toptree_count(cur, CORE)) if (cores_free == toptree_count(cur, CORE))
toptree_move(cur, node); toptree_move(cur, node);
...@@ -291,6 +291,20 @@ static struct toptree *toptree_new(int id, int nodes) ...@@ -291,6 +291,20 @@ static struct toptree *toptree_new(int id, int nodes)
panic("NUMA emulation could not allocate topology"); panic("NUMA emulation could not allocate topology");
} }
/*
* Allocate and initialize core to node mapping
*/
static void create_core_to_node_map(void)
{
int i;
emu_cores = kzalloc(sizeof(*emu_cores), GFP_KERNEL);
if (emu_cores == NULL)
panic("Could not allocate cores to node memory");
for (i = 0; i < ARRAY_SIZE(emu_cores->to_node_id); i++)
emu_cores->to_node_id[i] = NODE_ID_FREE;
}
/* /*
* Move cores from physical topology into NUMA target topology * Move cores from physical topology into NUMA target topology
* and try to keep as much of the physical topology as possible. * and try to keep as much of the physical topology as possible.
...@@ -299,8 +313,10 @@ static struct toptree *toptree_to_numa(struct toptree *phys) ...@@ -299,8 +313,10 @@ static struct toptree *toptree_to_numa(struct toptree *phys)
{ {
static int first = 1; static int first = 1;
struct toptree *numa; struct toptree *numa;
int cores_total;
cores_per_node_target = (cores_total + cores_free(phys)) / emu_nodes; cores_total = emu_cores->total + cores_free(phys);
emu_cores->per_node_target = cores_total / emu_nodes;
numa = toptree_new(TOPTREE_ID_NUMA, emu_nodes); numa = toptree_new(TOPTREE_ID_NUMA, emu_nodes);
if (first) { if (first) {
toptree_to_numa_first(numa, phys); toptree_to_numa_first(numa, phys);
...@@ -386,8 +402,8 @@ static void print_node_to_core_map(void) ...@@ -386,8 +402,8 @@ static void print_node_to_core_map(void)
printk(KERN_DEBUG "NUMA node to core mapping\n"); printk(KERN_DEBUG "NUMA node to core mapping\n");
for (nid = 0; nid < emu_nodes; nid++) { for (nid = 0; nid < emu_nodes; nid++) {
printk(KERN_DEBUG " node %3d: ", nid); printk(KERN_DEBUG " node %3d: ", nid);
for (cid = 0; cid < ARRAY_SIZE(cores_to_node_id); cid++) { for (cid = 0; cid < ARRAY_SIZE(emu_cores->to_node_id); cid++) {
if (cores_to_node_id[cid] == nid) if (emu_cores->to_node_id[cid] == nid)
printk(KERN_CONT "%d ", cid); printk(KERN_CONT "%d ", cid);
} }
printk(KERN_CONT "\n"); printk(KERN_CONT "\n");
...@@ -404,6 +420,8 @@ static void emu_update_cpu_topology(void) ...@@ -404,6 +420,8 @@ static void emu_update_cpu_topology(void)
{ {
struct toptree *phys, *numa; struct toptree *phys, *numa;
if (emu_cores == NULL)
create_core_to_node_map();
phys = toptree_from_topology(); phys = toptree_from_topology();
numa = toptree_to_numa(phys); numa = toptree_to_numa(phys);
toptree_free(phys); toptree_free(phys);
...@@ -443,12 +461,8 @@ static int emu_setup_nodes_adjust(int nodes) ...@@ -443,12 +461,8 @@ static int emu_setup_nodes_adjust(int nodes)
*/ */
static void emu_setup(void) static void emu_setup(void)
{ {
int i;
emu_size = emu_setup_size_adjust(emu_size); emu_size = emu_setup_size_adjust(emu_size);
emu_nodes = emu_setup_nodes_adjust(emu_nodes); emu_nodes = emu_setup_nodes_adjust(emu_nodes);
for (i = 0; i < ARRAY_SIZE(cores_to_node_id); i++)
cores_to_node_id[i] = NODE_ID_FREE;
pr_info("Creating %d nodes with memory stripe size %ld MB\n", pr_info("Creating %d nodes with memory stripe size %ld MB\n",
emu_nodes, emu_size >> 20); emu_nodes, emu_size >> 20);
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment