[PATCH 21/21] X86_64, UV: Fix incorrect nodes and pnodes for cpuless and memoryless nodes

2016-04-29 Thread Mike Travis
From: Dimitri Sivanich 

This patch fixes the problem of incorrect nodes and pnodes being returned
when referring to nodes that either have no cpus (AKA "headless") or no
memory.

Signed-off-by: Dimitri Sivanich 
Signed-off-by: Mike Travis 
Tested-by: John Estabrook 
Tested-by: Gary Kroening 
Tested-by: Nathan Zimmer 
---
 arch/x86/kernel/apic/x2apic_uv_x.c |   58 +
 1 file changed, 28 insertions(+), 30 deletions(-)

--- linux.orig/arch/x86/kernel/apic/x2apic_uv_x.c
+++ linux/arch/x86/kernel/apic/x2apic_uv_x.c
@@ -1225,8 +1225,7 @@ static void __init decode_uv_systab(void
  */
 static __init void boot_init_possible_blades(struct uv_hub_info_s *hub_info)
 {
-   size_t bytes;
-   int blade, i, j, uv_pb = 0, num_nodes = num_possible_nodes();
+   int i, uv_pb = 0;
 
pr_info("UV: NODE_PRESENT_DEPTH = %d\n", UVH_NODE_PRESENT_TABLE_DEPTH);
for (i = 0; i < UVH_NODE_PRESENT_TABLE_DEPTH; i++) {
@@ -1240,28 +1239,6 @@ static __init void boot_init_possible_bl
}
if (uv_possible_blades != uv_pb)
uv_possible_blades = uv_pb;
-
-   bytes = num_nodes * sizeof(_node_to_pnode[0]);
-   _node_to_pnode = kmalloc(bytes, GFP_KERNEL);
-   BUG_ON(!_node_to_pnode);
-
-   for (blade = 0, i = 0; i < UVH_NODE_PRESENT_TABLE_DEPTH; i++) {
-   unsigned short pnode;
-   unsigned long present =
-   uv_read_local_mmr(UVH_NODE_PRESENT_TABLE + i * 8);
-
-   for (j = 0; j < 64; j++) {
-   if (!test_bit(j, ))
-   continue;
-   pnode = (i * 64 + j) & hub_info->pnode_mask;
-   _node_to_pnode[blade++] = pnode;
-   }
-   if (blade > num_nodes) {
-   pr_err("UV: blade count(%d) exceeds node count(%d)!\n",
-   blade, num_nodes);
-   BUG();
-   }
-   }
 }
 
 static void __init build_socket_tables(void)
@@ -1442,7 +1419,6 @@ void __init uv_system_init(void)
bytes = sizeof(struct uv_hub_info_s);
for_each_node(nodeid) {
struct uv_hub_info_s *new_hub;
-   unsigned short pnode;
 
if (__uv_hub_info_list[nodeid]) {
pr_err("UV: Node %d UV HUB already initialized!?\n",
@@ -1460,10 +1436,11 @@ void __init uv_system_init(void)
BUG_ON(!new_hub);
*new_hub = hub_info;
 
-   pnode = _node_to_pnode[nodeid];
-   min_pnode = min(pnode, min_pnode);
-   max_pnode = max(pnode, max_pnode);
-   new_hub->pnode = pnode;
+   /* Use information from GAM table if available */
+   if (_node_to_pnode)
+   new_hub->pnode = _node_to_pnode[nodeid];
+   else/* Fill in during cpu loop */
+   new_hub->pnode = 0x;
new_hub->numa_blade_id = uv_node_to_blade_id(nodeid);
new_hub->memory_nid = -1;
new_hub->nr_possible_cpus = 0;
@@ -1473,18 +1450,39 @@ void __init uv_system_init(void)
/* Initialize per cpu info */
for_each_possible_cpu(cpu) {
int apicid = per_cpu(x86_cpu_to_apicid, cpu);
+   int numa_node_id;
+   unsigned short pnode;
 
nodeid = cpu_to_node(cpu);
+   numa_node_id = numa_cpu_node(cpu);
+   pnode = uv_apicid_to_pnode(apicid);
+
uv_cpu_info_per(cpu)->p_uv_hub_info = uv_hub_info_list(nodeid);
uv_cpu_info_per(cpu)->blade_cpu_id =
uv_cpu_hub_info(cpu)->nr_possible_cpus++;
if (uv_cpu_hub_info(cpu)->memory_nid == -1)
uv_cpu_hub_info(cpu)->memory_nid = cpu_to_node(cpu);
+   if (nodeid != numa_node_id &&   /* init memoryless node */
+   uv_hub_info_list(numa_node_id)->pnode == 0x)
+   uv_hub_info_list(numa_node_id)->pnode = pnode;
+   else if (uv_cpu_hub_info(cpu)->pnode == 0x)
+   uv_cpu_hub_info(cpu)->pnode = pnode;
uv_cpu_scir_info(cpu)->offset = uv_scir_offset(apicid);
}
 
-   /* Display per node info */
for_each_node(nodeid) {
+   unsigned short pnode = uv_hub_info_list(nodeid)->pnode;
+
+   /* Add pnode info for pre-GAM list nodes without cpus */
+   if (pnode == 0x) {
+   unsigned long paddr;
+
+   paddr = node_start_pfn(nodeid) << PAGE_SHIFT;
+   pnode = uv_gpa_to_pnode(uv_soc_phys_ram_to_gpa(paddr));
+   uv_hub_info_list(nodeid)->pnode = pnode;
+   }
+   min_pnode = min(pnode, 

[PATCH 21/21] X86_64, UV: Fix incorrect nodes and pnodes for cpuless and memoryless nodes

2016-04-29 Thread Mike Travis
From: Dimitri Sivanich 

This patch fixes the problem of incorrect nodes and pnodes being returned
when referring to nodes that either have no cpus (AKA "headless") or no
memory.

Signed-off-by: Dimitri Sivanich 
Signed-off-by: Mike Travis 
Tested-by: John Estabrook 
Tested-by: Gary Kroening 
Tested-by: Nathan Zimmer 
---
 arch/x86/kernel/apic/x2apic_uv_x.c |   58 +
 1 file changed, 28 insertions(+), 30 deletions(-)

--- linux.orig/arch/x86/kernel/apic/x2apic_uv_x.c
+++ linux/arch/x86/kernel/apic/x2apic_uv_x.c
@@ -1225,8 +1225,7 @@ static void __init decode_uv_systab(void
  */
 static __init void boot_init_possible_blades(struct uv_hub_info_s *hub_info)
 {
-   size_t bytes;
-   int blade, i, j, uv_pb = 0, num_nodes = num_possible_nodes();
+   int i, uv_pb = 0;
 
pr_info("UV: NODE_PRESENT_DEPTH = %d\n", UVH_NODE_PRESENT_TABLE_DEPTH);
for (i = 0; i < UVH_NODE_PRESENT_TABLE_DEPTH; i++) {
@@ -1240,28 +1239,6 @@ static __init void boot_init_possible_bl
}
if (uv_possible_blades != uv_pb)
uv_possible_blades = uv_pb;
-
-   bytes = num_nodes * sizeof(_node_to_pnode[0]);
-   _node_to_pnode = kmalloc(bytes, GFP_KERNEL);
-   BUG_ON(!_node_to_pnode);
-
-   for (blade = 0, i = 0; i < UVH_NODE_PRESENT_TABLE_DEPTH; i++) {
-   unsigned short pnode;
-   unsigned long present =
-   uv_read_local_mmr(UVH_NODE_PRESENT_TABLE + i * 8);
-
-   for (j = 0; j < 64; j++) {
-   if (!test_bit(j, ))
-   continue;
-   pnode = (i * 64 + j) & hub_info->pnode_mask;
-   _node_to_pnode[blade++] = pnode;
-   }
-   if (blade > num_nodes) {
-   pr_err("UV: blade count(%d) exceeds node count(%d)!\n",
-   blade, num_nodes);
-   BUG();
-   }
-   }
 }
 
 static void __init build_socket_tables(void)
@@ -1442,7 +1419,6 @@ void __init uv_system_init(void)
bytes = sizeof(struct uv_hub_info_s);
for_each_node(nodeid) {
struct uv_hub_info_s *new_hub;
-   unsigned short pnode;
 
if (__uv_hub_info_list[nodeid]) {
pr_err("UV: Node %d UV HUB already initialized!?\n",
@@ -1460,10 +1436,11 @@ void __init uv_system_init(void)
BUG_ON(!new_hub);
*new_hub = hub_info;
 
-   pnode = _node_to_pnode[nodeid];
-   min_pnode = min(pnode, min_pnode);
-   max_pnode = max(pnode, max_pnode);
-   new_hub->pnode = pnode;
+   /* Use information from GAM table if available */
+   if (_node_to_pnode)
+   new_hub->pnode = _node_to_pnode[nodeid];
+   else/* Fill in during cpu loop */
+   new_hub->pnode = 0x;
new_hub->numa_blade_id = uv_node_to_blade_id(nodeid);
new_hub->memory_nid = -1;
new_hub->nr_possible_cpus = 0;
@@ -1473,18 +1450,39 @@ void __init uv_system_init(void)
/* Initialize per cpu info */
for_each_possible_cpu(cpu) {
int apicid = per_cpu(x86_cpu_to_apicid, cpu);
+   int numa_node_id;
+   unsigned short pnode;
 
nodeid = cpu_to_node(cpu);
+   numa_node_id = numa_cpu_node(cpu);
+   pnode = uv_apicid_to_pnode(apicid);
+
uv_cpu_info_per(cpu)->p_uv_hub_info = uv_hub_info_list(nodeid);
uv_cpu_info_per(cpu)->blade_cpu_id =
uv_cpu_hub_info(cpu)->nr_possible_cpus++;
if (uv_cpu_hub_info(cpu)->memory_nid == -1)
uv_cpu_hub_info(cpu)->memory_nid = cpu_to_node(cpu);
+   if (nodeid != numa_node_id &&   /* init memoryless node */
+   uv_hub_info_list(numa_node_id)->pnode == 0x)
+   uv_hub_info_list(numa_node_id)->pnode = pnode;
+   else if (uv_cpu_hub_info(cpu)->pnode == 0x)
+   uv_cpu_hub_info(cpu)->pnode = pnode;
uv_cpu_scir_info(cpu)->offset = uv_scir_offset(apicid);
}
 
-   /* Display per node info */
for_each_node(nodeid) {
+   unsigned short pnode = uv_hub_info_list(nodeid)->pnode;
+
+   /* Add pnode info for pre-GAM list nodes without cpus */
+   if (pnode == 0x) {
+   unsigned long paddr;
+
+   paddr = node_start_pfn(nodeid) << PAGE_SHIFT;
+   pnode = uv_gpa_to_pnode(uv_soc_phys_ram_to_gpa(paddr));
+   uv_hub_info_list(nodeid)->pnode = pnode;
+   }
+   min_pnode = min(pnode, min_pnode);
+   max_pnode = max(pnode, max_pnode);
pr_info("UV: UVHUB 

[PATCH 21/21] X86_64, UV: Fix incorrect nodes and pnodes for cpuless and memoryless nodes

2016-04-28 Thread Mike Travis
From: Dimitri Sivanich 

This patch fixes the problem of incorrect nodes and pnodes being returned
when referring to nodes that either have no cpus (AKA "headless") or no
memory.

Signed-off-by: Dimitri Sivanich 
Signed-off-by: Mike Travis 
Tested-by: John Estabrook 
Tested-by: Gary Kroening 
Tested-by: Nathan Zimmer 
---
 arch/x86/kernel/apic/x2apic_uv_x.c |   58 +
 1 file changed, 28 insertions(+), 30 deletions(-)

--- linux.orig/arch/x86/kernel/apic/x2apic_uv_x.c
+++ linux/arch/x86/kernel/apic/x2apic_uv_x.c
@@ -1252,8 +1252,7 @@ static void __init decode_uv_systab(void
  */
 static __init void boot_init_possible_blades(struct uv_hub_info_s *hub_info)
 {
-   size_t bytes;
-   int blade, i, j, uv_pb = 0, num_nodes = num_possible_nodes();
+   int i, uv_pb = 0;
 
pr_info("UV: NODE_PRESENT_DEPTH = %d\n", UVH_NODE_PRESENT_TABLE_DEPTH);
for (i = 0; i < UVH_NODE_PRESENT_TABLE_DEPTH; i++) {
@@ -1267,28 +1266,6 @@ static __init void boot_init_possible_bl
}
if (uv_possible_blades != uv_pb)
uv_possible_blades = uv_pb;
-
-   bytes = num_nodes * sizeof(_node_to_pnode[0]);
-   _node_to_pnode = kmalloc(bytes, GFP_KERNEL);
-   BUG_ON(!_node_to_pnode);
-
-   for (blade = 0, i = 0; i < UVH_NODE_PRESENT_TABLE_DEPTH; i++) {
-   unsigned short pnode;
-   unsigned long present =
-   uv_read_local_mmr(UVH_NODE_PRESENT_TABLE + i * 8);
-
-   for (j = 0; j < 64; j++) {
-   if (!test_bit(j, ))
-   continue;
-   pnode = (i * 64 + j) & hub_info->pnode_mask;
-   _node_to_pnode[blade++] = pnode;
-   }
-   if (blade > num_nodes) {
-   pr_err("UV: blade count(%d) exceeds node count(%d)!\n",
-   blade, num_nodes);
-   BUG();
-   }
-   }
 }
 
 static void __init build_socket_tables(void)
@@ -1469,7 +1446,6 @@ void __init uv_system_init(void)
bytes = sizeof(struct uv_hub_info_s);
for_each_node(nodeid) {
struct uv_hub_info_s *new_hub;
-   unsigned short pnode;
 
if (__uv_hub_info_list[nodeid]) {
pr_err("UV: Node %d UV HUB already initialized!?\n",
@@ -1487,10 +1463,11 @@ void __init uv_system_init(void)
BUG_ON(!new_hub);
*new_hub = hub_info;
 
-   pnode = _node_to_pnode[nodeid];
-   min_pnode = min(pnode, min_pnode);
-   max_pnode = max(pnode, max_pnode);
-   new_hub->pnode = pnode;
+   /* Use information from GAM table if available */
+   if (_node_to_pnode)
+   new_hub->pnode = _node_to_pnode[nodeid];
+   else/* Fill in during cpu loop */
+   new_hub->pnode = 0x;
new_hub->numa_blade_id = uv_node_to_blade_id(nodeid);
new_hub->memory_nid = -1;
new_hub->nr_possible_cpus = 0;
@@ -1500,18 +1477,39 @@ void __init uv_system_init(void)
/* Initialize per cpu info */
for_each_possible_cpu(cpu) {
int apicid = per_cpu(x86_cpu_to_apicid, cpu);
+   int numa_node_id;
+   unsigned short pnode;
 
nodeid = cpu_to_node(cpu);
+   numa_node_id = numa_cpu_node(cpu);
+   pnode = uv_apicid_to_pnode(apicid);
+
uv_cpu_info_per(cpu)->p_uv_hub_info = uv_hub_info_list(nodeid);
uv_cpu_info_per(cpu)->blade_cpu_id =
uv_cpu_hub_info(cpu)->nr_possible_cpus++;
if (uv_cpu_hub_info(cpu)->memory_nid == -1)
uv_cpu_hub_info(cpu)->memory_nid = cpu_to_node(cpu);
+   if (nodeid != numa_node_id &&   /* init memoryless node */
+   uv_hub_info_list(numa_node_id)->pnode == 0x)
+   uv_hub_info_list(numa_node_id)->pnode = pnode;
+   else if (uv_cpu_hub_info(cpu)->pnode == 0x)
+   uv_cpu_hub_info(cpu)->pnode = pnode;
uv_cpu_scir_info(cpu)->offset = uv_scir_offset(apicid);
}
 
-   /* Display per node info */
for_each_node(nodeid) {
+   unsigned short pnode = uv_hub_info_list(nodeid)->pnode;
+
+   /* Add pnode info for pre-GAM list nodes without cpus */
+   if (pnode == 0x) {
+   unsigned long paddr;
+
+   paddr = node_start_pfn(nodeid) << PAGE_SHIFT;
+   pnode = uv_gpa_to_pnode(uv_soc_phys_ram_to_gpa(paddr));
+   uv_hub_info_list(nodeid)->pnode = pnode;
+   }
+   min_pnode = min(pnode, 

[PATCH 21/21] X86_64, UV: Fix incorrect nodes and pnodes for cpuless and memoryless nodes

2016-04-28 Thread Mike Travis
From: Dimitri Sivanich 

This patch fixes the problem of incorrect nodes and pnodes being returned
when referring to nodes that either have no cpus (AKA "headless") or no
memory.

Signed-off-by: Dimitri Sivanich 
Signed-off-by: Mike Travis 
Tested-by: John Estabrook 
Tested-by: Gary Kroening 
Tested-by: Nathan Zimmer 
---
 arch/x86/kernel/apic/x2apic_uv_x.c |   58 +
 1 file changed, 28 insertions(+), 30 deletions(-)

--- linux.orig/arch/x86/kernel/apic/x2apic_uv_x.c
+++ linux/arch/x86/kernel/apic/x2apic_uv_x.c
@@ -1252,8 +1252,7 @@ static void __init decode_uv_systab(void
  */
 static __init void boot_init_possible_blades(struct uv_hub_info_s *hub_info)
 {
-   size_t bytes;
-   int blade, i, j, uv_pb = 0, num_nodes = num_possible_nodes();
+   int i, uv_pb = 0;
 
pr_info("UV: NODE_PRESENT_DEPTH = %d\n", UVH_NODE_PRESENT_TABLE_DEPTH);
for (i = 0; i < UVH_NODE_PRESENT_TABLE_DEPTH; i++) {
@@ -1267,28 +1266,6 @@ static __init void boot_init_possible_bl
}
if (uv_possible_blades != uv_pb)
uv_possible_blades = uv_pb;
-
-   bytes = num_nodes * sizeof(_node_to_pnode[0]);
-   _node_to_pnode = kmalloc(bytes, GFP_KERNEL);
-   BUG_ON(!_node_to_pnode);
-
-   for (blade = 0, i = 0; i < UVH_NODE_PRESENT_TABLE_DEPTH; i++) {
-   unsigned short pnode;
-   unsigned long present =
-   uv_read_local_mmr(UVH_NODE_PRESENT_TABLE + i * 8);
-
-   for (j = 0; j < 64; j++) {
-   if (!test_bit(j, ))
-   continue;
-   pnode = (i * 64 + j) & hub_info->pnode_mask;
-   _node_to_pnode[blade++] = pnode;
-   }
-   if (blade > num_nodes) {
-   pr_err("UV: blade count(%d) exceeds node count(%d)!\n",
-   blade, num_nodes);
-   BUG();
-   }
-   }
 }
 
 static void __init build_socket_tables(void)
@@ -1469,7 +1446,6 @@ void __init uv_system_init(void)
bytes = sizeof(struct uv_hub_info_s);
for_each_node(nodeid) {
struct uv_hub_info_s *new_hub;
-   unsigned short pnode;
 
if (__uv_hub_info_list[nodeid]) {
pr_err("UV: Node %d UV HUB already initialized!?\n",
@@ -1487,10 +1463,11 @@ void __init uv_system_init(void)
BUG_ON(!new_hub);
*new_hub = hub_info;
 
-   pnode = _node_to_pnode[nodeid];
-   min_pnode = min(pnode, min_pnode);
-   max_pnode = max(pnode, max_pnode);
-   new_hub->pnode = pnode;
+   /* Use information from GAM table if available */
+   if (_node_to_pnode)
+   new_hub->pnode = _node_to_pnode[nodeid];
+   else/* Fill in during cpu loop */
+   new_hub->pnode = 0x;
new_hub->numa_blade_id = uv_node_to_blade_id(nodeid);
new_hub->memory_nid = -1;
new_hub->nr_possible_cpus = 0;
@@ -1500,18 +1477,39 @@ void __init uv_system_init(void)
/* Initialize per cpu info */
for_each_possible_cpu(cpu) {
int apicid = per_cpu(x86_cpu_to_apicid, cpu);
+   int numa_node_id;
+   unsigned short pnode;
 
nodeid = cpu_to_node(cpu);
+   numa_node_id = numa_cpu_node(cpu);
+   pnode = uv_apicid_to_pnode(apicid);
+
uv_cpu_info_per(cpu)->p_uv_hub_info = uv_hub_info_list(nodeid);
uv_cpu_info_per(cpu)->blade_cpu_id =
uv_cpu_hub_info(cpu)->nr_possible_cpus++;
if (uv_cpu_hub_info(cpu)->memory_nid == -1)
uv_cpu_hub_info(cpu)->memory_nid = cpu_to_node(cpu);
+   if (nodeid != numa_node_id &&   /* init memoryless node */
+   uv_hub_info_list(numa_node_id)->pnode == 0x)
+   uv_hub_info_list(numa_node_id)->pnode = pnode;
+   else if (uv_cpu_hub_info(cpu)->pnode == 0x)
+   uv_cpu_hub_info(cpu)->pnode = pnode;
uv_cpu_scir_info(cpu)->offset = uv_scir_offset(apicid);
}
 
-   /* Display per node info */
for_each_node(nodeid) {
+   unsigned short pnode = uv_hub_info_list(nodeid)->pnode;
+
+   /* Add pnode info for pre-GAM list nodes without cpus */
+   if (pnode == 0x) {
+   unsigned long paddr;
+
+   paddr = node_start_pfn(nodeid) << PAGE_SHIFT;
+   pnode = uv_gpa_to_pnode(uv_soc_phys_ram_to_gpa(paddr));
+   uv_hub_info_list(nodeid)->pnode = pnode;
+   }
+   min_pnode = min(pnode, min_pnode);
+   max_pnode = max(pnode, max_pnode);
pr_info("UV: UVHUB 

[PATCH 21/21] X86_64, UV: Fix incorrect nodes and pnodes for cpuless and memoryless nodes

2016-04-27 Thread Mike Travis
From: Dimitri Sivanich 

This patch fixes the problem of incorrect nodes and pnodes being returned when
referring to nodes that either have no cpus (AKA "headless") or no memory.

Signed-off-by: Dimitri Sivanich 
Signed-off-by: Mike Travis 
Tested-by: John Estabrook 
Tested-by: Gary Kroening 
---
 arch/x86/kernel/apic/x2apic_uv_x.c |   57 +
 1 file changed, 27 insertions(+), 30 deletions(-)

--- linux.orig/arch/x86/kernel/apic/x2apic_uv_x.c
+++ linux/arch/x86/kernel/apic/x2apic_uv_x.c
@@ -1192,8 +1192,7 @@ static void __init decode_uv_systab(void
  */
 static __init void boot_init_possible_blades(struct uv_hub_info_s *hub_info)
 {
-   size_t bytes;
-   int blade, i, j, uv_pb = 0, num_nodes = num_possible_nodes();
+   int i, uv_pb = 0;
 
pr_info("UV: NODE_PRESENT_DEPTH = %d\n", UVH_NODE_PRESENT_TABLE_DEPTH);
for (i = 0; i < UVH_NODE_PRESENT_TABLE_DEPTH; i++) {
@@ -1207,28 +1206,6 @@ static __init void boot_init_possible_bl
}
if (uv_possible_blades != uv_pb)
uv_possible_blades = uv_pb;
-
-   bytes = num_nodes * sizeof(_node_to_pnode[0]);
-   _node_to_pnode = kmalloc(bytes, GFP_KERNEL);
-   BUG_ON(!_node_to_pnode);
-
-   for (blade = 0, i = 0; i < UVH_NODE_PRESENT_TABLE_DEPTH; i++) {
-   unsigned short pnode;
-   unsigned long present =
-   uv_read_local_mmr(UVH_NODE_PRESENT_TABLE + i * 8);
-
-   for (j = 0; j < 64; j++) {
-   if (!test_bit(j, ))
-   continue;
-   pnode = (i * 64 + j) & hub_info->pnode_mask;
-   _node_to_pnode[blade++] = pnode;
-   }
-   if (blade > num_nodes) {
-   pr_err("UV: blade count(%d) exceeds node count(%d)!\n",
-   blade, num_nodes);
-   BUG();
-   }
-   }
 }
 
 static void __init build_socket_tables(void)
@@ -1409,7 +1386,6 @@ void __init uv_system_init(void)
bytes = sizeof(struct uv_hub_info_s);
for_each_node(nodeid) {
struct uv_hub_info_s *new_hub;
-   unsigned short pnode;
 
if (__uv_hub_info_list[nodeid]) {
pr_err("UV: Node %d UV HUB already initialized!?\n",
@@ -1427,10 +1403,11 @@ void __init uv_system_init(void)
BUG_ON(!new_hub);
*new_hub = hub_info;
 
-   pnode = _node_to_pnode[nodeid];
-   min_pnode = min(pnode, min_pnode);
-   max_pnode = max(pnode, max_pnode);
-   new_hub->pnode = pnode;
+   /* Use information from GAM table if available */
+   if (_node_to_pnode)
+   new_hub->pnode = _node_to_pnode[nodeid];
+   else/* Fill in during cpu loop */
+   new_hub->pnode = 0x;
new_hub->numa_blade_id = uv_node_to_blade_id(nodeid);
new_hub->memory_nid = -1;
new_hub->nr_possible_cpus = 0;
@@ -1440,18 +1417,38 @@ void __init uv_system_init(void)
/* Initialize per cpu info */
for_each_possible_cpu(cpu) {
int apicid = per_cpu(x86_cpu_to_apicid, cpu);
+   int numa_node_id;
+   unsigned short pnode;
 
nodeid = cpu_to_node(cpu);
+   numa_node_id = numa_cpu_node(cpu);
+   pnode = uv_apicid_to_pnode(apicid);
+
uv_cpu_info_per(cpu)->p_uv_hub_info = uv_hub_info_list(nodeid);
uv_cpu_info_per(cpu)->blade_cpu_id =
uv_cpu_hub_info(cpu)->nr_possible_cpus++;
if (uv_cpu_hub_info(cpu)->memory_nid == -1)
uv_cpu_hub_info(cpu)->memory_nid = cpu_to_node(cpu);
+   if (nodeid != numa_node_id &&   /* init memoryless node */
+   uv_hub_info_list(numa_node_id)->pnode == 0x)
+   uv_hub_info_list(numa_node_id)->pnode = pnode;
+   else if (uv_cpu_hub_info(cpu)->pnode == 0x)
+   uv_cpu_hub_info(cpu)->pnode = pnode;
uv_cpu_scir_info(cpu)->offset = uv_scir_offset(apicid);
}
 
-   /* Display per node info */
for_each_node(nodeid) {
+   unsigned short pnode = uv_hub_info_list(nodeid)->pnode;
+
+   /* Add pnode info for pre-GAM list nodes without cpus */
+   if (pnode == 0x) {
+   unsigned long paddr;
+   paddr = node_start_pfn(nodeid) << PAGE_SHIFT;
+   pnode = uv_gpa_to_pnode(uv_soc_phys_ram_to_gpa(paddr));
+   uv_hub_info_list(nodeid)->pnode = pnode;
+   }
+   min_pnode = min(pnode, min_pnode);
+   max_pnode = 

[PATCH 21/21] X86_64, UV: Fix incorrect nodes and pnodes for cpuless and memoryless nodes

2016-04-27 Thread Mike Travis
From: Dimitri Sivanich 

This patch fixes the problem of incorrect nodes and pnodes being returned when
referring to nodes that either have no cpus (AKA "headless") or no memory.

Signed-off-by: Dimitri Sivanich 
Signed-off-by: Mike Travis 
Tested-by: John Estabrook 
Tested-by: Gary Kroening 
---
 arch/x86/kernel/apic/x2apic_uv_x.c |   57 +
 1 file changed, 27 insertions(+), 30 deletions(-)

--- linux.orig/arch/x86/kernel/apic/x2apic_uv_x.c
+++ linux/arch/x86/kernel/apic/x2apic_uv_x.c
@@ -1192,8 +1192,7 @@ static void __init decode_uv_systab(void
  */
 static __init void boot_init_possible_blades(struct uv_hub_info_s *hub_info)
 {
-   size_t bytes;
-   int blade, i, j, uv_pb = 0, num_nodes = num_possible_nodes();
+   int i, uv_pb = 0;
 
pr_info("UV: NODE_PRESENT_DEPTH = %d\n", UVH_NODE_PRESENT_TABLE_DEPTH);
for (i = 0; i < UVH_NODE_PRESENT_TABLE_DEPTH; i++) {
@@ -1207,28 +1206,6 @@ static __init void boot_init_possible_bl
}
if (uv_possible_blades != uv_pb)
uv_possible_blades = uv_pb;
-
-   bytes = num_nodes * sizeof(_node_to_pnode[0]);
-   _node_to_pnode = kmalloc(bytes, GFP_KERNEL);
-   BUG_ON(!_node_to_pnode);
-
-   for (blade = 0, i = 0; i < UVH_NODE_PRESENT_TABLE_DEPTH; i++) {
-   unsigned short pnode;
-   unsigned long present =
-   uv_read_local_mmr(UVH_NODE_PRESENT_TABLE + i * 8);
-
-   for (j = 0; j < 64; j++) {
-   if (!test_bit(j, ))
-   continue;
-   pnode = (i * 64 + j) & hub_info->pnode_mask;
-   _node_to_pnode[blade++] = pnode;
-   }
-   if (blade > num_nodes) {
-   pr_err("UV: blade count(%d) exceeds node count(%d)!\n",
-   blade, num_nodes);
-   BUG();
-   }
-   }
 }
 
 static void __init build_socket_tables(void)
@@ -1409,7 +1386,6 @@ void __init uv_system_init(void)
bytes = sizeof(struct uv_hub_info_s);
for_each_node(nodeid) {
struct uv_hub_info_s *new_hub;
-   unsigned short pnode;
 
if (__uv_hub_info_list[nodeid]) {
pr_err("UV: Node %d UV HUB already initialized!?\n",
@@ -1427,10 +1403,11 @@ void __init uv_system_init(void)
BUG_ON(!new_hub);
*new_hub = hub_info;
 
-   pnode = _node_to_pnode[nodeid];
-   min_pnode = min(pnode, min_pnode);
-   max_pnode = max(pnode, max_pnode);
-   new_hub->pnode = pnode;
+   /* Use information from GAM table if available */
+   if (_node_to_pnode)
+   new_hub->pnode = _node_to_pnode[nodeid];
+   else/* Fill in during cpu loop */
+   new_hub->pnode = 0x;
new_hub->numa_blade_id = uv_node_to_blade_id(nodeid);
new_hub->memory_nid = -1;
new_hub->nr_possible_cpus = 0;
@@ -1440,18 +1417,38 @@ void __init uv_system_init(void)
/* Initialize per cpu info */
for_each_possible_cpu(cpu) {
int apicid = per_cpu(x86_cpu_to_apicid, cpu);
+   int numa_node_id;
+   unsigned short pnode;
 
nodeid = cpu_to_node(cpu);
+   numa_node_id = numa_cpu_node(cpu);
+   pnode = uv_apicid_to_pnode(apicid);
+
uv_cpu_info_per(cpu)->p_uv_hub_info = uv_hub_info_list(nodeid);
uv_cpu_info_per(cpu)->blade_cpu_id =
uv_cpu_hub_info(cpu)->nr_possible_cpus++;
if (uv_cpu_hub_info(cpu)->memory_nid == -1)
uv_cpu_hub_info(cpu)->memory_nid = cpu_to_node(cpu);
+   if (nodeid != numa_node_id &&   /* init memoryless node */
+   uv_hub_info_list(numa_node_id)->pnode == 0x)
+   uv_hub_info_list(numa_node_id)->pnode = pnode;
+   else if (uv_cpu_hub_info(cpu)->pnode == 0x)
+   uv_cpu_hub_info(cpu)->pnode = pnode;
uv_cpu_scir_info(cpu)->offset = uv_scir_offset(apicid);
}
 
-   /* Display per node info */
for_each_node(nodeid) {
+   unsigned short pnode = uv_hub_info_list(nodeid)->pnode;
+
+   /* Add pnode info for pre-GAM list nodes without cpus */
+   if (pnode == 0x) {
+   unsigned long paddr;
+   paddr = node_start_pfn(nodeid) << PAGE_SHIFT;
+   pnode = uv_gpa_to_pnode(uv_soc_phys_ram_to_gpa(paddr));
+   uv_hub_info_list(nodeid)->pnode = pnode;
+   }
+   min_pnode = min(pnode, min_pnode);
+   max_pnode = max(pnode, max_pnode);
pr_info("UV: UVHUB node:%2d pn:%02x nrcpus:%d\n",