aboutsummaryrefslogtreecommitdiffstats
path: root/arch/sparc64/kernel/irq.c
diff options
context:
space:
mode:
Diffstat (limited to 'arch/sparc64/kernel/irq.c')
-rw-r--r--arch/sparc64/kernel/irq.c345
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
44static void distribute_irqs(void); 46static 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
139extern unsigned long real_hard_smp_processor_id(void);
140
141static 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. */
140void enable_irq(unsigned int irq) 177void 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
335unsigned 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
297static void atomic_bucket_insert(struct ino_bucket *bucket) 366static 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
649extern irqreturn_t floppy_interrupt(int, void *, struct pt_regs *);; 730extern 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 */
652extern unsigned char *pdma_vaddr; 733extern 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);
727static int retarget_one_irq(struct irqaction *p, int goal_cpu) 808static 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
849void init_irqwork_curcpu(void) 928void 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" 935static 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. */ 948static 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
958static 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
975static 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
992static 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. */
1015void __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.