aboutsummaryrefslogtreecommitdiffstats
path: root/arch/ia64/kernel/iosapic.c
diff options
context:
space:
mode:
Diffstat (limited to 'arch/ia64/kernel/iosapic.c')
-rw-r--r--arch/ia64/kernel/iosapic.c301
1 files changed, 135 insertions, 166 deletions
diff --git a/arch/ia64/kernel/iosapic.c b/arch/ia64/kernel/iosapic.c
index 37f46527d233..522b13d0bde3 100644
--- a/arch/ia64/kernel/iosapic.c
+++ b/arch/ia64/kernel/iosapic.c
@@ -209,9 +209,7 @@ gsi_to_irq (unsigned int gsi)
209 * and Linux irq numbers... 209 * and Linux irq numbers...
210 */ 210 */
211 spin_lock_irqsave(&iosapic_lock, flags); 211 spin_lock_irqsave(&iosapic_lock, flags);
212 { 212 irq = _gsi_to_vector(gsi);
213 irq = _gsi_to_vector(gsi);
214 }
215 spin_unlock_irqrestore(&iosapic_lock, flags); 213 spin_unlock_irqrestore(&iosapic_lock, flags);
216 214
217 return irq; 215 return irq;
@@ -322,15 +320,12 @@ mask_irq (unsigned int irq)
322 return; /* not an IOSAPIC interrupt! */ 320 return; /* not an IOSAPIC interrupt! */
323 321
324 spin_lock_irqsave(&iosapic_lock, flags); 322 spin_lock_irqsave(&iosapic_lock, flags);
325 { 323 /* set only the mask bit */
326 /* set only the mask bit */ 324 low32 = iosapic_intr_info[vec].low32 |= IOSAPIC_MASK;
327 low32 = iosapic_intr_info[vec].low32 |= IOSAPIC_MASK; 325 list_for_each_entry(rte, &iosapic_intr_info[vec].rtes, rte_list) {
328 list_for_each_entry(rte, &iosapic_intr_info[vec].rtes, 326 addr = rte->addr;
329 rte_list) { 327 rte_index = rte->rte_index;
330 addr = rte->addr; 328 iosapic_write(addr, IOSAPIC_RTE_LOW(rte_index), low32);
331 rte_index = rte->rte_index;
332 iosapic_write(addr, IOSAPIC_RTE_LOW(rte_index), low32);
333 }
334 } 329 }
335 spin_unlock_irqrestore(&iosapic_lock, flags); 330 spin_unlock_irqrestore(&iosapic_lock, flags);
336} 331}
@@ -349,14 +344,11 @@ unmask_irq (unsigned int irq)
349 return; /* not an IOSAPIC interrupt! */ 344 return; /* not an IOSAPIC interrupt! */
350 345
351 spin_lock_irqsave(&iosapic_lock, flags); 346 spin_lock_irqsave(&iosapic_lock, flags);
352 { 347 low32 = iosapic_intr_info[vec].low32 &= ~IOSAPIC_MASK;
353 low32 = iosapic_intr_info[vec].low32 &= ~IOSAPIC_MASK; 348 list_for_each_entry(rte, &iosapic_intr_info[vec].rtes, rte_list) {
354 list_for_each_entry(rte, &iosapic_intr_info[vec].rtes, 349 addr = rte->addr;
355 rte_list) { 350 rte_index = rte->rte_index;
356 addr = rte->addr; 351 iosapic_write(addr, IOSAPIC_RTE_LOW(rte_index), low32);
357 rte_index = rte->rte_index;
358 iosapic_write(addr, IOSAPIC_RTE_LOW(rte_index), low32);
359 }
360 } 352 }
361 spin_unlock_irqrestore(&iosapic_lock, flags); 353 spin_unlock_irqrestore(&iosapic_lock, flags);
362} 354}
@@ -391,28 +383,21 @@ iosapic_set_affinity (unsigned int irq, cpumask_t mask)
391 high32 = dest << IOSAPIC_DEST_SHIFT; 383 high32 = dest << IOSAPIC_DEST_SHIFT;
392 384
393 spin_lock_irqsave(&iosapic_lock, flags); 385 spin_lock_irqsave(&iosapic_lock, flags);
394 { 386 low32 = iosapic_intr_info[vec].low32 & ~(7 << IOSAPIC_DELIVERY_SHIFT);
395 low32 = iosapic_intr_info[vec].low32 & 387 if (redir)
396 ~(7 << IOSAPIC_DELIVERY_SHIFT); 388 /* change delivery mode to lowest priority */
397 389 low32 |= (IOSAPIC_LOWEST_PRIORITY << IOSAPIC_DELIVERY_SHIFT);
398 if (redir) 390 else
399 /* change delivery mode to lowest priority */ 391 /* change delivery mode to fixed */
400 low32 |= (IOSAPIC_LOWEST_PRIORITY << 392 low32 |= (IOSAPIC_FIXED << IOSAPIC_DELIVERY_SHIFT);
401 IOSAPIC_DELIVERY_SHIFT); 393
402 else 394 iosapic_intr_info[vec].low32 = low32;
403 /* change delivery mode to fixed */ 395 iosapic_intr_info[vec].dest = dest;
404 low32 |= (IOSAPIC_FIXED << IOSAPIC_DELIVERY_SHIFT); 396 list_for_each_entry(rte, &iosapic_intr_info[vec].rtes, rte_list) {
405 397 addr = rte->addr;
406 iosapic_intr_info[vec].low32 = low32; 398 rte_index = rte->rte_index;
407 iosapic_intr_info[vec].dest = dest; 399 iosapic_write(addr, IOSAPIC_RTE_HIGH(rte_index), high32);
408 list_for_each_entry(rte, &iosapic_intr_info[vec].rtes, 400 iosapic_write(addr, IOSAPIC_RTE_LOW(rte_index), low32);
409 rte_list) {
410 addr = rte->addr;
411 rte_index = rte->rte_index;
412 iosapic_write(addr, IOSAPIC_RTE_HIGH(rte_index),
413 high32);
414 iosapic_write(addr, IOSAPIC_RTE_LOW(rte_index), low32);
415 }
416 } 401 }
417 spin_unlock_irqrestore(&iosapic_lock, flags); 402 spin_unlock_irqrestore(&iosapic_lock, flags);
418#endif 403#endif
@@ -797,14 +782,12 @@ again:
797 * don't touch the RTE. 782 * don't touch the RTE.
798 */ 783 */
799 spin_lock_irqsave(&iosapic_lock, flags); 784 spin_lock_irqsave(&iosapic_lock, flags);
800 { 785 vector = gsi_to_vector(gsi);
801 vector = gsi_to_vector(gsi); 786 if (vector > 0) {
802 if (vector > 0) { 787 rte = gsi_vector_to_rte(gsi, vector);
803 rte = gsi_vector_to_rte(gsi, vector); 788 rte->refcnt++;
804 rte->refcnt++; 789 spin_unlock_irqrestore(&iosapic_lock, flags);
805 spin_unlock_irqrestore(&iosapic_lock, flags); 790 return vector;
806 return vector;
807 }
808 } 791 }
809 spin_unlock_irqrestore(&iosapic_lock, flags); 792 spin_unlock_irqrestore(&iosapic_lock, flags);
810 793
@@ -818,35 +801,31 @@ again:
818 801
819 spin_lock_irqsave(&irq_desc[vector].lock, flags); 802 spin_lock_irqsave(&irq_desc[vector].lock, flags);
820 spin_lock(&iosapic_lock); 803 spin_lock(&iosapic_lock);
821 { 804 if (gsi_to_vector(gsi) > 0) {
822 if (gsi_to_vector(gsi) > 0) { 805 if (list_empty(&iosapic_intr_info[vector].rtes))
823 if (list_empty(&iosapic_intr_info[vector].rtes)) 806 free_irq_vector(vector);
824 free_irq_vector(vector); 807 spin_unlock(&iosapic_lock);
825 spin_unlock(&iosapic_lock); 808 spin_unlock_irqrestore(&irq_desc[vector].lock, flags);
826 spin_unlock_irqrestore(&irq_desc[vector].lock, 809 goto again;
827 flags); 810 }
828 goto again;
829 }
830
831 dest = get_target_cpu(gsi, vector);
832 err = register_intr(gsi, vector, IOSAPIC_LOWEST_PRIORITY,
833 polarity, trigger);
834 if (err < 0) {
835 spin_unlock(&iosapic_lock);
836 spin_unlock_irqrestore(&irq_desc[vector].lock,
837 flags);
838 return err;
839 }
840 811
841 /* 812 dest = get_target_cpu(gsi, vector);
842 * If the vector is shared and already unmasked for 813 err = register_intr(gsi, vector, IOSAPIC_LOWEST_PRIORITY,
843 * other interrupt sources, don't mask it. 814 polarity, trigger);
844 */ 815 if (err < 0) {
845 low32 = iosapic_intr_info[vector].low32; 816 spin_unlock(&iosapic_lock);
846 if (vector_is_shared(vector) && !(low32 & IOSAPIC_MASK)) 817 spin_unlock_irqrestore(&irq_desc[vector].lock, flags);
847 mask = 0; 818 return err;
848 set_rte(gsi, vector, dest, mask);
849 } 819 }
820
821 /*
822 * If the vector is shared and already unmasked for other
823 * interrupt sources, don't mask it.
824 */
825 low32 = iosapic_intr_info[vector].low32;
826 if (vector_is_shared(vector) && !(low32 & IOSAPIC_MASK))
827 mask = 0;
828 set_rte(gsi, vector, dest, mask);
850 spin_unlock(&iosapic_lock); 829 spin_unlock(&iosapic_lock);
851 spin_unlock_irqrestore(&irq_desc[vector].lock, flags); 830 spin_unlock_irqrestore(&irq_desc[vector].lock, flags);
852 831
@@ -886,69 +865,64 @@ iosapic_unregister_intr (unsigned int gsi)
886 idesc = irq_desc + irq; 865 idesc = irq_desc + irq;
887 spin_lock_irqsave(&idesc->lock, flags); 866 spin_lock_irqsave(&idesc->lock, flags);
888 spin_lock(&iosapic_lock); 867 spin_lock(&iosapic_lock);
889 { 868 if ((rte = gsi_vector_to_rte(gsi, vector)) == NULL) {
890 if ((rte = gsi_vector_to_rte(gsi, vector)) == NULL) { 869 printk(KERN_ERR "iosapic_unregister_intr(%u) unbalanced\n",
891 printk(KERN_ERR 870 gsi);
892 "iosapic_unregister_intr(%u) unbalanced\n", 871 WARN_ON(1);
893 gsi); 872 goto out;
894 WARN_ON(1); 873 }
895 goto out;
896 }
897 874
898 if (--rte->refcnt > 0) 875 if (--rte->refcnt > 0)
899 goto out; 876 goto out;
900 877
901 /* Mask the interrupt */ 878 /* Mask the interrupt */
902 low32 = iosapic_intr_info[vector].low32 | IOSAPIC_MASK; 879 low32 = iosapic_intr_info[vector].low32 | IOSAPIC_MASK;
903 iosapic_write(rte->addr, IOSAPIC_RTE_LOW(rte->rte_index), 880 iosapic_write(rte->addr, IOSAPIC_RTE_LOW(rte->rte_index), low32);
904 low32);
905 881
906 /* Remove the rte entry from the list */ 882 /* Remove the rte entry from the list */
907 list_del(&rte->rte_list); 883 list_del(&rte->rte_list);
908 iosapic_intr_info[vector].count--; 884 iosapic_intr_info[vector].count--;
909 iosapic_free_rte(rte); 885 iosapic_free_rte(rte);
910 index = find_iosapic(gsi); 886 index = find_iosapic(gsi);
911 iosapic_lists[index].rtes_inuse--; 887 iosapic_lists[index].rtes_inuse--;
912 WARN_ON(iosapic_lists[index].rtes_inuse < 0); 888 WARN_ON(iosapic_lists[index].rtes_inuse < 0);
913
914 trigger = iosapic_intr_info[vector].trigger;
915 polarity = iosapic_intr_info[vector].polarity;
916 dest = iosapic_intr_info[vector].dest;
917 printk(KERN_INFO
918 "GSI %u (%s, %s) -> CPU %d (0x%04x)"
919 " vector %d unregistered\n",
920 gsi, (trigger == IOSAPIC_EDGE ? "edge" : "level"),
921 (polarity == IOSAPIC_POL_HIGH ? "high" : "low"),
922 cpu_logical_id(dest), dest, vector);
923 889
924 if (list_empty(&iosapic_intr_info[vector].rtes)) { 890 trigger = iosapic_intr_info[vector].trigger;
925 /* Sanity check */ 891 polarity = iosapic_intr_info[vector].polarity;
926 BUG_ON(iosapic_intr_info[vector].count); 892 dest = iosapic_intr_info[vector].dest;
893 printk(KERN_INFO
894 "GSI %u (%s, %s) -> CPU %d (0x%04x) vector %d unregistered\n",
895 gsi, (trigger == IOSAPIC_EDGE ? "edge" : "level"),
896 (polarity == IOSAPIC_POL_HIGH ? "high" : "low"),
897 cpu_logical_id(dest), dest, vector);
927 898
928 /* Clear the interrupt controller descriptor */ 899 if (list_empty(&iosapic_intr_info[vector].rtes)) {
929 idesc->chip = &no_irq_type; 900 /* Sanity check */
901 BUG_ON(iosapic_intr_info[vector].count);
902
903 /* Clear the interrupt controller descriptor */
904 idesc->chip = &no_irq_type;
930 905
931#ifdef CONFIG_SMP 906#ifdef CONFIG_SMP
932 /* Clear affinity */ 907 /* Clear affinity */
933 cpus_setall(idesc->affinity); 908 cpus_setall(idesc->affinity);
934#endif 909#endif
935 910
936 /* Clear the interrupt information */ 911 /* Clear the interrupt information */
937 memset(&iosapic_intr_info[vector], 0, 912 memset(&iosapic_intr_info[vector], 0,
938 sizeof(struct iosapic_intr_info)); 913 sizeof(struct iosapic_intr_info));
939 iosapic_intr_info[vector].low32 |= IOSAPIC_MASK; 914 iosapic_intr_info[vector].low32 |= IOSAPIC_MASK;
940 INIT_LIST_HEAD(&iosapic_intr_info[vector].rtes); 915 INIT_LIST_HEAD(&iosapic_intr_info[vector].rtes);
941
942 if (idesc->action) {
943 printk(KERN_ERR
944 "interrupt handlers still exist on"
945 "IRQ %u\n", irq);
946 WARN_ON(1);
947 }
948 916
949 /* Free the interrupt vector */ 917 if (idesc->action) {
950 free_irq_vector(vector); 918 printk(KERN_ERR
919 "interrupt handlers still exist on IRQ %u\n",
920 irq);
921 WARN_ON(1);
951 } 922 }
923
924 /* Free the interrupt vector */
925 free_irq_vector(vector);
952 } 926 }
953 out: 927 out:
954 spin_unlock(&iosapic_lock); 928 spin_unlock(&iosapic_lock);
@@ -1108,31 +1082,29 @@ iosapic_init (unsigned long phys_addr, unsigned int gsi_base)
1108 unsigned long flags; 1082 unsigned long flags;
1109 1083
1110 spin_lock_irqsave(&iosapic_lock, flags); 1084 spin_lock_irqsave(&iosapic_lock, flags);
1111 { 1085 addr = ioremap(phys_addr, 0);
1112 addr = ioremap(phys_addr, 0); 1086 ver = iosapic_version(addr);
1113 ver = iosapic_version(addr);
1114 1087
1115 if ((err = iosapic_check_gsi_range(gsi_base, ver))) { 1088 if ((err = iosapic_check_gsi_range(gsi_base, ver))) {
1116 iounmap(addr); 1089 iounmap(addr);
1117 spin_unlock_irqrestore(&iosapic_lock, flags); 1090 spin_unlock_irqrestore(&iosapic_lock, flags);
1118 return err; 1091 return err;
1119 } 1092 }
1120 1093
1121 /* 1094 /*
1122 * The MAX_REDIR register holds the highest input pin 1095 * The MAX_REDIR register holds the highest input pin number
1123 * number (starting from 0). 1096 * (starting from 0). We add 1 so that we can use it for
1124 * We add 1 so that we can use it for number of pins (= RTEs) 1097 * number of pins (= RTEs)
1125 */ 1098 */
1126 num_rte = ((ver >> 16) & 0xff) + 1; 1099 num_rte = ((ver >> 16) & 0xff) + 1;
1127 1100
1128 index = iosapic_alloc(); 1101 index = iosapic_alloc();
1129 iosapic_lists[index].addr = addr; 1102 iosapic_lists[index].addr = addr;
1130 iosapic_lists[index].gsi_base = gsi_base; 1103 iosapic_lists[index].gsi_base = gsi_base;
1131 iosapic_lists[index].num_rte = num_rte; 1104 iosapic_lists[index].num_rte = num_rte;
1132#ifdef CONFIG_NUMA 1105#ifdef CONFIG_NUMA
1133 iosapic_lists[index].node = MAX_NUMNODES; 1106 iosapic_lists[index].node = MAX_NUMNODES;
1134#endif 1107#endif
1135 }
1136 spin_unlock_irqrestore(&iosapic_lock, flags); 1108 spin_unlock_irqrestore(&iosapic_lock, flags);
1137 1109
1138 if ((gsi_base == 0) && pcat_compat) { 1110 if ((gsi_base == 0) && pcat_compat) {
@@ -1157,25 +1129,22 @@ iosapic_remove (unsigned int gsi_base)
1157 unsigned long flags; 1129 unsigned long flags;
1158 1130
1159 spin_lock_irqsave(&iosapic_lock, flags); 1131 spin_lock_irqsave(&iosapic_lock, flags);
1160 { 1132 index = find_iosapic(gsi_base);
1161 index = find_iosapic(gsi_base); 1133 if (index < 0) {
1162 if (index < 0) { 1134 printk(KERN_WARNING "%s: No IOSAPIC for GSI base %u\n",
1163 printk(KERN_WARNING "%s: No IOSAPIC for GSI base %u\n", 1135 __FUNCTION__, gsi_base);
1164 __FUNCTION__, gsi_base); 1136 goto out;
1165 goto out; 1137 }
1166 }
1167
1168 if (iosapic_lists[index].rtes_inuse) {
1169 err = -EBUSY;
1170 printk(KERN_WARNING
1171 "%s: IOSAPIC for GSI base %u is busy\n",
1172 __FUNCTION__, gsi_base);
1173 goto out;
1174 }
1175 1138
1176 iounmap(iosapic_lists[index].addr); 1139 if (iosapic_lists[index].rtes_inuse) {
1177 iosapic_free(index); 1140 err = -EBUSY;
1141 printk(KERN_WARNING "%s: IOSAPIC for GSI base %u is busy\n",
1142 __FUNCTION__, gsi_base);
1143 goto out;
1178 } 1144 }
1145
1146 iounmap(iosapic_lists[index].addr);
1147 iosapic_free(index);
1179 out: 1148 out:
1180 spin_unlock_irqrestore(&iosapic_lock, flags); 1149 spin_unlock_irqrestore(&iosapic_lock, flags);
1181 return err; 1150 return err;