diff options
author | Mike Travis <travis@sgi.com> | 2008-12-17 18:21:39 -0500 |
---|---|---|
committer | Ingo Molnar <mingo@elte.hu> | 2008-12-18 05:59:24 -0500 |
commit | a775a38b1353161a6d7af86b667d6523c12c1a37 (patch) | |
tree | 3bb16b2345b1090bacbd06c64ca62298a5fc23d6 /arch/x86/kernel | |
parent | 9a3d8f735eee90bb5b1351983e946bc637041c01 (diff) |
x86: fix cpu_mask_to_apicid_and to include cpu_online_mask
Impact: fix potential APIC crash
In determining the destination apicid, there are usually three cpumasks
that are considered: the incoming cpumask arg, cfg->domain and the
cpu_online_mask. Since we are just introducing the cpu_mask_to_apicid_and
function, make sure it includes the cpu_online_mask in it's evaluation.
[Added with this patch.]
There are two io_apic.c functions that did not previously use the
cpu_online_mask: setup_IO_APIC_irq and msi_compose_msg. Both of these
simply used cpu_mask_to_apicid(cfg->domain & TARGET_CPUS), and all but
one arch (NUMAQ[*]) returns only online cpus in the TARGET_CPUS mask,
so the behavior is identical for all cases.
[*: NUMAQ bug?]
Note that alloc_cpumask_var is only used for the 32-bit cases where
it's highly likely that the cpumask set size will be small and therefore
CPUMASK_OFFSTACK=n. But if that's not the case, failing the allocate
will cause the same return value as the default.
Signed-off-by: Mike Travis <travis@sgi.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
Diffstat (limited to 'arch/x86/kernel')
-rw-r--r-- | arch/x86/kernel/genapic_flat_64.c | 4 | ||||
-rw-r--r-- | arch/x86/kernel/genx2apic_cluster.c | 4 | ||||
-rw-r--r-- | arch/x86/kernel/genx2apic_phys.c | 4 | ||||
-rw-r--r-- | arch/x86/kernel/genx2apic_uv_x.c | 4 |
4 files changed, 12 insertions, 4 deletions
diff --git a/arch/x86/kernel/genapic_flat_64.c b/arch/x86/kernel/genapic_flat_64.c index 7fa5f49c2dda..34185488e4fb 100644 --- a/arch/x86/kernel/genapic_flat_64.c +++ b/arch/x86/kernel/genapic_flat_64.c | |||
@@ -276,7 +276,9 @@ physflat_cpu_mask_to_apicid_and(const struct cpumask *cpumask, | |||
276 | * We're using fixed IRQ delivery, can only return one phys APIC ID. | 276 | * We're using fixed IRQ delivery, can only return one phys APIC ID. |
277 | * May as well be the first. | 277 | * May as well be the first. |
278 | */ | 278 | */ |
279 | cpu = cpumask_any_and(cpumask, andmask); | 279 | for_each_cpu_and(cpu, cpumask, andmask) |
280 | if (cpumask_test_cpu(cpu, cpu_online_mask)) | ||
281 | break; | ||
280 | if (cpu < nr_cpu_ids) | 282 | if (cpu < nr_cpu_ids) |
281 | return per_cpu(x86_cpu_to_apicid, cpu); | 283 | return per_cpu(x86_cpu_to_apicid, cpu); |
282 | return BAD_APICID; | 284 | return BAD_APICID; |
diff --git a/arch/x86/kernel/genx2apic_cluster.c b/arch/x86/kernel/genx2apic_cluster.c index 4716a0c9f936..d451c9b9fdff 100644 --- a/arch/x86/kernel/genx2apic_cluster.c +++ b/arch/x86/kernel/genx2apic_cluster.c | |||
@@ -133,7 +133,9 @@ static unsigned int x2apic_cpu_mask_to_apicid_and(const struct cpumask *cpumask, | |||
133 | * We're using fixed IRQ delivery, can only return one phys APIC ID. | 133 | * We're using fixed IRQ delivery, can only return one phys APIC ID. |
134 | * May as well be the first. | 134 | * May as well be the first. |
135 | */ | 135 | */ |
136 | cpu = cpumask_any_and(cpumask, andmask); | 136 | for_each_cpu_and(cpu, cpumask, andmask) |
137 | if (cpumask_test_cpu(cpu, cpu_online_mask)) | ||
138 | break; | ||
137 | if (cpu < nr_cpu_ids) | 139 | if (cpu < nr_cpu_ids) |
138 | return per_cpu(x86_cpu_to_apicid, cpu); | 140 | return per_cpu(x86_cpu_to_apicid, cpu); |
139 | return BAD_APICID; | 141 | return BAD_APICID; |
diff --git a/arch/x86/kernel/genx2apic_phys.c b/arch/x86/kernel/genx2apic_phys.c index b255507884f2..62895cf315ff 100644 --- a/arch/x86/kernel/genx2apic_phys.c +++ b/arch/x86/kernel/genx2apic_phys.c | |||
@@ -132,7 +132,9 @@ static unsigned int x2apic_cpu_mask_to_apicid_and(const struct cpumask *cpumask, | |||
132 | * We're using fixed IRQ delivery, can only return one phys APIC ID. | 132 | * We're using fixed IRQ delivery, can only return one phys APIC ID. |
133 | * May as well be the first. | 133 | * May as well be the first. |
134 | */ | 134 | */ |
135 | cpu = cpumask_any_and(cpumask, andmask); | 135 | for_each_cpu_and(cpu, cpumask, andmask) |
136 | if (cpumask_test_cpu(cpu, cpu_online_mask)) | ||
137 | break; | ||
136 | if (cpu < nr_cpu_ids) | 138 | if (cpu < nr_cpu_ids) |
137 | return per_cpu(x86_cpu_to_apicid, cpu); | 139 | return per_cpu(x86_cpu_to_apicid, cpu); |
138 | return BAD_APICID; | 140 | return BAD_APICID; |
diff --git a/arch/x86/kernel/genx2apic_uv_x.c b/arch/x86/kernel/genx2apic_uv_x.c index 3984682cd849..0e88be11227d 100644 --- a/arch/x86/kernel/genx2apic_uv_x.c +++ b/arch/x86/kernel/genx2apic_uv_x.c | |||
@@ -188,7 +188,9 @@ static unsigned int uv_cpu_mask_to_apicid_and(const struct cpumask *cpumask, | |||
188 | * We're using fixed IRQ delivery, can only return one phys APIC ID. | 188 | * We're using fixed IRQ delivery, can only return one phys APIC ID. |
189 | * May as well be the first. | 189 | * May as well be the first. |
190 | */ | 190 | */ |
191 | cpu = cpumask_any_and(cpumask, andmask); | 191 | for_each_cpu_and(cpu, cpumask, andmask) |
192 | if (cpumask_test_cpu(cpu, cpu_online_mask)) | ||
193 | break; | ||
192 | if (cpu < nr_cpu_ids) | 194 | if (cpu < nr_cpu_ids) |
193 | return per_cpu(x86_cpu_to_apicid, cpu); | 195 | return per_cpu(x86_cpu_to_apicid, cpu); |
194 | return BAD_APICID; | 196 | return BAD_APICID; |