diff options
author | Suresh Siddha <suresh.b.siddha@intel.com> | 2008-07-10 14:16:49 -0400 |
---|---|---|
committer | Ingo Molnar <mingo@elte.hu> | 2008-07-12 02:44:59 -0400 |
commit | 1b374e4d6f8b3eb2fcd034fcc24ea8ba1dfde7aa (patch) | |
tree | faf5aa00e344e473957206bc82ffbb746e438d0b /arch/x86/kernel | |
parent | 2d7a66d02e11af9ab8e16c76d22767e622b4e3d7 (diff) |
x64, x2apic/intr-remap: basic apic ops support
Introduce basic apic operations which handle the apic programming. This
will be used later to introduce another specific operations for x2apic.
For the perfomance critial accesses like IPI's, EOI etc, we use the
native operations as they are already referenced by different
indirections like genapic, irq_chip etc.
64bit Paravirt ops can also define their apic operations accordingly.
Signed-off-by: Suresh Siddha <suresh.b.siddha@intel.com>
Cc: akpm@linux-foundation.org
Cc: arjan@linux.intel.com
Cc: andi@firstfloor.org
Cc: ebiederm@xmission.com
Cc: jbarnes@virtuousgeek.org
Cc: steiner@sgi.com
Signed-off-by: Ingo Molnar <mingo@elte.hu>
Diffstat (limited to 'arch/x86/kernel')
-rw-r--r-- | arch/x86/kernel/apic_32.c | 6 | ||||
-rw-r--r-- | arch/x86/kernel/apic_64.c | 34 | ||||
-rw-r--r-- | arch/x86/kernel/io_apic_64.c | 8 | ||||
-rw-r--r-- | arch/x86/kernel/paravirt.c | 8 | ||||
-rw-r--r-- | arch/x86/kernel/smpboot.c | 28 |
5 files changed, 58 insertions, 26 deletions
diff --git a/arch/x86/kernel/apic_32.c b/arch/x86/kernel/apic_32.c index 3e58b676d23b..2a83c07bd887 100644 --- a/arch/x86/kernel/apic_32.c +++ b/arch/x86/kernel/apic_32.c | |||
@@ -145,6 +145,12 @@ static int modern_apic(void) | |||
145 | return lapic_get_version() >= 0x14; | 145 | return lapic_get_version() >= 0x14; |
146 | } | 146 | } |
147 | 147 | ||
148 | void apic_icr_write(u32 low, u32 id) | ||
149 | { | ||
150 | apic_write_around(APIC_ICR2, SET_APIC_DEST_FIELD(id)); | ||
151 | apic_write_around(APIC_ICR, low); | ||
152 | } | ||
153 | |||
148 | void apic_wait_icr_idle(void) | 154 | void apic_wait_icr_idle(void) |
149 | { | 155 | { |
150 | while (apic_read(APIC_ICR) & APIC_ICR_BUSY) | 156 | while (apic_read(APIC_ICR) & APIC_ICR_BUSY) |
diff --git a/arch/x86/kernel/apic_64.c b/arch/x86/kernel/apic_64.c index 3963f590c3d4..9bb040689b31 100644 --- a/arch/x86/kernel/apic_64.c +++ b/arch/x86/kernel/apic_64.c | |||
@@ -119,13 +119,13 @@ static int modern_apic(void) | |||
119 | return lapic_get_version() >= 0x14; | 119 | return lapic_get_version() >= 0x14; |
120 | } | 120 | } |
121 | 121 | ||
122 | void apic_wait_icr_idle(void) | 122 | void xapic_wait_icr_idle(void) |
123 | { | 123 | { |
124 | while (apic_read(APIC_ICR) & APIC_ICR_BUSY) | 124 | while (apic_read(APIC_ICR) & APIC_ICR_BUSY) |
125 | cpu_relax(); | 125 | cpu_relax(); |
126 | } | 126 | } |
127 | 127 | ||
128 | u32 safe_apic_wait_icr_idle(void) | 128 | u32 safe_xapic_wait_icr_idle(void) |
129 | { | 129 | { |
130 | u32 send_status; | 130 | u32 send_status; |
131 | int timeout; | 131 | int timeout; |
@@ -141,6 +141,36 @@ u32 safe_apic_wait_icr_idle(void) | |||
141 | return send_status; | 141 | return send_status; |
142 | } | 142 | } |
143 | 143 | ||
144 | void xapic_icr_write(u32 low, u32 id) | ||
145 | { | ||
146 | apic_write(APIC_ICR2, id << 24); | ||
147 | apic_write(APIC_ICR, low); | ||
148 | } | ||
149 | |||
150 | u64 xapic_icr_read(void) | ||
151 | { | ||
152 | u32 icr1, icr2; | ||
153 | |||
154 | icr2 = apic_read(APIC_ICR2); | ||
155 | icr1 = apic_read(APIC_ICR); | ||
156 | |||
157 | return (icr1 | ((u64)icr2 << 32)); | ||
158 | } | ||
159 | |||
160 | static struct apic_ops xapic_ops = { | ||
161 | .read = native_apic_mem_read, | ||
162 | .write = native_apic_mem_write, | ||
163 | .write_atomic = native_apic_mem_write_atomic, | ||
164 | .icr_read = xapic_icr_read, | ||
165 | .icr_write = xapic_icr_write, | ||
166 | .wait_icr_idle = xapic_wait_icr_idle, | ||
167 | .safe_wait_icr_idle = safe_xapic_wait_icr_idle, | ||
168 | }; | ||
169 | |||
170 | struct apic_ops __read_mostly *apic_ops = &xapic_ops; | ||
171 | |||
172 | EXPORT_SYMBOL_GPL(apic_ops); | ||
173 | |||
144 | /** | 174 | /** |
145 | * enable_NMI_through_LVT0 - enable NMI through local vector table 0 | 175 | * enable_NMI_through_LVT0 - enable NMI through local vector table 0 |
146 | */ | 176 | */ |
diff --git a/arch/x86/kernel/io_apic_64.c b/arch/x86/kernel/io_apic_64.c index 84dd63c13d63..b62d42ef9283 100644 --- a/arch/x86/kernel/io_apic_64.c +++ b/arch/x86/kernel/io_apic_64.c | |||
@@ -1157,6 +1157,7 @@ static __apicdebuginit void print_APIC_bitfield (int base) | |||
1157 | void __apicdebuginit print_local_APIC(void * dummy) | 1157 | void __apicdebuginit print_local_APIC(void * dummy) |
1158 | { | 1158 | { |
1159 | unsigned int v, ver, maxlvt; | 1159 | unsigned int v, ver, maxlvt; |
1160 | unsigned long icr; | ||
1160 | 1161 | ||
1161 | if (apic_verbosity == APIC_QUIET) | 1162 | if (apic_verbosity == APIC_QUIET) |
1162 | return; | 1163 | return; |
@@ -1200,10 +1201,9 @@ void __apicdebuginit print_local_APIC(void * dummy) | |||
1200 | v = apic_read(APIC_ESR); | 1201 | v = apic_read(APIC_ESR); |
1201 | printk(KERN_DEBUG "... APIC ESR: %08x\n", v); | 1202 | printk(KERN_DEBUG "... APIC ESR: %08x\n", v); |
1202 | 1203 | ||
1203 | v = apic_read(APIC_ICR); | 1204 | icr = apic_icr_read(); |
1204 | printk(KERN_DEBUG "... APIC ICR: %08x\n", v); | 1205 | printk(KERN_DEBUG "... APIC ICR: %08x\n", icr); |
1205 | v = apic_read(APIC_ICR2); | 1206 | printk(KERN_DEBUG "... APIC ICR2: %08x\n", icr >> 32); |
1206 | printk(KERN_DEBUG "... APIC ICR2: %08x\n", v); | ||
1207 | 1207 | ||
1208 | v = apic_read(APIC_LVTT); | 1208 | v = apic_read(APIC_LVTT); |
1209 | printk(KERN_DEBUG "... APIC LVTT: %08x\n", v); | 1209 | printk(KERN_DEBUG "... APIC LVTT: %08x\n", v); |
diff --git a/arch/x86/kernel/paravirt.c b/arch/x86/kernel/paravirt.c index e0f571d58c19..b80105a0f474 100644 --- a/arch/x86/kernel/paravirt.c +++ b/arch/x86/kernel/paravirt.c | |||
@@ -360,9 +360,11 @@ struct pv_cpu_ops pv_cpu_ops = { | |||
360 | 360 | ||
361 | struct pv_apic_ops pv_apic_ops = { | 361 | struct pv_apic_ops pv_apic_ops = { |
362 | #ifdef CONFIG_X86_LOCAL_APIC | 362 | #ifdef CONFIG_X86_LOCAL_APIC |
363 | .apic_write = native_apic_write, | 363 | #ifnded CONFIG_X86_64 |
364 | .apic_write_atomic = native_apic_write_atomic, | 364 | .apic_write = native_apic_mem_write, |
365 | .apic_read = native_apic_read, | 365 | .apic_write_atomic = native_apic_mem_write_atomic, |
366 | .apic_read = native_apic_mem_read, | ||
367 | #endif | ||
366 | .setup_boot_clock = setup_boot_APIC_clock, | 368 | .setup_boot_clock = setup_boot_APIC_clock, |
367 | .setup_secondary_clock = setup_secondary_APIC_clock, | 369 | .setup_secondary_clock = setup_secondary_APIC_clock, |
368 | .startup_ipi_hook = paravirt_nop, | 370 | .startup_ipi_hook = paravirt_nop, |
diff --git a/arch/x86/kernel/smpboot.c b/arch/x86/kernel/smpboot.c index f35c2d8016ac..c55263b3df02 100644 --- a/arch/x86/kernel/smpboot.c +++ b/arch/x86/kernel/smpboot.c | |||
@@ -123,7 +123,6 @@ EXPORT_PER_CPU_SYMBOL(cpu_info); | |||
123 | 123 | ||
124 | static atomic_t init_deasserted; | 124 | static atomic_t init_deasserted; |
125 | 125 | ||
126 | static int boot_cpu_logical_apicid; | ||
127 | 126 | ||
128 | /* representing cpus for which sibling maps can be computed */ | 127 | /* representing cpus for which sibling maps can be computed */ |
129 | static cpumask_t cpu_sibling_setup_map; | 128 | static cpumask_t cpu_sibling_setup_map; |
@@ -165,6 +164,8 @@ static void unmap_cpu_to_node(int cpu) | |||
165 | #endif | 164 | #endif |
166 | 165 | ||
167 | #ifdef CONFIG_X86_32 | 166 | #ifdef CONFIG_X86_32 |
167 | static int boot_cpu_logical_apicid; | ||
168 | |||
168 | u8 cpu_2_logical_apicid[NR_CPUS] __read_mostly = | 169 | u8 cpu_2_logical_apicid[NR_CPUS] __read_mostly = |
169 | { [0 ... NR_CPUS-1] = BAD_APICID }; | 170 | { [0 ... NR_CPUS-1] = BAD_APICID }; |
170 | 171 | ||
@@ -546,8 +547,7 @@ static inline void __inquire_remote_apic(int apicid) | |||
546 | printk(KERN_CONT | 547 | printk(KERN_CONT |
547 | "a previous APIC delivery may have failed\n"); | 548 | "a previous APIC delivery may have failed\n"); |
548 | 549 | ||
549 | apic_write_around(APIC_ICR2, SET_APIC_DEST_FIELD(apicid)); | 550 | apic_icr_write(APIC_DM_REMRD | regs[i], apicid); |
550 | apic_write_around(APIC_ICR, APIC_DM_REMRD | regs[i]); | ||
551 | 551 | ||
552 | timeout = 0; | 552 | timeout = 0; |
553 | do { | 553 | do { |
@@ -579,11 +579,9 @@ wakeup_secondary_cpu(int logical_apicid, unsigned long start_eip) | |||
579 | int maxlvt; | 579 | int maxlvt; |
580 | 580 | ||
581 | /* Target chip */ | 581 | /* Target chip */ |
582 | apic_write_around(APIC_ICR2, SET_APIC_DEST_FIELD(logical_apicid)); | ||
583 | |||
584 | /* Boot on the stack */ | 582 | /* Boot on the stack */ |
585 | /* Kick the second */ | 583 | /* Kick the second */ |
586 | apic_write_around(APIC_ICR, APIC_DM_NMI | APIC_DEST_LOGICAL); | 584 | apic_icr_write(APIC_DM_NMI | APIC_DEST_LOGICAL, logical_apicid); |
587 | 585 | ||
588 | Dprintk("Waiting for send to finish...\n"); | 586 | Dprintk("Waiting for send to finish...\n"); |
589 | send_status = safe_apic_wait_icr_idle(); | 587 | send_status = safe_apic_wait_icr_idle(); |
@@ -639,13 +637,11 @@ wakeup_secondary_cpu(int phys_apicid, unsigned long start_eip) | |||
639 | /* | 637 | /* |
640 | * Turn INIT on target chip | 638 | * Turn INIT on target chip |
641 | */ | 639 | */ |
642 | apic_write_around(APIC_ICR2, SET_APIC_DEST_FIELD(phys_apicid)); | ||
643 | |||
644 | /* | 640 | /* |
645 | * Send IPI | 641 | * Send IPI |
646 | */ | 642 | */ |
647 | apic_write_around(APIC_ICR, APIC_INT_LEVELTRIG | APIC_INT_ASSERT | 643 | apic_icr_write(APIC_INT_LEVELTRIG | APIC_INT_ASSERT | APIC_DM_INIT, |
648 | | APIC_DM_INIT); | 644 | phys_apicid); |
649 | 645 | ||
650 | Dprintk("Waiting for send to finish...\n"); | 646 | Dprintk("Waiting for send to finish...\n"); |
651 | send_status = safe_apic_wait_icr_idle(); | 647 | send_status = safe_apic_wait_icr_idle(); |
@@ -655,10 +651,8 @@ wakeup_secondary_cpu(int phys_apicid, unsigned long start_eip) | |||
655 | Dprintk("Deasserting INIT.\n"); | 651 | Dprintk("Deasserting INIT.\n"); |
656 | 652 | ||
657 | /* Target chip */ | 653 | /* Target chip */ |
658 | apic_write_around(APIC_ICR2, SET_APIC_DEST_FIELD(phys_apicid)); | ||
659 | |||
660 | /* Send IPI */ | 654 | /* Send IPI */ |
661 | apic_write_around(APIC_ICR, APIC_INT_LEVELTRIG | APIC_DM_INIT); | 655 | apic_icr_write(APIC_INT_LEVELTRIG | APIC_DM_INIT, phys_apicid); |
662 | 656 | ||
663 | Dprintk("Waiting for send to finish...\n"); | 657 | Dprintk("Waiting for send to finish...\n"); |
664 | send_status = safe_apic_wait_icr_idle(); | 658 | send_status = safe_apic_wait_icr_idle(); |
@@ -703,12 +697,10 @@ wakeup_secondary_cpu(int phys_apicid, unsigned long start_eip) | |||
703 | */ | 697 | */ |
704 | 698 | ||
705 | /* Target chip */ | 699 | /* Target chip */ |
706 | apic_write_around(APIC_ICR2, SET_APIC_DEST_FIELD(phys_apicid)); | ||
707 | |||
708 | /* Boot on the stack */ | 700 | /* Boot on the stack */ |
709 | /* Kick the second */ | 701 | /* Kick the second */ |
710 | apic_write_around(APIC_ICR, APIC_DM_STARTUP | 702 | apic_icr_write(APIC_DM_STARTUP | (start_eip >> 12), |
711 | | (start_eip >> 12)); | 703 | phys_apicid); |
712 | 704 | ||
713 | /* | 705 | /* |
714 | * Give the other CPU some time to accept the IPI. | 706 | * Give the other CPU some time to accept the IPI. |
@@ -1147,7 +1139,9 @@ void __init native_smp_prepare_cpus(unsigned int max_cpus) | |||
1147 | * Setup boot CPU information | 1139 | * Setup boot CPU information |
1148 | */ | 1140 | */ |
1149 | smp_store_cpu_info(0); /* Final full version of the data */ | 1141 | smp_store_cpu_info(0); /* Final full version of the data */ |
1142 | #ifdef CONFIG_X86_32 | ||
1150 | boot_cpu_logical_apicid = logical_smp_processor_id(); | 1143 | boot_cpu_logical_apicid = logical_smp_processor_id(); |
1144 | #endif | ||
1151 | current_thread_info()->cpu = 0; /* needed? */ | 1145 | current_thread_info()->cpu = 0; /* needed? */ |
1152 | set_cpu_sibling_map(0); | 1146 | set_cpu_sibling_map(0); |
1153 | 1147 | ||