diff options
Diffstat (limited to 'arch/x86_64/kernel/io_apic.c')
-rw-r--r-- | arch/x86_64/kernel/io_apic.c | 78 |
1 files changed, 73 insertions, 5 deletions
diff --git a/arch/x86_64/kernel/io_apic.c b/arch/x86_64/kernel/io_apic.c index 8dede0bd2267..48593f6b708f 100644 --- a/arch/x86_64/kernel/io_apic.c +++ b/arch/x86_64/kernel/io_apic.c | |||
@@ -36,6 +36,7 @@ | |||
36 | #include <acpi/acpi_bus.h> | 36 | #include <acpi/acpi_bus.h> |
37 | #endif | 37 | #endif |
38 | 38 | ||
39 | #include <asm/idle.h> | ||
39 | #include <asm/io.h> | 40 | #include <asm/io.h> |
40 | #include <asm/smp.h> | 41 | #include <asm/smp.h> |
41 | #include <asm/desc.h> | 42 | #include <asm/desc.h> |
@@ -49,7 +50,10 @@ | |||
49 | 50 | ||
50 | struct irq_cfg { | 51 | struct irq_cfg { |
51 | cpumask_t domain; | 52 | cpumask_t domain; |
53 | cpumask_t old_domain; | ||
54 | unsigned move_cleanup_count; | ||
52 | u8 vector; | 55 | u8 vector; |
56 | u8 move_in_progress : 1; | ||
53 | }; | 57 | }; |
54 | 58 | ||
55 | /* irq_cfg is indexed by the sum of all RTEs in all I/O APICs. */ | 59 | /* irq_cfg is indexed by the sum of all RTEs in all I/O APICs. */ |
@@ -652,7 +656,6 @@ static int __assign_irq_vector(int irq, cpumask_t mask) | |||
652 | * 0x80, because int 0x80 is hm, kind of importantish. ;) | 656 | * 0x80, because int 0x80 is hm, kind of importantish. ;) |
653 | */ | 657 | */ |
654 | static int current_vector = FIRST_DEVICE_VECTOR, current_offset = 0; | 658 | static int current_vector = FIRST_DEVICE_VECTOR, current_offset = 0; |
655 | cpumask_t old_mask = CPU_MASK_NONE; | ||
656 | unsigned int old_vector; | 659 | unsigned int old_vector; |
657 | int cpu; | 660 | int cpu; |
658 | struct irq_cfg *cfg; | 661 | struct irq_cfg *cfg; |
@@ -663,18 +666,20 @@ static int __assign_irq_vector(int irq, cpumask_t mask) | |||
663 | /* Only try and allocate irqs on cpus that are present */ | 666 | /* Only try and allocate irqs on cpus that are present */ |
664 | cpus_and(mask, mask, cpu_online_map); | 667 | cpus_and(mask, mask, cpu_online_map); |
665 | 668 | ||
669 | if ((cfg->move_in_progress) || cfg->move_cleanup_count) | ||
670 | return -EBUSY; | ||
671 | |||
666 | old_vector = cfg->vector; | 672 | old_vector = cfg->vector; |
667 | if (old_vector) { | 673 | if (old_vector) { |
668 | cpumask_t tmp; | 674 | cpumask_t tmp; |
669 | cpus_and(tmp, cfg->domain, mask); | 675 | cpus_and(tmp, cfg->domain, mask); |
670 | if (!cpus_empty(tmp)) | 676 | if (!cpus_empty(tmp)) |
671 | return 0; | 677 | return 0; |
672 | cpus_and(old_mask, cfg->domain, cpu_online_map); | ||
673 | } | 678 | } |
674 | 679 | ||
675 | for_each_cpu_mask(cpu, mask) { | 680 | for_each_cpu_mask(cpu, mask) { |
676 | cpumask_t domain, new_mask; | 681 | cpumask_t domain, new_mask; |
677 | int new_cpu, old_cpu; | 682 | int new_cpu; |
678 | int vector, offset; | 683 | int vector, offset; |
679 | 684 | ||
680 | domain = vector_allocation_domain(cpu); | 685 | domain = vector_allocation_domain(cpu); |
@@ -699,8 +704,10 @@ next: | |||
699 | /* Found one! */ | 704 | /* Found one! */ |
700 | current_vector = vector; | 705 | current_vector = vector; |
701 | current_offset = offset; | 706 | current_offset = offset; |
702 | for_each_cpu_mask(old_cpu, old_mask) | 707 | if (old_vector) { |
703 | per_cpu(vector_irq, old_cpu)[old_vector] = -1; | 708 | cfg->move_in_progress = 1; |
709 | cfg->old_domain = cfg->domain; | ||
710 | } | ||
704 | for_each_cpu_mask(new_cpu, new_mask) | 711 | for_each_cpu_mask(new_cpu, new_mask) |
705 | per_cpu(vector_irq, new_cpu)[vector] = irq; | 712 | per_cpu(vector_irq, new_cpu)[vector] = irq; |
706 | cfg->vector = vector; | 713 | cfg->vector = vector; |
@@ -1360,8 +1367,68 @@ static int ioapic_retrigger_irq(unsigned int irq) | |||
1360 | * races. | 1367 | * races. |
1361 | */ | 1368 | */ |
1362 | 1369 | ||
1370 | #ifdef CONFIG_SMP | ||
1371 | asmlinkage void smp_irq_move_cleanup_interrupt(void) | ||
1372 | { | ||
1373 | unsigned vector, me; | ||
1374 | ack_APIC_irq(); | ||
1375 | exit_idle(); | ||
1376 | irq_enter(); | ||
1377 | |||
1378 | me = smp_processor_id(); | ||
1379 | for (vector = FIRST_EXTERNAL_VECTOR; vector < NR_VECTORS; vector++) { | ||
1380 | unsigned int irq; | ||
1381 | struct irq_desc *desc; | ||
1382 | struct irq_cfg *cfg; | ||
1383 | irq = __get_cpu_var(vector_irq)[vector]; | ||
1384 | if (irq >= NR_IRQS) | ||
1385 | continue; | ||
1386 | |||
1387 | desc = irq_desc + irq; | ||
1388 | cfg = irq_cfg + irq; | ||
1389 | spin_lock(&desc->lock); | ||
1390 | if (!cfg->move_cleanup_count) | ||
1391 | goto unlock; | ||
1392 | |||
1393 | if ((vector == cfg->vector) && cpu_isset(me, cfg->domain)) | ||
1394 | goto unlock; | ||
1395 | |||
1396 | __get_cpu_var(vector_irq)[vector] = -1; | ||
1397 | cfg->move_cleanup_count--; | ||
1398 | unlock: | ||
1399 | spin_unlock(&desc->lock); | ||
1400 | } | ||
1401 | |||
1402 | irq_exit(); | ||
1403 | } | ||
1404 | |||
1405 | static void irq_complete_move(unsigned int irq) | ||
1406 | { | ||
1407 | struct irq_cfg *cfg = irq_cfg + irq; | ||
1408 | unsigned vector, me; | ||
1409 | |||
1410 | if (likely(!cfg->move_in_progress)) | ||
1411 | return; | ||
1412 | |||
1413 | vector = ~get_irq_regs()->orig_rax; | ||
1414 | me = smp_processor_id(); | ||
1415 | if ((vector == cfg->vector) && | ||
1416 | cpu_isset(smp_processor_id(), cfg->domain)) { | ||
1417 | cpumask_t cleanup_mask; | ||
1418 | |||
1419 | cpus_and(cleanup_mask, cfg->old_domain, cpu_online_map); | ||
1420 | cfg->move_cleanup_count = cpus_weight(cleanup_mask); | ||
1421 | send_IPI_mask(cleanup_mask, IRQ_MOVE_CLEANUP_VECTOR); | ||
1422 | cfg->move_in_progress = 0; | ||
1423 | } | ||
1424 | } | ||
1425 | #else | ||
1426 | static inline void irq_complete_move(unsigned int irq) {} | ||
1427 | #endif | ||
1428 | |||
1363 | static void ack_apic_edge(unsigned int irq) | 1429 | static void ack_apic_edge(unsigned int irq) |
1364 | { | 1430 | { |
1431 | irq_complete_move(irq); | ||
1365 | move_native_irq(irq); | 1432 | move_native_irq(irq); |
1366 | ack_APIC_irq(); | 1433 | ack_APIC_irq(); |
1367 | } | 1434 | } |
@@ -1370,6 +1437,7 @@ static void ack_apic_level(unsigned int irq) | |||
1370 | { | 1437 | { |
1371 | int do_unmask_irq = 0; | 1438 | int do_unmask_irq = 0; |
1372 | 1439 | ||
1440 | irq_complete_move(irq); | ||
1373 | #if defined(CONFIG_GENERIC_PENDING_IRQ) || defined(CONFIG_IRQBALANCE) | 1441 | #if defined(CONFIG_GENERIC_PENDING_IRQ) || defined(CONFIG_IRQBALANCE) |
1374 | /* If we are moving the irq we need to mask it */ | 1442 | /* If we are moving the irq we need to mask it */ |
1375 | if (unlikely(irq_desc[irq].status & IRQ_MOVE_PENDING)) { | 1443 | if (unlikely(irq_desc[irq].status & IRQ_MOVE_PENDING)) { |