提交 c21f515c 编写于 作者: J John Allen 提交者: Michael Ellerman

powerpc/pseries: Make the acquire/release of the drc for memory a seperate step

When adding and removing LMBs we should make the acquire/release of
the DRC a separate step to allow for a few improvements. First
this will ensure that LMBs removed during a remove by count operation
are all available if a error occurs and we need to add them back. By
first removeing all the LMBs from the kernel before releasing their
DRCs the LMBs are available to add back should an error occur.

Also, this will allow for faster re-add operations of memory for
PRRN event handling since we can skip the unneeded step of having
to release the DRC and the acquire it back.
Signed-off-by: NNathan Fontenot <nfont@linux.vnet.ibm.com>
Signed-off-by: NJohn Allen <jallen@linux.vnet.ibm.com>
Signed-off-by: NMichael Ellerman <mpe@ellerman.id.au>
上级 e71ff89c
...@@ -446,9 +446,7 @@ static int dlpar_remove_lmb(struct of_drconf_cell *lmb) ...@@ -446,9 +446,7 @@ static int dlpar_remove_lmb(struct of_drconf_cell *lmb)
/* Update memory regions for memory remove */ /* Update memory regions for memory remove */
memblock_remove(lmb->base_addr, block_sz); memblock_remove(lmb->base_addr, block_sz);
dlpar_release_drc(lmb->drc_index);
dlpar_remove_device_tree_lmb(lmb); dlpar_remove_device_tree_lmb(lmb);
return 0; return 0;
} }
...@@ -516,6 +514,7 @@ static int dlpar_memory_remove_by_count(u32 lmbs_to_remove, ...@@ -516,6 +514,7 @@ static int dlpar_memory_remove_by_count(u32 lmbs_to_remove,
if (!lmbs[i].reserved) if (!lmbs[i].reserved)
continue; continue;
dlpar_release_drc(lmbs[i].drc_index);
pr_info("Memory at %llx was hot-removed\n", pr_info("Memory at %llx was hot-removed\n",
lmbs[i].base_addr); lmbs[i].base_addr);
...@@ -545,6 +544,9 @@ static int dlpar_memory_remove_by_index(u32 drc_index, struct property *prop) ...@@ -545,6 +544,9 @@ static int dlpar_memory_remove_by_index(u32 drc_index, struct property *prop)
if (lmbs[i].drc_index == drc_index) { if (lmbs[i].drc_index == drc_index) {
lmb_found = 1; lmb_found = 1;
rc = dlpar_remove_lmb(&lmbs[i]); rc = dlpar_remove_lmb(&lmbs[i]);
if (!rc)
dlpar_release_drc(lmbs[i].drc_index);
break; break;
} }
} }
...@@ -599,10 +601,6 @@ static int dlpar_add_lmb(struct of_drconf_cell *lmb) ...@@ -599,10 +601,6 @@ static int dlpar_add_lmb(struct of_drconf_cell *lmb)
if (lmb->flags & DRCONF_MEM_ASSIGNED) if (lmb->flags & DRCONF_MEM_ASSIGNED)
return -EINVAL; return -EINVAL;
rc = dlpar_acquire_drc(lmb->drc_index);
if (rc)
return rc;
rc = dlpar_add_device_tree_lmb(lmb); rc = dlpar_add_device_tree_lmb(lmb);
if (rc) { if (rc) {
pr_err("Couldn't update device tree for drc index %x\n", pr_err("Couldn't update device tree for drc index %x\n",
...@@ -618,12 +616,10 @@ static int dlpar_add_lmb(struct of_drconf_cell *lmb) ...@@ -618,12 +616,10 @@ static int dlpar_add_lmb(struct of_drconf_cell *lmb)
/* Add the memory */ /* Add the memory */
rc = add_memory(nid, lmb->base_addr, block_sz); rc = add_memory(nid, lmb->base_addr, block_sz);
if (rc) { if (rc)
dlpar_remove_device_tree_lmb(lmb); dlpar_remove_device_tree_lmb(lmb);
dlpar_release_drc(lmb->drc_index); else
} else {
lmb->flags |= DRCONF_MEM_ASSIGNED; lmb->flags |= DRCONF_MEM_ASSIGNED;
}
return rc; return rc;
} }
...@@ -655,10 +651,16 @@ static int dlpar_memory_add_by_count(u32 lmbs_to_add, struct property *prop) ...@@ -655,10 +651,16 @@ static int dlpar_memory_add_by_count(u32 lmbs_to_add, struct property *prop)
return -EINVAL; return -EINVAL;
for (i = 0; i < num_lmbs && lmbs_to_add != lmbs_added; i++) { for (i = 0; i < num_lmbs && lmbs_to_add != lmbs_added; i++) {
rc = dlpar_add_lmb(&lmbs[i]); rc = dlpar_acquire_drc(lmbs[i].drc_index);
if (rc) if (rc)
continue; continue;
rc = dlpar_add_lmb(&lmbs[i]);
if (rc) {
dlpar_release_drc(lmbs[i].drc_index);
continue;
}
lmbs_added++; lmbs_added++;
/* Mark this lmb so we can remove it later if all of the /* Mark this lmb so we can remove it later if all of the
...@@ -678,6 +680,8 @@ static int dlpar_memory_add_by_count(u32 lmbs_to_add, struct property *prop) ...@@ -678,6 +680,8 @@ static int dlpar_memory_add_by_count(u32 lmbs_to_add, struct property *prop)
if (rc) if (rc)
pr_err("Failed to remove LMB, drc index %x\n", pr_err("Failed to remove LMB, drc index %x\n",
be32_to_cpu(lmbs[i].drc_index)); be32_to_cpu(lmbs[i].drc_index));
else
dlpar_release_drc(lmbs[i].drc_index);
} }
rc = -EINVAL; rc = -EINVAL;
} else { } else {
...@@ -711,7 +715,13 @@ static int dlpar_memory_add_by_index(u32 drc_index, struct property *prop) ...@@ -711,7 +715,13 @@ static int dlpar_memory_add_by_index(u32 drc_index, struct property *prop)
for (i = 0; i < num_lmbs; i++) { for (i = 0; i < num_lmbs; i++) {
if (lmbs[i].drc_index == drc_index) { if (lmbs[i].drc_index == drc_index) {
lmb_found = 1; lmb_found = 1;
rc = dlpar_add_lmb(&lmbs[i]); rc = dlpar_acquire_drc(lmbs[i].drc_index);
if (!rc) {
rc = dlpar_add_lmb(&lmbs[i]);
if (rc)
dlpar_release_drc(lmbs[i].drc_index);
}
break; break;
} }
} }
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册