Merge branch 'net-smc-add-event-based-framework-for-LLC-msgs'
Karsten Graul says: ==================== net/smc: add event-based framework for LLC msgs These patches are the next step towards SMC-R link failover support. They add a new framework to handle Link Layer Control (LLC) messages and adapt the existing code to use the new framework. ==================== Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
commit
fe4f961eb5
108
net/smc/af_smc.c
108
net/smc/af_smc.c
|
@ -382,22 +382,24 @@ static int smcr_lgr_reg_rmbs(struct smc_link_group *lgr,
|
||||||
static int smcr_clnt_conf_first_link(struct smc_sock *smc)
|
static int smcr_clnt_conf_first_link(struct smc_sock *smc)
|
||||||
{
|
{
|
||||||
struct smc_link *link = smc->conn.lnk;
|
struct smc_link *link = smc->conn.lnk;
|
||||||
int rest;
|
struct smc_llc_qentry *qentry;
|
||||||
int rc;
|
int rc;
|
||||||
|
|
||||||
|
link->lgr->type = SMC_LGR_SINGLE;
|
||||||
|
|
||||||
/* receive CONFIRM LINK request from server over RoCE fabric */
|
/* receive CONFIRM LINK request from server over RoCE fabric */
|
||||||
rest = wait_for_completion_interruptible_timeout(
|
qentry = smc_llc_wait(link->lgr, NULL, SMC_LLC_WAIT_TIME,
|
||||||
&link->llc_confirm,
|
SMC_LLC_CONFIRM_LINK);
|
||||||
SMC_LLC_WAIT_FIRST_TIME);
|
if (!qentry) {
|
||||||
if (rest <= 0) {
|
|
||||||
struct smc_clc_msg_decline dclc;
|
struct smc_clc_msg_decline dclc;
|
||||||
|
|
||||||
rc = smc_clc_wait_msg(smc, &dclc, sizeof(dclc),
|
rc = smc_clc_wait_msg(smc, &dclc, sizeof(dclc),
|
||||||
SMC_CLC_DECLINE, CLC_WAIT_TIME_SHORT);
|
SMC_CLC_DECLINE, CLC_WAIT_TIME_SHORT);
|
||||||
return rc == -EAGAIN ? SMC_CLC_DECL_TIMEOUT_CL : rc;
|
return rc == -EAGAIN ? SMC_CLC_DECL_TIMEOUT_CL : rc;
|
||||||
}
|
}
|
||||||
|
rc = smc_llc_eval_conf_link(qentry, SMC_LLC_REQ);
|
||||||
if (link->llc_confirm_rc)
|
smc_llc_flow_qentry_del(&link->lgr->llc_flow_lcl);
|
||||||
|
if (rc)
|
||||||
return SMC_CLC_DECL_RMBE_EC;
|
return SMC_CLC_DECL_RMBE_EC;
|
||||||
|
|
||||||
rc = smc_ib_modify_qp_rts(link);
|
rc = smc_ib_modify_qp_rts(link);
|
||||||
|
@ -409,31 +411,30 @@ static int smcr_clnt_conf_first_link(struct smc_sock *smc)
|
||||||
if (smcr_link_reg_rmb(link, smc->conn.rmb_desc, false))
|
if (smcr_link_reg_rmb(link, smc->conn.rmb_desc, false))
|
||||||
return SMC_CLC_DECL_ERR_REGRMB;
|
return SMC_CLC_DECL_ERR_REGRMB;
|
||||||
|
|
||||||
|
/* confirm_rkey is implicit on 1st contact */
|
||||||
|
smc->conn.rmb_desc->is_conf_rkey = true;
|
||||||
|
|
||||||
/* send CONFIRM LINK response over RoCE fabric */
|
/* send CONFIRM LINK response over RoCE fabric */
|
||||||
rc = smc_llc_send_confirm_link(link, SMC_LLC_RESP);
|
rc = smc_llc_send_confirm_link(link, SMC_LLC_RESP);
|
||||||
if (rc < 0)
|
if (rc < 0)
|
||||||
return SMC_CLC_DECL_TIMEOUT_CL;
|
return SMC_CLC_DECL_TIMEOUT_CL;
|
||||||
|
|
||||||
/* receive ADD LINK request from server over RoCE fabric */
|
smc_llc_link_active(link);
|
||||||
rest = wait_for_completion_interruptible_timeout(&link->llc_add,
|
|
||||||
SMC_LLC_WAIT_TIME);
|
/* optional 2nd link, receive ADD LINK request from server */
|
||||||
if (rest <= 0) {
|
qentry = smc_llc_wait(link->lgr, NULL, SMC_LLC_WAIT_TIME,
|
||||||
|
SMC_LLC_ADD_LINK);
|
||||||
|
if (!qentry) {
|
||||||
struct smc_clc_msg_decline dclc;
|
struct smc_clc_msg_decline dclc;
|
||||||
|
|
||||||
rc = smc_clc_wait_msg(smc, &dclc, sizeof(dclc),
|
rc = smc_clc_wait_msg(smc, &dclc, sizeof(dclc),
|
||||||
SMC_CLC_DECLINE, CLC_WAIT_TIME_SHORT);
|
SMC_CLC_DECLINE, CLC_WAIT_TIME_SHORT);
|
||||||
return rc == -EAGAIN ? SMC_CLC_DECL_TIMEOUT_AL : rc;
|
if (rc == -EAGAIN)
|
||||||
|
rc = 0; /* no DECLINE received, go with one link */
|
||||||
|
return rc;
|
||||||
}
|
}
|
||||||
|
smc_llc_flow_qentry_clr(&link->lgr->llc_flow_lcl);
|
||||||
/* send add link reject message, only one link supported for now */
|
/* tbd: call smc_llc_cli_add_link(link, qentry); */
|
||||||
rc = smc_llc_send_add_link(link,
|
|
||||||
link->smcibdev->mac[link->ibport - 1],
|
|
||||||
link->gid, SMC_LLC_RESP);
|
|
||||||
if (rc < 0)
|
|
||||||
return SMC_CLC_DECL_TIMEOUT_AL;
|
|
||||||
|
|
||||||
smc_llc_link_active(link);
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -613,8 +614,8 @@ static int smc_connect_rdma(struct smc_sock *smc,
|
||||||
struct smc_clc_msg_accept_confirm *aclc,
|
struct smc_clc_msg_accept_confirm *aclc,
|
||||||
struct smc_init_info *ini)
|
struct smc_init_info *ini)
|
||||||
{
|
{
|
||||||
|
int i, reason_code = 0;
|
||||||
struct smc_link *link;
|
struct smc_link *link;
|
||||||
int reason_code = 0;
|
|
||||||
|
|
||||||
ini->is_smcd = false;
|
ini->is_smcd = false;
|
||||||
ini->ib_lcl = &aclc->lcl;
|
ini->ib_lcl = &aclc->lcl;
|
||||||
|
@ -627,10 +628,28 @@ static int smc_connect_rdma(struct smc_sock *smc,
|
||||||
mutex_unlock(&smc_client_lgr_pending);
|
mutex_unlock(&smc_client_lgr_pending);
|
||||||
return reason_code;
|
return reason_code;
|
||||||
}
|
}
|
||||||
link = smc->conn.lnk;
|
|
||||||
|
|
||||||
smc_conn_save_peer_info(smc, aclc);
|
smc_conn_save_peer_info(smc, aclc);
|
||||||
|
|
||||||
|
if (ini->cln_first_contact == SMC_FIRST_CONTACT) {
|
||||||
|
link = smc->conn.lnk;
|
||||||
|
} else {
|
||||||
|
/* set link that was assigned by server */
|
||||||
|
link = NULL;
|
||||||
|
for (i = 0; i < SMC_LINKS_PER_LGR_MAX; i++) {
|
||||||
|
struct smc_link *l = &smc->conn.lgr->lnk[i];
|
||||||
|
|
||||||
|
if (l->peer_qpn == ntoh24(aclc->qpn)) {
|
||||||
|
link = l;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!link)
|
||||||
|
return smc_connect_abort(smc, SMC_CLC_DECL_NOSRVLINK,
|
||||||
|
ini->cln_first_contact);
|
||||||
|
smc->conn.lnk = link;
|
||||||
|
}
|
||||||
|
|
||||||
/* create send buffer and rmb */
|
/* create send buffer and rmb */
|
||||||
if (smc_buf_create(smc, false))
|
if (smc_buf_create(smc, false))
|
||||||
return smc_connect_abort(smc, SMC_CLC_DECL_MEM,
|
return smc_connect_abort(smc, SMC_CLC_DECL_MEM,
|
||||||
|
@ -666,7 +685,9 @@ static int smc_connect_rdma(struct smc_sock *smc,
|
||||||
|
|
||||||
if (ini->cln_first_contact == SMC_FIRST_CONTACT) {
|
if (ini->cln_first_contact == SMC_FIRST_CONTACT) {
|
||||||
/* QP confirmation over RoCE fabric */
|
/* QP confirmation over RoCE fabric */
|
||||||
|
smc_llc_flow_initiate(link->lgr, SMC_LLC_FLOW_ADD_LINK);
|
||||||
reason_code = smcr_clnt_conf_first_link(smc);
|
reason_code = smcr_clnt_conf_first_link(smc);
|
||||||
|
smc_llc_flow_stop(link->lgr, &link->lgr->llc_flow_lcl);
|
||||||
if (reason_code)
|
if (reason_code)
|
||||||
return smc_connect_abort(smc, reason_code,
|
return smc_connect_abort(smc, reason_code,
|
||||||
ini->cln_first_contact);
|
ini->cln_first_contact);
|
||||||
|
@ -1019,9 +1040,11 @@ void smc_close_non_accepted(struct sock *sk)
|
||||||
static int smcr_serv_conf_first_link(struct smc_sock *smc)
|
static int smcr_serv_conf_first_link(struct smc_sock *smc)
|
||||||
{
|
{
|
||||||
struct smc_link *link = smc->conn.lnk;
|
struct smc_link *link = smc->conn.lnk;
|
||||||
int rest;
|
struct smc_llc_qentry *qentry;
|
||||||
int rc;
|
int rc;
|
||||||
|
|
||||||
|
link->lgr->type = SMC_LGR_SINGLE;
|
||||||
|
|
||||||
if (smcr_link_reg_rmb(link, smc->conn.rmb_desc, false))
|
if (smcr_link_reg_rmb(link, smc->conn.rmb_desc, false))
|
||||||
return SMC_CLC_DECL_ERR_REGRMB;
|
return SMC_CLC_DECL_ERR_REGRMB;
|
||||||
|
|
||||||
|
@ -1031,40 +1054,27 @@ static int smcr_serv_conf_first_link(struct smc_sock *smc)
|
||||||
return SMC_CLC_DECL_TIMEOUT_CL;
|
return SMC_CLC_DECL_TIMEOUT_CL;
|
||||||
|
|
||||||
/* receive CONFIRM LINK response from client over the RoCE fabric */
|
/* receive CONFIRM LINK response from client over the RoCE fabric */
|
||||||
rest = wait_for_completion_interruptible_timeout(
|
qentry = smc_llc_wait(link->lgr, link, SMC_LLC_WAIT_TIME,
|
||||||
&link->llc_confirm_resp,
|
SMC_LLC_CONFIRM_LINK);
|
||||||
SMC_LLC_WAIT_FIRST_TIME);
|
if (!qentry) {
|
||||||
if (rest <= 0) {
|
|
||||||
struct smc_clc_msg_decline dclc;
|
struct smc_clc_msg_decline dclc;
|
||||||
|
|
||||||
rc = smc_clc_wait_msg(smc, &dclc, sizeof(dclc),
|
rc = smc_clc_wait_msg(smc, &dclc, sizeof(dclc),
|
||||||
SMC_CLC_DECLINE, CLC_WAIT_TIME_SHORT);
|
SMC_CLC_DECLINE, CLC_WAIT_TIME_SHORT);
|
||||||
return rc == -EAGAIN ? SMC_CLC_DECL_TIMEOUT_CL : rc;
|
return rc == -EAGAIN ? SMC_CLC_DECL_TIMEOUT_CL : rc;
|
||||||
}
|
}
|
||||||
|
rc = smc_llc_eval_conf_link(qentry, SMC_LLC_RESP);
|
||||||
if (link->llc_confirm_resp_rc)
|
smc_llc_flow_qentry_del(&link->lgr->llc_flow_lcl);
|
||||||
|
if (rc)
|
||||||
return SMC_CLC_DECL_RMBE_EC;
|
return SMC_CLC_DECL_RMBE_EC;
|
||||||
|
|
||||||
/* send ADD LINK request to client over the RoCE fabric */
|
/* confirm_rkey is implicit on 1st contact */
|
||||||
rc = smc_llc_send_add_link(link,
|
smc->conn.rmb_desc->is_conf_rkey = true;
|
||||||
link->smcibdev->mac[link->ibport - 1],
|
|
||||||
link->gid, SMC_LLC_REQ);
|
|
||||||
if (rc < 0)
|
|
||||||
return SMC_CLC_DECL_TIMEOUT_AL;
|
|
||||||
|
|
||||||
/* receive ADD LINK response from client over the RoCE fabric */
|
|
||||||
rest = wait_for_completion_interruptible_timeout(&link->llc_add_resp,
|
|
||||||
SMC_LLC_WAIT_TIME);
|
|
||||||
if (rest <= 0) {
|
|
||||||
struct smc_clc_msg_decline dclc;
|
|
||||||
|
|
||||||
rc = smc_clc_wait_msg(smc, &dclc, sizeof(dclc),
|
|
||||||
SMC_CLC_DECLINE, CLC_WAIT_TIME_SHORT);
|
|
||||||
return rc == -EAGAIN ? SMC_CLC_DECL_TIMEOUT_AL : rc;
|
|
||||||
}
|
|
||||||
|
|
||||||
smc_llc_link_active(link);
|
smc_llc_link_active(link);
|
||||||
|
|
||||||
|
/* initial contact - try to establish second link */
|
||||||
|
/* tbd: call smc_llc_srv_add_link(link); */
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1240,7 +1250,9 @@ static int smc_listen_rdma_finish(struct smc_sock *new_smc,
|
||||||
goto decline;
|
goto decline;
|
||||||
}
|
}
|
||||||
/* QP confirmation over RoCE fabric */
|
/* QP confirmation over RoCE fabric */
|
||||||
|
smc_llc_flow_initiate(link->lgr, SMC_LLC_FLOW_ADD_LINK);
|
||||||
reason_code = smcr_serv_conf_first_link(new_smc);
|
reason_code = smcr_serv_conf_first_link(new_smc);
|
||||||
|
smc_llc_flow_stop(link->lgr, &link->lgr->llc_flow_lcl);
|
||||||
if (reason_code)
|
if (reason_code)
|
||||||
goto decline;
|
goto decline;
|
||||||
}
|
}
|
||||||
|
|
|
@ -45,6 +45,7 @@
|
||||||
#define SMC_CLC_DECL_GETVLANERR 0x03080000 /* err to get vlan id of ip device*/
|
#define SMC_CLC_DECL_GETVLANERR 0x03080000 /* err to get vlan id of ip device*/
|
||||||
#define SMC_CLC_DECL_ISMVLANERR 0x03090000 /* err to reg vlan id on ism dev */
|
#define SMC_CLC_DECL_ISMVLANERR 0x03090000 /* err to reg vlan id on ism dev */
|
||||||
#define SMC_CLC_DECL_NOACTLINK 0x030a0000 /* no active smc-r link in lgr */
|
#define SMC_CLC_DECL_NOACTLINK 0x030a0000 /* no active smc-r link in lgr */
|
||||||
|
#define SMC_CLC_DECL_NOSRVLINK 0x030b0000 /* SMC-R link from srv not found */
|
||||||
#define SMC_CLC_DECL_SYNCERR 0x04000000 /* synchronization error */
|
#define SMC_CLC_DECL_SYNCERR 0x04000000 /* synchronization error */
|
||||||
#define SMC_CLC_DECL_PEERDECL 0x05000000 /* peer declined during handshake */
|
#define SMC_CLC_DECL_PEERDECL 0x05000000 /* peer declined during handshake */
|
||||||
#define SMC_CLC_DECL_INTERR 0x09990000 /* internal error */
|
#define SMC_CLC_DECL_INTERR 0x09990000 /* internal error */
|
||||||
|
|
|
@ -200,7 +200,6 @@ static int smcr_link_send_delete(struct smc_link *lnk, bool orderly)
|
||||||
{
|
{
|
||||||
if (lnk->state == SMC_LNK_ACTIVE &&
|
if (lnk->state == SMC_LNK_ACTIVE &&
|
||||||
!smc_llc_send_delete_link(lnk, SMC_LLC_REQ, orderly)) {
|
!smc_llc_send_delete_link(lnk, SMC_LLC_REQ, orderly)) {
|
||||||
smc_llc_link_deleting(lnk);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
return -ENOTCONN;
|
return -ENOTCONN;
|
||||||
|
@ -263,6 +262,7 @@ static void smc_lgr_free_work(struct work_struct *work)
|
||||||
if (smc_link_usable(lnk))
|
if (smc_link_usable(lnk))
|
||||||
lnk->state = SMC_LNK_INACTIVE;
|
lnk->state = SMC_LNK_INACTIVE;
|
||||||
}
|
}
|
||||||
|
wake_up_interruptible_all(&lgr->llc_waiter);
|
||||||
}
|
}
|
||||||
smc_lgr_free(lgr);
|
smc_lgr_free(lgr);
|
||||||
}
|
}
|
||||||
|
@ -445,13 +445,11 @@ out:
|
||||||
}
|
}
|
||||||
|
|
||||||
static void smcr_buf_unuse(struct smc_buf_desc *rmb_desc,
|
static void smcr_buf_unuse(struct smc_buf_desc *rmb_desc,
|
||||||
struct smc_link *lnk)
|
struct smc_link_group *lgr)
|
||||||
{
|
{
|
||||||
struct smc_link_group *lgr = lnk->lgr;
|
|
||||||
|
|
||||||
if (rmb_desc->is_conf_rkey && !list_empty(&lgr->list)) {
|
if (rmb_desc->is_conf_rkey && !list_empty(&lgr->list)) {
|
||||||
/* unregister rmb with peer */
|
/* unregister rmb with peer */
|
||||||
smc_llc_do_delete_rkey(lnk, rmb_desc);
|
smc_llc_do_delete_rkey(lgr, rmb_desc);
|
||||||
rmb_desc->is_conf_rkey = false;
|
rmb_desc->is_conf_rkey = false;
|
||||||
}
|
}
|
||||||
if (rmb_desc->is_reg_err) {
|
if (rmb_desc->is_reg_err) {
|
||||||
|
@ -474,7 +472,7 @@ static void smc_buf_unuse(struct smc_connection *conn,
|
||||||
if (conn->rmb_desc && lgr->is_smcd)
|
if (conn->rmb_desc && lgr->is_smcd)
|
||||||
conn->rmb_desc->used = 0;
|
conn->rmb_desc->used = 0;
|
||||||
else if (conn->rmb_desc)
|
else if (conn->rmb_desc)
|
||||||
smcr_buf_unuse(conn->rmb_desc, conn->lnk);
|
smcr_buf_unuse(conn->rmb_desc, lgr);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* remove a finished connection from its link group */
|
/* remove a finished connection from its link group */
|
||||||
|
@ -696,6 +694,7 @@ static void smc_lgr_cleanup(struct smc_link_group *lgr)
|
||||||
if (smc_link_usable(lnk))
|
if (smc_link_usable(lnk))
|
||||||
lnk->state = SMC_LNK_INACTIVE;
|
lnk->state = SMC_LNK_INACTIVE;
|
||||||
}
|
}
|
||||||
|
wake_up_interruptible_all(&lgr->llc_waiter);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -767,8 +766,7 @@ void smc_port_terminate(struct smc_ib_device *smcibdev, u8 ibport)
|
||||||
continue;
|
continue;
|
||||||
/* tbd - terminate only when no more links are active */
|
/* tbd - terminate only when no more links are active */
|
||||||
for (i = 0; i < SMC_LINKS_PER_LGR_MAX; i++) {
|
for (i = 0; i < SMC_LINKS_PER_LGR_MAX; i++) {
|
||||||
if (!smc_link_usable(&lgr->lnk[i]) ||
|
if (!smc_link_usable(&lgr->lnk[i]))
|
||||||
lgr->lnk[i].state == SMC_LNK_DELETING)
|
|
||||||
continue;
|
continue;
|
||||||
if (lgr->lnk[i].smcibdev == smcibdev &&
|
if (lgr->lnk[i].smcibdev == smcibdev &&
|
||||||
lgr->lnk[i].ibport == ibport) {
|
lgr->lnk[i].ibport == ibport) {
|
||||||
|
@ -1167,7 +1165,6 @@ static int smcr_buf_map_usable_links(struct smc_link_group *lgr,
|
||||||
if (!smc_link_usable(lnk))
|
if (!smc_link_usable(lnk))
|
||||||
continue;
|
continue;
|
||||||
if (smcr_buf_map_link(buf_desc, is_rmb, lnk)) {
|
if (smcr_buf_map_link(buf_desc, is_rmb, lnk)) {
|
||||||
smcr_buf_unuse(buf_desc, lnk);
|
|
||||||
rc = -ENOMEM;
|
rc = -ENOMEM;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
@ -1273,6 +1270,7 @@ static int __smc_buf_create(struct smc_sock *smc, bool is_smcd, bool is_rmb)
|
||||||
|
|
||||||
if (!is_smcd) {
|
if (!is_smcd) {
|
||||||
if (smcr_buf_map_usable_links(lgr, buf_desc, is_rmb)) {
|
if (smcr_buf_map_usable_links(lgr, buf_desc, is_rmb)) {
|
||||||
|
smcr_buf_unuse(buf_desc, lgr);
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1368,6 +1366,53 @@ static inline int smc_rmb_reserve_rtoken_idx(struct smc_link_group *lgr)
|
||||||
return -ENOSPC;
|
return -ENOSPC;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int smc_rtoken_find_by_link(struct smc_link_group *lgr, int lnk_idx,
|
||||||
|
u32 rkey)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < SMC_RMBS_PER_LGR_MAX; i++) {
|
||||||
|
if (test_bit(i, lgr->rtokens_used_mask) &&
|
||||||
|
lgr->rtokens[i][lnk_idx].rkey == rkey)
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
return -ENOENT;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* set rtoken for a new link to an existing rmb */
|
||||||
|
void smc_rtoken_set(struct smc_link_group *lgr, int link_idx, int link_idx_new,
|
||||||
|
__be32 nw_rkey_known, __be64 nw_vaddr, __be32 nw_rkey)
|
||||||
|
{
|
||||||
|
int rtok_idx;
|
||||||
|
|
||||||
|
rtok_idx = smc_rtoken_find_by_link(lgr, link_idx, ntohl(nw_rkey_known));
|
||||||
|
if (rtok_idx == -ENOENT)
|
||||||
|
return;
|
||||||
|
lgr->rtokens[rtok_idx][link_idx_new].rkey = ntohl(nw_rkey);
|
||||||
|
lgr->rtokens[rtok_idx][link_idx_new].dma_addr = be64_to_cpu(nw_vaddr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* set rtoken for a new link whose link_id is given */
|
||||||
|
void smc_rtoken_set2(struct smc_link_group *lgr, int rtok_idx, int link_id,
|
||||||
|
__be64 nw_vaddr, __be32 nw_rkey)
|
||||||
|
{
|
||||||
|
u64 dma_addr = be64_to_cpu(nw_vaddr);
|
||||||
|
u32 rkey = ntohl(nw_rkey);
|
||||||
|
bool found = false;
|
||||||
|
int link_idx;
|
||||||
|
|
||||||
|
for (link_idx = 0; link_idx < SMC_LINKS_PER_LGR_MAX; link_idx++) {
|
||||||
|
if (lgr->lnk[link_idx].link_id == link_id) {
|
||||||
|
found = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!found)
|
||||||
|
return;
|
||||||
|
lgr->rtokens[rtok_idx][link_idx].rkey = rkey;
|
||||||
|
lgr->rtokens[rtok_idx][link_idx].dma_addr = dma_addr;
|
||||||
|
}
|
||||||
|
|
||||||
/* add a new rtoken from peer */
|
/* add a new rtoken from peer */
|
||||||
int smc_rtoken_add(struct smc_link *lnk, __be64 nw_vaddr, __be32 nw_rkey)
|
int smc_rtoken_add(struct smc_link *lnk, __be64 nw_vaddr, __be32 nw_rkey)
|
||||||
{
|
{
|
||||||
|
|
|
@ -36,7 +36,6 @@ enum smc_link_state { /* possible states of a link */
|
||||||
SMC_LNK_INACTIVE, /* link is inactive */
|
SMC_LNK_INACTIVE, /* link is inactive */
|
||||||
SMC_LNK_ACTIVATING, /* link is being activated */
|
SMC_LNK_ACTIVATING, /* link is being activated */
|
||||||
SMC_LNK_ACTIVE, /* link is active */
|
SMC_LNK_ACTIVE, /* link is active */
|
||||||
SMC_LNK_DELETING, /* link is being deleted */
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#define SMC_WR_BUF_SIZE 48 /* size of work request buffer */
|
#define SMC_WR_BUF_SIZE 48 /* size of work request buffer */
|
||||||
|
@ -120,20 +119,9 @@ struct smc_link {
|
||||||
struct smc_link_group *lgr; /* parent link group */
|
struct smc_link_group *lgr; /* parent link group */
|
||||||
|
|
||||||
enum smc_link_state state; /* state of link */
|
enum smc_link_state state; /* state of link */
|
||||||
struct completion llc_confirm; /* wait for rx of conf link */
|
|
||||||
struct completion llc_confirm_resp; /* wait 4 rx of cnf lnk rsp */
|
|
||||||
int llc_confirm_rc; /* rc from confirm link msg */
|
|
||||||
int llc_confirm_resp_rc; /* rc from conf_resp msg */
|
|
||||||
struct completion llc_add; /* wait for rx of add link */
|
|
||||||
struct completion llc_add_resp; /* wait for rx of add link rsp*/
|
|
||||||
struct delayed_work llc_testlink_wrk; /* testlink worker */
|
struct delayed_work llc_testlink_wrk; /* testlink worker */
|
||||||
struct completion llc_testlink_resp; /* wait for rx of testlink */
|
struct completion llc_testlink_resp; /* wait for rx of testlink */
|
||||||
int llc_testlink_time; /* testlink interval */
|
int llc_testlink_time; /* testlink interval */
|
||||||
struct completion llc_confirm_rkey_resp; /* w4 rx of cnf rkey */
|
|
||||||
int llc_confirm_rkey_resp_rc; /* rc from cnf rkey */
|
|
||||||
struct completion llc_delete_rkey_resp; /* w4 rx of del rkey */
|
|
||||||
int llc_delete_rkey_resp_rc; /* rc from del rkey */
|
|
||||||
struct mutex llc_delete_rkey_mutex; /* serialize usage */
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/* For now we just allow one parallel link per link group. The SMC protocol
|
/* For now we just allow one parallel link per link group. The SMC protocol
|
||||||
|
@ -197,6 +185,28 @@ struct smc_rtoken { /* address/key of remote RMB */
|
||||||
|
|
||||||
struct smcd_dev;
|
struct smcd_dev;
|
||||||
|
|
||||||
|
enum smc_lgr_type { /* redundancy state of lgr */
|
||||||
|
SMC_LGR_NONE, /* no active links, lgr to be deleted */
|
||||||
|
SMC_LGR_SINGLE, /* 1 active RNIC on each peer */
|
||||||
|
SMC_LGR_SYMMETRIC, /* 2 active RNICs on each peer */
|
||||||
|
SMC_LGR_ASYMMETRIC_PEER, /* local has 2, peer 1 active RNICs */
|
||||||
|
SMC_LGR_ASYMMETRIC_LOCAL, /* local has 1, peer 2 active RNICs */
|
||||||
|
};
|
||||||
|
|
||||||
|
enum smc_llc_flowtype {
|
||||||
|
SMC_LLC_FLOW_NONE = 0,
|
||||||
|
SMC_LLC_FLOW_ADD_LINK = 2,
|
||||||
|
SMC_LLC_FLOW_DEL_LINK = 4,
|
||||||
|
SMC_LLC_FLOW_RKEY = 6,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct smc_llc_qentry;
|
||||||
|
|
||||||
|
struct smc_llc_flow {
|
||||||
|
enum smc_llc_flowtype type;
|
||||||
|
struct smc_llc_qentry *qentry;
|
||||||
|
};
|
||||||
|
|
||||||
struct smc_link_group {
|
struct smc_link_group {
|
||||||
struct list_head list;
|
struct list_head list;
|
||||||
struct rb_root conns_all; /* connection tree */
|
struct rb_root conns_all; /* connection tree */
|
||||||
|
@ -232,12 +242,24 @@ struct smc_link_group {
|
||||||
DECLARE_BITMAP(rtokens_used_mask, SMC_RMBS_PER_LGR_MAX);
|
DECLARE_BITMAP(rtokens_used_mask, SMC_RMBS_PER_LGR_MAX);
|
||||||
/* used rtoken elements */
|
/* used rtoken elements */
|
||||||
u8 next_link_id;
|
u8 next_link_id;
|
||||||
|
enum smc_lgr_type type;
|
||||||
|
/* redundancy state */
|
||||||
struct list_head llc_event_q;
|
struct list_head llc_event_q;
|
||||||
/* queue for llc events */
|
/* queue for llc events */
|
||||||
spinlock_t llc_event_q_lock;
|
spinlock_t llc_event_q_lock;
|
||||||
/* protects llc_event_q */
|
/* protects llc_event_q */
|
||||||
struct work_struct llc_event_work;
|
struct work_struct llc_event_work;
|
||||||
/* llc event worker */
|
/* llc event worker */
|
||||||
|
wait_queue_head_t llc_waiter;
|
||||||
|
/* w4 next llc event */
|
||||||
|
struct smc_llc_flow llc_flow_lcl;
|
||||||
|
/* llc local control field */
|
||||||
|
struct smc_llc_flow llc_flow_rmt;
|
||||||
|
/* llc remote control field */
|
||||||
|
struct smc_llc_qentry *delayed_event;
|
||||||
|
/* arrived when flow active */
|
||||||
|
spinlock_t llc_flow_lock;
|
||||||
|
/* protects llc flow */
|
||||||
int llc_testlink_time;
|
int llc_testlink_time;
|
||||||
/* link keep alive time */
|
/* link keep alive time */
|
||||||
};
|
};
|
||||||
|
@ -329,6 +351,10 @@ int smc_rmb_rtoken_handling(struct smc_connection *conn, struct smc_link *link,
|
||||||
struct smc_clc_msg_accept_confirm *clc);
|
struct smc_clc_msg_accept_confirm *clc);
|
||||||
int smc_rtoken_add(struct smc_link *lnk, __be64 nw_vaddr, __be32 nw_rkey);
|
int smc_rtoken_add(struct smc_link *lnk, __be64 nw_vaddr, __be32 nw_rkey);
|
||||||
int smc_rtoken_delete(struct smc_link *lnk, __be32 nw_rkey);
|
int smc_rtoken_delete(struct smc_link *lnk, __be32 nw_rkey);
|
||||||
|
void smc_rtoken_set(struct smc_link_group *lgr, int link_idx, int link_idx_new,
|
||||||
|
__be32 nw_rkey_known, __be64 nw_vaddr, __be32 nw_rkey);
|
||||||
|
void smc_rtoken_set2(struct smc_link_group *lgr, int rtok_idx, int link_id,
|
||||||
|
__be64 nw_vaddr, __be32 nw_rkey);
|
||||||
void smc_sndbuf_sync_sg_for_cpu(struct smc_connection *conn);
|
void smc_sndbuf_sync_sg_for_cpu(struct smc_connection *conn);
|
||||||
void smc_sndbuf_sync_sg_for_device(struct smc_connection *conn);
|
void smc_sndbuf_sync_sg_for_device(struct smc_connection *conn);
|
||||||
void smc_rmb_sync_sg_for_cpu(struct smc_connection *conn);
|
void smc_rmb_sync_sg_for_cpu(struct smc_connection *conn);
|
||||||
|
|
|
@ -98,13 +98,8 @@ struct smc_llc_msg_confirm_rkey { /* type 0x06 */
|
||||||
u8 reserved;
|
u8 reserved;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct smc_llc_msg_confirm_rkey_cont { /* type 0x08 */
|
|
||||||
struct smc_llc_hdr hd;
|
|
||||||
u8 num_rkeys;
|
|
||||||
struct smc_rmb_rtoken rtoken[SMC_LLC_RKEYS_PER_MSG];
|
|
||||||
};
|
|
||||||
|
|
||||||
#define SMC_LLC_DEL_RKEY_MAX 8
|
#define SMC_LLC_DEL_RKEY_MAX 8
|
||||||
|
#define SMC_LLC_FLAG_RKEY_RETRY 0x10
|
||||||
#define SMC_LLC_FLAG_RKEY_NEG 0x20
|
#define SMC_LLC_FLAG_RKEY_NEG 0x20
|
||||||
|
|
||||||
struct smc_llc_msg_delete_rkey { /* type 0x09 */
|
struct smc_llc_msg_delete_rkey { /* type 0x09 */
|
||||||
|
@ -122,7 +117,6 @@ union smc_llc_msg {
|
||||||
struct smc_llc_msg_del_link delete_link;
|
struct smc_llc_msg_del_link delete_link;
|
||||||
|
|
||||||
struct smc_llc_msg_confirm_rkey confirm_rkey;
|
struct smc_llc_msg_confirm_rkey confirm_rkey;
|
||||||
struct smc_llc_msg_confirm_rkey_cont confirm_rkey_cont;
|
|
||||||
struct smc_llc_msg_delete_rkey delete_rkey;
|
struct smc_llc_msg_delete_rkey delete_rkey;
|
||||||
|
|
||||||
struct smc_llc_msg_test_link test_link;
|
struct smc_llc_msg_test_link test_link;
|
||||||
|
@ -140,6 +134,154 @@ struct smc_llc_qentry {
|
||||||
union smc_llc_msg msg;
|
union smc_llc_msg msg;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct smc_llc_qentry *smc_llc_flow_qentry_clr(struct smc_llc_flow *flow)
|
||||||
|
{
|
||||||
|
struct smc_llc_qentry *qentry = flow->qentry;
|
||||||
|
|
||||||
|
flow->qentry = NULL;
|
||||||
|
return qentry;
|
||||||
|
}
|
||||||
|
|
||||||
|
void smc_llc_flow_qentry_del(struct smc_llc_flow *flow)
|
||||||
|
{
|
||||||
|
struct smc_llc_qentry *qentry;
|
||||||
|
|
||||||
|
if (flow->qentry) {
|
||||||
|
qentry = flow->qentry;
|
||||||
|
flow->qentry = NULL;
|
||||||
|
kfree(qentry);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void smc_llc_flow_qentry_set(struct smc_llc_flow *flow,
|
||||||
|
struct smc_llc_qentry *qentry)
|
||||||
|
{
|
||||||
|
flow->qentry = qentry;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* try to start a new llc flow, initiated by an incoming llc msg */
|
||||||
|
static bool smc_llc_flow_start(struct smc_llc_flow *flow,
|
||||||
|
struct smc_llc_qentry *qentry)
|
||||||
|
{
|
||||||
|
struct smc_link_group *lgr = qentry->link->lgr;
|
||||||
|
|
||||||
|
spin_lock_bh(&lgr->llc_flow_lock);
|
||||||
|
if (flow->type) {
|
||||||
|
/* a flow is already active */
|
||||||
|
if ((qentry->msg.raw.hdr.common.type == SMC_LLC_ADD_LINK ||
|
||||||
|
qentry->msg.raw.hdr.common.type == SMC_LLC_DELETE_LINK) &&
|
||||||
|
!lgr->delayed_event) {
|
||||||
|
lgr->delayed_event = qentry;
|
||||||
|
} else {
|
||||||
|
/* forget this llc request */
|
||||||
|
kfree(qentry);
|
||||||
|
}
|
||||||
|
spin_unlock_bh(&lgr->llc_flow_lock);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
switch (qentry->msg.raw.hdr.common.type) {
|
||||||
|
case SMC_LLC_ADD_LINK:
|
||||||
|
flow->type = SMC_LLC_FLOW_ADD_LINK;
|
||||||
|
break;
|
||||||
|
case SMC_LLC_DELETE_LINK:
|
||||||
|
flow->type = SMC_LLC_FLOW_DEL_LINK;
|
||||||
|
break;
|
||||||
|
case SMC_LLC_CONFIRM_RKEY:
|
||||||
|
case SMC_LLC_DELETE_RKEY:
|
||||||
|
flow->type = SMC_LLC_FLOW_RKEY;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
flow->type = SMC_LLC_FLOW_NONE;
|
||||||
|
}
|
||||||
|
if (qentry == lgr->delayed_event)
|
||||||
|
lgr->delayed_event = NULL;
|
||||||
|
spin_unlock_bh(&lgr->llc_flow_lock);
|
||||||
|
smc_llc_flow_qentry_set(flow, qentry);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* start a new local llc flow, wait till current flow finished */
|
||||||
|
int smc_llc_flow_initiate(struct smc_link_group *lgr,
|
||||||
|
enum smc_llc_flowtype type)
|
||||||
|
{
|
||||||
|
enum smc_llc_flowtype allowed_remote = SMC_LLC_FLOW_NONE;
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
/* all flows except confirm_rkey and delete_rkey are exclusive,
|
||||||
|
* confirm/delete rkey flows can run concurrently (local and remote)
|
||||||
|
*/
|
||||||
|
if (type == SMC_LLC_FLOW_RKEY)
|
||||||
|
allowed_remote = SMC_LLC_FLOW_RKEY;
|
||||||
|
again:
|
||||||
|
if (list_empty(&lgr->list))
|
||||||
|
return -ENODEV;
|
||||||
|
spin_lock_bh(&lgr->llc_flow_lock);
|
||||||
|
if (lgr->llc_flow_lcl.type == SMC_LLC_FLOW_NONE &&
|
||||||
|
(lgr->llc_flow_rmt.type == SMC_LLC_FLOW_NONE ||
|
||||||
|
lgr->llc_flow_rmt.type == allowed_remote)) {
|
||||||
|
lgr->llc_flow_lcl.type = type;
|
||||||
|
spin_unlock_bh(&lgr->llc_flow_lock);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
spin_unlock_bh(&lgr->llc_flow_lock);
|
||||||
|
rc = wait_event_interruptible_timeout(lgr->llc_waiter,
|
||||||
|
(lgr->llc_flow_lcl.type == SMC_LLC_FLOW_NONE &&
|
||||||
|
(lgr->llc_flow_rmt.type == SMC_LLC_FLOW_NONE ||
|
||||||
|
lgr->llc_flow_rmt.type == allowed_remote)),
|
||||||
|
SMC_LLC_WAIT_TIME);
|
||||||
|
if (!rc)
|
||||||
|
return -ETIMEDOUT;
|
||||||
|
goto again;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* finish the current llc flow */
|
||||||
|
void smc_llc_flow_stop(struct smc_link_group *lgr, struct smc_llc_flow *flow)
|
||||||
|
{
|
||||||
|
spin_lock_bh(&lgr->llc_flow_lock);
|
||||||
|
memset(flow, 0, sizeof(*flow));
|
||||||
|
flow->type = SMC_LLC_FLOW_NONE;
|
||||||
|
spin_unlock_bh(&lgr->llc_flow_lock);
|
||||||
|
if (!list_empty(&lgr->list) && lgr->delayed_event &&
|
||||||
|
flow == &lgr->llc_flow_lcl)
|
||||||
|
schedule_work(&lgr->llc_event_work);
|
||||||
|
else
|
||||||
|
wake_up_interruptible(&lgr->llc_waiter);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* lnk is optional and used for early wakeup when link goes down, useful in
|
||||||
|
* cases where we wait for a response on the link after we sent a request
|
||||||
|
*/
|
||||||
|
struct smc_llc_qentry *smc_llc_wait(struct smc_link_group *lgr,
|
||||||
|
struct smc_link *lnk,
|
||||||
|
int time_out, u8 exp_msg)
|
||||||
|
{
|
||||||
|
struct smc_llc_flow *flow = &lgr->llc_flow_lcl;
|
||||||
|
|
||||||
|
wait_event_interruptible_timeout(lgr->llc_waiter,
|
||||||
|
(flow->qentry ||
|
||||||
|
(lnk && !smc_link_usable(lnk)) ||
|
||||||
|
list_empty(&lgr->list)),
|
||||||
|
time_out);
|
||||||
|
if (!flow->qentry ||
|
||||||
|
(lnk && !smc_link_usable(lnk)) || list_empty(&lgr->list)) {
|
||||||
|
smc_llc_flow_qentry_del(flow);
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
if (exp_msg && flow->qentry->msg.raw.hdr.common.type != exp_msg) {
|
||||||
|
if (exp_msg == SMC_LLC_ADD_LINK &&
|
||||||
|
flow->qentry->msg.raw.hdr.common.type ==
|
||||||
|
SMC_LLC_DELETE_LINK) {
|
||||||
|
/* flow_start will delay the unexpected msg */
|
||||||
|
smc_llc_flow_start(&lgr->llc_flow_lcl,
|
||||||
|
smc_llc_flow_qentry_clr(flow));
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
smc_llc_flow_qentry_del(flow);
|
||||||
|
}
|
||||||
|
out:
|
||||||
|
return flow->qentry;
|
||||||
|
}
|
||||||
|
|
||||||
/********************************** send *************************************/
|
/********************************** send *************************************/
|
||||||
|
|
||||||
struct smc_llc_tx_pend {
|
struct smc_llc_tx_pend {
|
||||||
|
@ -221,27 +363,44 @@ int smc_llc_send_confirm_link(struct smc_link *link,
|
||||||
}
|
}
|
||||||
|
|
||||||
/* send LLC confirm rkey request */
|
/* send LLC confirm rkey request */
|
||||||
static int smc_llc_send_confirm_rkey(struct smc_link *link,
|
static int smc_llc_send_confirm_rkey(struct smc_link *send_link,
|
||||||
struct smc_buf_desc *rmb_desc)
|
struct smc_buf_desc *rmb_desc)
|
||||||
{
|
{
|
||||||
struct smc_llc_msg_confirm_rkey *rkeyllc;
|
struct smc_llc_msg_confirm_rkey *rkeyllc;
|
||||||
struct smc_wr_tx_pend_priv *pend;
|
struct smc_wr_tx_pend_priv *pend;
|
||||||
struct smc_wr_buf *wr_buf;
|
struct smc_wr_buf *wr_buf;
|
||||||
int rc;
|
struct smc_link *link;
|
||||||
|
int i, rc, rtok_ix;
|
||||||
|
|
||||||
rc = smc_llc_add_pending_send(link, &wr_buf, &pend);
|
rc = smc_llc_add_pending_send(send_link, &wr_buf, &pend);
|
||||||
if (rc)
|
if (rc)
|
||||||
return rc;
|
return rc;
|
||||||
rkeyllc = (struct smc_llc_msg_confirm_rkey *)wr_buf;
|
rkeyllc = (struct smc_llc_msg_confirm_rkey *)wr_buf;
|
||||||
memset(rkeyllc, 0, sizeof(*rkeyllc));
|
memset(rkeyllc, 0, sizeof(*rkeyllc));
|
||||||
rkeyllc->hd.common.type = SMC_LLC_CONFIRM_RKEY;
|
rkeyllc->hd.common.type = SMC_LLC_CONFIRM_RKEY;
|
||||||
rkeyllc->hd.length = sizeof(struct smc_llc_msg_confirm_rkey);
|
rkeyllc->hd.length = sizeof(struct smc_llc_msg_confirm_rkey);
|
||||||
|
|
||||||
|
rtok_ix = 1;
|
||||||
|
for (i = 0; i < SMC_LINKS_PER_LGR_MAX; i++) {
|
||||||
|
link = &send_link->lgr->lnk[i];
|
||||||
|
if (link->state == SMC_LNK_ACTIVE && link != send_link) {
|
||||||
|
rkeyllc->rtoken[rtok_ix].link_id = link->link_id;
|
||||||
|
rkeyllc->rtoken[rtok_ix].rmb_key =
|
||||||
|
htonl(rmb_desc->mr_rx[link->link_idx]->rkey);
|
||||||
|
rkeyllc->rtoken[rtok_ix].rmb_vaddr = cpu_to_be64(
|
||||||
|
(u64)sg_dma_address(
|
||||||
|
rmb_desc->sgt[link->link_idx].sgl));
|
||||||
|
rtok_ix++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/* rkey of send_link is in rtoken[0] */
|
||||||
|
rkeyllc->rtoken[0].num_rkeys = rtok_ix - 1;
|
||||||
rkeyllc->rtoken[0].rmb_key =
|
rkeyllc->rtoken[0].rmb_key =
|
||||||
htonl(rmb_desc->mr_rx[link->link_idx]->rkey);
|
htonl(rmb_desc->mr_rx[send_link->link_idx]->rkey);
|
||||||
rkeyllc->rtoken[0].rmb_vaddr = cpu_to_be64(
|
rkeyllc->rtoken[0].rmb_vaddr = cpu_to_be64(
|
||||||
(u64)sg_dma_address(rmb_desc->sgt[link->link_idx].sgl));
|
(u64)sg_dma_address(rmb_desc->sgt[send_link->link_idx].sgl));
|
||||||
/* send llc message */
|
/* send llc message */
|
||||||
rc = smc_wr_tx_send(link, pend);
|
rc = smc_wr_tx_send(send_link, pend);
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -380,54 +539,12 @@ static int smc_llc_send_message(struct smc_link *link, void *llcbuf)
|
||||||
|
|
||||||
/********************************* receive ***********************************/
|
/********************************* receive ***********************************/
|
||||||
|
|
||||||
static void smc_llc_rx_confirm_link(struct smc_link *link,
|
|
||||||
struct smc_llc_msg_confirm_link *llc)
|
|
||||||
{
|
|
||||||
struct smc_link_group *lgr = smc_get_lgr(link);
|
|
||||||
int conf_rc = 0;
|
|
||||||
|
|
||||||
/* RMBE eyecatchers are not supported */
|
|
||||||
if (!(llc->hd.flags & SMC_LLC_FLAG_NO_RMBE_EYEC))
|
|
||||||
conf_rc = ENOTSUPP;
|
|
||||||
|
|
||||||
if (lgr->role == SMC_CLNT &&
|
|
||||||
link->state == SMC_LNK_ACTIVATING) {
|
|
||||||
link->llc_confirm_rc = conf_rc;
|
|
||||||
link->link_id = llc->link_num;
|
|
||||||
complete(&link->llc_confirm);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void smc_llc_rx_add_link(struct smc_link *link,
|
|
||||||
struct smc_llc_msg_add_link *llc)
|
|
||||||
{
|
|
||||||
struct smc_link_group *lgr = smc_get_lgr(link);
|
|
||||||
|
|
||||||
if (link->state == SMC_LNK_ACTIVATING) {
|
|
||||||
complete(&link->llc_add);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (lgr->role == SMC_SERV) {
|
|
||||||
smc_llc_prep_add_link(llc, link,
|
|
||||||
link->smcibdev->mac[link->ibport - 1],
|
|
||||||
link->gid, SMC_LLC_REQ);
|
|
||||||
|
|
||||||
} else {
|
|
||||||
smc_llc_prep_add_link(llc, link,
|
|
||||||
link->smcibdev->mac[link->ibport - 1],
|
|
||||||
link->gid, SMC_LLC_RESP);
|
|
||||||
}
|
|
||||||
smc_llc_send_message(link, llc);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void smc_llc_rx_delete_link(struct smc_link *link,
|
static void smc_llc_rx_delete_link(struct smc_link *link,
|
||||||
struct smc_llc_msg_del_link *llc)
|
struct smc_llc_msg_del_link *llc)
|
||||||
{
|
{
|
||||||
struct smc_link_group *lgr = smc_get_lgr(link);
|
struct smc_link_group *lgr = smc_get_lgr(link);
|
||||||
|
|
||||||
smc_lgr_forget(lgr);
|
smc_lgr_forget(lgr);
|
||||||
smc_llc_link_deleting(link);
|
|
||||||
if (lgr->role == SMC_SERV) {
|
if (lgr->role == SMC_SERV) {
|
||||||
/* client asks to delete this link, send request */
|
/* client asks to delete this link, send request */
|
||||||
smc_llc_prep_delete_link(llc, link, SMC_LLC_REQ, true);
|
smc_llc_prep_delete_link(llc, link, SMC_LLC_REQ, true);
|
||||||
|
@ -439,57 +556,68 @@ static void smc_llc_rx_delete_link(struct smc_link *link,
|
||||||
smc_lgr_terminate_sched(lgr);
|
smc_lgr_terminate_sched(lgr);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void smc_llc_rx_test_link(struct smc_link *link,
|
/* process a confirm_rkey request from peer, remote flow */
|
||||||
struct smc_llc_msg_test_link *llc)
|
static void smc_llc_rmt_conf_rkey(struct smc_link_group *lgr)
|
||||||
{
|
{
|
||||||
|
struct smc_llc_msg_confirm_rkey *llc;
|
||||||
|
struct smc_llc_qentry *qentry;
|
||||||
|
struct smc_link *link;
|
||||||
|
int num_entries;
|
||||||
|
int rk_idx;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
qentry = lgr->llc_flow_rmt.qentry;
|
||||||
|
llc = &qentry->msg.confirm_rkey;
|
||||||
|
link = qentry->link;
|
||||||
|
|
||||||
|
num_entries = llc->rtoken[0].num_rkeys;
|
||||||
|
/* first rkey entry is for receiving link */
|
||||||
|
rk_idx = smc_rtoken_add(link,
|
||||||
|
llc->rtoken[0].rmb_vaddr,
|
||||||
|
llc->rtoken[0].rmb_key);
|
||||||
|
if (rk_idx < 0)
|
||||||
|
goto out_err;
|
||||||
|
|
||||||
|
for (i = 1; i <= min_t(u8, num_entries, SMC_LLC_RKEYS_PER_MSG - 1); i++)
|
||||||
|
smc_rtoken_set2(lgr, rk_idx, llc->rtoken[i].link_id,
|
||||||
|
llc->rtoken[i].rmb_vaddr,
|
||||||
|
llc->rtoken[i].rmb_key);
|
||||||
|
/* max links is 3 so there is no need to support conf_rkey_cont msgs */
|
||||||
|
goto out;
|
||||||
|
out_err:
|
||||||
|
llc->hd.flags |= SMC_LLC_FLAG_RKEY_NEG;
|
||||||
|
llc->hd.flags |= SMC_LLC_FLAG_RKEY_RETRY;
|
||||||
|
out:
|
||||||
llc->hd.flags |= SMC_LLC_FLAG_RESP;
|
llc->hd.flags |= SMC_LLC_FLAG_RESP;
|
||||||
smc_llc_send_message(link, llc);
|
smc_llc_send_message(link, &qentry->msg);
|
||||||
|
smc_llc_flow_qentry_del(&lgr->llc_flow_rmt);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void smc_llc_rx_confirm_rkey(struct smc_link *link,
|
/* process a delete_rkey request from peer, remote flow */
|
||||||
struct smc_llc_msg_confirm_rkey *llc)
|
static void smc_llc_rmt_delete_rkey(struct smc_link_group *lgr)
|
||||||
{
|
|
||||||
int rc;
|
|
||||||
|
|
||||||
rc = smc_rtoken_add(link,
|
|
||||||
llc->rtoken[0].rmb_vaddr,
|
|
||||||
llc->rtoken[0].rmb_key);
|
|
||||||
|
|
||||||
/* ignore rtokens for other links, we have only one link */
|
|
||||||
|
|
||||||
llc->hd.flags |= SMC_LLC_FLAG_RESP;
|
|
||||||
if (rc < 0)
|
|
||||||
llc->hd.flags |= SMC_LLC_FLAG_RKEY_NEG;
|
|
||||||
smc_llc_send_message(link, llc);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void smc_llc_rx_confirm_rkey_cont(struct smc_link *link,
|
|
||||||
struct smc_llc_msg_confirm_rkey_cont *llc)
|
|
||||||
{
|
|
||||||
/* ignore rtokens for other links, we have only one link */
|
|
||||||
llc->hd.flags |= SMC_LLC_FLAG_RESP;
|
|
||||||
smc_llc_send_message(link, llc);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void smc_llc_rx_delete_rkey(struct smc_link *link,
|
|
||||||
struct smc_llc_msg_delete_rkey *llc)
|
|
||||||
{
|
{
|
||||||
|
struct smc_llc_msg_delete_rkey *llc;
|
||||||
|
struct smc_llc_qentry *qentry;
|
||||||
|
struct smc_link *link;
|
||||||
u8 err_mask = 0;
|
u8 err_mask = 0;
|
||||||
int i, max;
|
int i, max;
|
||||||
|
|
||||||
|
qentry = lgr->llc_flow_rmt.qentry;
|
||||||
|
llc = &qentry->msg.delete_rkey;
|
||||||
|
link = qentry->link;
|
||||||
|
|
||||||
max = min_t(u8, llc->num_rkeys, SMC_LLC_DEL_RKEY_MAX);
|
max = min_t(u8, llc->num_rkeys, SMC_LLC_DEL_RKEY_MAX);
|
||||||
for (i = 0; i < max; i++) {
|
for (i = 0; i < max; i++) {
|
||||||
if (smc_rtoken_delete(link, llc->rkey[i]))
|
if (smc_rtoken_delete(link, llc->rkey[i]))
|
||||||
err_mask |= 1 << (SMC_LLC_DEL_RKEY_MAX - 1 - i);
|
err_mask |= 1 << (SMC_LLC_DEL_RKEY_MAX - 1 - i);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (err_mask) {
|
if (err_mask) {
|
||||||
llc->hd.flags |= SMC_LLC_FLAG_RKEY_NEG;
|
llc->hd.flags |= SMC_LLC_FLAG_RKEY_NEG;
|
||||||
llc->err_mask = err_mask;
|
llc->err_mask = err_mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
llc->hd.flags |= SMC_LLC_FLAG_RESP;
|
llc->hd.flags |= SMC_LLC_FLAG_RESP;
|
||||||
smc_llc_send_message(link, llc);
|
smc_llc_send_message(link, &qentry->msg);
|
||||||
|
smc_llc_flow_qentry_del(&lgr->llc_flow_rmt);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* flush the llc event queue */
|
/* flush the llc event queue */
|
||||||
|
@ -509,32 +637,66 @@ static void smc_llc_event_handler(struct smc_llc_qentry *qentry)
|
||||||
{
|
{
|
||||||
union smc_llc_msg *llc = &qentry->msg;
|
union smc_llc_msg *llc = &qentry->msg;
|
||||||
struct smc_link *link = qentry->link;
|
struct smc_link *link = qentry->link;
|
||||||
|
struct smc_link_group *lgr = link->lgr;
|
||||||
|
|
||||||
if (!smc_link_usable(link))
|
if (!smc_link_usable(link))
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
switch (llc->raw.hdr.common.type) {
|
switch (llc->raw.hdr.common.type) {
|
||||||
case SMC_LLC_TEST_LINK:
|
case SMC_LLC_TEST_LINK:
|
||||||
smc_llc_rx_test_link(link, &llc->test_link);
|
llc->test_link.hd.flags |= SMC_LLC_FLAG_RESP;
|
||||||
break;
|
smc_llc_send_message(link, llc);
|
||||||
case SMC_LLC_CONFIRM_LINK:
|
|
||||||
smc_llc_rx_confirm_link(link, &llc->confirm_link);
|
|
||||||
break;
|
break;
|
||||||
case SMC_LLC_ADD_LINK:
|
case SMC_LLC_ADD_LINK:
|
||||||
smc_llc_rx_add_link(link, &llc->add_link);
|
if (list_empty(&lgr->list))
|
||||||
|
goto out; /* lgr is terminating */
|
||||||
|
if (lgr->role == SMC_CLNT) {
|
||||||
|
if (lgr->llc_flow_lcl.type == SMC_LLC_FLOW_ADD_LINK) {
|
||||||
|
/* a flow is waiting for this message */
|
||||||
|
smc_llc_flow_qentry_set(&lgr->llc_flow_lcl,
|
||||||
|
qentry);
|
||||||
|
wake_up_interruptible(&lgr->llc_waiter);
|
||||||
|
} else if (smc_llc_flow_start(&lgr->llc_flow_lcl,
|
||||||
|
qentry)) {
|
||||||
|
/* tbd: schedule_work(&lgr->llc_add_link_work); */
|
||||||
|
}
|
||||||
|
} else if (smc_llc_flow_start(&lgr->llc_flow_lcl, qentry)) {
|
||||||
|
/* as smc server, handle client suggestion */
|
||||||
|
/* tbd: schedule_work(&lgr->llc_add_link_work); */
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
case SMC_LLC_CONFIRM_LINK:
|
||||||
|
if (lgr->llc_flow_lcl.type != SMC_LLC_FLOW_NONE) {
|
||||||
|
/* a flow is waiting for this message */
|
||||||
|
smc_llc_flow_qentry_set(&lgr->llc_flow_lcl, qentry);
|
||||||
|
wake_up_interruptible(&lgr->llc_waiter);
|
||||||
|
return;
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case SMC_LLC_DELETE_LINK:
|
case SMC_LLC_DELETE_LINK:
|
||||||
smc_llc_rx_delete_link(link, &llc->delete_link);
|
smc_llc_rx_delete_link(link, &llc->delete_link);
|
||||||
break;
|
break;
|
||||||
case SMC_LLC_CONFIRM_RKEY:
|
case SMC_LLC_CONFIRM_RKEY:
|
||||||
smc_llc_rx_confirm_rkey(link, &llc->confirm_rkey);
|
/* new request from remote, assign to remote flow */
|
||||||
break;
|
if (smc_llc_flow_start(&lgr->llc_flow_rmt, qentry)) {
|
||||||
|
/* process here, does not wait for more llc msgs */
|
||||||
|
smc_llc_rmt_conf_rkey(lgr);
|
||||||
|
smc_llc_flow_stop(lgr, &lgr->llc_flow_rmt);
|
||||||
|
}
|
||||||
|
return;
|
||||||
case SMC_LLC_CONFIRM_RKEY_CONT:
|
case SMC_LLC_CONFIRM_RKEY_CONT:
|
||||||
smc_llc_rx_confirm_rkey_cont(link, &llc->confirm_rkey_cont);
|
/* not used because max links is 3, and 3 rkeys fit into
|
||||||
|
* one CONFIRM_RKEY message
|
||||||
|
*/
|
||||||
break;
|
break;
|
||||||
case SMC_LLC_DELETE_RKEY:
|
case SMC_LLC_DELETE_RKEY:
|
||||||
smc_llc_rx_delete_rkey(link, &llc->delete_rkey);
|
/* new request from remote, assign to remote flow */
|
||||||
break;
|
if (smc_llc_flow_start(&lgr->llc_flow_rmt, qentry)) {
|
||||||
|
/* process here, does not wait for more llc msgs */
|
||||||
|
smc_llc_rmt_delete_rkey(lgr);
|
||||||
|
smc_llc_flow_stop(lgr, &lgr->llc_flow_rmt);
|
||||||
|
}
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
out:
|
out:
|
||||||
kfree(qentry);
|
kfree(qentry);
|
||||||
|
@ -547,6 +709,16 @@ static void smc_llc_event_work(struct work_struct *work)
|
||||||
llc_event_work);
|
llc_event_work);
|
||||||
struct smc_llc_qentry *qentry;
|
struct smc_llc_qentry *qentry;
|
||||||
|
|
||||||
|
if (!lgr->llc_flow_lcl.type && lgr->delayed_event) {
|
||||||
|
if (smc_link_usable(lgr->delayed_event->link)) {
|
||||||
|
smc_llc_event_handler(lgr->delayed_event);
|
||||||
|
} else {
|
||||||
|
qentry = lgr->delayed_event;
|
||||||
|
lgr->delayed_event = NULL;
|
||||||
|
kfree(qentry);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
again:
|
again:
|
||||||
spin_lock_bh(&lgr->llc_event_q_lock);
|
spin_lock_bh(&lgr->llc_event_q_lock);
|
||||||
if (!list_empty(&lgr->llc_event_q)) {
|
if (!list_empty(&lgr->llc_event_q)) {
|
||||||
|
@ -561,80 +733,75 @@ again:
|
||||||
}
|
}
|
||||||
|
|
||||||
/* process llc responses in tasklet context */
|
/* process llc responses in tasklet context */
|
||||||
static void smc_llc_rx_response(struct smc_link *link, union smc_llc_msg *llc)
|
static void smc_llc_rx_response(struct smc_link *link,
|
||||||
|
struct smc_llc_qentry *qentry)
|
||||||
{
|
{
|
||||||
int rc = 0;
|
u8 llc_type = qentry->msg.raw.hdr.common.type;
|
||||||
|
|
||||||
switch (llc->raw.hdr.common.type) {
|
switch (llc_type) {
|
||||||
case SMC_LLC_TEST_LINK:
|
case SMC_LLC_TEST_LINK:
|
||||||
if (link->state == SMC_LNK_ACTIVE)
|
if (link->state == SMC_LNK_ACTIVE)
|
||||||
complete(&link->llc_testlink_resp);
|
complete(&link->llc_testlink_resp);
|
||||||
break;
|
break;
|
||||||
case SMC_LLC_CONFIRM_LINK:
|
|
||||||
if (!(llc->raw.hdr.flags & SMC_LLC_FLAG_NO_RMBE_EYEC))
|
|
||||||
rc = ENOTSUPP;
|
|
||||||
if (link->lgr->role == SMC_SERV &&
|
|
||||||
link->state == SMC_LNK_ACTIVATING) {
|
|
||||||
link->llc_confirm_resp_rc = rc;
|
|
||||||
complete(&link->llc_confirm_resp);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case SMC_LLC_ADD_LINK:
|
case SMC_LLC_ADD_LINK:
|
||||||
if (link->state == SMC_LNK_ACTIVATING)
|
case SMC_LLC_CONFIRM_LINK:
|
||||||
complete(&link->llc_add_resp);
|
case SMC_LLC_CONFIRM_RKEY:
|
||||||
break;
|
case SMC_LLC_DELETE_RKEY:
|
||||||
|
/* assign responses to the local flow, we requested them */
|
||||||
|
smc_llc_flow_qentry_set(&link->lgr->llc_flow_lcl, qentry);
|
||||||
|
wake_up_interruptible(&link->lgr->llc_waiter);
|
||||||
|
return;
|
||||||
case SMC_LLC_DELETE_LINK:
|
case SMC_LLC_DELETE_LINK:
|
||||||
if (link->lgr->role == SMC_SERV)
|
if (link->lgr->role == SMC_SERV)
|
||||||
smc_lgr_schedule_free_work_fast(link->lgr);
|
smc_lgr_schedule_free_work_fast(link->lgr);
|
||||||
break;
|
break;
|
||||||
case SMC_LLC_CONFIRM_RKEY:
|
|
||||||
link->llc_confirm_rkey_resp_rc = llc->raw.hdr.flags &
|
|
||||||
SMC_LLC_FLAG_RKEY_NEG;
|
|
||||||
complete(&link->llc_confirm_rkey_resp);
|
|
||||||
break;
|
|
||||||
case SMC_LLC_CONFIRM_RKEY_CONT:
|
case SMC_LLC_CONFIRM_RKEY_CONT:
|
||||||
/* unused as long as we don't send this type of msg */
|
/* not used because max links is 3 */
|
||||||
break;
|
|
||||||
case SMC_LLC_DELETE_RKEY:
|
|
||||||
link->llc_delete_rkey_resp_rc = llc->raw.hdr.flags &
|
|
||||||
SMC_LLC_FLAG_RKEY_NEG;
|
|
||||||
complete(&link->llc_delete_rkey_resp);
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
kfree(qentry);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* copy received msg and add it to the event queue */
|
static void smc_llc_enqueue(struct smc_link *link, union smc_llc_msg *llc)
|
||||||
static void smc_llc_rx_handler(struct ib_wc *wc, void *buf)
|
|
||||||
{
|
{
|
||||||
struct smc_link *link = (struct smc_link *)wc->qp->qp_context;
|
|
||||||
struct smc_link_group *lgr = link->lgr;
|
struct smc_link_group *lgr = link->lgr;
|
||||||
struct smc_llc_qentry *qentry;
|
struct smc_llc_qentry *qentry;
|
||||||
union smc_llc_msg *llc = buf;
|
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
|
||||||
if (wc->byte_len < sizeof(*llc))
|
|
||||||
return; /* short message */
|
|
||||||
if (llc->raw.hdr.length != sizeof(*llc))
|
|
||||||
return; /* invalid message */
|
|
||||||
|
|
||||||
/* process responses immediately */
|
|
||||||
if (llc->raw.hdr.flags & SMC_LLC_FLAG_RESP) {
|
|
||||||
smc_llc_rx_response(link, llc);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
qentry = kmalloc(sizeof(*qentry), GFP_ATOMIC);
|
qentry = kmalloc(sizeof(*qentry), GFP_ATOMIC);
|
||||||
if (!qentry)
|
if (!qentry)
|
||||||
return;
|
return;
|
||||||
qentry->link = link;
|
qentry->link = link;
|
||||||
INIT_LIST_HEAD(&qentry->list);
|
INIT_LIST_HEAD(&qentry->list);
|
||||||
memcpy(&qentry->msg, llc, sizeof(union smc_llc_msg));
|
memcpy(&qentry->msg, llc, sizeof(union smc_llc_msg));
|
||||||
|
|
||||||
|
/* process responses immediately */
|
||||||
|
if (llc->raw.hdr.flags & SMC_LLC_FLAG_RESP) {
|
||||||
|
smc_llc_rx_response(link, qentry);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* add requests to event queue */
|
||||||
spin_lock_irqsave(&lgr->llc_event_q_lock, flags);
|
spin_lock_irqsave(&lgr->llc_event_q_lock, flags);
|
||||||
list_add_tail(&qentry->list, &lgr->llc_event_q);
|
list_add_tail(&qentry->list, &lgr->llc_event_q);
|
||||||
spin_unlock_irqrestore(&lgr->llc_event_q_lock, flags);
|
spin_unlock_irqrestore(&lgr->llc_event_q_lock, flags);
|
||||||
schedule_work(&link->lgr->llc_event_work);
|
schedule_work(&link->lgr->llc_event_work);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* copy received msg and add it to the event queue */
|
||||||
|
static void smc_llc_rx_handler(struct ib_wc *wc, void *buf)
|
||||||
|
{
|
||||||
|
struct smc_link *link = (struct smc_link *)wc->qp->qp_context;
|
||||||
|
union smc_llc_msg *llc = buf;
|
||||||
|
|
||||||
|
if (wc->byte_len < sizeof(*llc))
|
||||||
|
return; /* short message */
|
||||||
|
if (llc->raw.hdr.length != sizeof(*llc))
|
||||||
|
return; /* invalid message */
|
||||||
|
|
||||||
|
smc_llc_enqueue(link, llc);
|
||||||
|
}
|
||||||
|
|
||||||
/***************************** worker, utils *********************************/
|
/***************************** worker, utils *********************************/
|
||||||
|
|
||||||
static void smc_llc_testlink_work(struct work_struct *work)
|
static void smc_llc_testlink_work(struct work_struct *work)
|
||||||
|
@ -676,6 +843,8 @@ void smc_llc_lgr_init(struct smc_link_group *lgr, struct smc_sock *smc)
|
||||||
INIT_WORK(&lgr->llc_event_work, smc_llc_event_work);
|
INIT_WORK(&lgr->llc_event_work, smc_llc_event_work);
|
||||||
INIT_LIST_HEAD(&lgr->llc_event_q);
|
INIT_LIST_HEAD(&lgr->llc_event_q);
|
||||||
spin_lock_init(&lgr->llc_event_q_lock);
|
spin_lock_init(&lgr->llc_event_q_lock);
|
||||||
|
spin_lock_init(&lgr->llc_flow_lock);
|
||||||
|
init_waitqueue_head(&lgr->llc_waiter);
|
||||||
lgr->llc_testlink_time = net->ipv4.sysctl_tcp_keepalive_time;
|
lgr->llc_testlink_time = net->ipv4.sysctl_tcp_keepalive_time;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -683,18 +852,16 @@ void smc_llc_lgr_init(struct smc_link_group *lgr, struct smc_sock *smc)
|
||||||
void smc_llc_lgr_clear(struct smc_link_group *lgr)
|
void smc_llc_lgr_clear(struct smc_link_group *lgr)
|
||||||
{
|
{
|
||||||
smc_llc_event_flush(lgr);
|
smc_llc_event_flush(lgr);
|
||||||
|
wake_up_interruptible_all(&lgr->llc_waiter);
|
||||||
cancel_work_sync(&lgr->llc_event_work);
|
cancel_work_sync(&lgr->llc_event_work);
|
||||||
|
if (lgr->delayed_event) {
|
||||||
|
kfree(lgr->delayed_event);
|
||||||
|
lgr->delayed_event = NULL;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int smc_llc_link_init(struct smc_link *link)
|
int smc_llc_link_init(struct smc_link *link)
|
||||||
{
|
{
|
||||||
init_completion(&link->llc_confirm);
|
|
||||||
init_completion(&link->llc_confirm_resp);
|
|
||||||
init_completion(&link->llc_add);
|
|
||||||
init_completion(&link->llc_add_resp);
|
|
||||||
init_completion(&link->llc_confirm_rkey_resp);
|
|
||||||
init_completion(&link->llc_delete_rkey_resp);
|
|
||||||
mutex_init(&link->llc_delete_rkey_mutex);
|
|
||||||
init_completion(&link->llc_testlink_resp);
|
init_completion(&link->llc_testlink_resp);
|
||||||
INIT_DELAYED_WORK(&link->llc_testlink_wrk, smc_llc_testlink_work);
|
INIT_DELAYED_WORK(&link->llc_testlink_wrk, smc_llc_testlink_work);
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -710,12 +877,6 @@ void smc_llc_link_active(struct smc_link *link)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void smc_llc_link_deleting(struct smc_link *link)
|
|
||||||
{
|
|
||||||
link->state = SMC_LNK_DELETING;
|
|
||||||
smc_wr_wakeup_tx_wait(link);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* called in worker context */
|
/* called in worker context */
|
||||||
void smc_llc_link_clear(struct smc_link *link)
|
void smc_llc_link_clear(struct smc_link *link)
|
||||||
{
|
{
|
||||||
|
@ -725,50 +886,74 @@ void smc_llc_link_clear(struct smc_link *link)
|
||||||
smc_wr_wakeup_tx_wait(link);
|
smc_wr_wakeup_tx_wait(link);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* register a new rtoken at the remote peer */
|
/* register a new rtoken at the remote peer (for all links) */
|
||||||
int smc_llc_do_confirm_rkey(struct smc_link *link,
|
int smc_llc_do_confirm_rkey(struct smc_link *send_link,
|
||||||
struct smc_buf_desc *rmb_desc)
|
struct smc_buf_desc *rmb_desc)
|
||||||
{
|
{
|
||||||
int rc;
|
struct smc_link_group *lgr = send_link->lgr;
|
||||||
|
struct smc_llc_qentry *qentry = NULL;
|
||||||
|
int rc = 0;
|
||||||
|
|
||||||
/* protected by mutex smc_create_lgr_pending */
|
rc = smc_llc_flow_initiate(lgr, SMC_LLC_FLOW_RKEY);
|
||||||
reinit_completion(&link->llc_confirm_rkey_resp);
|
|
||||||
rc = smc_llc_send_confirm_rkey(link, rmb_desc);
|
|
||||||
if (rc)
|
if (rc)
|
||||||
return rc;
|
return rc;
|
||||||
|
rc = smc_llc_send_confirm_rkey(send_link, rmb_desc);
|
||||||
|
if (rc)
|
||||||
|
goto out;
|
||||||
/* receive CONFIRM RKEY response from server over RoCE fabric */
|
/* receive CONFIRM RKEY response from server over RoCE fabric */
|
||||||
rc = wait_for_completion_interruptible_timeout(
|
qentry = smc_llc_wait(lgr, send_link, SMC_LLC_WAIT_TIME,
|
||||||
&link->llc_confirm_rkey_resp, SMC_LLC_WAIT_TIME);
|
SMC_LLC_CONFIRM_RKEY);
|
||||||
if (rc <= 0 || link->llc_confirm_rkey_resp_rc)
|
if (!qentry || (qentry->msg.raw.hdr.flags & SMC_LLC_FLAG_RKEY_NEG))
|
||||||
return -EFAULT;
|
rc = -EFAULT;
|
||||||
return 0;
|
out:
|
||||||
|
if (qentry)
|
||||||
|
smc_llc_flow_qentry_del(&lgr->llc_flow_lcl);
|
||||||
|
smc_llc_flow_stop(lgr, &lgr->llc_flow_lcl);
|
||||||
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* unregister an rtoken at the remote peer */
|
/* unregister an rtoken at the remote peer */
|
||||||
int smc_llc_do_delete_rkey(struct smc_link *link,
|
int smc_llc_do_delete_rkey(struct smc_link_group *lgr,
|
||||||
struct smc_buf_desc *rmb_desc)
|
struct smc_buf_desc *rmb_desc)
|
||||||
{
|
{
|
||||||
|
struct smc_llc_qentry *qentry = NULL;
|
||||||
|
struct smc_link *send_link;
|
||||||
int rc = 0;
|
int rc = 0;
|
||||||
|
|
||||||
mutex_lock(&link->llc_delete_rkey_mutex);
|
send_link = smc_llc_usable_link(lgr);
|
||||||
if (link->state != SMC_LNK_ACTIVE)
|
if (!send_link)
|
||||||
goto out;
|
return -ENOLINK;
|
||||||
reinit_completion(&link->llc_delete_rkey_resp);
|
|
||||||
rc = smc_llc_send_delete_rkey(link, rmb_desc);
|
rc = smc_llc_flow_initiate(lgr, SMC_LLC_FLOW_RKEY);
|
||||||
|
if (rc)
|
||||||
|
return rc;
|
||||||
|
/* protected by llc_flow control */
|
||||||
|
rc = smc_llc_send_delete_rkey(send_link, rmb_desc);
|
||||||
if (rc)
|
if (rc)
|
||||||
goto out;
|
goto out;
|
||||||
/* receive DELETE RKEY response from server over RoCE fabric */
|
/* receive DELETE RKEY response from server over RoCE fabric */
|
||||||
rc = wait_for_completion_interruptible_timeout(
|
qentry = smc_llc_wait(lgr, send_link, SMC_LLC_WAIT_TIME,
|
||||||
&link->llc_delete_rkey_resp, SMC_LLC_WAIT_TIME);
|
SMC_LLC_DELETE_RKEY);
|
||||||
if (rc <= 0 || link->llc_delete_rkey_resp_rc)
|
if (!qentry || (qentry->msg.raw.hdr.flags & SMC_LLC_FLAG_RKEY_NEG))
|
||||||
rc = -EFAULT;
|
rc = -EFAULT;
|
||||||
else
|
|
||||||
rc = 0;
|
|
||||||
out:
|
out:
|
||||||
mutex_unlock(&link->llc_delete_rkey_mutex);
|
if (qentry)
|
||||||
|
smc_llc_flow_qentry_del(&lgr->llc_flow_lcl);
|
||||||
|
smc_llc_flow_stop(lgr, &lgr->llc_flow_lcl);
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* evaluate confirm link request or response */
|
||||||
|
int smc_llc_eval_conf_link(struct smc_llc_qentry *qentry,
|
||||||
|
enum smc_llc_reqresp type)
|
||||||
|
{
|
||||||
|
if (type == SMC_LLC_REQ) /* SMC server assigns link_id */
|
||||||
|
qentry->link->link_id = qentry->msg.confirm_link.link_num;
|
||||||
|
if (!(qentry->msg.raw.hdr.flags & SMC_LLC_FLAG_NO_RMBE_EYEC))
|
||||||
|
return -ENOTSUPP;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/***************************** init, exit, misc ******************************/
|
/***************************** init, exit, misc ******************************/
|
||||||
|
|
||||||
static struct smc_wr_rx_handler smc_llc_rx_handlers[] = {
|
static struct smc_wr_rx_handler smc_llc_rx_handlers[] = {
|
||||||
|
|
|
@ -57,12 +57,21 @@ void smc_llc_lgr_init(struct smc_link_group *lgr, struct smc_sock *smc);
|
||||||
void smc_llc_lgr_clear(struct smc_link_group *lgr);
|
void smc_llc_lgr_clear(struct smc_link_group *lgr);
|
||||||
int smc_llc_link_init(struct smc_link *link);
|
int smc_llc_link_init(struct smc_link *link);
|
||||||
void smc_llc_link_active(struct smc_link *link);
|
void smc_llc_link_active(struct smc_link *link);
|
||||||
void smc_llc_link_deleting(struct smc_link *link);
|
|
||||||
void smc_llc_link_clear(struct smc_link *link);
|
void smc_llc_link_clear(struct smc_link *link);
|
||||||
int smc_llc_do_confirm_rkey(struct smc_link *link,
|
int smc_llc_do_confirm_rkey(struct smc_link *send_link,
|
||||||
struct smc_buf_desc *rmb_desc);
|
struct smc_buf_desc *rmb_desc);
|
||||||
int smc_llc_do_delete_rkey(struct smc_link *link,
|
int smc_llc_do_delete_rkey(struct smc_link_group *lgr,
|
||||||
struct smc_buf_desc *rmb_desc);
|
struct smc_buf_desc *rmb_desc);
|
||||||
|
int smc_llc_flow_initiate(struct smc_link_group *lgr,
|
||||||
|
enum smc_llc_flowtype type);
|
||||||
|
void smc_llc_flow_stop(struct smc_link_group *lgr, struct smc_llc_flow *flow);
|
||||||
|
int smc_llc_eval_conf_link(struct smc_llc_qentry *qentry,
|
||||||
|
enum smc_llc_reqresp type);
|
||||||
|
struct smc_llc_qentry *smc_llc_wait(struct smc_link_group *lgr,
|
||||||
|
struct smc_link *lnk,
|
||||||
|
int time_out, u8 exp_msg);
|
||||||
|
struct smc_llc_qentry *smc_llc_flow_qentry_clr(struct smc_llc_flow *flow);
|
||||||
|
void smc_llc_flow_qentry_del(struct smc_llc_flow *flow);
|
||||||
int smc_llc_init(void) __init;
|
int smc_llc_init(void) __init;
|
||||||
|
|
||||||
#endif /* SMC_LLC_H */
|
#endif /* SMC_LLC_H */
|
||||||
|
|
Loading…
Reference in New Issue