Skip to content

Commit 06c9922

Browse files
hmynenityreld
authored andcommitted
drmgr/pci: Do not add partner device if exists in the device tree
For PCI hotplug interface, the partner device will be added or removed if configured with the primary device add / remove. Whereas for PHB interface, drmgr notifies the user to add / remove partner device if configured with the primary device. So there is a possibility of partner PCI node exists in the device tree when PCI interface for ADD is executed. The current ADD code in drslot_chrp_pci.c does not check whether the partner device node is present and may add / enable partner device again which may give EEH errors. See the following sequence to get this scenario: drmgr -r -c phb -s "PHB 1336" --> Remove primary PHB node drmgr -r -c pci -s "U50EE.001.WZS000E-P3-C24-R2" --> Remove partner PCI node drmgr -a -c phb -s "PHB 1336" --> Add primary PHB node drmgr -a -c pci -s "U50EE.001.WZS000E-P3-C24-R2" --> Add partner PCI node and can try to add primary PCI node again. In this case "U50EE.001.WZS000E-P3-C24-R1". This patch fixes the issue by checking the device node in the device tree and add the device if does not present and remove only if the corresponding device node exists. Fixes: 4e6670d ("drmgr/pci: Add multipath partner device support for hotplug add") Signed-off-by: Haren Myneni <[email protected]> [tyreld: fixed up white space and replace/remove phrasing] Signed-off-by: Tyrel Datwyler <[email protected]>
1 parent a560285 commit 06c9922

File tree

1 file changed

+52
-20
lines changed

1 file changed

+52
-20
lines changed

src/drmgr/drslot_chrp_pci.c

