net/smc: fix final cleanup sequence for SMCD devices
authorUrsula Braun <ubraun@linux.ibm.com>
Thu, 14 Nov 2019 12:02:40 +0000 (13:02 +0100)
committerDavid S. Miller <davem@davemloft.net>
Fri, 15 Nov 2019 20:28:28 +0000 (12:28 -0800)
If peer announces shutdown, use the link group terminate worker for
local cleanup of link groups and connections to terminate link group
in proper context.

Make sure link groups are cleaned up first before destroying the
event queue of the SMCD device, because link group cleanup may
raise events.

Send signal shutdown only if peer has not done it already.

Send socket abort or close only, if peer has not already announced
shutdown.

Signed-off-by: Ursula Braun <ubraun@linux.ibm.com>
Signed-off-by: Karsten Graul <kgraul@linux.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
net/smc/smc_cdc.c
net/smc/smc_core.c
net/smc/smc_core.h
net/smc/smc_ism.c

index 7dc07ec2379be019c95331c9fb2566b06632ab57..164f1584861b84dd1ee4aaf64efd52425912e08f 100644 (file)
@@ -131,6 +131,9 @@ int smc_cdc_get_slot_and_msg_send(struct smc_connection *conn)
 {
        int rc;
 
+       if (!conn->lgr || (conn->lgr->is_smcd && conn->lgr->peer_shutdown))
+               return -EPIPE;
+
        if (conn->lgr->is_smcd) {
                spin_lock_bh(&conn->send_lock);
                rc = smcd_cdc_msg_send(conn);
index 0d92456729ab9464072a4cdcad881974fd590afa..561f069b30de91b9ec27ce3a56a3a98b495de696 100644 (file)
@@ -275,6 +275,7 @@ static int smc_lgr_create(struct smc_sock *smc, struct smc_init_info *ini)
                lgr->smcd = ini->ism_dev;
                lgr_list = &ini->ism_dev->lgr_list;
                lgr_lock = &lgr->smcd->lgr_lock;
+               lgr->peer_shutdown = 0;
        } else {
                /* SMC-R specific settings */
                get_device(&ini->ib_dev->ibdev->dev);
@@ -514,11 +515,16 @@ static void smc_conn_kill(struct smc_connection *conn)
 {
        struct smc_sock *smc = container_of(conn, struct smc_sock, conn);
 
-       smc_close_abort(conn);
+       if (conn->lgr->is_smcd && conn->lgr->peer_shutdown)
+               conn->local_tx_ctrl.conn_state_flags.peer_conn_abort = 1;
+       else
+               smc_close_abort(conn);
        conn->killed = 1;
+       smc->sk.sk_err = ECONNABORTED;
        smc_sk_wake_ups(smc);
+       if (conn->lgr->is_smcd)
+               tasklet_kill(&conn->rx_tsklet);
        smc_lgr_unregister_conn(conn);
-       smc->sk.sk_err = ECONNABORTED;
        smc_close_active_abort(smc);
 }
 
@@ -604,6 +610,8 @@ void smc_smcd_terminate(struct smcd_dev *dev, u64 peer_gid, unsigned short vlan)
        list_for_each_entry_safe(lgr, l, &dev->lgr_list, list) {
                if ((!peer_gid || lgr->peer_gid == peer_gid) &&
                    (vlan == VLAN_VID_MASK || lgr->vlan_id == vlan)) {
+                       if (peer_gid) /* peer triggered termination */
+                               lgr->peer_shutdown = 1;
                        list_move(&lgr->list, &lgr_free_list);
                }
        }
@@ -612,11 +620,7 @@ void smc_smcd_terminate(struct smcd_dev *dev, u64 peer_gid, unsigned short vlan)
        /* cancel the regular free workers and actually free lgrs */
        list_for_each_entry_safe(lgr, l, &lgr_free_list, list) {
                list_del_init(&lgr->list);
-               __smc_lgr_terminate(lgr);
-               cancel_delayed_work_sync(&lgr->free_work);
-               if (!peer_gid && vlan == VLAN_VID_MASK) /* dev terminated? */
-                       smc_ism_signal_shutdown(lgr);
-               smc_lgr_free(lgr);
+               schedule_work(&lgr->terminate_work);
        }
 }
 
index e6fd1ed42064ccaf0898e32c8ada175bc3216c18..097ceba86caf98e711fddc3268117bd11fcb587b 100644 (file)
@@ -228,6 +228,8 @@ struct smc_link_group {
                                                /* Peer GID (remote) */
                        struct smcd_dev         *smcd;
                                                /* ISM device for VLAN reg. */
+                       u8                      peer_shutdown : 1;
+                                               /* peer triggered shutdownn */
                };
        };
 };
index ee7340898cb4d534b3f9ccd6607ab05850796767..18946e95a3be64a8377b14860ccffdb65c725011 100644 (file)
@@ -226,6 +226,9 @@ int smc_ism_signal_shutdown(struct smc_link_group *lgr)
        int rc;
        union smcd_sw_event_info ev_info;
 
+       if (lgr->peer_shutdown)
+               return 0;
+
        memcpy(ev_info.uid, lgr->id, SMC_LGR_ID_SIZE);
        ev_info.vlan_id = lgr->vlan_id;
        ev_info.code = ISM_EVENT_REQUEST;
@@ -313,12 +316,12 @@ EXPORT_SYMBOL_GPL(smcd_register_dev);
 void smcd_unregister_dev(struct smcd_dev *smcd)
 {
        spin_lock(&smcd_dev_list.lock);
-       list_del(&smcd->list);
+       list_del_init(&smcd->list);
        spin_unlock(&smcd_dev_list.lock);
        smcd->going_away = 1;
+       smc_smcd_terminate(smcd, 0, VLAN_VID_MASK);
        flush_workqueue(smcd->event_wq);
        destroy_workqueue(smcd->event_wq);
-       smc_smcd_terminate(smcd, 0, VLAN_VID_MASK);
 
        device_del(&smcd->dev);
 }