提交 91b265bf 编写于 作者: M Martin K. Petersen 提交者: Christoph Hellwig

mpt3sas: Rework the MSI-X grouping code

On systems with a non power-of-two CPU count the existing MSI-X grouping
code failed to distribute interrupts correctly. Rework the code to
handle arbitrary processor counts.

Also remove the hardcoded upper limit on the number of processors so we
can boot on large systems.
Signed-off-by: NMartin K. Petersen <martin.petersen@oracle.com>
Acked-by: NSreekanth Reddy <Sreekanth.reddy@avagotech.com>
Signed-off-by: NChristoph Hellwig <hch@lst.de>
上级 cbbb7b31
...@@ -1624,66 +1624,35 @@ _base_request_irq(struct MPT3SAS_ADAPTER *ioc, u8 index, u32 vector) ...@@ -1624,66 +1624,35 @@ _base_request_irq(struct MPT3SAS_ADAPTER *ioc, u8 index, u32 vector)
static void static void
_base_assign_reply_queues(struct MPT3SAS_ADAPTER *ioc) _base_assign_reply_queues(struct MPT3SAS_ADAPTER *ioc)
{ {
struct adapter_reply_queue *reply_q; unsigned int cpu, nr_cpus, nr_msix, index = 0;
int cpu_id;
int cpu_grouping, loop, grouping, grouping_mod;
int reply_queue;
if (!_base_is_controller_msix_enabled(ioc)) if (!_base_is_controller_msix_enabled(ioc))
return; return;
memset(ioc->cpu_msix_table, 0, ioc->cpu_msix_table_sz); memset(ioc->cpu_msix_table, 0, ioc->cpu_msix_table_sz);
/* NUMA Hardware bug workaround - drop to less reply queues */ nr_cpus = num_online_cpus();
if (ioc->reply_queue_count > ioc->facts.MaxMSIxVectors) { nr_msix = ioc->reply_queue_count = min(ioc->reply_queue_count,
ioc->reply_queue_count = ioc->facts.MaxMSIxVectors; ioc->facts.MaxMSIxVectors);
reply_queue = 0; if (!nr_msix)
list_for_each_entry(reply_q, &ioc->reply_queue_list, list) { return;
reply_q->msix_index = reply_queue;
if (++reply_queue == ioc->reply_queue_count)
reply_queue = 0;
}
}
/* when there are more cpus than available msix vectors, cpu = cpumask_first(cpu_online_mask);
* then group cpus togeather on same irq
*/ do {
if (ioc->cpu_count > ioc->msix_vector_count) { unsigned int i, group = nr_cpus / nr_msix;
grouping = ioc->cpu_count / ioc->msix_vector_count;
grouping_mod = ioc->cpu_count % ioc->msix_vector_count; if (index < nr_cpus % nr_msix)
if (grouping < 2 || (grouping == 2 && !grouping_mod)) group++;
cpu_grouping = 2;
else if (grouping < 4 || (grouping == 4 && !grouping_mod)) for (i = 0 ; i < group ; i++) {
cpu_grouping = 4; ioc->cpu_msix_table[cpu] = index;
else if (grouping < 8 || (grouping == 8 && !grouping_mod)) cpu = cpumask_next(cpu, cpu_online_mask);
cpu_grouping = 8;
else
cpu_grouping = 16;
} else
cpu_grouping = 0;
loop = 0;
reply_q = list_entry(ioc->reply_queue_list.next,
struct adapter_reply_queue, list);
for_each_online_cpu(cpu_id) {
if (!cpu_grouping) {
ioc->cpu_msix_table[cpu_id] = reply_q->msix_index;
reply_q = list_entry(reply_q->list.next,
struct adapter_reply_queue, list);
} else {
if (loop < cpu_grouping) {
ioc->cpu_msix_table[cpu_id] =
reply_q->msix_index;
loop++;
} else {
reply_q = list_entry(reply_q->list.next,
struct adapter_reply_queue, list);
ioc->cpu_msix_table[cpu_id] =
reply_q->msix_index;
loop = 1;
}
}
} }
index++;
} while (cpu < nr_cpus);
} }
/** /**
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册