+52-20
Original file line numberDiff line numberDiff line change
@@ -536,12 +536,15 @@ static int do_insert_card_work(struct dr_node *node, bool partner_device)
536536
* Find the partner DRC index and retrieve the partner node.
537537
*/
538538
static struct dr_node *
539-
find_partner_node(struct dr_node *node, struct dr_node *all_nodes)
539+
find_partner_node(struct dr_node *node, struct dr_node *all_nodes,
540+
int *dt_entry_present)
540541
{
541542
struct dr_node *partner_node = NULL;
542543
uint32_t partner_index = 0;
543544
int rc;
544545

546+
*dt_entry_present = 0;
547+
545548
/*
546549
* Expect the partner device only for the PCI node
547550
*/
@@ -550,8 +553,20 @@ find_partner_node(struct dr_node *node, struct dr_node *all_nodes)
550553

551554
/* Find the multipath partner device index if available */
552555
rc = get_my_partner_drc_index(node->children, &partner_index);
553-
if (!rc)
556+
if (!rc) {
554557
partner_node = find_slot(NULL, partner_index, all_nodes, 1);
558+
/*
559+
* Do not add if the partner entry is already present.
560+
* Nothing to remove if the partner node does not exists.
561+
*/
562+
if (partner_node && partner_node->children) {
563+
uint32_t index;
564+
rc = get_my_drc_index(partner_node->children->ofdt_path,
565+
&index);
566+
if (!rc && (partner_index == index))
567+
*dt_entry_present = 1;
568+
}
569+
}
555570

556571
return partner_node;
557572
}
@@ -626,6 +641,7 @@ static int insert_add_work(struct dr_node *node, bool partner_device)
626641
static int do_add(struct dr_node *all_nodes)
627642
{
628643
struct dr_node *node, *partner_node = NULL;
644+
int dt_entry;
629645
int rc;
630646

631647
node = find_slot(usr_drc_name, 0, all_nodes, 0);
@@ -641,8 +657,8 @@ static int do_add(struct dr_node *all_nodes)
641657
if (rc <= 0)
642658
return rc;
643659

644-
partner_node = find_partner_node(node, all_nodes);
645-
if (partner_node) {
660+
partner_node = find_partner_node(node, all_nodes, &dt_entry);
661+
if (partner_node && !dt_entry) {
646662
printf("<%s> and <%s> are\nmultipath partner devices. "
647663
"So <%s> is\nalso added.\n", node->drc_name,
648664
partner_node->drc_name, partner_node->drc_name);
@@ -839,16 +855,25 @@ static int remove_work(struct dr_node *node, bool partner_device)
839855
static int do_remove(struct dr_node *all_nodes)
840856
{
841857
struct dr_node *node, *partner_node = NULL;
858+
int dt_entry;
842859

843860
node = find_slot(usr_drc_name, 0, all_nodes, 0);
844861
if (node == NULL)
845862
return -1;
846863

847-
partner_node = find_partner_node(node, all_nodes);
848-
if (partner_node)
849-
printf("<%s> and <%s> are\nmultipath partner devices. "
850-
"So <%s> will\nbe also removed.\n", node->drc_name,
851-
partner_node->drc_name, partner_node->drc_name);
864+
partner_node = find_partner_node(node, all_nodes, &dt_entry);
865+
if (partner_node) {
866+
/*
867+
* Remove partner device if exists in the device tree.
868+
*/
869+
if (dt_entry)
870+
printf("<%s> and <%s> are\nmultipath partner devices. "
871+
"So <%s> will\nalso be removed.\n",
872+
node->drc_name, partner_node->drc_name,
873+
partner_node->drc_name);
874+
else
875+
partner_node = NULL;
876+
}
852877

853878
/* Remove the specified slot and update the device-tree */
854879
if (remove_work(node, false))
@@ -939,18 +964,23 @@ static int replace_add_work(struct dr_node *node, bool partner_device)
939964
static int do_replace(struct dr_node *all_nodes)
940965
{
941966
struct dr_node *repl_node, *partner_node = NULL;
967+
int dt_entry;
942968
int rc;
943969

944970
repl_node = find_slot(usr_drc_name, 0, all_nodes, 0);
945971
if (repl_node == NULL)
946972
return -1;
947973

948-
partner_node = find_partner_node(repl_node, all_nodes);
949-
if (partner_node)
950-
printf("<%s> and <%s> are\nmultipath partner devices. "
951-
"So <%s> will\nbe also replaced.\n",
952-
repl_node->drc_name, partner_node->drc_name,
953-
partner_node->drc_name);
974+
partner_node = find_partner_node(repl_node, all_nodes, &dt_entry);
975+
if (partner_node) {
976+
if (dt_entry)
977+
printf("<%s> and <%s> are\nmultipath partner devices. "
978+
"So <%s> will\nalso be replaced.\n",
979+
repl_node->drc_name, partner_node->drc_name,
980+
partner_node->drc_name);
981+
else
982+
partner_node = NULL;
983+
}
954984

955985
/* Call the routine which does the work of getting the node info,
956986
* then removing it from the OF device tree.
@@ -972,9 +1002,11 @@ static int do_replace(struct dr_node *all_nodes)
9721002
if (rc <= 0)
9731003
return rc;
9741004

975-
rc = replace_add_work(partner_node, true);
976-
if (rc <= 0)
977-
return rc;
1005+
if (partner_node) {
1006+
rc = replace_add_work(partner_node, true);
1007+
if (rc <= 0)
1008+
return rc;
1009+
}
9781010

9791011
if (repl_node->post_replace_processing) {
9801012
int prompt_save = usr_prompt;
@@ -988,8 +1020,8 @@ static int do_replace(struct dr_node *all_nodes)
9881020
if (remove_work(repl_node, false))
9891021
return -1;
9901022

991-
partner_node = find_partner_node(repl_node, node);
992-
if (partner_node) {
1023+
partner_node = find_partner_node(repl_node, node, &dt_entry);
1024+
if (partner_node && dt_entry) {
9931025
if (remove_work(partner_node, true))
9941026
return -1;
9951027
}

0 commit comments

Comments
 (0)