diff options
Diffstat (limited to 'arch/sparc64/kernel/irq.c')
-rw-r--r-- | arch/sparc64/kernel/irq.c | 345 |
1 files changed, 254 insertions, 91 deletions
diff --git a/arch/sparc64/kernel/irq.c b/arch/sparc64/kernel/irq.c index 233526ba3abe..11e645c9ec50 100644 --- a/arch/sparc64/kernel/irq.c +++ b/arch/sparc64/kernel/irq.c | |||
@@ -21,6 +21,7 @@ | |||
21 | #include <linux/delay.h> | 21 | #include <linux/delay.h> |
22 | #include <linux/proc_fs.h> | 22 | #include <linux/proc_fs.h> |
23 | #include <linux/seq_file.h> | 23 | #include <linux/seq_file.h> |
24 | #include <linux/bootmem.h> | ||
24 | 25 | ||
25 | #include <asm/ptrace.h> | 26 | #include <asm/ptrace.h> |
26 | #include <asm/processor.h> | 27 | #include <asm/processor.h> |
@@ -39,6 +40,7 @@ | |||
39 | #include <asm/cache.h> | 40 | #include <asm/cache.h> |
40 | #include <asm/cpudata.h> | 41 | #include <asm/cpudata.h> |
41 | #include <asm/auxio.h> | 42 | #include <asm/auxio.h> |
43 | #include <asm/head.h> | ||
42 | 44 | ||
43 | #ifdef CONFIG_SMP | 45 | #ifdef CONFIG_SMP |
44 | static void distribute_irqs(void); | 46 | static void distribute_irqs(void); |
@@ -115,9 +117,7 @@ int show_interrupts(struct seq_file *p, void *v) | |||
115 | #ifndef CONFIG_SMP | 117 | #ifndef CONFIG_SMP |
116 | seq_printf(p, "%10u ", kstat_irqs(i)); | 118 | seq_printf(p, "%10u ", kstat_irqs(i)); |
117 | #else | 119 | #else |
118 | for (j = 0; j < NR_CPUS; j++) { | 120 | for_each_online_cpu(j) { |
119 | if (!cpu_online(j)) | ||
120 | continue; | ||
121 | seq_printf(p, "%10u ", | 121 | seq_printf(p, "%10u ", |
122 | kstat_cpu(j).irqs[i]); | 122 | kstat_cpu(j).irqs[i]); |
123 | } | 123 | } |
@@ -136,12 +136,48 @@ out_unlock: | |||
136 | return 0; | 136 | return 0; |
137 | } | 137 | } |
138 | 138 | ||
139 | extern unsigned long real_hard_smp_processor_id(void); | ||
140 | |||
141 | static unsigned int sun4u_compute_tid(unsigned long imap, unsigned long cpuid) | ||
142 | { | ||
143 | unsigned int tid; | ||
144 | |||
145 | if (this_is_starfire) { | ||
146 | tid = starfire_translate(imap, cpuid); | ||
147 | tid <<= IMAP_TID_SHIFT; | ||
148 | tid &= IMAP_TID_UPA; | ||
149 | } else { | ||
150 | if (tlb_type == cheetah || tlb_type == cheetah_plus) { | ||
151 | unsigned long ver; | ||
152 | |||
153 | __asm__ ("rdpr %%ver, %0" : "=r" (ver)); | ||
154 | if ((ver >> 32UL) == __JALAPENO_ID || | ||
155 | (ver >> 32UL) == __SERRANO_ID) { | ||
156 | tid = cpuid << IMAP_TID_SHIFT; | ||
157 | tid &= IMAP_TID_JBUS; | ||
158 | } else { | ||
159 | unsigned int a = cpuid & 0x1f; | ||
160 | unsigned int n = (cpuid >> 5) & 0x1f; | ||
161 | |||
162 | tid = ((a << IMAP_AID_SHIFT) | | ||
163 | (n << IMAP_NID_SHIFT)); | ||
164 | tid &= (IMAP_AID_SAFARI | | ||
165 | IMAP_NID_SAFARI);; | ||
166 | } | ||
167 | } else { | ||
168 | tid = cpuid << IMAP_TID_SHIFT; | ||
169 | tid &= IMAP_TID_UPA; | ||
170 | } | ||
171 | } | ||
172 | |||
173 | return tid; | ||
174 | } | ||
175 | |||
139 | /* Now these are always passed a true fully specified sun4u INO. */ | 176 | /* Now these are always passed a true fully specified sun4u INO. */ |
140 | void enable_irq(unsigned int irq) | 177 | void enable_irq(unsigned int irq) |
141 | { | 178 | { |
142 | struct ino_bucket *bucket = __bucket(irq); | 179 | struct ino_bucket *bucket = __bucket(irq); |
143 | unsigned long imap; | 180 | unsigned long imap, cpuid; |
144 | unsigned long tid; | ||
145 | 181 | ||
146 | imap = bucket->imap; | 182 | imap = bucket->imap; |
147 | if (imap == 0UL) | 183 | if (imap == 0UL) |
@@ -149,47 +185,38 @@ void enable_irq(unsigned int irq) | |||
149 | 185 | ||
150 | preempt_disable(); | 186 | preempt_disable(); |
151 | 187 | ||
152 | if (tlb_type == cheetah || tlb_type == cheetah_plus) { | 188 | /* This gets the physical processor ID, even on uniprocessor, |
153 | unsigned long ver; | 189 | * so we can always program the interrupt target correctly. |
154 | 190 | */ | |
155 | __asm__ ("rdpr %%ver, %0" : "=r" (ver)); | 191 | cpuid = real_hard_smp_processor_id(); |
156 | if ((ver >> 32) == 0x003e0016) { | 192 | |
157 | /* We set it to our JBUS ID. */ | 193 | if (tlb_type == hypervisor) { |
158 | __asm__ __volatile__("ldxa [%%g0] %1, %0" | 194 | unsigned int ino = __irq_ino(irq); |
159 | : "=r" (tid) | 195 | int err; |
160 | : "i" (ASI_JBUS_CONFIG)); | 196 | |
161 | tid = ((tid & (0x1fUL<<17)) << 9); | 197 | err = sun4v_intr_settarget(ino, cpuid); |
162 | tid &= IMAP_TID_JBUS; | 198 | if (err != HV_EOK) |
163 | } else { | 199 | printk("sun4v_intr_settarget(%x,%lu): err(%d)\n", |
164 | /* We set it to our Safari AID. */ | 200 | ino, cpuid, err); |
165 | __asm__ __volatile__("ldxa [%%g0] %1, %0" | 201 | err = sun4v_intr_setenabled(ino, HV_INTR_ENABLED); |
166 | : "=r" (tid) | 202 | if (err != HV_EOK) |
167 | : "i" (ASI_SAFARI_CONFIG)); | 203 | printk("sun4v_intr_setenabled(%x): err(%d)\n", |
168 | tid = ((tid & (0x3ffUL<<17)) << 9); | 204 | ino, err); |
169 | tid &= IMAP_AID_SAFARI; | ||
170 | } | ||
171 | } else if (this_is_starfire == 0) { | ||
172 | /* We set it to our UPA MID. */ | ||
173 | __asm__ __volatile__("ldxa [%%g0] %1, %0" | ||
174 | : "=r" (tid) | ||
175 | : "i" (ASI_UPA_CONFIG)); | ||
176 | tid = ((tid & UPA_CONFIG_MID) << 9); | ||
177 | tid &= IMAP_TID_UPA; | ||
178 | } else { | 205 | } else { |
179 | tid = (starfire_translate(imap, smp_processor_id()) << 26); | 206 | unsigned int tid = sun4u_compute_tid(imap, cpuid); |
180 | tid &= IMAP_TID_UPA; | 207 | |
208 | /* NOTE NOTE NOTE, IGN and INO are read-only, IGN is a product | ||
209 | * of this SYSIO's preconfigured IGN in the SYSIO Control | ||
210 | * Register, the hardware just mirrors that value here. | ||
211 | * However for Graphics and UPA Slave devices the full | ||
212 | * IMAP_INR field can be set by the programmer here. | ||
213 | * | ||
214 | * Things like FFB can now be handled via the new IRQ | ||
215 | * mechanism. | ||
216 | */ | ||
217 | upa_writel(tid | IMAP_VALID, imap); | ||
181 | } | 218 | } |
182 | 219 | ||
183 | /* NOTE NOTE NOTE, IGN and INO are read-only, IGN is a product | ||
184 | * of this SYSIO's preconfigured IGN in the SYSIO Control | ||
185 | * Register, the hardware just mirrors that value here. | ||
186 | * However for Graphics and UPA Slave devices the full | ||
187 | * IMAP_INR field can be set by the programmer here. | ||
188 | * | ||
189 | * Things like FFB can now be handled via the new IRQ mechanism. | ||
190 | */ | ||
191 | upa_writel(tid | IMAP_VALID, imap); | ||
192 | |||
193 | preempt_enable(); | 220 | preempt_enable(); |
194 | } | 221 | } |
195 | 222 | ||
@@ -201,16 +228,26 @@ void disable_irq(unsigned int irq) | |||
201 | 228 | ||
202 | imap = bucket->imap; | 229 | imap = bucket->imap; |
203 | if (imap != 0UL) { | 230 | if (imap != 0UL) { |
204 | u32 tmp; | 231 | if (tlb_type == hypervisor) { |
232 | unsigned int ino = __irq_ino(irq); | ||
233 | int err; | ||
234 | |||
235 | err = sun4v_intr_setenabled(ino, HV_INTR_DISABLED); | ||
236 | if (err != HV_EOK) | ||
237 | printk("sun4v_intr_setenabled(%x): " | ||
238 | "err(%d)\n", ino, err); | ||
239 | } else { | ||
240 | u32 tmp; | ||
205 | 241 | ||
206 | /* NOTE: We do not want to futz with the IRQ clear registers | 242 | /* NOTE: We do not want to futz with the IRQ clear registers |
207 | * and move the state to IDLE, the SCSI code does call | 243 | * and move the state to IDLE, the SCSI code does call |
208 | * disable_irq() to assure atomicity in the queue cmd | 244 | * disable_irq() to assure atomicity in the queue cmd |
209 | * SCSI adapter driver code. Thus we'd lose interrupts. | 245 | * SCSI adapter driver code. Thus we'd lose interrupts. |
210 | */ | 246 | */ |
211 | tmp = upa_readl(imap); | 247 | tmp = upa_readl(imap); |
212 | tmp &= ~IMAP_VALID; | 248 | tmp &= ~IMAP_VALID; |
213 | upa_writel(tmp, imap); | 249 | upa_writel(tmp, imap); |
250 | } | ||
214 | } | 251 | } |
215 | } | 252 | } |
216 | 253 | ||
@@ -248,6 +285,8 @@ unsigned int build_irq(int pil, int inofixup, unsigned long iclr, unsigned long | |||
248 | return __irq(&pil0_dummy_bucket); | 285 | return __irq(&pil0_dummy_bucket); |
249 | } | 286 | } |
250 | 287 | ||
288 | BUG_ON(tlb_type == hypervisor); | ||
289 | |||
251 | /* RULE: Both must be specified in all other cases. */ | 290 | /* RULE: Both must be specified in all other cases. */ |
252 | if (iclr == 0UL || imap == 0UL) { | 291 | if (iclr == 0UL || imap == 0UL) { |
253 | prom_printf("Invalid build_irq %d %d %016lx %016lx\n", | 292 | prom_printf("Invalid build_irq %d %d %016lx %016lx\n", |
@@ -275,12 +314,11 @@ unsigned int build_irq(int pil, int inofixup, unsigned long iclr, unsigned long | |||
275 | goto out; | 314 | goto out; |
276 | } | 315 | } |
277 | 316 | ||
278 | bucket->irq_info = kmalloc(sizeof(struct irq_desc), GFP_ATOMIC); | 317 | bucket->irq_info = kzalloc(sizeof(struct irq_desc), GFP_ATOMIC); |
279 | if (!bucket->irq_info) { | 318 | if (!bucket->irq_info) { |
280 | prom_printf("IRQ: Error, kmalloc(irq_desc) failed.\n"); | 319 | prom_printf("IRQ: Error, kmalloc(irq_desc) failed.\n"); |
281 | prom_halt(); | 320 | prom_halt(); |
282 | } | 321 | } |
283 | memset(bucket->irq_info, 0, sizeof(struct irq_desc)); | ||
284 | 322 | ||
285 | /* Ok, looks good, set it up. Don't touch the irq_chain or | 323 | /* Ok, looks good, set it up. Don't touch the irq_chain or |
286 | * the pending flag. | 324 | * the pending flag. |
@@ -294,6 +332,37 @@ out: | |||
294 | return __irq(bucket); | 332 | return __irq(bucket); |
295 | } | 333 | } |
296 | 334 | ||
335 | unsigned int sun4v_build_irq(u32 devhandle, unsigned int devino, int pil, unsigned char flags) | ||
336 | { | ||
337 | struct ino_bucket *bucket; | ||
338 | unsigned long sysino; | ||
339 | |||
340 | sysino = sun4v_devino_to_sysino(devhandle, devino); | ||
341 | |||
342 | bucket = &ivector_table[sysino]; | ||
343 | |||
344 | /* Catch accidental accesses to these things. IMAP/ICLR handling | ||
345 | * is done by hypervisor calls on sun4v platforms, not by direct | ||
346 | * register accesses. | ||
347 | * | ||
348 | * But we need to make them look unique for the disable_irq() logic | ||
349 | * in free_irq(). | ||
350 | */ | ||
351 | bucket->imap = ~0UL - sysino; | ||
352 | bucket->iclr = ~0UL - sysino; | ||
353 | |||
354 | bucket->pil = pil; | ||
355 | bucket->flags = flags; | ||
356 | |||
357 | bucket->irq_info = kzalloc(sizeof(struct irq_desc), GFP_ATOMIC); | ||
358 | if (!bucket->irq_info) { | ||
359 | prom_printf("IRQ: Error, kmalloc(irq_desc) failed.\n"); | ||
360 | prom_halt(); | ||
361 | } | ||
362 | |||
363 | return __irq(bucket); | ||
364 | } | ||
365 | |||
297 | static void atomic_bucket_insert(struct ino_bucket *bucket) | 366 | static void atomic_bucket_insert(struct ino_bucket *bucket) |
298 | { | 367 | { |
299 | unsigned long pstate; | 368 | unsigned long pstate; |
@@ -482,7 +551,6 @@ void free_irq(unsigned int irq, void *dev_id) | |||
482 | bucket = __bucket(irq); | 551 | bucket = __bucket(irq); |
483 | if (bucket != &pil0_dummy_bucket) { | 552 | if (bucket != &pil0_dummy_bucket) { |
484 | struct irq_desc *desc = bucket->irq_info; | 553 | struct irq_desc *desc = bucket->irq_info; |
485 | unsigned long imap = bucket->imap; | ||
486 | int ent, i; | 554 | int ent, i; |
487 | 555 | ||
488 | for (i = 0; i < MAX_IRQ_DESC_ACTION; i++) { | 556 | for (i = 0; i < MAX_IRQ_DESC_ACTION; i++) { |
@@ -495,6 +563,8 @@ void free_irq(unsigned int irq, void *dev_id) | |||
495 | } | 563 | } |
496 | 564 | ||
497 | if (!desc->action_active_mask) { | 565 | if (!desc->action_active_mask) { |
566 | unsigned long imap = bucket->imap; | ||
567 | |||
498 | /* This unique interrupt source is now inactive. */ | 568 | /* This unique interrupt source is now inactive. */ |
499 | bucket->flags &= ~IBF_ACTIVE; | 569 | bucket->flags &= ~IBF_ACTIVE; |
500 | 570 | ||
@@ -592,7 +662,18 @@ static void process_bucket(int irq, struct ino_bucket *bp, struct pt_regs *regs) | |||
592 | break; | 662 | break; |
593 | } | 663 | } |
594 | if (bp->pil != 0) { | 664 | if (bp->pil != 0) { |
595 | upa_writel(ICLR_IDLE, bp->iclr); | 665 | if (tlb_type == hypervisor) { |
666 | unsigned int ino = __irq_ino(bp); | ||
667 | int err; | ||
668 | |||
669 | err = sun4v_intr_setstate(ino, HV_INTR_STATE_IDLE); | ||
670 | if (err != HV_EOK) | ||
671 | printk("sun4v_intr_setstate(%x): " | ||
672 | "err(%d)\n", ino, err); | ||
673 | } else { | ||
674 | upa_writel(ICLR_IDLE, bp->iclr); | ||
675 | } | ||
676 | |||
596 | /* Test and add entropy */ | 677 | /* Test and add entropy */ |
597 | if (random & SA_SAMPLE_RANDOM) | 678 | if (random & SA_SAMPLE_RANDOM) |
598 | add_interrupt_randomness(irq); | 679 | add_interrupt_randomness(irq); |
@@ -646,7 +727,7 @@ void handler_irq(int irq, struct pt_regs *regs) | |||
646 | } | 727 | } |
647 | 728 | ||
648 | #ifdef CONFIG_BLK_DEV_FD | 729 | #ifdef CONFIG_BLK_DEV_FD |
649 | extern irqreturn_t floppy_interrupt(int, void *, struct pt_regs *);; | 730 | extern irqreturn_t floppy_interrupt(int, void *, struct pt_regs *); |
650 | 731 | ||
651 | /* XXX No easy way to include asm/floppy.h XXX */ | 732 | /* XXX No easy way to include asm/floppy.h XXX */ |
652 | extern unsigned char *pdma_vaddr; | 733 | extern unsigned char *pdma_vaddr; |
@@ -694,7 +775,7 @@ irqreturn_t sparc_floppy_irq(int irq, void *dev_cookie, struct pt_regs *regs) | |||
694 | val = readb(auxio_register); | 775 | val = readb(auxio_register); |
695 | val |= AUXIO_AUX1_FTCNT; | 776 | val |= AUXIO_AUX1_FTCNT; |
696 | writeb(val, auxio_register); | 777 | writeb(val, auxio_register); |
697 | val &= AUXIO_AUX1_FTCNT; | 778 | val &= ~AUXIO_AUX1_FTCNT; |
698 | writeb(val, auxio_register); | 779 | writeb(val, auxio_register); |
699 | 780 | ||
700 | doing_pdma = 0; | 781 | doing_pdma = 0; |
@@ -727,25 +808,23 @@ EXPORT_SYMBOL(probe_irq_off); | |||
727 | static int retarget_one_irq(struct irqaction *p, int goal_cpu) | 808 | static int retarget_one_irq(struct irqaction *p, int goal_cpu) |
728 | { | 809 | { |
729 | struct ino_bucket *bucket = get_ino_in_irqaction(p) + ivector_table; | 810 | struct ino_bucket *bucket = get_ino_in_irqaction(p) + ivector_table; |
730 | unsigned long imap = bucket->imap; | ||
731 | unsigned int tid; | ||
732 | 811 | ||
733 | while (!cpu_online(goal_cpu)) { | 812 | while (!cpu_online(goal_cpu)) { |
734 | if (++goal_cpu >= NR_CPUS) | 813 | if (++goal_cpu >= NR_CPUS) |
735 | goal_cpu = 0; | 814 | goal_cpu = 0; |
736 | } | 815 | } |
737 | 816 | ||
738 | if (tlb_type == cheetah || tlb_type == cheetah_plus) { | 817 | if (tlb_type == hypervisor) { |
739 | tid = goal_cpu << 26; | 818 | unsigned int ino = __irq_ino(bucket); |
740 | tid &= IMAP_AID_SAFARI; | 819 | |
741 | } else if (this_is_starfire == 0) { | 820 | sun4v_intr_settarget(ino, goal_cpu); |
742 | tid = goal_cpu << 26; | 821 | sun4v_intr_setenabled(ino, HV_INTR_ENABLED); |
743 | tid &= IMAP_TID_UPA; | ||
744 | } else { | 822 | } else { |
745 | tid = (starfire_translate(imap, goal_cpu) << 26); | 823 | unsigned long imap = bucket->imap; |
746 | tid &= IMAP_TID_UPA; | 824 | unsigned int tid = sun4u_compute_tid(imap, goal_cpu); |
825 | |||
826 | upa_writel(tid | IMAP_VALID, imap); | ||
747 | } | 827 | } |
748 | upa_writel(tid | IMAP_VALID, imap); | ||
749 | 828 | ||
750 | do { | 829 | do { |
751 | if (++goal_cpu >= NR_CPUS) | 830 | if (++goal_cpu >= NR_CPUS) |
@@ -848,33 +927,114 @@ static void kill_prom_timer(void) | |||
848 | 927 | ||
849 | void init_irqwork_curcpu(void) | 928 | void init_irqwork_curcpu(void) |
850 | { | 929 | { |
851 | register struct irq_work_struct *workp asm("o2"); | ||
852 | register unsigned long tmp asm("o3"); | ||
853 | int cpu = hard_smp_processor_id(); | 930 | int cpu = hard_smp_processor_id(); |
854 | 931 | ||
855 | memset(__irq_work + cpu, 0, sizeof(*workp)); | 932 | memset(__irq_work + cpu, 0, sizeof(struct irq_work_struct)); |
856 | 933 | } | |
857 | /* Make sure we are called with PSTATE_IE disabled. */ | 934 | |
858 | __asm__ __volatile__("rdpr %%pstate, %0\n\t" | 935 | static void __cpuinit register_one_mondo(unsigned long paddr, unsigned long type) |
859 | : "=r" (tmp)); | 936 | { |
860 | if (tmp & PSTATE_IE) { | 937 | unsigned long num_entries = 128; |
861 | prom_printf("BUG: init_irqwork_curcpu() called with " | 938 | unsigned long status; |
862 | "PSTATE_IE enabled, bailing.\n"); | 939 | |
863 | __asm__ __volatile__("mov %%i7, %0\n\t" | 940 | status = sun4v_cpu_qconf(type, paddr, num_entries); |
864 | : "=r" (tmp)); | 941 | if (status != HV_EOK) { |
865 | prom_printf("BUG: Called from %lx\n", tmp); | 942 | prom_printf("SUN4V: sun4v_cpu_qconf(%lu:%lx:%lu) failed, " |
943 | "err %lu\n", type, paddr, num_entries, status); | ||
866 | prom_halt(); | 944 | prom_halt(); |
867 | } | 945 | } |
946 | } | ||
868 | 947 | ||
869 | /* Set interrupt globals. */ | 948 | static void __cpuinit sun4v_register_mondo_queues(int this_cpu) |
870 | workp = &__irq_work[cpu]; | 949 | { |
871 | __asm__ __volatile__( | 950 | struct trap_per_cpu *tb = &trap_block[this_cpu]; |
872 | "rdpr %%pstate, %0\n\t" | 951 | |
873 | "wrpr %0, %1, %%pstate\n\t" | 952 | register_one_mondo(tb->cpu_mondo_pa, HV_CPU_QUEUE_CPU_MONDO); |
874 | "mov %2, %%g6\n\t" | 953 | register_one_mondo(tb->dev_mondo_pa, HV_CPU_QUEUE_DEVICE_MONDO); |
875 | "wrpr %0, 0x0, %%pstate\n\t" | 954 | register_one_mondo(tb->resum_mondo_pa, HV_CPU_QUEUE_RES_ERROR); |
876 | : "=&r" (tmp) | 955 | register_one_mondo(tb->nonresum_mondo_pa, HV_CPU_QUEUE_NONRES_ERROR); |
877 | : "i" (PSTATE_IG), "r" (workp)); | 956 | } |
957 | |||
958 | static void __cpuinit alloc_one_mondo(unsigned long *pa_ptr, int use_bootmem) | ||
959 | { | ||
960 | void *page; | ||
961 | |||
962 | if (use_bootmem) | ||
963 | page = alloc_bootmem_low_pages(PAGE_SIZE); | ||
964 | else | ||
965 | page = (void *) get_zeroed_page(GFP_ATOMIC); | ||
966 | |||
967 | if (!page) { | ||
968 | prom_printf("SUN4V: Error, cannot allocate mondo queue.\n"); | ||
969 | prom_halt(); | ||
970 | } | ||
971 | |||
972 | *pa_ptr = __pa(page); | ||
973 | } | ||
974 | |||
975 | static void __cpuinit alloc_one_kbuf(unsigned long *pa_ptr, int use_bootmem) | ||
976 | { | ||
977 | void *page; | ||
978 | |||
979 | if (use_bootmem) | ||
980 | page = alloc_bootmem_low_pages(PAGE_SIZE); | ||
981 | else | ||
982 | page = (void *) get_zeroed_page(GFP_ATOMIC); | ||
983 | |||
984 | if (!page) { | ||
985 | prom_printf("SUN4V: Error, cannot allocate kbuf page.\n"); | ||
986 | prom_halt(); | ||
987 | } | ||
988 | |||
989 | *pa_ptr = __pa(page); | ||
990 | } | ||
991 | |||
992 | static void __cpuinit init_cpu_send_mondo_info(struct trap_per_cpu *tb, int use_bootmem) | ||
993 | { | ||
994 | #ifdef CONFIG_SMP | ||
995 | void *page; | ||
996 | |||
997 | BUILD_BUG_ON((NR_CPUS * sizeof(u16)) > (PAGE_SIZE - 64)); | ||
998 | |||
999 | if (use_bootmem) | ||
1000 | page = alloc_bootmem_low_pages(PAGE_SIZE); | ||
1001 | else | ||
1002 | page = (void *) get_zeroed_page(GFP_ATOMIC); | ||
1003 | |||
1004 | if (!page) { | ||
1005 | prom_printf("SUN4V: Error, cannot allocate cpu mondo page.\n"); | ||
1006 | prom_halt(); | ||
1007 | } | ||
1008 | |||
1009 | tb->cpu_mondo_block_pa = __pa(page); | ||
1010 | tb->cpu_list_pa = __pa(page + 64); | ||
1011 | #endif | ||
1012 | } | ||
1013 | |||
1014 | /* Allocate and register the mondo and error queues for this cpu. */ | ||
1015 | void __cpuinit sun4v_init_mondo_queues(int use_bootmem, int cpu, int alloc, int load) | ||
1016 | { | ||
1017 | struct trap_per_cpu *tb = &trap_block[cpu]; | ||
1018 | |||
1019 | if (alloc) { | ||
1020 | alloc_one_mondo(&tb->cpu_mondo_pa, use_bootmem); | ||
1021 | alloc_one_mondo(&tb->dev_mondo_pa, use_bootmem); | ||
1022 | alloc_one_mondo(&tb->resum_mondo_pa, use_bootmem); | ||
1023 | alloc_one_kbuf(&tb->resum_kernel_buf_pa, use_bootmem); | ||
1024 | alloc_one_mondo(&tb->nonresum_mondo_pa, use_bootmem); | ||
1025 | alloc_one_kbuf(&tb->nonresum_kernel_buf_pa, use_bootmem); | ||
1026 | |||
1027 | init_cpu_send_mondo_info(tb, use_bootmem); | ||
1028 | } | ||
1029 | |||
1030 | if (load) { | ||
1031 | if (cpu != hard_smp_processor_id()) { | ||
1032 | prom_printf("SUN4V: init mondo on cpu %d not %d\n", | ||
1033 | cpu, hard_smp_processor_id()); | ||
1034 | prom_halt(); | ||
1035 | } | ||
1036 | sun4v_register_mondo_queues(cpu); | ||
1037 | } | ||
878 | } | 1038 | } |
879 | 1039 | ||
880 | /* Only invoked on boot processor. */ | 1040 | /* Only invoked on boot processor. */ |
@@ -884,6 +1044,9 @@ void __init init_IRQ(void) | |||
884 | kill_prom_timer(); | 1044 | kill_prom_timer(); |
885 | memset(&ivector_table[0], 0, sizeof(ivector_table)); | 1045 | memset(&ivector_table[0], 0, sizeof(ivector_table)); |
886 | 1046 | ||
1047 | if (tlb_type == hypervisor) | ||
1048 | sun4v_init_mondo_queues(1, hard_smp_processor_id(), 1, 1); | ||
1049 | |||
887 | /* We need to clear any IRQ's pending in the soft interrupt | 1050 | /* We need to clear any IRQ's pending in the soft interrupt |
888 | * registers, a spurious one could be left around from the | 1051 | * registers, a spurious one could be left around from the |
889 | * PROM timer which we just disabled. | 1052 | * PROM timer which we just disabled. |