Merge branch 'x86/urgent' into x86/platform, to resolve conflict
authorIngo Molnar <mingo@kernel.org>
Thu, 5 May 2016 07:56:10 +0000 (09:56 +0200)
committerIngo Molnar <mingo@kernel.org>
Thu, 5 May 2016 07:56:10 +0000 (09:56 +0200)
Conflicts:
arch/x86/kernel/apic/x2apic_uv_x.c

Signed-off-by: Ingo Molnar <mingo@kernel.org>
1  2 
arch/x86/kernel/apic/x2apic_uv_x.c

index aa6de8bad360b5a85c9f6e4dbd7a1298ffbf3a22,d7ce96a7dacaedc630a0e9a5ef5b761edbd0ffc7..29003154fafd2277e4e014f5c1ac019f1a446c32
@@@ -1035,468 -871,146 +1035,466 @@@ void uv_cpu_init(void
                set_x2apic_extra_bits(uv_hub_info->pnode);
  }
  
 -void __init uv_system_init(void)
 +struct mn {
 +      unsigned char   m_val;
 +      unsigned char   n_val;
 +      unsigned char   m_shift;
 +      unsigned char   n_lshift;
 +};
 +
 +static void get_mn(struct mn *mnp)
  {
 -      union uvh_rh_gam_config_mmr_u  m_n_config;
 -      union uvh_node_id_u node_id;
 -      unsigned long gnode_upper, lowmem_redir_base, lowmem_redir_size;
 -      int bytes, nid, cpu, lcpu, pnode, blade, i, j, m_val, n_val;
 -      int gnode_extra, min_pnode = 999999, max_pnode = -1;
 -      unsigned long mmr_base, present, paddr;
 -      unsigned short pnode_mask;
 -      unsigned char n_lshift;
 -      char *hub = (is_uv1_hub() ? "UV100/1000" :
 -                  (is_uv2_hub() ? "UV2000/3000" :
 -                  (is_uv3_hub() ? "UV300" : NULL)));
 +      union uvh_rh_gam_config_mmr_u m_n_config;
 +      union uv3h_gr0_gam_gr_config_u m_gr_config;
  
 -      if (!hub) {
 -              pr_err("UV: Unknown/unsupported UV hub\n");
 -              return;
 +      m_n_config.v = uv_read_local_mmr(UVH_RH_GAM_CONFIG_MMR);
 +      mnp->n_val = m_n_config.s.n_skt;
 +      if (is_uv4_hub()) {
 +              mnp->m_val = 0;
 +              mnp->n_lshift = 0;
 +      } else if (is_uv3_hub()) {
 +              mnp->m_val = m_n_config.s3.m_skt;
 +              m_gr_config.v = uv_read_local_mmr(UV3H_GR0_GAM_GR_CONFIG);
 +              mnp->n_lshift = m_gr_config.s3.m_skt;
 +      } else if (is_uv2_hub()) {
 +              mnp->m_val = m_n_config.s2.m_skt;
 +              mnp->n_lshift = mnp->m_val == 40 ? 40 : 39;
 +      } else if (is_uv1_hub()) {
 +              mnp->m_val = m_n_config.s1.m_skt;
 +              mnp->n_lshift = mnp->m_val;
        }
 -      pr_info("UV: Found %s hub\n", hub);
 +      mnp->m_shift = mnp->m_val ? 64 - mnp->m_val : 0;
 +}
  
 -      map_low_mmrs();
 +void __init uv_init_hub_info(struct uv_hub_info_s *hub_info)
 +{
 +      struct mn mn = {0};     /* avoid unitialized warnings */
 +      union uvh_node_id_u node_id;
  
 -      m_n_config.v = uv_read_local_mmr(UVH_RH_GAM_CONFIG_MMR );
 -      m_val = m_n_config.s.m_skt;
 -      n_val = m_n_config.s.n_skt;
 -      pnode_mask = (1 << n_val) - 1;
 -      n_lshift = get_n_lshift(m_val);
 -      mmr_base =
 -          uv_read_local_mmr(UVH_RH_GAM_MMR_OVERLAY_CONFIG_MMR) &
 -          ~UV_MMR_ENABLE;
 +      get_mn(&mn);
 +      hub_info->m_val = mn.m_val;
 +      hub_info->n_val = mn.n_val;
 +      hub_info->m_shift = mn.m_shift;
 +      hub_info->n_lshift = mn.n_lshift ? mn.n_lshift : 0;
 +
 +      hub_info->hub_revision = uv_hub_info->hub_revision;
 +      hub_info->pnode_mask = uv_cpuid.pnode_mask;
 +      hub_info->min_pnode = _min_pnode;
 +      hub_info->min_socket = _min_socket;
 +      hub_info->pnode_to_socket = _pnode_to_socket;
 +      hub_info->socket_to_node = _socket_to_node;
 +      hub_info->socket_to_pnode = _socket_to_pnode;
 +      hub_info->gr_table_len = _gr_table_len;
 +      hub_info->gr_table = _gr_table;
 +      hub_info->gpa_mask = mn.m_val ?
 +              (1UL << (mn.m_val + mn.n_val)) - 1 :
 +              (1UL << uv_cpuid.gpa_shift) - 1;
  
        node_id.v = uv_read_local_mmr(UVH_NODE_ID);
 -      gnode_extra = (node_id.s.node_id & ~((1 << n_val) - 1)) >> 1;
 -      gnode_upper = ((unsigned long)gnode_extra  << m_val);
 -      pr_info("UV: N:%d M:%d pnode_mask:0x%x gnode_upper/extra:0x%lx/0x%x n_lshift 0x%x\n",
 -                      n_val, m_val, pnode_mask, gnode_upper, gnode_extra,
 -                      n_lshift);
 +      hub_info->gnode_extra =
 +              (node_id.s.node_id & ~((1 << mn.n_val) - 1)) >> 1;
 +
 +      hub_info->gnode_upper =
 +              ((unsigned long)hub_info->gnode_extra << mn.m_val);
 +
 +      if (uv_gp_table) {
 +              hub_info->global_mmr_base = uv_gp_table->mmr_base;
 +              hub_info->global_mmr_shift = uv_gp_table->mmr_shift;
 +              hub_info->global_gru_base = uv_gp_table->gru_base;
 +              hub_info->global_gru_shift = uv_gp_table->gru_shift;
 +              hub_info->gpa_shift = uv_gp_table->gpa_shift;
 +              hub_info->gpa_mask = (1UL << hub_info->gpa_shift) - 1;
 +      } else {
 +              hub_info->global_mmr_base =
 +                      uv_read_local_mmr(UVH_RH_GAM_MMR_OVERLAY_CONFIG_MMR) &
 +                                      ~UV_MMR_ENABLE;
 +              hub_info->global_mmr_shift = _UV_GLOBAL_MMR64_PNODE_SHIFT;
 +      }
  
 -      pr_info("UV: global MMR base 0x%lx\n", mmr_base);
 +      get_lowmem_redirect(
 +              &hub_info->lowmem_remap_base, &hub_info->lowmem_remap_top);
  
 -      for(i = 0; i < UVH_NODE_PRESENT_TABLE_DEPTH; i++)
 -              uv_possible_blades +=
 -                hweight64(uv_read_local_mmr( UVH_NODE_PRESENT_TABLE + i * 8));
 +      hub_info->apic_pnode_shift = uv_cpuid.socketid_shift;
  
 -      /* uv_num_possible_blades() is really the hub count */
 -      pr_info("UV: Found %d blades, %d hubs\n",
 -                      is_uv1_hub() ? uv_num_possible_blades() :
 -                      (uv_num_possible_blades() + 1) / 2,
 -                      uv_num_possible_blades());
 +      /* show system specific info */
 +      pr_info("UV: N:%d M:%d m_shift:%d n_lshift:%d\n",
 +              hub_info->n_val, hub_info->m_val,
 +              hub_info->m_shift, hub_info->n_lshift);
  
 -      bytes = sizeof(struct uv_blade_info) * uv_num_possible_blades();
 -      uv_blade_info = kzalloc(bytes, GFP_KERNEL);
 -      BUG_ON(!uv_blade_info);
 +      pr_info("UV: gpa_mask/shift:0x%lx/%d pnode_mask:0x%x apic_pns:%d\n",
 +              hub_info->gpa_mask, hub_info->gpa_shift,
 +              hub_info->pnode_mask, hub_info->apic_pnode_shift);
  
 -      for (blade = 0; blade < uv_num_possible_blades(); blade++)
 -              uv_blade_info[blade].memory_nid = -1;
 +      pr_info("UV: mmr_base/shift:0x%lx/%ld gru_base/shift:0x%lx/%ld\n",
 +              hub_info->global_mmr_base, hub_info->global_mmr_shift,
 +              hub_info->global_gru_base, hub_info->global_gru_shift);
  
 -      get_lowmem_redirect(&lowmem_redir_base, &lowmem_redir_size);
 +      pr_info("UV: gnode_upper:0x%lx gnode_extra:0x%x\n",
 +              hub_info->gnode_upper, hub_info->gnode_extra);
 +}
 +
 +static void __init decode_gam_params(unsigned long ptr)
 +{
 +      uv_gp_table = (struct uv_gam_parameters *)ptr;
 +
 +      pr_info("UV: GAM Params...\n");
 +      pr_info("UV: mmr_base/shift:0x%llx/%d gru_base/shift:0x%llx/%d gpa_shift:%d\n",
 +              uv_gp_table->mmr_base, uv_gp_table->mmr_shift,
 +              uv_gp_table->gru_base, uv_gp_table->gru_shift,
 +              uv_gp_table->gpa_shift);
 +}
 +
 +static void __init decode_gam_rng_tbl(unsigned long ptr)
 +{
 +      struct uv_gam_range_entry *gre = (struct uv_gam_range_entry *)ptr;
 +      unsigned long lgre = 0;
 +      int index = 0;
 +      int sock_min = 999999, pnode_min = 99999;
 +      int sock_max = -1, pnode_max = -1;
 +
 +      uv_gre_table = gre;
 +      for (; gre->type != UV_GAM_RANGE_TYPE_UNUSED; gre++) {
 +              if (!index) {
 +                      pr_info("UV: GAM Range Table...\n");
 +                      pr_info("UV:  # %20s %14s %5s %4s %5s %3s %2s %3s\n",
 +                              "Range", "", "Size", "Type", "NASID",
 +                              "SID", "PN", "PXM");
 +              }
 +              pr_info(
 +              "UV: %2d: 0x%014lx-0x%014lx %5luG %3d   %04x  %02x %02x %3d\n",
 +                      index++,
 +                      (unsigned long)lgre << UV_GAM_RANGE_SHFT,
 +                      (unsigned long)gre->limit << UV_GAM_RANGE_SHFT,
 +                      ((unsigned long)(gre->limit - lgre)) >>
 +                              (30 - UV_GAM_RANGE_SHFT), /* 64M -> 1G */
 +                      gre->type, gre->nasid, gre->sockid,
 +                      gre->pnode, gre->pxm);
 +
 +              lgre = gre->limit;
 +              if (sock_min > gre->sockid)
 +                      sock_min = gre->sockid;
 +              if (sock_max < gre->sockid)
 +                      sock_max = gre->sockid;
 +              if (pnode_min > gre->pnode)
 +                      pnode_min = gre->pnode;
 +              if (pnode_max < gre->pnode)
 +                      pnode_max = gre->pnode;
 +      }
 +      _min_socket = sock_min;
 +      _max_socket = sock_max;
 +      _min_pnode = pnode_min;
 +      _max_pnode = pnode_max;
 +      _gr_table_len = index;
 +      pr_info(
 +      "UV: GRT: %d entries, sockets(min:%x,max:%x) pnodes(min:%x,max:%x)\n",
 +              index, _min_socket, _max_socket, _min_pnode, _max_pnode);
 +}
 +
 +static void __init decode_uv_systab(void)
 +{
 +      struct uv_systab *st;
 +      int i;
 +
 +      st = uv_systab;
 +      if ((!st || st->revision < UV_SYSTAB_VERSION_UV4) && !is_uv4_hub())
 +              return;
 +      if (st->revision != UV_SYSTAB_VERSION_UV4_LATEST) {
 +              pr_crit(
 +              "UV: BIOS UVsystab version(%x) mismatch, expecting(%x)\n",
 +                      st->revision, UV_SYSTAB_VERSION_UV4_LATEST);
 +              BUG();
 +      }
 +
 +      for (i = 0; st->entry[i].type != UV_SYSTAB_TYPE_UNUSED; i++) {
 +              unsigned long ptr = st->entry[i].offset;
  
 -      bytes = sizeof(uv_node_to_blade[0]) * num_possible_nodes();
 -      uv_node_to_blade = kmalloc(bytes, GFP_KERNEL);
 -      BUG_ON(!uv_node_to_blade);
 -      memset(uv_node_to_blade, 255, bytes);
 +              if (!ptr)
 +                      continue;
 +
 +              ptr = ptr + (unsigned long)st;
 +
 +              switch (st->entry[i].type) {
 +              case UV_SYSTAB_TYPE_GAM_PARAMS:
 +                      decode_gam_params(ptr);
 +                      break;
 +
 +              case UV_SYSTAB_TYPE_GAM_RNG_TBL:
 +                      decode_gam_rng_tbl(ptr);
 +                      break;
 +              }
 +      }
 +}
  
 -      bytes = sizeof(uv_cpu_to_blade[0]) * num_possible_cpus();
 -      uv_cpu_to_blade = kmalloc(bytes, GFP_KERNEL);
 -      BUG_ON(!uv_cpu_to_blade);
 -      memset(uv_cpu_to_blade, 255, bytes);
 +/*
 + * Setup physical blade translations from UVH_NODE_PRESENT_TABLE
 + * .. NB: UVH_NODE_PRESENT_TABLE is going away,
 + * .. being replaced by GAM Range Table
 + */
 +static __init void boot_init_possible_blades(struct uv_hub_info_s *hub_info)
 +{
 +      int i, uv_pb = 0;
  
 -      blade = 0;
 +      pr_info("UV: NODE_PRESENT_DEPTH = %d\n", UVH_NODE_PRESENT_TABLE_DEPTH);
        for (i = 0; i < UVH_NODE_PRESENT_TABLE_DEPTH; i++) {
 -              present = uv_read_local_mmr(UVH_NODE_PRESENT_TABLE + i * 8);
 -              for (j = 0; j < 64; j++) {
 -                      if (!test_bit(j, &present))
 -                              continue;
 -                      pnode = (i * 64 + j) & pnode_mask;
 -                      uv_blade_info[blade].pnode = pnode;
 -                      uv_blade_info[blade].nr_possible_cpus = 0;
 -                      uv_blade_info[blade].nr_online_cpus = 0;
 -                      spin_lock_init(&uv_blade_info[blade].nmi_lock);
 -                      min_pnode = min(pnode, min_pnode);
 -                      max_pnode = max(pnode, max_pnode);
 -                      blade++;
 +              unsigned long np;
 +
 +              np = uv_read_local_mmr(UVH_NODE_PRESENT_TABLE + i * 8);
 +              if (np)
 +                      pr_info("UV: NODE_PRESENT(%d) = 0x%016lx\n", i, np);
 +
 +              uv_pb += hweight64(np);
 +      }
 +      if (uv_possible_blades != uv_pb)
 +              uv_possible_blades = uv_pb;
 +}
 +
 +static void __init build_socket_tables(void)
 +{
 +      struct uv_gam_range_entry *gre = uv_gre_table;
 +      int num, nump;
 +      int cpu, i, lnid;
 +      int minsock = _min_socket;
 +      int maxsock = _max_socket;
 +      int minpnode = _min_pnode;
 +      int maxpnode = _max_pnode;
 +      size_t bytes;
 +
 +      if (!gre) {
 +              if (is_uv1_hub() || is_uv2_hub() || is_uv3_hub()) {
 +                      pr_info("UV: No UVsystab socket table, ignoring\n");
 +                      return;         /* not required */
                }
 +              pr_crit(
 +              "UV: Error: UVsystab address translations not available!\n");
 +              BUG();
 +      }
 +
 +      /* build socket id -> node id, pnode */
 +      num = maxsock - minsock + 1;
 +      bytes = num * sizeof(_socket_to_node[0]);
 +      _socket_to_node = kmalloc(bytes, GFP_KERNEL);
 +      _socket_to_pnode = kmalloc(bytes, GFP_KERNEL);
 +
 +      nump = maxpnode - minpnode + 1;
 +      bytes = nump * sizeof(_pnode_to_socket[0]);
 +      _pnode_to_socket = kmalloc(bytes, GFP_KERNEL);
 +      BUG_ON(!_socket_to_node || !_socket_to_pnode || !_pnode_to_socket);
 +
 +      for (i = 0; i < num; i++)
 +              _socket_to_node[i] = _socket_to_pnode[i] = SOCK_EMPTY;
 +
 +      for (i = 0; i < nump; i++)
 +              _pnode_to_socket[i] = SOCK_EMPTY;
 +
 +      /* fill in pnode/node/addr conversion list values */
 +      pr_info("UV: GAM Building socket/pnode/pxm conversion tables\n");
 +      for (; gre->type != UV_GAM_RANGE_TYPE_UNUSED; gre++) {
 +              if (gre->type == UV_GAM_RANGE_TYPE_HOLE)
 +                      continue;
 +              i = gre->sockid - minsock;
 +              if (_socket_to_pnode[i] != SOCK_EMPTY)
 +                      continue;       /* duplicate */
 +              _socket_to_pnode[i] = gre->pnode;
 +              _socket_to_node[i] = gre->pxm;
 +
 +              i = gre->pnode - minpnode;
 +              _pnode_to_socket[i] = gre->sockid;
 +
 +              pr_info(
 +              "UV: sid:%02x type:%d nasid:%04x pn:%02x pxm:%2d pn2s:%2x\n",
 +                      gre->sockid, gre->type, gre->nasid,
 +                      _socket_to_pnode[gre->sockid - minsock],
 +                      _socket_to_node[gre->sockid - minsock],
 +                      _pnode_to_socket[gre->pnode - minpnode]);
        }
  
 -      uv_bios_init();
 +      /* check socket -> node values */
 +      lnid = -1;
 +      for_each_present_cpu(cpu) {
 +              int nid = cpu_to_node(cpu);
 +              int apicid, sockid;
 +
 +              if (lnid == nid)
 +                      continue;
 +              lnid = nid;
 +              apicid = per_cpu(x86_cpu_to_apicid, cpu);
 +              sockid = apicid >> uv_cpuid.socketid_shift;
 +              i = sockid - minsock;
 +
 +              if (nid != _socket_to_node[i]) {
 +                      pr_warn(
 +                      "UV: %02x: type:%d socket:%02x PXM:%02x != node:%2d\n",
 +                              i, sockid, gre->type, _socket_to_node[i], nid);
 +                      _socket_to_node[i] = nid;
 +              }
 +      }
 +
 +      /* Setup physical blade to pnode translation from GAM Range Table */
 +      bytes = num_possible_nodes() * sizeof(_node_to_pnode[0]);
 +      _node_to_pnode = kmalloc(bytes, GFP_KERNEL);
 +      BUG_ON(!_node_to_pnode);
 +
 +      for (lnid = 0; lnid < num_possible_nodes(); lnid++) {
 +              unsigned short sockid;
 +
 +              for (sockid = minsock; sockid <= maxsock; sockid++) {
 +                      if (lnid == _socket_to_node[sockid - minsock]) {
 +                              _node_to_pnode[lnid] =
 +                                      _socket_to_pnode[sockid - minsock];
 +                              break;
 +                      }
 +              }
 +              if (sockid > maxsock) {
 +                      pr_err("UV: socket for node %d not found!\n", lnid);
 +                      BUG();
 +              }
 +      }
 +
 +      /*
 +       * If socket id == pnode or socket id == node for all nodes,
 +       *   system runs faster by removing corresponding conversion table.
 +       */
 +      pr_info("UV: Checking socket->node/pnode for identity maps\n");
 +      if (minsock == 0) {
 +              for (i = 0; i < num; i++)
 +                      if (_socket_to_node[i] == SOCK_EMPTY ||
 +                              i != _socket_to_node[i])
 +                              break;
 +              if (i >= num) {
 +                      kfree(_socket_to_node);
 +                      _socket_to_node = NULL;
 +                      pr_info("UV: 1:1 socket_to_node table removed\n");
 +              }
 +      }
 +      if (minsock == minpnode) {
 +              for (i = 0; i < num; i++)
 +                      if (_socket_to_pnode[i] != SOCK_EMPTY &&
 +                              _socket_to_pnode[i] != i + minpnode)
 +                              break;
 +              if (i >= num) {
 +                      kfree(_socket_to_pnode);
 +                      _socket_to_pnode = NULL;
 +                      pr_info("UV: 1:1 socket_to_pnode table removed\n");
 +              }
 +      }
 +}
 +
 +void __init uv_system_init(void)
 +{
 +      struct uv_hub_info_s hub_info = {0};
 +      int bytes, cpu, nodeid;
 +      unsigned short min_pnode = 9999, max_pnode = 0;
 +      char *hub = is_uv4_hub() ? "UV400" :
 +                  is_uv3_hub() ? "UV300" :
 +                  is_uv2_hub() ? "UV2000/3000" :
 +                  is_uv1_hub() ? "UV100/1000" : NULL;
 +
 +      if (!hub) {
 +              pr_err("UV: Unknown/unsupported UV hub\n");
 +              return;
 +      }
 +      pr_info("UV: Found %s hub\n", hub);
 +
-       /* We now only need to map the MMRs on UV1 */
-       if (is_uv1_hub())
-               map_low_mmrs();
++      map_low_mmrs();
 +
 +      uv_bios_init();                 /* get uv_systab for decoding */
 +      decode_uv_systab();
 +      build_socket_tables();
 +      build_uv_gr_table();
 +      uv_init_hub_info(&hub_info);
 +      uv_possible_blades = num_possible_nodes();
 +      if (!_node_to_pnode)
 +              boot_init_possible_blades(&hub_info);
 +
 +      /* uv_num_possible_blades() is really the hub count */
 +      pr_info("UV: Found %d hubs, %d nodes, %d cpus\n",
 +                      uv_num_possible_blades(),
 +                      num_possible_nodes(),
 +                      num_possible_cpus());
 +
        uv_bios_get_sn_info(0, &uv_type, &sn_partition_id, &sn_coherency_id,
                            &sn_region_size, &system_serial_number);
 +      hub_info.coherency_domain_number = sn_coherency_id;
        uv_rtc_init();
  
 -      for_each_present_cpu(cpu) {
 -              int apicid = per_cpu(x86_cpu_to_apicid, cpu);
 +      bytes = sizeof(void *) * uv_num_possible_blades();
 +      __uv_hub_info_list = kzalloc(bytes, GFP_KERNEL);
 +      BUG_ON(!__uv_hub_info_list);
  
 -              nid = cpu_to_node(cpu);
 -              /*
 -               * apic_pnode_shift must be set before calling uv_apicid_to_pnode();
 -               */
 -              uv_cpu_hub_info(cpu)->pnode_mask = pnode_mask;
 -              uv_cpu_hub_info(cpu)->apic_pnode_shift = uvh_apicid.s.pnode_shift;
 -              uv_cpu_hub_info(cpu)->hub_revision = uv_hub_info->hub_revision;
 +      bytes = sizeof(struct uv_hub_info_s);
 +      for_each_node(nodeid) {
 +              struct uv_hub_info_s *new_hub;
  
 -              uv_cpu_hub_info(cpu)->m_shift = 64 - m_val;
 -              uv_cpu_hub_info(cpu)->n_lshift = n_lshift;
 +              if (__uv_hub_info_list[nodeid]) {
 +                      pr_err("UV: Node %d UV HUB already initialized!?\n",
 +                              nodeid);
 +                      BUG();
 +              }
 +
 +              /* Allocate new per hub info list */
 +              new_hub = (nodeid == 0) ?
 +                      &uv_hub_info_node0 :
 +                      kzalloc_node(bytes, GFP_KERNEL, nodeid);
 +              BUG_ON(!new_hub);
 +              __uv_hub_info_list[nodeid] = new_hub;
 +              new_hub = uv_hub_info_list(nodeid);
 +              BUG_ON(!new_hub);
 +              *new_hub = hub_info;
 +
 +              /* 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 = 0xffff;
 +              new_hub->numa_blade_id = uv_node_to_blade_id(nodeid);
 +              new_hub->memory_nid = -1;
 +              new_hub->nr_possible_cpus = 0;
 +              new_hub->nr_online_cpus = 0;
 +      }
  
 +      /* 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);
 -              blade = boot_pnode_to_blade(pnode);
 -              lcpu = uv_blade_info[blade].nr_possible_cpus;
 -              uv_blade_info[blade].nr_possible_cpus++;
 -
 -              /* Any node on the blade, else will contain -1. */
 -              uv_blade_info[blade].memory_nid = nid;
 -
 -              uv_cpu_hub_info(cpu)->lowmem_remap_base = lowmem_redir_base;
 -              uv_cpu_hub_info(cpu)->lowmem_remap_top = lowmem_redir_size;
 -              uv_cpu_hub_info(cpu)->m_val = m_val;
 -              uv_cpu_hub_info(cpu)->n_val = n_val;
 -              uv_cpu_hub_info(cpu)->numa_blade_id = blade;
 -              uv_cpu_hub_info(cpu)->blade_processor_id = lcpu;
 -              uv_cpu_hub_info(cpu)->pnode = pnode;
 -              uv_cpu_hub_info(cpu)->gpa_mask = (1UL << (m_val + n_val)) - 1;
 -              uv_cpu_hub_info(cpu)->gnode_upper = gnode_upper;
 -              uv_cpu_hub_info(cpu)->gnode_extra = gnode_extra;
 -              uv_cpu_hub_info(cpu)->global_mmr_base = mmr_base;
 -              uv_cpu_hub_info(cpu)->coherency_domain_number = sn_coherency_id;
 -              uv_cpu_hub_info(cpu)->scir.offset = uv_scir_offset(apicid);
 -              uv_node_to_blade[nid] = blade;
 -              uv_cpu_to_blade[cpu] = blade;
 +
 +              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 == 0xffff)
 +                      uv_hub_info_list(numa_node_id)->pnode = pnode;
 +              else if (uv_cpu_hub_info(cpu)->pnode == 0xffff)
 +                      uv_cpu_hub_info(cpu)->pnode = pnode;
 +              uv_cpu_scir_info(cpu)->offset = uv_scir_offset(apicid);
        }
  
 -      /* Add blade/pnode info for nodes without cpus */
 -      for_each_online_node(nid) {
 -              if (uv_node_to_blade[nid] >= 0)
 -                      continue;
 -              paddr = node_start_pfn(nid) << PAGE_SHIFT;
 -              pnode = uv_gpa_to_pnode(uv_soc_phys_ram_to_gpa(paddr));
 -              blade = boot_pnode_to_blade(pnode);
 -              uv_node_to_blade[nid] = blade;
 +      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 == 0xffff) {
 +                      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",
 +                      nodeid,
 +                      uv_hub_info_list(nodeid)->pnode,
 +                      uv_hub_info_list(nodeid)->nr_possible_cpus);
        }
  
 +      pr_info("UV: min_pnode:%02x max_pnode:%02x\n", min_pnode, max_pnode);
        map_gru_high(max_pnode);
        map_mmr_high(max_pnode);
        map_mmioh_high(min_pnode, max_pnode);