提交 5a52a7ac 编写于 作者: S Sebastian Sanchez 提交者: Doug Ledford

IB/hfi1: NULL pointer dereference when freeing rhashtable

A NULL pointer dereference occurs when the driver
is unloaded, and the SDMA rhashtable is freed if
the rhashtable_init() function has not been called.
Prevent this by changing sdma_rht to be a pointer
to a dynamically allocated hash table. The NULL-ness
of the pointer serves as an indication that the hash
table was initialized and that it needs to be
destroyed.

Fixes: 0cb2aa69 ("IB/hfi1: Add sysfs interface for affinity setup")
Reviewed-by: NMike Marciniszyn <mike.marciniszyn@intel.com>
Signed-off-by: NSebastian Sanchez <sebastian.sanchez@intel.com>
Signed-off-by: NDennis Dalessandro <dennis.dalessandro@intel.com>
Signed-off-by: NDoug Ledford <dledford@redhat.com>
上级 8688426b
...@@ -1167,7 +1167,7 @@ struct hfi1_devdata { ...@@ -1167,7 +1167,7 @@ struct hfi1_devdata {
bool eprom_available; /* true if EPROM is available for this device */ bool eprom_available; /* true if EPROM is available for this device */
bool aspm_supported; /* Does HW support ASPM */ bool aspm_supported; /* Does HW support ASPM */
bool aspm_enabled; /* ASPM state: enabled/disabled */ bool aspm_enabled; /* ASPM state: enabled/disabled */
struct rhashtable sdma_rht; struct rhashtable *sdma_rht;
struct kobject kobj; struct kobject kobj;
}; };
......
...@@ -868,7 +868,7 @@ struct sdma_engine *sdma_select_user_engine(struct hfi1_devdata *dd, ...@@ -868,7 +868,7 @@ struct sdma_engine *sdma_select_user_engine(struct hfi1_devdata *dd,
cpu_id = smp_processor_id(); cpu_id = smp_processor_id();
rcu_read_lock(); rcu_read_lock();
rht_node = rhashtable_lookup_fast(&dd->sdma_rht, &cpu_id, rht_node = rhashtable_lookup_fast(dd->sdma_rht, &cpu_id,
sdma_rht_params); sdma_rht_params);
if (rht_node && rht_node->map[vl]) { if (rht_node && rht_node->map[vl]) {
...@@ -962,7 +962,7 @@ ssize_t sdma_set_cpu_to_sde_map(struct sdma_engine *sde, const char *buf, ...@@ -962,7 +962,7 @@ ssize_t sdma_set_cpu_to_sde_map(struct sdma_engine *sde, const char *buf,
continue; continue;
} }
rht_node = rhashtable_lookup_fast(&dd->sdma_rht, &cpu, rht_node = rhashtable_lookup_fast(dd->sdma_rht, &cpu,
sdma_rht_params); sdma_rht_params);
if (!rht_node) { if (!rht_node) {
rht_node = kzalloc(sizeof(*rht_node), GFP_KERNEL); rht_node = kzalloc(sizeof(*rht_node), GFP_KERNEL);
...@@ -982,7 +982,7 @@ ssize_t sdma_set_cpu_to_sde_map(struct sdma_engine *sde, const char *buf, ...@@ -982,7 +982,7 @@ ssize_t sdma_set_cpu_to_sde_map(struct sdma_engine *sde, const char *buf,
rht_node->map[vl]->ctr = 1; rht_node->map[vl]->ctr = 1;
rht_node->map[vl]->sde[0] = sde; rht_node->map[vl]->sde[0] = sde;
ret = rhashtable_insert_fast(&dd->sdma_rht, ret = rhashtable_insert_fast(dd->sdma_rht,
&rht_node->node, &rht_node->node,
sdma_rht_params); sdma_rht_params);
if (ret) { if (ret) {
...@@ -1025,7 +1025,7 @@ ssize_t sdma_set_cpu_to_sde_map(struct sdma_engine *sde, const char *buf, ...@@ -1025,7 +1025,7 @@ ssize_t sdma_set_cpu_to_sde_map(struct sdma_engine *sde, const char *buf,
if (cpumask_test_cpu(cpu, mask)) if (cpumask_test_cpu(cpu, mask))
continue; continue;
rht_node = rhashtable_lookup_fast(&dd->sdma_rht, &cpu, rht_node = rhashtable_lookup_fast(dd->sdma_rht, &cpu,
sdma_rht_params); sdma_rht_params);
if (rht_node) { if (rht_node) {
bool empty = true; bool empty = true;
...@@ -1049,7 +1049,7 @@ ssize_t sdma_set_cpu_to_sde_map(struct sdma_engine *sde, const char *buf, ...@@ -1049,7 +1049,7 @@ ssize_t sdma_set_cpu_to_sde_map(struct sdma_engine *sde, const char *buf,
} }
if (empty) { if (empty) {
ret = rhashtable_remove_fast(&dd->sdma_rht, ret = rhashtable_remove_fast(dd->sdma_rht,
&rht_node->node, &rht_node->node,
sdma_rht_params); sdma_rht_params);
WARN_ON(ret); WARN_ON(ret);
...@@ -1108,7 +1108,7 @@ void sdma_seqfile_dump_cpu_list(struct seq_file *s, ...@@ -1108,7 +1108,7 @@ void sdma_seqfile_dump_cpu_list(struct seq_file *s,
struct sdma_rht_node *rht_node; struct sdma_rht_node *rht_node;
int i, j; int i, j;
rht_node = rhashtable_lookup_fast(&dd->sdma_rht, &cpuid, rht_node = rhashtable_lookup_fast(dd->sdma_rht, &cpuid,
sdma_rht_params); sdma_rht_params);
if (!rht_node) if (!rht_node)
return; return;
...@@ -1322,6 +1322,12 @@ static void sdma_clean(struct hfi1_devdata *dd, size_t num_engines) ...@@ -1322,6 +1322,12 @@ static void sdma_clean(struct hfi1_devdata *dd, size_t num_engines)
synchronize_rcu(); synchronize_rcu();
kfree(dd->per_sdma); kfree(dd->per_sdma);
dd->per_sdma = NULL; dd->per_sdma = NULL;
if (dd->sdma_rht) {
rhashtable_free_and_destroy(dd->sdma_rht, sdma_rht_free, NULL);
kfree(dd->sdma_rht);
dd->sdma_rht = NULL;
}
} }
/** /**
...@@ -1341,12 +1347,14 @@ int sdma_init(struct hfi1_devdata *dd, u8 port) ...@@ -1341,12 +1347,14 @@ int sdma_init(struct hfi1_devdata *dd, u8 port)
{ {
unsigned this_idx; unsigned this_idx;
struct sdma_engine *sde; struct sdma_engine *sde;
struct rhashtable *tmp_sdma_rht;
u16 descq_cnt; u16 descq_cnt;
void *curr_head; void *curr_head;
struct hfi1_pportdata *ppd = dd->pport + port; struct hfi1_pportdata *ppd = dd->pport + port;
u32 per_sdma_credits; u32 per_sdma_credits;
uint idle_cnt = sdma_idle_cnt; uint idle_cnt = sdma_idle_cnt;
size_t num_engines = dd->chip_sdma_engines; size_t num_engines = dd->chip_sdma_engines;
int ret = -ENOMEM;
if (!HFI1_CAP_IS_KSET(SDMA)) { if (!HFI1_CAP_IS_KSET(SDMA)) {
HFI1_CAP_CLEAR(SDMA_AHG); HFI1_CAP_CLEAR(SDMA_AHG);
...@@ -1378,7 +1386,7 @@ int sdma_init(struct hfi1_devdata *dd, u8 port) ...@@ -1378,7 +1386,7 @@ int sdma_init(struct hfi1_devdata *dd, u8 port)
/* alloc memory for array of send engines */ /* alloc memory for array of send engines */
dd->per_sdma = kcalloc(num_engines, sizeof(*dd->per_sdma), GFP_KERNEL); dd->per_sdma = kcalloc(num_engines, sizeof(*dd->per_sdma), GFP_KERNEL);
if (!dd->per_sdma) if (!dd->per_sdma)
return -ENOMEM; return ret;
idle_cnt = ns_to_cclock(dd, idle_cnt); idle_cnt = ns_to_cclock(dd, idle_cnt);
if (!sdma_desct_intr) if (!sdma_desct_intr)
...@@ -1507,18 +1515,27 @@ int sdma_init(struct hfi1_devdata *dd, u8 port) ...@@ -1507,18 +1515,27 @@ int sdma_init(struct hfi1_devdata *dd, u8 port)
dd->flags |= HFI1_HAS_SEND_DMA; dd->flags |= HFI1_HAS_SEND_DMA;
dd->flags |= idle_cnt ? HFI1_HAS_SDMA_TIMEOUT : 0; dd->flags |= idle_cnt ? HFI1_HAS_SDMA_TIMEOUT : 0;
dd->num_sdma = num_engines; dd->num_sdma = num_engines;
if (sdma_map_init(dd, port, ppd->vls_operational, NULL)) ret = sdma_map_init(dd, port, ppd->vls_operational, NULL);
if (ret < 0)
goto bail;
tmp_sdma_rht = kzalloc(sizeof(*tmp_sdma_rht), GFP_KERNEL);
if (!tmp_sdma_rht) {
ret = -ENOMEM;
goto bail; goto bail;
}
if (rhashtable_init(&dd->sdma_rht, &sdma_rht_params)) ret = rhashtable_init(tmp_sdma_rht, &sdma_rht_params);
if (ret < 0)
goto bail; goto bail;
dd->sdma_rht = tmp_sdma_rht;
dd_dev_info(dd, "SDMA num_sdma: %u\n", dd->num_sdma); dd_dev_info(dd, "SDMA num_sdma: %u\n", dd->num_sdma);
return 0; return 0;
bail: bail:
sdma_clean(dd, num_engines); sdma_clean(dd, num_engines);
return -ENOMEM; return ret;
} }
/** /**
...@@ -1604,7 +1621,6 @@ void sdma_exit(struct hfi1_devdata *dd) ...@@ -1604,7 +1621,6 @@ void sdma_exit(struct hfi1_devdata *dd)
sdma_finalput(&sde->state); sdma_finalput(&sde->state);
} }
sdma_clean(dd, dd->num_sdma); sdma_clean(dd, dd->num_sdma);
rhashtable_free_and_destroy(&dd->sdma_rht, sdma_rht_free, NULL);
} }
/* /*
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册