diff options
Diffstat (limited to 'fs/smb/client/smbdirect.c')
| -rw-r--r-- | fs/smb/client/smbdirect.c | 103 | 
1 files changed, 65 insertions, 38 deletions
| diff --git a/fs/smb/client/smbdirect.c b/fs/smb/client/smbdirect.c index 49e2df3ad1f0..85a4c55b61b8 100644 --- a/fs/smb/client/smbdirect.c +++ b/fs/smb/client/smbdirect.c @@ -172,6 +172,7 @@ static void smbd_disconnect_wake_up_all(struct smbdirect_socket *sc)  	 * in order to notice the broken connection.  	 */  	wake_up_all(&sc->status_wait); +	wake_up_all(&sc->send_io.lcredits.wait_queue);  	wake_up_all(&sc->send_io.credits.wait_queue);  	wake_up_all(&sc->send_io.pending.dec_wait_queue);  	wake_up_all(&sc->send_io.pending.zero_wait_queue); @@ -495,6 +496,7 @@ static void send_done(struct ib_cq *cq, struct ib_wc *wc)  	struct smbdirect_send_io *request =  		container_of(wc->wr_cqe, struct smbdirect_send_io, cqe);  	struct smbdirect_socket *sc = request->socket; +	int lcredits = 0;  	log_rdma_send(INFO, "smbdirect_send_io 0x%p completed wc->status=%s\n",  		request, ib_wc_status_msg(wc->status)); @@ -504,22 +506,24 @@ static void send_done(struct ib_cq *cq, struct ib_wc *wc)  			request->sge[i].addr,  			request->sge[i].length,  			DMA_TO_DEVICE); +	mempool_free(request, sc->send_io.mem.pool); +	lcredits += 1;  	if (wc->status != IB_WC_SUCCESS || wc->opcode != IB_WC_SEND) {  		if (wc->status != IB_WC_WR_FLUSH_ERR)  			log_rdma_send(ERR, "wc->status=%s wc->opcode=%d\n",  				ib_wc_status_msg(wc->status), wc->opcode); -		mempool_free(request, sc->send_io.mem.pool);  		smbd_disconnect_rdma_connection(sc);  		return;  	} +	atomic_add(lcredits, &sc->send_io.lcredits.count); +	wake_up(&sc->send_io.lcredits.wait_queue); +  	if (atomic_dec_and_test(&sc->send_io.pending.count))  		wake_up(&sc->send_io.pending.zero_wait_queue);  	wake_up(&sc->send_io.pending.dec_wait_queue); - -	mempool_free(request, sc->send_io.mem.pool);  }  static void dump_smbdirect_negotiate_resp(struct smbdirect_negotiate_resp *resp) @@ -567,6 +571,7 @@ static bool process_negotiation_response(  		log_rdma_event(ERR, "error: credits_granted==0\n");  		return false;  	} +	atomic_set(&sc->send_io.lcredits.count, sp->send_credit_target);  	atomic_set(&sc->send_io.credits.count, le16_to_cpu(packet->credits_granted));  	if (le32_to_cpu(packet->preferred_send_size) > sp->max_recv_size) { @@ -1114,6 +1119,24 @@ static int smbd_post_send_iter(struct smbdirect_socket *sc,  	struct smbdirect_data_transfer *packet;  	int new_credits = 0; +wait_lcredit: +	/* Wait for local send credits */ +	rc = wait_event_interruptible(sc->send_io.lcredits.wait_queue, +		atomic_read(&sc->send_io.lcredits.count) > 0 || +		sc->status != SMBDIRECT_SOCKET_CONNECTED); +	if (rc) +		goto err_wait_lcredit; + +	if (sc->status != SMBDIRECT_SOCKET_CONNECTED) { +		log_outgoing(ERR, "disconnected not sending on wait_credit\n"); +		rc = -EAGAIN; +		goto err_wait_lcredit; +	} +	if (unlikely(atomic_dec_return(&sc->send_io.lcredits.count) < 0)) { +		atomic_inc(&sc->send_io.lcredits.count); +		goto wait_lcredit; +	} +  wait_credit:  	/* Wait for send credits. A SMBD packet needs one credit */  	rc = wait_event_interruptible(sc->send_io.credits.wait_queue, @@ -1132,23 +1155,6 @@ wait_credit:  		goto wait_credit;  	} -wait_send_queue: -	wait_event(sc->send_io.pending.dec_wait_queue, -		atomic_read(&sc->send_io.pending.count) < sp->send_credit_target || -		sc->status != SMBDIRECT_SOCKET_CONNECTED); - -	if (sc->status != SMBDIRECT_SOCKET_CONNECTED) { -		log_outgoing(ERR, "disconnected not sending on wait_send_queue\n"); -		rc = -EAGAIN; -		goto err_wait_send_queue; -	} - -	if (unlikely(atomic_inc_return(&sc->send_io.pending.count) > -				sp->send_credit_target)) { -		atomic_dec(&sc->send_io.pending.count); -		goto wait_send_queue; -	} -  	request = mempool_alloc(sc->send_io.mem.pool, GFP_KERNEL);  	if (!request) {  		rc = -ENOMEM; @@ -1229,10 +1235,21 @@ wait_send_queue:  		     le32_to_cpu(packet->data_length),  		     le32_to_cpu(packet->remaining_data_length)); +	/* +	 * Now that we got a local and a remote credit +	 * we add us as pending +	 */ +	atomic_inc(&sc->send_io.pending.count); +  	rc = smbd_post_send(sc, request);  	if (!rc)  		return 0; +	if (atomic_dec_and_test(&sc->send_io.pending.count)) +		wake_up(&sc->send_io.pending.zero_wait_queue); + +	wake_up(&sc->send_io.pending.dec_wait_queue); +  err_dma:  	for (i = 0; i < request->num_sge; i++)  		if (request->sge[i].addr) @@ -1246,14 +1263,14 @@ err_dma:  	atomic_sub(new_credits, &sc->recv_io.credits.count);  err_alloc: -	if (atomic_dec_and_test(&sc->send_io.pending.count)) -		wake_up(&sc->send_io.pending.zero_wait_queue); - -err_wait_send_queue: -	/* roll back send credits and pending */  	atomic_inc(&sc->send_io.credits.count); +	wake_up(&sc->send_io.credits.wait_queue);  err_wait_credit: +	atomic_inc(&sc->send_io.lcredits.count); +	wake_up(&sc->send_io.lcredits.wait_queue); + +err_wait_lcredit:  	return rc;  } @@ -1767,6 +1784,7 @@ static struct smbd_connection *_smbd_get_connection(  	struct smbdirect_socket *sc;  	struct smbdirect_socket_parameters *sp;  	struct rdma_conn_param conn_param; +	struct ib_qp_cap qp_cap;  	struct ib_qp_init_attr qp_attr;  	struct sockaddr_in *addr_in = (struct sockaddr_in *) dstaddr;  	struct ib_port_immutable port_immutable; @@ -1838,6 +1856,25 @@ static struct smbd_connection *_smbd_get_connection(  		goto config_failed;  	} +	sp->responder_resources = +		min_t(u8, sp->responder_resources, +		      sc->ib.dev->attrs.max_qp_rd_atom); +	log_rdma_mr(INFO, "responder_resources=%d\n", +		sp->responder_resources); + +	/* +	 * We use allocate sp->responder_resources * 2 MRs +	 * and each MR needs WRs for REG and INV, so +	 * we use '* 4'. +	 * +	 * +1 for ib_drain_qp() +	 */ +	memset(&qp_cap, 0, sizeof(qp_cap)); +	qp_cap.max_send_wr = sp->send_credit_target + sp->responder_resources * 4 + 1; +	qp_cap.max_recv_wr = sp->recv_credit_max + 1; +	qp_cap.max_send_sge = SMBDIRECT_SEND_IO_MAX_SGE; +	qp_cap.max_recv_sge = SMBDIRECT_RECV_IO_MAX_SGE; +  	sc->ib.pd = ib_alloc_pd(sc->ib.dev, 0);  	if (IS_ERR(sc->ib.pd)) {  		rc = PTR_ERR(sc->ib.pd); @@ -1848,7 +1885,7 @@ static struct smbd_connection *_smbd_get_connection(  	sc->ib.send_cq =  		ib_alloc_cq_any(sc->ib.dev, sc, -				sp->send_credit_target, IB_POLL_SOFTIRQ); +				qp_cap.max_send_wr, IB_POLL_SOFTIRQ);  	if (IS_ERR(sc->ib.send_cq)) {  		sc->ib.send_cq = NULL;  		goto alloc_cq_failed; @@ -1856,7 +1893,7 @@ static struct smbd_connection *_smbd_get_connection(  	sc->ib.recv_cq =  		ib_alloc_cq_any(sc->ib.dev, sc, -				sp->recv_credit_max, IB_POLL_SOFTIRQ); +				qp_cap.max_recv_wr, IB_POLL_SOFTIRQ);  	if (IS_ERR(sc->ib.recv_cq)) {  		sc->ib.recv_cq = NULL;  		goto alloc_cq_failed; @@ -1865,11 +1902,7 @@ static struct smbd_connection *_smbd_get_connection(  	memset(&qp_attr, 0, sizeof(qp_attr));  	qp_attr.event_handler = smbd_qp_async_error_upcall;  	qp_attr.qp_context = sc; -	qp_attr.cap.max_send_wr = sp->send_credit_target; -	qp_attr.cap.max_recv_wr = sp->recv_credit_max; -	qp_attr.cap.max_send_sge = SMBDIRECT_SEND_IO_MAX_SGE; -	qp_attr.cap.max_recv_sge = SMBDIRECT_RECV_IO_MAX_SGE; -	qp_attr.cap.max_inline_data = 0; +	qp_attr.cap = qp_cap;  	qp_attr.sq_sig_type = IB_SIGNAL_REQ_WR;  	qp_attr.qp_type = IB_QPT_RC;  	qp_attr.send_cq = sc->ib.send_cq; @@ -1883,12 +1916,6 @@ static struct smbd_connection *_smbd_get_connection(  	}  	sc->ib.qp = sc->rdma.cm_id->qp; -	sp->responder_resources = -		min_t(u8, sp->responder_resources, -		      sc->ib.dev->attrs.max_qp_rd_atom); -	log_rdma_mr(INFO, "responder_resources=%d\n", -		sp->responder_resources); -  	memset(&conn_param, 0, sizeof(conn_param));  	conn_param.initiator_depth = sp->initiator_depth;  	conn_param.responder_resources = sp->responder_resources; | 
