aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86/kernel/apic
diff options
context:
space:
mode:
authorSuresh Siddha <suresh.b.siddha@intel.com>2012-06-25 16:38:29 -0400
committerIngo Molnar <mingo@kernel.org>2012-07-06 05:00:23 -0400
commitd872818dbbeed1bccf58c7f8c7db432154c802f9 (patch)
tree76d505388da3ad4b9d6fdcf48297c38165563a0a /arch/x86/kernel/apic
parent1ac322d0b169c95ce34d55b3ed6d40ce1a5f3a02 (diff)
x86/apic/x2apic: Use multiple cluster members for the irq destination only with the explicit affinity
During boot or driver load etc, interrupt destination is setup using default target cpu's. Later the user (irqbalance etc) or the driver (irq_set_affinity/ irq_set_affinity_hint) can request the interrupt to be migrated to some specific set of cpu's. In the x2apic cluster routing, for the default scenario use single cpu as the interrupt destination and when there is an explicit interrupt affinity request, route the interrupt to multiple members of a x2apic cluster specified in the cpumask of the migration request. This will minmize the vector pressure when there are lot of interrupt sources and relatively few x2apic clusters (for example a single socket server). This will allow the performance critical interrupts to be routed to multiple cpu's in the x2apic cluster (irqbalance for example uses the cache siblings etc while specifying the interrupt destination) and allow non-critical interrupts to be serviced by a single logical cpu. Signed-off-by: Suresh Siddha <suresh.b.siddha@intel.com> Acked-by: Yinghai Lu <yinghai@kernel.org> Acked-by: Alexander Gordeev <agordeev@redhat.com> Acked-by: Cyrill Gorcunov <gorcunov@openvz.org> Link: http://lkml.kernel.org/r/1340656709-11423-4-git-send-email-suresh.b.siddha@intel.com Signed-off-by: Ingo Molnar <mingo@kernel.org>
Diffstat (limited to 'arch/x86/kernel/apic')
-rw-r--r--arch/x86/kernel/apic/x2apic_cluster.c21
1 files changed, 19 insertions, 2 deletions
diff --git a/arch/x86/kernel/apic/x2apic_cluster.c b/arch/x86/kernel/apic/x2apic_cluster.c
index bde78d0098a4..c88baa4ff0e5 100644
--- a/arch/x86/kernel/apic/x2apic_cluster.c
+++ b/arch/x86/kernel/apic/x2apic_cluster.c
@@ -209,13 +209,30 @@ static int x2apic_cluster_probe(void)
209 return 0; 209 return 0;
210} 210}
211 211
212static const struct cpumask *x2apic_cluster_target_cpus(void)
213{
214 return cpu_all_mask;
215}
216
212/* 217/*
213 * Each x2apic cluster is an allocation domain. 218 * Each x2apic cluster is an allocation domain.
214 */ 219 */
215static void cluster_vector_allocation_domain(int cpu, struct cpumask *retmask, 220static void cluster_vector_allocation_domain(int cpu, struct cpumask *retmask,
216 const struct cpumask *mask) 221 const struct cpumask *mask)
217{ 222{
218 cpumask_and(retmask, mask, per_cpu(cpus_in_cluster, cpu)); 223 /*
224 * To minimize vector pressure, default case of boot, device bringup
225 * etc will use a single cpu for the interrupt destination.
226 *
227 * On explicit migration requests coming from irqbalance etc,
228 * interrupts will be routed to the x2apic cluster (cluster-id
229 * derived from the first cpu in the mask) members specified
230 * in the mask.
231 */
232 if (mask == x2apic_cluster_target_cpus())
233 cpumask_copy(retmask, cpumask_of(cpu));
234 else
235 cpumask_and(retmask, mask, per_cpu(cpus_in_cluster, cpu));
219} 236}
220 237
221static struct apic apic_x2apic_cluster = { 238static struct apic apic_x2apic_cluster = {
@@ -229,7 +246,7 @@ static struct apic apic_x2apic_cluster = {
229 .irq_delivery_mode = dest_LowestPrio, 246 .irq_delivery_mode = dest_LowestPrio,
230 .irq_dest_mode = 1, /* logical */ 247 .irq_dest_mode = 1, /* logical */
231 248
232 .target_cpus = online_target_cpus, 249 .target_cpus = x2apic_cluster_target_cpus,
233 .disable_esr = 0, 250 .disable_esr = 0,
234 .dest_logical = APIC_DEST_LOGICAL, 251 .dest_logical = APIC_DEST_LOGICAL,
235 .check_apicid_used = NULL, 252 .check_apicid_used = NULL,