ibmvnic: restore adapter state on failed reset

In a failed reset, driver could end up in VNIC_PROBED or VNIC_CLOSED
state and cannot recover in subsequent resets, leaving it offline.
This patch restores the adapter state to reset_state, the original
state when reset was called.

Fixes: b27507bb59 ("net/ibmvnic: unlock rtnl_lock in reset so linkwatch_event can run")
Fixes: 2770a7984d ("ibmvnic: Introduce hard reset recovery")
Signed-off-by: Dany Madden <drt@linux.ibm.com>
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
This commit is contained in:
Dany Madden 2020-11-25 18:04:27 -06:00 коммит произвёл Jakub Kicinski
Родитель 9281cf2d58
Коммит 0cb4bc66ba
1 изменённых файлов: 36 добавлений и 31 удалений

Просмотреть файл

@ -1857,7 +1857,7 @@ static int do_change_param_reset(struct ibmvnic_adapter *adapter,
if (reset_state == VNIC_OPEN) { if (reset_state == VNIC_OPEN) {
rc = __ibmvnic_close(netdev); rc = __ibmvnic_close(netdev);
if (rc) if (rc)
return rc; goto out;
} }
release_resources(adapter); release_resources(adapter);
@ -1875,24 +1875,25 @@ static int do_change_param_reset(struct ibmvnic_adapter *adapter,
} }
rc = ibmvnic_reset_init(adapter, true); rc = ibmvnic_reset_init(adapter, true);
if (rc) if (rc) {
return IBMVNIC_INIT_FAILED; rc = IBMVNIC_INIT_FAILED;
goto out;
}
/* If the adapter was in PROBE state prior to the reset, /* If the adapter was in PROBE state prior to the reset,
* exit here. * exit here.
*/ */
if (reset_state == VNIC_PROBED) if (reset_state == VNIC_PROBED)
return 0; goto out;
rc = ibmvnic_login(netdev); rc = ibmvnic_login(netdev);
if (rc) { if (rc) {
adapter->state = reset_state; goto out;
return rc;
} }
rc = init_resources(adapter); rc = init_resources(adapter);
if (rc) if (rc)
return rc; goto out;
ibmvnic_disable_irqs(adapter); ibmvnic_disable_irqs(adapter);
@ -1902,8 +1903,10 @@ static int do_change_param_reset(struct ibmvnic_adapter *adapter,
return 0; return 0;
rc = __ibmvnic_open(netdev); rc = __ibmvnic_open(netdev);
if (rc) if (rc) {
return IBMVNIC_OPEN_FAILED; rc = IBMVNIC_OPEN_FAILED;
goto out;
}
/* refresh device's multicast list */ /* refresh device's multicast list */
ibmvnic_set_multi(netdev); ibmvnic_set_multi(netdev);
@ -1912,7 +1915,10 @@ static int do_change_param_reset(struct ibmvnic_adapter *adapter,
for (i = 0; i < adapter->req_rx_queues; i++) for (i = 0; i < adapter->req_rx_queues; i++)
napi_schedule(&adapter->napi[i]); napi_schedule(&adapter->napi[i]);
return 0; out:
if (rc)
adapter->state = reset_state;
return rc;
} }
/** /**
@ -2015,7 +2021,6 @@ static int do_reset(struct ibmvnic_adapter *adapter,
rc = ibmvnic_login(netdev); rc = ibmvnic_login(netdev);
if (rc) { if (rc) {
adapter->state = reset_state;
goto out; goto out;
} }
@ -2083,6 +2088,9 @@ static int do_reset(struct ibmvnic_adapter *adapter,
rc = 0; rc = 0;
out: out:
/* restore the adapter state if reset failed */
if (rc)
adapter->state = reset_state;
rtnl_unlock(); rtnl_unlock();
return rc; return rc;
@ -2115,43 +2123,46 @@ static int do_hard_reset(struct ibmvnic_adapter *adapter,
if (rc) { if (rc) {
netdev_err(adapter->netdev, netdev_err(adapter->netdev,
"Couldn't initialize crq. rc=%d\n", rc); "Couldn't initialize crq. rc=%d\n", rc);
return rc; goto out;
} }
rc = ibmvnic_reset_init(adapter, false); rc = ibmvnic_reset_init(adapter, false);
if (rc) if (rc)
return rc; goto out;
/* If the adapter was in PROBE state prior to the reset, /* If the adapter was in PROBE state prior to the reset,
* exit here. * exit here.
*/ */
if (reset_state == VNIC_PROBED) if (reset_state == VNIC_PROBED)
return 0; goto out;
rc = ibmvnic_login(netdev); rc = ibmvnic_login(netdev);
if (rc) { if (rc)
adapter->state = VNIC_PROBED; goto out;
return 0;
}
rc = init_resources(adapter); rc = init_resources(adapter);
if (rc) if (rc)
return rc; goto out;
ibmvnic_disable_irqs(adapter); ibmvnic_disable_irqs(adapter);
adapter->state = VNIC_CLOSED; adapter->state = VNIC_CLOSED;
if (reset_state == VNIC_CLOSED) if (reset_state == VNIC_CLOSED)
return 0; goto out;
rc = __ibmvnic_open(netdev); rc = __ibmvnic_open(netdev);
if (rc) if (rc) {
return IBMVNIC_OPEN_FAILED; rc = IBMVNIC_OPEN_FAILED;
goto out;
}
call_netdevice_notifiers(NETDEV_NOTIFY_PEERS, netdev); call_netdevice_notifiers(NETDEV_NOTIFY_PEERS, netdev);
call_netdevice_notifiers(NETDEV_RESEND_IGMP, netdev); call_netdevice_notifiers(NETDEV_RESEND_IGMP, netdev);
out:
return 0; /* restore adapter state if reset failed */
if (rc)
adapter->state = reset_state;
return rc;
} }
static struct ibmvnic_rwi *get_next_rwi(struct ibmvnic_adapter *adapter) static struct ibmvnic_rwi *get_next_rwi(struct ibmvnic_adapter *adapter)
@ -2235,13 +2246,7 @@ static void __ibmvnic_reset(struct work_struct *work)
rc = do_reset(adapter, rwi, reset_state); rc = do_reset(adapter, rwi, reset_state);
} }
kfree(rwi); kfree(rwi);
if (rc == IBMVNIC_OPEN_FAILED) {
if (list_empty(&adapter->rwi_list))
adapter->state = VNIC_CLOSED;
else
adapter->state = reset_state;
rc = 0;
}
if (rc) if (rc)
netdev_dbg(adapter->netdev, "Reset failed, rc=%d\n", rc); netdev_dbg(adapter->netdev, "Reset failed, rc=%d\n", rc);