ip27-smp.c 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225
  1. /*
  2. * This file is subject to the terms and conditions of the GNU General
  3. * Public License. See the file "COPYING" in the main directory of this
  4. * archive for more details.
  5. *
  6. * Copyright (C) 2000 - 2001 by Kanoj Sarcar (kanoj@sgi.com)
  7. * Copyright (C) 2000 - 2001 by Silicon Graphics, Inc.
  8. */
  9. #include <linux/init.h>
  10. #include <linux/sched.h>
  11. #include <linux/nodemask.h>
  12. #include <asm/page.h>
  13. #include <asm/processor.h>
  14. #include <asm/sn/arch.h>
  15. #include <asm/sn/gda.h>
  16. #include <asm/sn/intr.h>
  17. #include <asm/sn/klconfig.h>
  18. #include <asm/sn/launch.h>
  19. #include <asm/sn/mapped_kernel.h>
  20. #include <asm/sn/sn_private.h>
  21. #include <asm/sn/types.h>
  22. #include <asm/sn/sn0/hubpi.h>
  23. #include <asm/sn/sn0/hubio.h>
  24. #include <asm/sn/sn0/ip27.h>
  25. /*
  26. * Takes as first input the PROM assigned cpu id, and the kernel
  27. * assigned cpu id as the second.
  28. */
  29. static void alloc_cpupda(cpuid_t cpu, int cpunum)
  30. {
  31. cnodeid_t node = get_cpu_cnode(cpu);
  32. nasid_t nasid = COMPACT_TO_NASID_NODEID(node);
  33. cputonasid(cpunum) = nasid;
  34. cpu_data[cpunum].p_nodeid = node;
  35. cputoslice(cpunum) = get_cpu_slice(cpu);
  36. }
  37. static nasid_t get_actual_nasid(lboard_t *brd)
  38. {
  39. klhub_t *hub;
  40. if (!brd)
  41. return INVALID_NASID;
  42. /* find out if we are a completely disabled brd. */
  43. hub = (klhub_t *)find_first_component(brd, KLSTRUCT_HUB);
  44. if (!hub)
  45. return INVALID_NASID;
  46. if (!(hub->hub_info.flags & KLINFO_ENABLE)) /* disabled node brd */
  47. return hub->hub_info.physid;
  48. else
  49. return brd->brd_nasid;
  50. }
  51. static int do_cpumask(cnodeid_t cnode, nasid_t nasid, int highest)
  52. {
  53. static int tot_cpus_found = 0;
  54. lboard_t *brd;
  55. klcpu_t *acpu;
  56. int cpus_found = 0;
  57. cpuid_t cpuid;
  58. brd = find_lboard((lboard_t *)KL_CONFIG_INFO(nasid), KLTYPE_IP27);
  59. do {
  60. acpu = (klcpu_t *)find_first_component(brd, KLSTRUCT_CPU);
  61. while (acpu) {
  62. cpuid = acpu->cpu_info.virtid;
  63. /* cnode is not valid for completely disabled brds */
  64. if (get_actual_nasid(brd) == brd->brd_nasid)
  65. cpuid_to_compact_node[cpuid] = cnode;
  66. if (cpuid > highest)
  67. highest = cpuid;
  68. /* Only let it join in if it's marked enabled */
  69. if ((acpu->cpu_info.flags & KLINFO_ENABLE) &&
  70. (tot_cpus_found != NR_CPUS)) {
  71. cpu_set(cpuid, phys_cpu_present_map);
  72. alloc_cpupda(cpuid, tot_cpus_found);
  73. cpus_found++;
  74. tot_cpus_found++;
  75. }
  76. acpu = (klcpu_t *)find_component(brd, (klinfo_t *)acpu,
  77. KLSTRUCT_CPU);
  78. }
  79. brd = KLCF_NEXT(brd);
  80. if (!brd)
  81. break;
  82. brd = find_lboard(brd, KLTYPE_IP27);
  83. } while (brd);
  84. return highest;
  85. }
  86. void cpu_node_probe(void)
  87. {
  88. int i, highest = 0;
  89. gda_t *gdap = GDA;
  90. /*
  91. * Initialize the arrays to invalid nodeid (-1)
  92. */
  93. for (i = 0; i < MAX_COMPACT_NODES; i++)
  94. compact_to_nasid_node[i] = INVALID_NASID;
  95. for (i = 0; i < MAX_NASIDS; i++)
  96. nasid_to_compact_node[i] = INVALID_CNODEID;
  97. for (i = 0; i < MAXCPUS; i++)
  98. cpuid_to_compact_node[i] = INVALID_CNODEID;
  99. /*
  100. * MCD - this whole "compact node" stuff can probably be dropped,
  101. * as we can handle sparse numbering now
  102. */
  103. nodes_clear(node_online_map);
  104. for (i = 0; i < MAX_COMPACT_NODES; i++) {
  105. nasid_t nasid = gdap->g_nasidtable[i];
  106. if (nasid == INVALID_NASID)
  107. break;
  108. compact_to_nasid_node[i] = nasid;
  109. nasid_to_compact_node[nasid] = i;
  110. node_set_online(num_online_nodes());
  111. highest = do_cpumask(i, nasid, highest);
  112. }
  113. printk("Discovered %d cpus on %d nodes\n", highest + 1, num_online_nodes());
  114. }
  115. static void intr_clear_bits(nasid_t nasid, volatile hubreg_t *pend,
  116. int base_level)
  117. {
  118. volatile hubreg_t bits;
  119. int i;
  120. /* Check pending interrupts */
  121. if ((bits = HUB_L(pend)) != 0)
  122. for (i = 0; i < N_INTPEND_BITS; i++)
  123. if (bits & (1 << i))
  124. LOCAL_HUB_CLR_INTR(base_level + i);
  125. }
  126. static void intr_clear_all(nasid_t nasid)
  127. {
  128. REMOTE_HUB_S(nasid, PI_INT_MASK0_A, 0);
  129. REMOTE_HUB_S(nasid, PI_INT_MASK0_B, 0);
  130. REMOTE_HUB_S(nasid, PI_INT_MASK1_A, 0);
  131. REMOTE_HUB_S(nasid, PI_INT_MASK1_B, 0);
  132. intr_clear_bits(nasid, REMOTE_HUB_ADDR(nasid, PI_INT_PEND0),
  133. INT_PEND0_BASELVL);
  134. intr_clear_bits(nasid, REMOTE_HUB_ADDR(nasid, PI_INT_PEND1),
  135. INT_PEND1_BASELVL);
  136. }
  137. void __init prom_prepare_cpus(unsigned int max_cpus)
  138. {
  139. cnodeid_t cnode;
  140. for_each_online_node(cnode)
  141. intr_clear_all(COMPACT_TO_NASID_NODEID(cnode));
  142. replicate_kernel_text();
  143. /*
  144. * Assumption to be fixed: we're always booted on logical / physical
  145. * processor 0. While we're always running on logical processor 0
  146. * this still means this is physical processor zero; it might for
  147. * example be disabled in the firwware.
  148. */
  149. alloc_cpupda(0, 0);
  150. }
  151. /*
  152. * Launch a slave into smp_bootstrap(). It doesn't take an argument, and we
  153. * set sp to the kernel stack of the newly created idle process, gp to the proc
  154. * struct so that current_thread_info() will work.
  155. */
  156. void __init prom_boot_secondary(int cpu, struct task_struct *idle)
  157. {
  158. unsigned long gp = (unsigned long) idle->thread_info;
  159. unsigned long sp = gp + THREAD_SIZE - 32;
  160. LAUNCH_SLAVE(cputonasid(cpu),cputoslice(cpu),
  161. (launch_proc_t)MAPPED_KERN_RW_TO_K0(smp_bootstrap),
  162. 0, (void *) sp, (void *) gp);
  163. }
  164. void prom_init_secondary(void)
  165. {
  166. per_cpu_init();
  167. local_irq_enable();
  168. }
  169. void __init prom_cpus_done(void)
  170. {
  171. }
  172. void prom_smp_finish(void)
  173. {
  174. }
  175. void core_send_ipi(int destid, unsigned int action)
  176. {
  177. int irq;
  178. switch (action) {
  179. case SMP_RESCHEDULE_YOURSELF:
  180. irq = CPU_RESCHED_A_IRQ;
  181. break;
  182. case SMP_CALL_FUNCTION:
  183. irq = CPU_CALL_A_IRQ;
  184. break;
  185. default:
  186. panic("sendintr");
  187. }
  188. irq += cputoslice(destid);
  189. /*
  190. * Convert the compact hub number to the NASID to get the correct
  191. * part of the address space. Then set the interrupt bit associated
  192. * with the CPU we want to send the interrupt to.
  193. */
  194. REMOTE_HUB_SEND_INTR(COMPACT_TO_NASID_NODEID(cpu_to_node(destid)), irq);
  195. }