mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-24 09:13:20 -05:00
net/smc: add SMC-D shutdown signal
When a SMC-D link group is freed, a shutdown signal should be sent to the peer to indicate that the link group is invalid. This patch adds the shutdown signal to the SMC code. Signed-off-by: Hans Wippel <hwippel@linux.ibm.com> Signed-off-by: Ursula Braun <ubraun@linux.ibm.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
ee05ff7af2
commit
0512f69e38
4 changed files with 43 additions and 14 deletions
|
@ -184,6 +184,8 @@ free:
|
||||||
|
|
||||||
if (!lgr->is_smcd && lnk->state != SMC_LNK_INACTIVE)
|
if (!lgr->is_smcd && lnk->state != SMC_LNK_INACTIVE)
|
||||||
smc_llc_link_inactive(lnk);
|
smc_llc_link_inactive(lnk);
|
||||||
|
if (lgr->is_smcd)
|
||||||
|
smc_ism_signal_shutdown(lgr);
|
||||||
smc_lgr_free(lgr);
|
smc_lgr_free(lgr);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -485,7 +487,7 @@ void smc_port_terminate(struct smc_ib_device *smcibdev, u8 ibport)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Called when SMC-D device is terminated or peer is lost */
|
/* Called when SMC-D device is terminated or peer is lost */
|
||||||
void smc_smcd_terminate(struct smcd_dev *dev, u64 peer_gid)
|
void smc_smcd_terminate(struct smcd_dev *dev, u64 peer_gid, unsigned short vlan)
|
||||||
{
|
{
|
||||||
struct smc_link_group *lgr, *l;
|
struct smc_link_group *lgr, *l;
|
||||||
LIST_HEAD(lgr_free_list);
|
LIST_HEAD(lgr_free_list);
|
||||||
|
@ -495,7 +497,7 @@ void smc_smcd_terminate(struct smcd_dev *dev, u64 peer_gid)
|
||||||
list_for_each_entry_safe(lgr, l, &smc_lgr_list.list, list) {
|
list_for_each_entry_safe(lgr, l, &smc_lgr_list.list, list) {
|
||||||
if (lgr->is_smcd && lgr->smcd == dev &&
|
if (lgr->is_smcd && lgr->smcd == dev &&
|
||||||
(!peer_gid || lgr->peer_gid == peer_gid) &&
|
(!peer_gid || lgr->peer_gid == peer_gid) &&
|
||||||
!list_empty(&lgr->list)) {
|
(vlan == VLAN_VID_MASK || lgr->vlan_id == vlan)) {
|
||||||
__smc_lgr_terminate(lgr);
|
__smc_lgr_terminate(lgr);
|
||||||
list_move(&lgr->list, &lgr_free_list);
|
list_move(&lgr->list, &lgr_free_list);
|
||||||
}
|
}
|
||||||
|
@ -506,6 +508,8 @@ void smc_smcd_terminate(struct smcd_dev *dev, u64 peer_gid)
|
||||||
list_for_each_entry_safe(lgr, l, &lgr_free_list, list) {
|
list_for_each_entry_safe(lgr, l, &lgr_free_list, list) {
|
||||||
list_del_init(&lgr->list);
|
list_del_init(&lgr->list);
|
||||||
cancel_delayed_work_sync(&lgr->free_work);
|
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);
|
smc_lgr_free(lgr);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1026,6 +1030,8 @@ void smc_core_exit(void)
|
||||||
smc_llc_link_inactive(lnk);
|
smc_llc_link_inactive(lnk);
|
||||||
}
|
}
|
||||||
cancel_delayed_work_sync(&lgr->free_work);
|
cancel_delayed_work_sync(&lgr->free_work);
|
||||||
|
if (lgr->is_smcd)
|
||||||
|
smc_ism_signal_shutdown(lgr);
|
||||||
smc_lgr_free(lgr); /* free link group */
|
smc_lgr_free(lgr); /* free link group */
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -247,7 +247,8 @@ void smc_lgr_free(struct smc_link_group *lgr);
|
||||||
void smc_lgr_forget(struct smc_link_group *lgr);
|
void smc_lgr_forget(struct smc_link_group *lgr);
|
||||||
void smc_lgr_terminate(struct smc_link_group *lgr);
|
void smc_lgr_terminate(struct smc_link_group *lgr);
|
||||||
void smc_port_terminate(struct smc_ib_device *smcibdev, u8 ibport);
|
void smc_port_terminate(struct smc_ib_device *smcibdev, u8 ibport);
|
||||||
void smc_smcd_terminate(struct smcd_dev *dev, u64 peer_gid);
|
void smc_smcd_terminate(struct smcd_dev *dev, u64 peer_gid,
|
||||||
|
unsigned short vlan);
|
||||||
int smc_buf_create(struct smc_sock *smc, bool is_smcd);
|
int smc_buf_create(struct smc_sock *smc, bool is_smcd);
|
||||||
int smc_uncompress_bufsize(u8 compressed);
|
int smc_uncompress_bufsize(u8 compressed);
|
||||||
int smc_rmb_rtoken_handling(struct smc_connection *conn,
|
int smc_rmb_rtoken_handling(struct smc_connection *conn,
|
||||||
|
|
|
@ -187,22 +187,28 @@ struct smc_ism_event_work {
|
||||||
#define ISM_EVENT_REQUEST 0x0001
|
#define ISM_EVENT_REQUEST 0x0001
|
||||||
#define ISM_EVENT_RESPONSE 0x0002
|
#define ISM_EVENT_RESPONSE 0x0002
|
||||||
#define ISM_EVENT_REQUEST_IR 0x00000001
|
#define ISM_EVENT_REQUEST_IR 0x00000001
|
||||||
|
#define ISM_EVENT_CODE_SHUTDOWN 0x80
|
||||||
#define ISM_EVENT_CODE_TESTLINK 0x83
|
#define ISM_EVENT_CODE_TESTLINK 0x83
|
||||||
|
|
||||||
|
union smcd_sw_event_info {
|
||||||
|
u64 info;
|
||||||
|
struct {
|
||||||
|
u8 uid[SMC_LGR_ID_SIZE];
|
||||||
|
unsigned short vlan_id;
|
||||||
|
u16 code;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
static void smcd_handle_sw_event(struct smc_ism_event_work *wrk)
|
static void smcd_handle_sw_event(struct smc_ism_event_work *wrk)
|
||||||
{
|
{
|
||||||
union {
|
union smcd_sw_event_info ev_info;
|
||||||
u64 info;
|
|
||||||
struct {
|
|
||||||
u32 uid;
|
|
||||||
unsigned short vlanid;
|
|
||||||
u16 code;
|
|
||||||
};
|
|
||||||
} ev_info;
|
|
||||||
|
|
||||||
|
ev_info.info = wrk->event.info;
|
||||||
switch (wrk->event.code) {
|
switch (wrk->event.code) {
|
||||||
|
case ISM_EVENT_CODE_SHUTDOWN: /* Peer shut down DMBs */
|
||||||
|
smc_smcd_terminate(wrk->smcd, wrk->event.tok, ev_info.vlan_id);
|
||||||
|
break;
|
||||||
case ISM_EVENT_CODE_TESTLINK: /* Activity timer */
|
case ISM_EVENT_CODE_TESTLINK: /* Activity timer */
|
||||||
ev_info.info = wrk->event.info;
|
|
||||||
if (ev_info.code == ISM_EVENT_REQUEST) {
|
if (ev_info.code == ISM_EVENT_REQUEST) {
|
||||||
ev_info.code = ISM_EVENT_RESPONSE;
|
ev_info.code = ISM_EVENT_RESPONSE;
|
||||||
wrk->smcd->ops->signal_event(wrk->smcd,
|
wrk->smcd->ops->signal_event(wrk->smcd,
|
||||||
|
@ -215,6 +221,21 @@ static void smcd_handle_sw_event(struct smc_ism_event_work *wrk)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int smc_ism_signal_shutdown(struct smc_link_group *lgr)
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
union smcd_sw_event_info ev_info;
|
||||||
|
|
||||||
|
memcpy(ev_info.uid, lgr->id, SMC_LGR_ID_SIZE);
|
||||||
|
ev_info.vlan_id = lgr->vlan_id;
|
||||||
|
ev_info.code = ISM_EVENT_REQUEST;
|
||||||
|
rc = lgr->smcd->ops->signal_event(lgr->smcd, lgr->peer_gid,
|
||||||
|
ISM_EVENT_REQUEST_IR,
|
||||||
|
ISM_EVENT_CODE_SHUTDOWN,
|
||||||
|
ev_info.info);
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
/* worker for SMC-D events */
|
/* worker for SMC-D events */
|
||||||
static void smc_ism_event_work(struct work_struct *work)
|
static void smc_ism_event_work(struct work_struct *work)
|
||||||
{
|
{
|
||||||
|
@ -223,7 +244,7 @@ static void smc_ism_event_work(struct work_struct *work)
|
||||||
|
|
||||||
switch (wrk->event.type) {
|
switch (wrk->event.type) {
|
||||||
case ISM_EVENT_GID: /* GID event, token is peer GID */
|
case ISM_EVENT_GID: /* GID event, token is peer GID */
|
||||||
smc_smcd_terminate(wrk->smcd, wrk->event.tok);
|
smc_smcd_terminate(wrk->smcd, wrk->event.tok, VLAN_VID_MASK);
|
||||||
break;
|
break;
|
||||||
case ISM_EVENT_DMB:
|
case ISM_EVENT_DMB:
|
||||||
break;
|
break;
|
||||||
|
@ -289,7 +310,7 @@ void smcd_unregister_dev(struct smcd_dev *smcd)
|
||||||
spin_unlock(&smcd_dev_list.lock);
|
spin_unlock(&smcd_dev_list.lock);
|
||||||
flush_workqueue(smcd->event_wq);
|
flush_workqueue(smcd->event_wq);
|
||||||
destroy_workqueue(smcd->event_wq);
|
destroy_workqueue(smcd->event_wq);
|
||||||
smc_smcd_terminate(smcd, 0);
|
smc_smcd_terminate(smcd, 0, VLAN_VID_MASK);
|
||||||
|
|
||||||
device_del(&smcd->dev);
|
device_del(&smcd->dev);
|
||||||
}
|
}
|
||||||
|
|
|
@ -45,4 +45,5 @@ int smc_ism_register_dmb(struct smc_link_group *lgr, int buf_size,
|
||||||
int smc_ism_unregister_dmb(struct smcd_dev *dev, struct smc_buf_desc *dmb_desc);
|
int smc_ism_unregister_dmb(struct smcd_dev *dev, struct smc_buf_desc *dmb_desc);
|
||||||
int smc_ism_write(struct smcd_dev *dev, const struct smc_ism_position *pos,
|
int smc_ism_write(struct smcd_dev *dev, const struct smc_ism_position *pos,
|
||||||
void *data, size_t len);
|
void *data, size_t len);
|
||||||
|
int smc_ism_signal_shutdown(struct smc_link_group *lgr);
|
||||||
#endif
|
#endif
|
||||||
|
|
Loading…
Add table
Reference in a new issue