| // SPDX-License-Identifier: GPL-2.0-or-later |
| /* |
| * Copyright (C) 2017, Microsoft Corporation. |
| * Copyright (C) 2018, LG Electronics. |
| * |
| * Author(s): Long Li <longli@microsoft.com>, |
| * Hyunchul Lee <hyc.lee@gmail.com> |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License as published by |
| * the Free Software Foundation; either version 2 of the License, or |
| * (at your option) any later version. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See |
| * the GNU General Public License for more details. |
| */ |
| |
| #define SUBMOD_NAME "smb_direct" |
| |
| #include <linux/kthread.h> |
| #include <linux/list.h> |
| #include <linux/mempool.h> |
| #include <linux/highmem.h> |
| #include <linux/scatterlist.h> |
| #include <rdma/ib_verbs.h> |
| #include <rdma/rdma_cm.h> |
| #include <rdma/rw.h> |
| |
| #include "glob.h" |
| #include "connection.h" |
| #include "smb_common.h" |
| #include "smbstatus.h" |
| #include "transport_rdma.h" |
| |
| #define SMB_DIRECT_PORT_IWARP 5445 |
| #define SMB_DIRECT_PORT_INFINIBAND 445 |
| |
| #define SMB_DIRECT_VERSION_LE cpu_to_le16(0x0100) |
| |
| /* SMB_DIRECT negotiation timeout in seconds */ |
| #define SMB_DIRECT_NEGOTIATE_TIMEOUT 120 |
| |
| #define SMB_DIRECT_MAX_SEND_SGES 8 |
| #define SMB_DIRECT_MAX_RECV_SGES 1 |
| |
| /* |
| * Default maximum number of RDMA read/write outstanding on this connection |
| * This value is possibly decreased during QP creation on hardware limit |
| */ |
| #define SMB_DIRECT_CM_INITIATOR_DEPTH 8 |
| |
| /* Maximum number of retries on data transfer operations */ |
| #define SMB_DIRECT_CM_RETRY 6 |
| /* No need to retry on Receiver Not Ready since SMB_DIRECT manages credits */ |
| #define SMB_DIRECT_CM_RNR_RETRY 0 |
| |
| /* |
| * User configurable initial values per SMB_DIRECT transport connection |
| * as defined in [MS-SMBD] 3.1.1.1 |
| * Those may change after a SMB_DIRECT negotiation |
| */ |
| |
| /* Set 445 port to SMB Direct port by default */ |
| static int smb_direct_port = SMB_DIRECT_PORT_INFINIBAND; |
| |
| /* The local peer's maximum number of credits to grant to the peer */ |
| static int smb_direct_receive_credit_max = 255; |
| |
| /* The remote peer's credit request of local peer */ |
| static int smb_direct_send_credit_target = 255; |
| |
| /* The maximum single message size can be sent to remote peer */ |
| static int smb_direct_max_send_size = 8192; |
| |
| /* The maximum fragmented upper-layer payload receive size supported */ |
| static int smb_direct_max_fragmented_recv_size = 1024 * 1024; |
| |
| /* The maximum single-message size which can be received */ |
| static int smb_direct_max_receive_size = 8192; |
| |
| static int smb_direct_max_read_write_size = SMBD_DEFAULT_IOSIZE; |
| |
| static LIST_HEAD(smb_direct_device_list); |
| static DEFINE_RWLOCK(smb_direct_device_lock); |
| |
| struct smb_direct_device { |
| struct ib_device *ib_dev; |
| struct list_head list; |
| }; |
| |
| static struct smb_direct_listener { |
| struct rdma_cm_id *cm_id; |
| } smb_direct_listener; |
| |
| static struct workqueue_struct *smb_direct_wq; |
| |
| enum smb_direct_status { |
| SMB_DIRECT_CS_NEW = 0, |
| SMB_DIRECT_CS_CONNECTED, |
| SMB_DIRECT_CS_DISCONNECTING, |
| SMB_DIRECT_CS_DISCONNECTED, |
| }; |
| |
| struct smb_direct_transport { |
| struct ksmbd_transport transport; |
| |
| enum smb_direct_status status; |
| bool full_packet_received; |
| wait_queue_head_t wait_status; |
| |
| struct rdma_cm_id *cm_id; |
| struct ib_cq *send_cq; |
| struct ib_cq *recv_cq; |
| struct ib_pd *pd; |
| struct ib_qp *qp; |
| |
| int max_send_size; |
| int max_recv_size; |
| int max_fragmented_send_size; |
| int max_fragmented_recv_size; |
| int max_rdma_rw_size; |
| |
| spinlock_t reassembly_queue_lock; |
| struct list_head reassembly_queue; |
| int reassembly_data_length; |
| int reassembly_queue_length; |
| int first_entry_offset; |
| wait_queue_head_t wait_reassembly_queue; |
| |
| spinlock_t receive_credit_lock; |
| int recv_credits; |
| int count_avail_recvmsg; |
| int recv_credit_max; |
| int recv_credit_target; |
| |
| spinlock_t recvmsg_queue_lock; |
| struct list_head recvmsg_queue; |
| |
| spinlock_t empty_recvmsg_queue_lock; |
| struct list_head empty_recvmsg_queue; |
| |
| int send_credit_target; |
| atomic_t send_credits; |
| spinlock_t lock_new_recv_credits; |
| int new_recv_credits; |
| int max_rw_credits; |
| int pages_per_rw_credit; |
| atomic_t rw_credits; |
| |
| wait_queue_head_t wait_send_credits; |
| wait_queue_head_t wait_rw_credits; |
| |
| mempool_t *sendmsg_mempool; |
| struct kmem_cache *sendmsg_cache; |
| mempool_t *recvmsg_mempool; |
| struct kmem_cache *recvmsg_cache; |
| |
| wait_queue_head_t wait_send_pending; |
| atomic_t send_pending; |
| |
| struct delayed_work post_recv_credits_work; |
| struct work_struct send_immediate_work; |
| struct work_struct disconnect_work; |
| |
| bool negotiation_requested; |
| }; |
| |
| #define KSMBD_TRANS(t) ((struct ksmbd_transport *)&((t)->transport)) |
| |
| enum { |
| SMB_DIRECT_MSG_NEGOTIATE_REQ = 0, |
| SMB_DIRECT_MSG_DATA_TRANSFER |
| }; |
| |
| static struct ksmbd_transport_ops ksmbd_smb_direct_transport_ops; |
| |
| struct smb_direct_send_ctx { |
| struct list_head msg_list; |
| int wr_cnt; |
| bool need_invalidate_rkey; |
| unsigned int remote_key; |
| }; |
| |
| struct smb_direct_sendmsg { |
| struct smb_direct_transport *transport; |
| struct ib_send_wr wr; |
| struct list_head list; |
| int num_sge; |
| struct ib_sge sge[SMB_DIRECT_MAX_SEND_SGES]; |
| struct ib_cqe cqe; |
| u8 packet[]; |
| }; |
| |
| struct smb_direct_recvmsg { |
| struct smb_direct_transport *transport; |
| struct list_head list; |
| int type; |
| struct ib_sge sge; |
| struct ib_cqe cqe; |
| bool first_segment; |
| u8 packet[]; |
| }; |
| |
| struct smb_direct_rdma_rw_msg { |
| struct smb_direct_transport *t; |
| struct ib_cqe cqe; |
| int status; |
| struct completion *completion; |
| struct list_head list; |
| struct rdma_rw_ctx rw_ctx; |
| struct sg_table sgt; |
| struct scatterlist sg_list[]; |
| }; |
| |
| void init_smbd_max_io_size(unsigned int sz) |
| { |
| sz = clamp_val(sz, SMBD_MIN_IOSIZE, SMBD_MAX_IOSIZE); |
| smb_direct_max_read_write_size = sz; |
| } |
| |
| unsigned int get_smbd_max_read_write_size(void) |
| { |
| return smb_direct_max_read_write_size; |
| } |
| |
| static inline int get_buf_page_count(void *buf, int size) |
| { |
| return DIV_ROUND_UP((uintptr_t)buf + size, PAGE_SIZE) - |
| (uintptr_t)buf / PAGE_SIZE; |
| } |
| |
| static void smb_direct_destroy_pools(struct smb_direct_transport *transport); |
| static void smb_direct_post_recv_credits(struct work_struct *work); |
| static int smb_direct_post_send_data(struct smb_direct_transport *t, |
| struct smb_direct_send_ctx *send_ctx, |
| struct kvec *iov, int niov, |
| int remaining_data_length); |
| |
| static inline struct smb_direct_transport * |
| smb_trans_direct_transfort(struct ksmbd_transport *t) |
| { |
| return container_of(t, struct smb_direct_transport, transport); |
| } |
| |
| static inline void |
| *smb_direct_recvmsg_payload(struct smb_direct_recvmsg *recvmsg) |
| { |
| return (void *)recvmsg->packet; |
| } |
| |
| static inline bool is_receive_credit_post_required(int receive_credits, |
| int avail_recvmsg_count) |
| { |
| return receive_credits <= (smb_direct_receive_credit_max >> 3) && |
| avail_recvmsg_count >= (receive_credits >> 2); |
| } |
| |
| static struct |
| smb_direct_recvmsg *get_free_recvmsg(struct smb_direct_transport *t) |
| { |
| struct smb_direct_recvmsg *recvmsg = NULL; |
| |
| spin_lock(&t->recvmsg_queue_lock); |
| if (!list_empty(&t->recvmsg_queue)) { |
| recvmsg = list_first_entry(&t->recvmsg_queue, |
| struct smb_direct_recvmsg, |
| list); |
| list_del(&recvmsg->list); |
| } |
| spin_unlock(&t->recvmsg_queue_lock); |
| return recvmsg; |
| } |
| |
| static void put_recvmsg(struct smb_direct_transport *t, |
| struct smb_direct_recvmsg *recvmsg) |
| { |
| ib_dma_unmap_single(t->cm_id->device, recvmsg->sge.addr, |
| recvmsg->sge.length, DMA_FROM_DEVICE); |
| |
| spin_lock(&t->recvmsg_queue_lock); |
| list_add(&recvmsg->list, &t->recvmsg_queue); |
| spin_unlock(&t->recvmsg_queue_lock); |
| } |
| |
| static struct |
| smb_direct_recvmsg *get_empty_recvmsg(struct smb_direct_transport *t) |
| { |
| struct smb_direct_recvmsg *recvmsg = NULL; |
| |
| spin_lock(&t->empty_recvmsg_queue_lock); |
| if (!list_empty(&t->empty_recvmsg_queue)) { |
| recvmsg = list_first_entry(&t->empty_recvmsg_queue, |
| struct smb_direct_recvmsg, list); |
| list_del(&recvmsg->list); |
| } |
| spin_unlock(&t->empty_recvmsg_queue_lock); |
| return recvmsg; |
| } |
| |
| static void put_empty_recvmsg(struct smb_direct_transport *t, |
| struct smb_direct_recvmsg *recvmsg) |
| { |
| ib_dma_unmap_single(t->cm_id->device, recvmsg->sge.addr, |
| recvmsg->sge.length, DMA_FROM_DEVICE); |
| |
| spin_lock(&t->empty_recvmsg_queue_lock); |
| list_add_tail(&recvmsg->list, &t->empty_recvmsg_queue); |
| spin_unlock(&t->empty_recvmsg_queue_lock); |
| } |
| |
| static void enqueue_reassembly(struct smb_direct_transport *t, |
| struct smb_direct_recvmsg *recvmsg, |
| int data_length) |
| { |
| spin_lock(&t->reassembly_queue_lock); |
| list_add_tail(&recvmsg->list, &t->reassembly_queue); |
| t->reassembly_queue_length++; |
| /* |
| * Make sure reassembly_data_length is updated after list and |
| * reassembly_queue_length are updated. On the dequeue side |
| * reassembly_data_length is checked without a lock to determine |
| * if reassembly_queue_length and list is up to date |
| */ |
| virt_wmb(); |
| t->reassembly_data_length += data_length; |
| spin_unlock(&t->reassembly_queue_lock); |
| } |
| |
| static struct smb_direct_recvmsg *get_first_reassembly(struct smb_direct_transport *t) |
| { |
| if (!list_empty(&t->reassembly_queue)) |
| return list_first_entry(&t->reassembly_queue, |
| struct smb_direct_recvmsg, list); |
| else |
| return NULL; |
| } |
| |
| static void smb_direct_disconnect_rdma_work(struct work_struct *work) |
| { |
| struct smb_direct_transport *t = |
| container_of(work, struct smb_direct_transport, |
| disconnect_work); |
| |
| if (t->status == SMB_DIRECT_CS_CONNECTED) { |
| t->status = SMB_DIRECT_CS_DISCONNECTING; |
| rdma_disconnect(t->cm_id); |
| } |
| } |
| |
| static void |
| smb_direct_disconnect_rdma_connection(struct smb_direct_transport *t) |
| { |
| if (t->status == SMB_DIRECT_CS_CONNECTED) |
| queue_work(smb_direct_wq, &t->disconnect_work); |
| } |
| |
| static void smb_direct_send_immediate_work(struct work_struct *work) |
| { |
| struct smb_direct_transport *t = container_of(work, |
| struct smb_direct_transport, send_immediate_work); |
| |
| if (t->status != SMB_DIRECT_CS_CONNECTED) |
| return; |
| |
| smb_direct_post_send_data(t, NULL, NULL, 0, 0); |
| } |
| |
| static struct smb_direct_transport *alloc_transport(struct rdma_cm_id *cm_id) |
| { |
| struct smb_direct_transport *t; |
| struct ksmbd_conn *conn; |
| |
| t = kzalloc(sizeof(*t), GFP_KERNEL); |
| if (!t) |
| return NULL; |
| |
| t->cm_id = cm_id; |
| cm_id->context = t; |
| |
| t->status = SMB_DIRECT_CS_NEW; |
| init_waitqueue_head(&t->wait_status); |
| |
| spin_lock_init(&t->reassembly_queue_lock); |
| INIT_LIST_HEAD(&t->reassembly_queue); |
| t->reassembly_data_length = 0; |
| t->reassembly_queue_length = 0; |
| init_waitqueue_head(&t->wait_reassembly_queue); |
| init_waitqueue_head(&t->wait_send_credits); |
| init_waitqueue_head(&t->wait_rw_credits); |
| |
| spin_lock_init(&t->receive_credit_lock); |
| spin_lock_init(&t->recvmsg_queue_lock); |
| INIT_LIST_HEAD(&t->recvmsg_queue); |
| |
| spin_lock_init(&t->empty_recvmsg_queue_lock); |
| INIT_LIST_HEAD(&t->empty_recvmsg_queue); |
| |
| init_waitqueue_head(&t->wait_send_pending); |
| atomic_set(&t->send_pending, 0); |
| |
| spin_lock_init(&t->lock_new_recv_credits); |
| |
| INIT_DELAYED_WORK(&t->post_recv_credits_work, |
| smb_direct_post_recv_credits); |
| INIT_WORK(&t->send_immediate_work, smb_direct_send_immediate_work); |
| INIT_WORK(&t->disconnect_work, smb_direct_disconnect_rdma_work); |
| |
| conn = ksmbd_conn_alloc(); |
| if (!conn) |
| goto err; |
| conn->transport = KSMBD_TRANS(t); |
| KSMBD_TRANS(t)->conn = conn; |
| KSMBD_TRANS(t)->ops = &ksmbd_smb_direct_transport_ops; |
| return t; |
| err: |
| kfree(t); |
| return NULL; |
| } |
| |
| static void free_transport(struct smb_direct_transport *t) |
| { |
| struct smb_direct_recvmsg *recvmsg; |
| |
| wake_up_interruptible(&t->wait_send_credits); |
| |
| ksmbd_debug(RDMA, "wait for all send posted to IB to finish\n"); |
| wait_event(t->wait_send_pending, |
| atomic_read(&t->send_pending) == 0); |
| |
| cancel_work_sync(&t->disconnect_work); |
| cancel_delayed_work_sync(&t->post_recv_credits_work); |
| cancel_work_sync(&t->send_immediate_work); |
| |
| if (t->qp) { |
| ib_drain_qp(t->qp); |
| ib_mr_pool_destroy(t->qp, &t->qp->rdma_mrs); |
| ib_destroy_qp(t->qp); |
| } |
| |
| ksmbd_debug(RDMA, "drain the reassembly queue\n"); |
| do { |
| spin_lock(&t->reassembly_queue_lock); |
| recvmsg = get_first_reassembly(t); |
| if (recvmsg) { |
| list_del(&recvmsg->list); |
| spin_unlock(&t->reassembly_queue_lock); |
| put_recvmsg(t, recvmsg); |
| } else { |
| spin_unlock(&t->reassembly_queue_lock); |
| } |
| } while (recvmsg); |
| t->reassembly_data_length = 0; |
| |
| if (t->send_cq) |
| ib_free_cq(t->send_cq); |
| if (t->recv_cq) |
| ib_free_cq(t->recv_cq); |
| if (t->pd) |
| ib_dealloc_pd(t->pd); |
| if (t->cm_id) |
| rdma_destroy_id(t->cm_id); |
| |
| smb_direct_destroy_pools(t); |
| ksmbd_conn_free(KSMBD_TRANS(t)->conn); |
| kfree(t); |
| } |
| |
| static struct smb_direct_sendmsg |
| *smb_direct_alloc_sendmsg(struct smb_direct_transport *t) |
| { |
| struct smb_direct_sendmsg *msg; |
| |
| msg = mempool_alloc(t->sendmsg_mempool, GFP_KERNEL); |
| if (!msg) |
| return ERR_PTR(-ENOMEM); |
| msg->transport = t; |
| INIT_LIST_HEAD(&msg->list); |
| msg->num_sge = 0; |
| return msg; |
| } |
| |
| static void smb_direct_free_sendmsg(struct smb_direct_transport *t, |
| struct smb_direct_sendmsg *msg) |
| { |
| int i; |
| |
| if (msg->num_sge > 0) { |
| ib_dma_unmap_single(t->cm_id->device, |
| msg->sge[0].addr, msg->sge[0].length, |
| DMA_TO_DEVICE); |
| for (i = 1; i < msg->num_sge; i++) |
| ib_dma_unmap_page(t->cm_id->device, |
| msg->sge[i].addr, msg->sge[i].length, |
| DMA_TO_DEVICE); |
| } |
| mempool_free(msg, t->sendmsg_mempool); |
| } |
| |
| static int smb_direct_check_recvmsg(struct smb_direct_recvmsg *recvmsg) |
| { |
| switch (recvmsg->type) { |
| case SMB_DIRECT_MSG_DATA_TRANSFER: { |
| struct smb_direct_data_transfer *req = |
| (struct smb_direct_data_transfer *)recvmsg->packet; |
| struct smb2_hdr *hdr = (struct smb2_hdr *)(recvmsg->packet |
| + le32_to_cpu(req->data_offset)); |
| ksmbd_debug(RDMA, |
| "CreditGranted: %u, CreditRequested: %u, DataLength: %u, RemainingDataLength: %u, SMB: %x, Command: %u\n", |
| le16_to_cpu(req->credits_granted), |
| le16_to_cpu(req->credits_requested), |
| req->data_length, req->remaining_data_length, |
| hdr->ProtocolId, hdr->Command); |
| break; |
| } |
| case SMB_DIRECT_MSG_NEGOTIATE_REQ: { |
| struct smb_direct_negotiate_req *req = |
| (struct smb_direct_negotiate_req *)recvmsg->packet; |
| ksmbd_debug(RDMA, |
| "MinVersion: %u, MaxVersion: %u, CreditRequested: %u, MaxSendSize: %u, MaxRecvSize: %u, MaxFragmentedSize: %u\n", |
| le16_to_cpu(req->min_version), |
| le16_to_cpu(req->max_version), |
| le16_to_cpu(req->credits_requested), |
| le32_to_cpu(req->preferred_send_size), |
| le32_to_cpu(req->max_receive_size), |
| le32_to_cpu(req->max_fragmented_size)); |
| if (le16_to_cpu(req->min_version) > 0x0100 || |
| le16_to_cpu(req->max_version) < 0x0100) |
| return -EOPNOTSUPP; |
| if (le16_to_cpu(req->credits_requested) <= 0 || |
| le32_to_cpu(req->max_receive_size) <= 128 || |
| le32_to_cpu(req->max_fragmented_size) <= |
| 128 * 1024) |
| return -ECONNABORTED; |
| |
| break; |
| } |
| default: |
| return -EINVAL; |
| } |
| return 0; |
| } |
| |
| static void recv_done(struct ib_cq *cq, struct ib_wc *wc) |
| { |
| struct smb_direct_recvmsg *recvmsg; |
| struct smb_direct_transport *t; |
| |
| recvmsg = container_of(wc->wr_cqe, struct smb_direct_recvmsg, cqe); |
| t = recvmsg->transport; |
| |
| if (wc->status != IB_WC_SUCCESS || wc->opcode != IB_WC_RECV) { |
| if (wc->status != IB_WC_WR_FLUSH_ERR) { |
| pr_err("Recv error. status='%s (%d)' opcode=%d\n", |
| ib_wc_status_msg(wc->status), wc->status, |
| wc->opcode); |
| smb_direct_disconnect_rdma_connection(t); |
| } |
| put_empty_recvmsg(t, recvmsg); |
| return; |
| } |
| |
| ksmbd_debug(RDMA, "Recv completed. status='%s (%d)', opcode=%d\n", |
| ib_wc_status_msg(wc->status), wc->status, |
| wc->opcode); |
| |
| ib_dma_sync_single_for_cpu(wc->qp->device, recvmsg->sge.addr, |
| recvmsg->sge.length, DMA_FROM_DEVICE); |
| |
| switch (recvmsg->type) { |
| case SMB_DIRECT_MSG_NEGOTIATE_REQ: |
| if (wc->byte_len < sizeof(struct smb_direct_negotiate_req)) { |
| put_empty_recvmsg(t, recvmsg); |
| return; |
| } |
| t->negotiation_requested = true; |
| t->full_packet_received = true; |
| t->status = SMB_DIRECT_CS_CONNECTED; |
| enqueue_reassembly(t, recvmsg, 0); |
| wake_up_interruptible(&t->wait_status); |
| break; |
| case SMB_DIRECT_MSG_DATA_TRANSFER: { |
| struct smb_direct_data_transfer *data_transfer = |
| (struct smb_direct_data_transfer *)recvmsg->packet; |
| unsigned int data_length; |
| int avail_recvmsg_count, receive_credits; |
| |
| if (wc->byte_len < |
| offsetof(struct smb_direct_data_transfer, padding)) { |
| put_empty_recvmsg(t, recvmsg); |
| return; |
| } |
| |
| data_length = le32_to_cpu(data_transfer->data_length); |
| if (data_length) { |
| if (wc->byte_len < sizeof(struct smb_direct_data_transfer) + |
| (u64)data_length) { |
| put_empty_recvmsg(t, recvmsg); |
| return; |
| } |
| |
| if (t->full_packet_received) |
| recvmsg->first_segment = true; |
| |
| if (le32_to_cpu(data_transfer->remaining_data_length)) |
| t->full_packet_received = false; |
| else |
| t->full_packet_received = true; |
| |
| enqueue_reassembly(t, recvmsg, (int)data_length); |
| wake_up_interruptible(&t->wait_reassembly_queue); |
| |
| spin_lock(&t->receive_credit_lock); |
| receive_credits = --(t->recv_credits); |
| avail_recvmsg_count = t->count_avail_recvmsg; |
| spin_unlock(&t->receive_credit_lock); |
| } else { |
| put_empty_recvmsg(t, recvmsg); |
| |
| spin_lock(&t->receive_credit_lock); |
| receive_credits = --(t->recv_credits); |
| avail_recvmsg_count = ++(t->count_avail_recvmsg); |
| spin_unlock(&t->receive_credit_lock); |
| } |
| |
| t->recv_credit_target = |
| le16_to_cpu(data_transfer->credits_requested); |
| atomic_add(le16_to_cpu(data_transfer->credits_granted), |
| &t->send_credits); |
| |
| if (le16_to_cpu(data_transfer->flags) & |
| SMB_DIRECT_RESPONSE_REQUESTED) |
| queue_work(smb_direct_wq, &t->send_immediate_work); |
| |
| if (atomic_read(&t->send_credits) > 0) |
| wake_up_interruptible(&t->wait_send_credits); |
| |
| if (is_receive_credit_post_required(receive_credits, avail_recvmsg_count)) |
| mod_delayed_work(smb_direct_wq, |
| &t->post_recv_credits_work, 0); |
| break; |
| } |
| default: |
| break; |
| } |
| } |
| |
| static int smb_direct_post_recv(struct smb_direct_transport *t, |
| struct smb_direct_recvmsg *recvmsg) |
| { |
| struct ib_recv_wr wr; |
| int ret; |
| |
| recvmsg->sge.addr = ib_dma_map_single(t->cm_id->device, |
| recvmsg->packet, t->max_recv_size, |
| DMA_FROM_DEVICE); |
| ret = ib_dma_mapping_error(t->cm_id->device, recvmsg->sge.addr); |
| if (ret) |
| return ret; |
| recvmsg->sge.length = t->max_recv_size; |
| recvmsg->sge.lkey = t->pd->local_dma_lkey; |
| recvmsg->cqe.done = recv_done; |
| |
| wr.wr_cqe = &recvmsg->cqe; |
| wr.next = NULL; |
| wr.sg_list = &recvmsg->sge; |
| wr.num_sge = 1; |
| |
| ret = ib_post_recv(t->qp, &wr, NULL); |
| if (ret) { |
| pr_err("Can't post recv: %d\n", ret); |
| ib_dma_unmap_single(t->cm_id->device, |
| recvmsg->sge.addr, recvmsg->sge.length, |
| DMA_FROM_DEVICE); |
| smb_direct_disconnect_rdma_connection(t); |
| return ret; |
| } |
| return ret; |
| } |
| |
| static int smb_direct_read(struct ksmbd_transport *t, char *buf, |
| unsigned int size) |
| { |
| struct smb_direct_recvmsg *recvmsg; |
| struct smb_direct_data_transfer *data_transfer; |
| int to_copy, to_read, data_read, offset; |
| u32 data_length, remaining_data_length, data_offset; |
| int rc; |
| struct smb_direct_transport *st = smb_trans_direct_transfort(t); |
| |
| again: |
| if (st->status != SMB_DIRECT_CS_CONNECTED) { |
| pr_err("disconnected\n"); |
| return -ENOTCONN; |
| } |
| |
| /* |
| * No need to hold the reassembly queue lock all the time as we are |
| * the only one reading from the front of the queue. The transport |
| * may add more entries to the back of the queue at the same time |
| */ |
| if (st->reassembly_data_length >= size) { |
| int queue_length; |
| int queue_removed = 0; |
| |
| /* |
| * Need to make sure reassembly_data_length is read before |
| * reading reassembly_queue_length and calling |
| * get_first_reassembly. This call is lock free |
| * as we never read at the end of the queue which are being |
| * updated in SOFTIRQ as more data is received |
| */ |
| virt_rmb(); |
| queue_length = st->reassembly_queue_length; |
| data_read = 0; |
| to_read = size; |
| offset = st->first_entry_offset; |
| while (data_read < size) { |
| recvmsg = get_first_reassembly(st); |
| data_transfer = smb_direct_recvmsg_payload(recvmsg); |
| data_length = le32_to_cpu(data_transfer->data_length); |
| remaining_data_length = |
| le32_to_cpu(data_transfer->remaining_data_length); |
| data_offset = le32_to_cpu(data_transfer->data_offset); |
| |
| /* |
| * The upper layer expects RFC1002 length at the |
| * beginning of the payload. Return it to indicate |
| * the total length of the packet. This minimize the |
| * change to upper layer packet processing logic. This |
| * will be eventually remove when an intermediate |
| * transport layer is added |
| */ |
| if (recvmsg->first_segment && size == 4) { |
| unsigned int rfc1002_len = |
| data_length + remaining_data_length; |
| *((__be32 *)buf) = cpu_to_be32(rfc1002_len); |
| data_read = 4; |
| recvmsg->first_segment = false; |
| ksmbd_debug(RDMA, |
| "returning rfc1002 length %d\n", |
| rfc1002_len); |
| goto read_rfc1002_done; |
| } |
| |
| to_copy = min_t(int, data_length - offset, to_read); |
| memcpy(buf + data_read, (char *)data_transfer + data_offset + offset, |
| to_copy); |
| |
| /* move on to the next buffer? */ |
| if (to_copy == data_length - offset) { |
| queue_length--; |
| /* |
| * No need to lock if we are not at the |
| * end of the queue |
| */ |
| if (queue_length) { |
| list_del(&recvmsg->list); |
| } else { |
| spin_lock_irq(&st->reassembly_queue_lock); |
| list_del(&recvmsg->list); |
| spin_unlock_irq(&st->reassembly_queue_lock); |
| } |
| queue_removed++; |
| put_recvmsg(st, recvmsg); |
| offset = 0; |
| } else { |
| offset += to_copy; |
| } |
| |
| to_read -= to_copy; |
| data_read += to_copy; |
| } |
| |
| spin_lock_irq(&st->reassembly_queue_lock); |
| st->reassembly_data_length -= data_read; |
| st->reassembly_queue_length -= queue_removed; |
| spin_unlock_irq(&st->reassembly_queue_lock); |
| |
| spin_lock(&st->receive_credit_lock); |
| st->count_avail_recvmsg += queue_removed; |
| if (is_receive_credit_post_required(st->recv_credits, st->count_avail_recvmsg)) { |
| spin_unlock(&st->receive_credit_lock); |
| mod_delayed_work(smb_direct_wq, |
| &st->post_recv_credits_work, 0); |
| } else { |
| spin_unlock(&st->receive_credit_lock); |
| } |
| |
| st->first_entry_offset = offset; |
| ksmbd_debug(RDMA, |
| "returning to thread data_read=%d reassembly_data_length=%d first_entry_offset=%d\n", |
| data_read, st->reassembly_data_length, |
| st->first_entry_offset); |
| read_rfc1002_done: |
| return data_read; |
| } |
| |
| ksmbd_debug(RDMA, "wait_event on more data\n"); |
| rc = wait_event_interruptible(st->wait_reassembly_queue, |
| st->reassembly_data_length >= size || |
| st->status != SMB_DIRECT_CS_CONNECTED); |
| if (rc) |
| return -EINTR; |
| |
| goto again; |
| } |
| |
| static void smb_direct_post_recv_credits(struct work_struct *work) |
| { |
| struct smb_direct_transport *t = container_of(work, |
| struct smb_direct_transport, post_recv_credits_work.work); |
| struct smb_direct_recvmsg *recvmsg; |
| int receive_credits, credits = 0; |
| int ret; |
| int use_free = 1; |
| |
| spin_lock(&t->receive_credit_lock); |
| receive_credits = t->recv_credits; |
| spin_unlock(&t->receive_credit_lock); |
| |
| if (receive_credits < t->recv_credit_target) { |
| while (true) { |
| if (use_free) |
| recvmsg = get_free_recvmsg(t); |
| else |
| recvmsg = get_empty_recvmsg(t); |
| if (!recvmsg) { |
| if (use_free) { |
| use_free = 0; |
| continue; |
| } else { |
| break; |
| } |
| } |
| |
| recvmsg->type = SMB_DIRECT_MSG_DATA_TRANSFER; |
| recvmsg->first_segment = false; |
| |
| ret = smb_direct_post_recv(t, recvmsg); |
| if (ret) { |
| pr_err("Can't post recv: %d\n", ret); |
| put_recvmsg(t, recvmsg); |
| break; |
| } |
| credits++; |
| } |
| } |
| |
| spin_lock(&t->receive_credit_lock); |
| t->recv_credits += credits; |
| t->count_avail_recvmsg -= credits; |
| spin_unlock(&t->receive_credit_lock); |
| |
| spin_lock(&t->lock_new_recv_credits); |
| t->new_recv_credits += credits; |
| spin_unlock(&t->lock_new_recv_credits); |
| |
| if (credits) |
| queue_work(smb_direct_wq, &t->send_immediate_work); |
| } |
| |
| static void send_done(struct ib_cq *cq, struct ib_wc *wc) |
| { |
| struct smb_direct_sendmsg *sendmsg, *sibling; |
| struct smb_direct_transport *t; |
| struct list_head *pos, *prev, *end; |
| |
| sendmsg = container_of(wc->wr_cqe, struct smb_direct_sendmsg, cqe); |
| t = sendmsg->transport; |
| |
| ksmbd_debug(RDMA, "Send completed. status='%s (%d)', opcode=%d\n", |
| ib_wc_status_msg(wc->status), wc->status, |
| wc->opcode); |
| |
| if (wc->status != IB_WC_SUCCESS || wc->opcode != IB_WC_SEND) { |
| pr_err("Send error. status='%s (%d)', opcode=%d\n", |
| ib_wc_status_msg(wc->status), wc->status, |
| wc->opcode); |
| smb_direct_disconnect_rdma_connection(t); |
| } |
| |
| if (atomic_dec_and_test(&t->send_pending)) |
| wake_up(&t->wait_send_pending); |
| |
| /* iterate and free the list of messages in reverse. the list's head |
| * is invalid. |
| */ |
| for (pos = &sendmsg->list, prev = pos->prev, end = sendmsg->list.next; |
| prev != end; pos = prev, prev = prev->prev) { |
| sibling = container_of(pos, struct smb_direct_sendmsg, list); |
| smb_direct_free_sendmsg(t, sibling); |
| } |
| |
| sibling = container_of(pos, struct smb_direct_sendmsg, list); |
| smb_direct_free_sendmsg(t, sibling); |
| } |
| |
| static int manage_credits_prior_sending(struct smb_direct_transport *t) |
| { |
| int new_credits; |
| |
| spin_lock(&t->lock_new_recv_credits); |
| new_credits = t->new_recv_credits; |
| t->new_recv_credits = 0; |
| spin_unlock(&t->lock_new_recv_credits); |
| |
| return new_credits; |
| } |
| |
| static int smb_direct_post_send(struct smb_direct_transport *t, |
| struct ib_send_wr *wr) |
| { |
| int ret; |
| |
| atomic_inc(&t->send_pending); |
| ret = ib_post_send(t->qp, wr, NULL); |
| if (ret) { |
| pr_err("failed to post send: %d\n", ret); |
| if (atomic_dec_and_test(&t->send_pending)) |
| wake_up(&t->wait_send_pending); |
| smb_direct_disconnect_rdma_connection(t); |
| } |
| return ret; |
| } |
| |
| static void smb_direct_send_ctx_init(struct smb_direct_transport *t, |
| struct smb_direct_send_ctx *send_ctx, |
| bool need_invalidate_rkey, |
| unsigned int remote_key) |
| { |
| INIT_LIST_HEAD(&send_ctx->msg_list); |
| send_ctx->wr_cnt = 0; |
| send_ctx->need_invalidate_rkey = need_invalidate_rkey; |
| send_ctx->remote_key = remote_key; |
| } |
| |
| static int smb_direct_flush_send_list(struct smb_direct_transport *t, |
| struct smb_direct_send_ctx *send_ctx, |
| bool is_last) |
| { |
| struct smb_direct_sendmsg *first, *last; |
| int ret; |
| |
| if (list_empty(&send_ctx->msg_list)) |
| return 0; |
| |
| first = list_first_entry(&send_ctx->msg_list, |
| struct smb_direct_sendmsg, |
| list); |
| last = list_last_entry(&send_ctx->msg_list, |
| struct smb_direct_sendmsg, |
| list); |
| |
| last->wr.send_flags = IB_SEND_SIGNALED; |
| last->wr.wr_cqe = &last->cqe; |
| if (is_last && send_ctx->need_invalidate_rkey) { |
| last->wr.opcode = IB_WR_SEND_WITH_INV; |
| last->wr.ex.invalidate_rkey = send_ctx->remote_key; |
| } |
| |
| ret = smb_direct_post_send(t, &first->wr); |
| if (!ret) { |
| smb_direct_send_ctx_init(t, send_ctx, |
| send_ctx->need_invalidate_rkey, |
| send_ctx->remote_key); |
| } else { |
| atomic_add(send_ctx->wr_cnt, &t->send_credits); |
| wake_up(&t->wait_send_credits); |
| list_for_each_entry_safe(first, last, &send_ctx->msg_list, |
| list) { |
| smb_direct_free_sendmsg(t, first); |
| } |
| } |
| return ret; |
| } |
| |
| static int wait_for_credits(struct smb_direct_transport *t, |
| wait_queue_head_t *waitq, atomic_t *total_credits, |
| int needed) |
| { |
| int ret; |
| |
| do { |
| if (atomic_sub_return(needed, total_credits) >= 0) |
| return 0; |
| |
| atomic_add(needed, total_credits); |
| ret = wait_event_interruptible(*waitq, |
| atomic_read(total_credits) >= needed || |
| t->status != SMB_DIRECT_CS_CONNECTED); |
| |
| if (t->status != SMB_DIRECT_CS_CONNECTED) |
| return -ENOTCONN; |
| else if (ret < 0) |
| return ret; |
| } while (true); |
| } |
| |
| static int wait_for_send_credits(struct smb_direct_transport *t, |
| struct smb_direct_send_ctx *send_ctx) |
| { |
| int ret; |
| |
| if (send_ctx && |
| (send_ctx->wr_cnt >= 16 || atomic_read(&t->send_credits) <= 1)) { |
| ret = smb_direct_flush_send_list(t, send_ctx, false); |
| if (ret) |
| return ret; |
| } |
| |
| return wait_for_credits(t, &t->wait_send_credits, &t->send_credits, 1); |
| } |
| |
| static int wait_for_rw_credits(struct smb_direct_transport *t, int credits) |
| { |
| return wait_for_credits(t, &t->wait_rw_credits, &t->rw_credits, credits); |
| } |
| |
| static int calc_rw_credits(struct smb_direct_transport *t, |
| char *buf, unsigned int len) |
| { |
| return DIV_ROUND_UP(get_buf_page_count(buf, len), |
| t->pages_per_rw_credit); |
| } |
| |
| static int smb_direct_create_header(struct smb_direct_transport *t, |
| int size, int remaining_data_length, |
| struct smb_direct_sendmsg **sendmsg_out) |
| { |
| struct smb_direct_sendmsg *sendmsg; |
| struct smb_direct_data_transfer *packet; |
| int header_length; |
| int ret; |
| |
| sendmsg = smb_direct_alloc_sendmsg(t); |
| if (IS_ERR(sendmsg)) |
| return PTR_ERR(sendmsg); |
| |
| /* Fill in the packet header */ |
| packet = (struct smb_direct_data_transfer *)sendmsg->packet; |
| packet->credits_requested = cpu_to_le16(t->send_credit_target); |
| packet->credits_granted = cpu_to_le16(manage_credits_prior_sending(t)); |
| |
| packet->flags = 0; |
| packet->reserved = 0; |
| if (!size) |
| packet->data_offset = 0; |
| else |
| packet->data_offset = cpu_to_le32(24); |
| packet->data_length = cpu_to_le32(size); |
| packet->remaining_data_length = cpu_to_le32(remaining_data_length); |
| packet->padding = 0; |
| |
| ksmbd_debug(RDMA, |
| "credits_requested=%d credits_granted=%d data_offset=%d data_length=%d remaining_data_length=%d\n", |
| le16_to_cpu(packet->credits_requested), |
| le16_to_cpu(packet->credits_granted), |
| le32_to_cpu(packet->data_offset), |
| le32_to_cpu(packet->data_length), |
| le32_to_cpu(packet->remaining_data_length)); |
| |
| /* Map the packet to DMA */ |
| header_length = sizeof(struct smb_direct_data_transfer); |
| /* If this is a packet without payload, don't send padding */ |
| if (!size) |
| header_length = |
| offsetof(struct smb_direct_data_transfer, padding); |
| |
| sendmsg->sge[0].addr = ib_dma_map_single(t->cm_id->device, |
| (void *)packet, |
| header_length, |
| DMA_TO_DEVICE); |
| ret = ib_dma_mapping_error(t->cm_id->device, sendmsg->sge[0].addr); |
| if (ret) { |
| smb_direct_free_sendmsg(t, sendmsg); |
| return ret; |
| } |
| |
| sendmsg->num_sge = 1; |
| sendmsg->sge[0].length = header_length; |
| sendmsg->sge[0].lkey = t->pd->local_dma_lkey; |
| |
| *sendmsg_out = sendmsg; |
| return 0; |
| } |
| |
| static int get_sg_list(void *buf, int size, struct scatterlist *sg_list, int nentries) |
| { |
| bool high = is_vmalloc_addr(buf); |
| struct page *page; |
| int offset, len; |
| int i = 0; |
| |
| if (size <= 0 || nentries < get_buf_page_count(buf, size)) |
| return -EINVAL; |
| |
| offset = offset_in_page(buf); |
| buf -= offset; |
| while (size > 0) { |
| len = min_t(int, PAGE_SIZE - offset, size); |
| if (high) |
| page = vmalloc_to_page(buf); |
| else |
| page = kmap_to_page(buf); |
| |
| if (!sg_list) |
| return -EINVAL; |
| sg_set_page(sg_list, page, len, offset); |
| sg_list = sg_next(sg_list); |
| |
| buf += PAGE_SIZE; |
| size -= len; |
| offset = 0; |
| i++; |
| } |
| return i; |
| } |
| |
| static int get_mapped_sg_list(struct ib_device *device, void *buf, int size, |
| struct scatterlist *sg_list, int nentries, |
| enum dma_data_direction dir) |
| { |
| int npages; |
| |
| npages = get_sg_list(buf, size, sg_list, nentries); |
| if (npages < 0) |
| return -EINVAL; |
| return ib_dma_map_sg(device, sg_list, npages, dir); |
| } |
| |
| static int post_sendmsg(struct smb_direct_transport *t, |
| struct smb_direct_send_ctx *send_ctx, |
| struct smb_direct_sendmsg *msg) |
| { |
| int i; |
| |
| for (i = 0; i < msg->num_sge; i++) |
| ib_dma_sync_single_for_device(t->cm_id->device, |
| msg->sge[i].addr, msg->sge[i].length, |
| DMA_TO_DEVICE); |
| |
| msg->cqe.done = send_done; |
| msg->wr.opcode = IB_WR_SEND; |
| msg->wr.sg_list = &msg->sge[0]; |
| msg->wr.num_sge = msg->num_sge; |
| msg->wr.next = NULL; |
| |
| if (send_ctx) { |
| msg->wr.wr_cqe = NULL; |
| msg->wr.send_flags = 0; |
| if (!list_empty(&send_ctx->msg_list)) { |
| struct smb_direct_sendmsg *last; |
| |
| last = list_last_entry(&send_ctx->msg_list, |
| struct smb_direct_sendmsg, |
| list); |
| last->wr.next = &msg->wr; |
| } |
| list_add_tail(&msg->list, &send_ctx->msg_list); |
| send_ctx->wr_cnt++; |
| return 0; |
| } |
| |
| msg->wr.wr_cqe = &msg->cqe; |
| msg->wr.send_flags = IB_SEND_SIGNALED; |
| return smb_direct_post_send(t, &msg->wr); |
| } |
| |
| static int smb_direct_post_send_data(struct smb_direct_transport *t, |
| struct smb_direct_send_ctx *send_ctx, |
| struct kvec *iov, int niov, |
| int remaining_data_length) |
| { |
| int i, j, ret; |
| struct smb_direct_sendmsg *msg; |
| int data_length; |
| struct scatterlist sg[SMB_DIRECT_MAX_SEND_SGES - 1]; |
| |
| ret = wait_for_send_credits(t, send_ctx); |
| if (ret) |
| return ret; |
| |
| data_length = 0; |
| for (i = 0; i < niov; i++) |
| data_length += iov[i].iov_len; |
| |
| ret = smb_direct_create_header(t, data_length, remaining_data_length, |
| &msg); |
| if (ret) { |
| atomic_inc(&t->send_credits); |
| return ret; |
| } |
| |
| for (i = 0; i < niov; i++) { |
| struct ib_sge *sge; |
| int sg_cnt; |
| |
| sg_init_table(sg, SMB_DIRECT_MAX_SEND_SGES - 1); |
| sg_cnt = get_mapped_sg_list(t->cm_id->device, |
| iov[i].iov_base, iov[i].iov_len, |
| sg, SMB_DIRECT_MAX_SEND_SGES - 1, |
| DMA_TO_DEVICE); |
| if (sg_cnt <= 0) { |
| pr_err("failed to map buffer\n"); |
| ret = -ENOMEM; |
| goto err; |
| } else if (sg_cnt + msg->num_sge > SMB_DIRECT_MAX_SEND_SGES) { |
| pr_err("buffer not fitted into sges\n"); |
| ret = -E2BIG; |
| ib_dma_unmap_sg(t->cm_id->device, sg, sg_cnt, |
| DMA_TO_DEVICE); |
| goto err; |
| } |
| |
| for (j = 0; j < sg_cnt; j++) { |
| sge = &msg->sge[msg->num_sge]; |
| sge->addr = sg_dma_address(&sg[j]); |
| sge->length = sg_dma_len(&sg[j]); |
| sge->lkey = t->pd->local_dma_lkey; |
| msg->num_sge++; |
| } |
| } |
| |
| ret = post_sendmsg(t, send_ctx, msg); |
| if (ret) |
| goto err; |
| return 0; |
| err: |
| smb_direct_free_sendmsg(t, msg); |
| atomic_inc(&t->send_credits); |
| return ret; |
| } |
| |
| static int smb_direct_writev(struct ksmbd_transport *t, |
| struct kvec *iov, int niovs, int buflen, |
| bool need_invalidate, unsigned int remote_key) |
| { |
| struct smb_direct_transport *st = smb_trans_direct_transfort(t); |
| int remaining_data_length; |
| int start, i, j; |
| int max_iov_size = st->max_send_size - |
| sizeof(struct smb_direct_data_transfer); |
| int ret; |
| struct kvec vec; |
| struct smb_direct_send_ctx send_ctx; |
| |
| if (st->status != SMB_DIRECT_CS_CONNECTED) |
| return -ENOTCONN; |
| |
| //FIXME: skip RFC1002 header.. |
| buflen -= 4; |
| iov[0].iov_base += 4; |
| iov[0].iov_len -= 4; |
| |
| remaining_data_length = buflen; |
| ksmbd_debug(RDMA, "Sending smb (RDMA): smb_len=%u\n", buflen); |
| |
| smb_direct_send_ctx_init(st, &send_ctx, need_invalidate, remote_key); |
| start = i = 0; |
| buflen = 0; |
| while (true) { |
| buflen += iov[i].iov_len; |
| if (buflen > max_iov_size) { |
| if (i > start) { |
| remaining_data_length -= |
| (buflen - iov[i].iov_len); |
| ret = smb_direct_post_send_data(st, &send_ctx, |
| &iov[start], i - start, |
| remaining_data_length); |
| if (ret) |
| goto done; |
| } else { |
| /* iov[start] is too big, break it */ |
| int nvec = (buflen + max_iov_size - 1) / |
| max_iov_size; |
| |
| for (j = 0; j < nvec; j++) { |
| vec.iov_base = |
| (char *)iov[start].iov_base + |
| j * max_iov_size; |
| vec.iov_len = |
| min_t(int, max_iov_size, |
| buflen - max_iov_size * j); |
| remaining_data_length -= vec.iov_len; |
| ret = smb_direct_post_send_data(st, &send_ctx, &vec, 1, |
| remaining_data_length); |
| if (ret) |
| goto done; |
| } |
| i++; |
| if (i == niovs) |
| break; |
| } |
| start = i; |
| buflen = 0; |
| } else { |
| i++; |
| if (i == niovs) { |
| /* send out all remaining vecs */ |
| remaining_data_length -= buflen; |
| ret = smb_direct_post_send_data(st, &send_ctx, |
| &iov[start], i - start, |
| remaining_data_length); |
| if (ret) |
| goto done; |
| break; |
| } |
| } |
| } |
| |
| done: |
| ret = smb_direct_flush_send_list(st, &send_ctx, true); |
| |
| /* |
| * As an optimization, we don't wait for individual I/O to finish |
| * before sending the next one. |
| * Send them all and wait for pending send count to get to 0 |
| * that means all the I/Os have been out and we are good to return |
| */ |
| |
| wait_event(st->wait_send_pending, |
| atomic_read(&st->send_pending) == 0); |
| return ret; |
| } |
| |
| static void smb_direct_free_rdma_rw_msg(struct smb_direct_transport *t, |
| struct smb_direct_rdma_rw_msg *msg, |
| enum dma_data_direction dir) |
| { |
| rdma_rw_ctx_destroy(&msg->rw_ctx, t->qp, t->qp->port, |
| msg->sgt.sgl, msg->sgt.nents, dir); |
| sg_free_table_chained(&msg->sgt, SG_CHUNK_SIZE); |
| kfree(msg); |
| } |
| |
| static void read_write_done(struct ib_cq *cq, struct ib_wc *wc, |
| enum dma_data_direction dir) |
| { |
| struct smb_direct_rdma_rw_msg *msg = container_of(wc->wr_cqe, |
| struct smb_direct_rdma_rw_msg, cqe); |
| struct smb_direct_transport *t = msg->t; |
| |
| if (wc->status != IB_WC_SUCCESS) { |
| msg->status = -EIO; |
| pr_err("read/write error. opcode = %d, status = %s(%d)\n", |
| wc->opcode, ib_wc_status_msg(wc->status), wc->status); |
| if (wc->status != IB_WC_WR_FLUSH_ERR) |
| smb_direct_disconnect_rdma_connection(t); |
| } |
| |
| complete(msg->completion); |
| } |
| |
| static void read_done(struct ib_cq *cq, struct ib_wc *wc) |
| { |
| read_write_done(cq, wc, DMA_FROM_DEVICE); |
| } |
| |
| static void write_done(struct ib_cq *cq, struct ib_wc *wc) |
| { |
| read_write_done(cq, wc, DMA_TO_DEVICE); |
| } |
| |
| static int smb_direct_rdma_xmit(struct smb_direct_transport *t, |
| void *buf, int buf_len, |
| struct smb2_buffer_desc_v1 *desc, |
| unsigned int desc_len, |
| bool is_read) |
| { |
| struct smb_direct_rdma_rw_msg *msg, *next_msg; |
| int i, ret; |
| DECLARE_COMPLETION_ONSTACK(completion); |
| struct ib_send_wr *first_wr; |
| LIST_HEAD(msg_list); |
| char *desc_buf; |
| int credits_needed; |
| unsigned int desc_buf_len; |
| size_t total_length = 0; |
| |
| if (t->status != SMB_DIRECT_CS_CONNECTED) |
| return -ENOTCONN; |
| |
| /* calculate needed credits */ |
| credits_needed = 0; |
| desc_buf = buf; |
| for (i = 0; i < desc_len / sizeof(*desc); i++) { |
| desc_buf_len = le32_to_cpu(desc[i].length); |
| |
| credits_needed += calc_rw_credits(t, desc_buf, desc_buf_len); |
| desc_buf += desc_buf_len; |
| total_length += desc_buf_len; |
| if (desc_buf_len == 0 || total_length > buf_len || |
| total_length > t->max_rdma_rw_size) |
| return -EINVAL; |
| } |
| |
| ksmbd_debug(RDMA, "RDMA %s, len %#x, needed credits %#x\n", |
| is_read ? "read" : "write", buf_len, credits_needed); |
| |
| ret = wait_for_rw_credits(t, credits_needed); |
| if (ret < 0) |
| return ret; |
| |
| /* build rdma_rw_ctx for each descriptor */ |
| desc_buf = buf; |
| for (i = 0; i < desc_len / sizeof(*desc); i++) { |
| msg = kzalloc(offsetof(struct smb_direct_rdma_rw_msg, sg_list) + |
| sizeof(struct scatterlist) * SG_CHUNK_SIZE, GFP_KERNEL); |
| if (!msg) { |
| ret = -ENOMEM; |
| goto out; |
| } |
| |
| desc_buf_len = le32_to_cpu(desc[i].length); |
| |
| msg->t = t; |
| msg->cqe.done = is_read ? read_done : write_done; |
| msg->completion = &completion; |
| |
| msg->sgt.sgl = &msg->sg_list[0]; |
| ret = sg_alloc_table_chained(&msg->sgt, |
| get_buf_page_count(desc_buf, desc_buf_len), |
| msg->sg_list, SG_CHUNK_SIZE); |
| if (ret) { |
| kfree(msg); |
| ret = -ENOMEM; |
| goto out; |
| } |
| |
| ret = get_sg_list(desc_buf, desc_buf_len, |
| msg->sgt.sgl, msg->sgt.orig_nents); |
| if (ret < 0) { |
| sg_free_table_chained(&msg->sgt, SG_CHUNK_SIZE); |
| kfree(msg); |
| goto out; |
| } |
| |
| ret = rdma_rw_ctx_init(&msg->rw_ctx, t->qp, t->qp->port, |
| msg->sgt.sgl, |
| get_buf_page_count(desc_buf, desc_buf_len), |
| 0, |
| le64_to_cpu(desc[i].offset), |
| le32_to_cpu(desc[i].token), |
| is_read ? DMA_FROM_DEVICE : DMA_TO_DEVICE); |
| if (ret < 0) { |
| pr_err("failed to init rdma_rw_ctx: %d\n", ret); |
| sg_free_table_chained(&msg->sgt, SG_CHUNK_SIZE); |
| kfree(msg); |
| goto out; |
| } |
| |
| list_add_tail(&msg->list, &msg_list); |
| desc_buf += desc_buf_len; |
| } |
| |
| /* concatenate work requests of rdma_rw_ctxs */ |
| first_wr = NULL; |
| list_for_each_entry_reverse(msg, &msg_list, list) { |
| first_wr = rdma_rw_ctx_wrs(&msg->rw_ctx, t->qp, t->qp->port, |
| &msg->cqe, first_wr); |
| } |
| |
| ret = ib_post_send(t->qp, first_wr, NULL); |
| if (ret) { |
| pr_err("failed to post send wr for RDMA R/W: %d\n", ret); |
| goto out; |
| } |
| |
| msg = list_last_entry(&msg_list, struct smb_direct_rdma_rw_msg, list); |
| wait_for_completion(&completion); |
| ret = msg->status; |
| out: |
| list_for_each_entry_safe(msg, next_msg, &msg_list, list) { |
| list_del(&msg->list); |
| smb_direct_free_rdma_rw_msg(t, msg, |
| is_read ? DMA_FROM_DEVICE : DMA_TO_DEVICE); |
| } |
| atomic_add(credits_needed, &t->rw_credits); |
| wake_up(&t->wait_rw_credits); |
| return ret; |
| } |
| |
| static int smb_direct_rdma_write(struct ksmbd_transport *t, |
| void *buf, unsigned int buflen, |
| struct smb2_buffer_desc_v1 *desc, |
| unsigned int desc_len) |
| { |
| return smb_direct_rdma_xmit(smb_trans_direct_transfort(t), buf, buflen, |
| desc, desc_len, false); |
| } |
| |
| static int smb_direct_rdma_read(struct ksmbd_transport *t, |
| void *buf, unsigned int buflen, |
| struct smb2_buffer_desc_v1 *desc, |
| unsigned int desc_len) |
| { |
| return smb_direct_rdma_xmit(smb_trans_direct_transfort(t), buf, buflen, |
| desc, desc_len, true); |
| } |
| |
| static void smb_direct_disconnect(struct ksmbd_transport *t) |
| { |
| struct smb_direct_transport *st = smb_trans_direct_transfort(t); |
| |
| ksmbd_debug(RDMA, "Disconnecting cm_id=%p\n", st->cm_id); |
| |
| smb_direct_disconnect_rdma_work(&st->disconnect_work); |
| wait_event_interruptible(st->wait_status, |
| st->status == SMB_DIRECT_CS_DISCONNECTED); |
| free_transport(st); |
| } |
| |
| static void smb_direct_shutdown(struct ksmbd_transport *t) |
| { |
| struct smb_direct_transport *st = smb_trans_direct_transfort(t); |
| |
| ksmbd_debug(RDMA, "smb-direct shutdown cm_id=%p\n", st->cm_id); |
| |
| smb_direct_disconnect_rdma_work(&st->disconnect_work); |
| } |
| |
| static int smb_direct_cm_handler(struct rdma_cm_id *cm_id, |
| struct rdma_cm_event *event) |
| { |
| struct smb_direct_transport *t = cm_id->context; |
| |
| ksmbd_debug(RDMA, "RDMA CM event. cm_id=%p event=%s (%d)\n", |
| cm_id, rdma_event_msg(event->event), event->event); |
| |
| switch (event->event) { |
| case RDMA_CM_EVENT_ESTABLISHED: { |
| t->status = SMB_DIRECT_CS_CONNECTED; |
| wake_up_interruptible(&t->wait_status); |
| break; |
| } |
| case RDMA_CM_EVENT_DEVICE_REMOVAL: |
| case RDMA_CM_EVENT_DISCONNECTED: { |
| t->status = SMB_DIRECT_CS_DISCONNECTED; |
| wake_up_interruptible(&t->wait_status); |
| wake_up_interruptible(&t->wait_reassembly_queue); |
| wake_up(&t->wait_send_credits); |
| break; |
| } |
| case RDMA_CM_EVENT_CONNECT_ERROR: { |
| t->status = SMB_DIRECT_CS_DISCONNECTED; |
| wake_up_interruptible(&t->wait_status); |
| break; |
| } |
| default: |
| pr_err("Unexpected RDMA CM event. cm_id=%p, event=%s (%d)\n", |
| cm_id, rdma_event_msg(event->event), |
| event->event); |
| break; |
| } |
| return 0; |
| } |
| |
| static void smb_direct_qpair_handler(struct ib_event *event, void *context) |
| { |
| struct smb_direct_transport *t = context; |
| |
| ksmbd_debug(RDMA, "Received QP event. cm_id=%p, event=%s (%d)\n", |
| t->cm_id, ib_event_msg(event->event), event->event); |
| |
| switch (event->event) { |
| case IB_EVENT_CQ_ERR: |
| case IB_EVENT_QP_FATAL: |
| smb_direct_disconnect_rdma_connection(t); |
| break; |
| default: |
| break; |
| } |
| } |
| |
| static int smb_direct_send_negotiate_response(struct smb_direct_transport *t, |
| int failed) |
| { |
| struct smb_direct_sendmsg *sendmsg; |
| struct smb_direct_negotiate_resp *resp; |
| int ret; |
| |
| sendmsg = smb_direct_alloc_sendmsg(t); |
| if (IS_ERR(sendmsg)) |
| return -ENOMEM; |
| |
| resp = (struct smb_direct_negotiate_resp *)sendmsg->packet; |
| if (failed) { |
| memset(resp, 0, sizeof(*resp)); |
| resp->min_version = cpu_to_le16(0x0100); |
| resp->max_version = cpu_to_le16(0x0100); |
| resp->status = STATUS_NOT_SUPPORTED; |
| } else { |
| resp->status = STATUS_SUCCESS; |
| resp->min_version = SMB_DIRECT_VERSION_LE; |
| resp->max_version = SMB_DIRECT_VERSION_LE; |
| resp->negotiated_version = SMB_DIRECT_VERSION_LE; |
| resp->reserved = 0; |
| resp->credits_requested = |
| cpu_to_le16(t->send_credit_target); |
| resp->credits_granted = cpu_to_le16(manage_credits_prior_sending(t)); |
| resp->max_readwrite_size = cpu_to_le32(t->max_rdma_rw_size); |
| resp->preferred_send_size = cpu_to_le32(t->max_send_size); |
| resp->max_receive_size = cpu_to_le32(t->max_recv_size); |
| resp->max_fragmented_size = |
| cpu_to_le32(t->max_fragmented_recv_size); |
| } |
| |
| sendmsg->sge[0].addr = ib_dma_map_single(t->cm_id->device, |
| (void *)resp, sizeof(*resp), |
| DMA_TO_DEVICE); |
| ret = ib_dma_mapping_error(t->cm_id->device, sendmsg->sge[0].addr); |
| if (ret) { |
| smb_direct_free_sendmsg(t, sendmsg); |
| return ret; |
| } |
| |
| sendmsg->num_sge = 1; |
| sendmsg->sge[0].length = sizeof(*resp); |
| sendmsg->sge[0].lkey = t->pd->local_dma_lkey; |
| |
| ret = post_sendmsg(t, NULL, sendmsg); |
| if (ret) { |
| smb_direct_free_sendmsg(t, sendmsg); |
| return ret; |
| } |
| |
| wait_event(t->wait_send_pending, |
| atomic_read(&t->send_pending) == 0); |
| return 0; |
| } |
| |
| static int smb_direct_accept_client(struct smb_direct_transport *t) |
| { |
| struct rdma_conn_param conn_param; |
| struct ib_port_immutable port_immutable; |
| u32 ird_ord_hdr[2]; |
| int ret; |
| |
| memset(&conn_param, 0, sizeof(conn_param)); |
| conn_param.initiator_depth = min_t(u8, t->cm_id->device->attrs.max_qp_rd_atom, |
| SMB_DIRECT_CM_INITIATOR_DEPTH); |
| conn_param.responder_resources = 0; |
| |
| t->cm_id->device->ops.get_port_immutable(t->cm_id->device, |
| t->cm_id->port_num, |
| &port_immutable); |
| if (port_immutable.core_cap_flags & RDMA_CORE_PORT_IWARP) { |
| ird_ord_hdr[0] = conn_param.responder_resources; |
| ird_ord_hdr[1] = 1; |
| conn_param.private_data = ird_ord_hdr; |
| conn_param.private_data_len = sizeof(ird_ord_hdr); |
| } else { |
| conn_param.private_data = NULL; |
| conn_param.private_data_len = 0; |
| } |
| conn_param.retry_count = SMB_DIRECT_CM_RETRY; |
| conn_param.rnr_retry_count = SMB_DIRECT_CM_RNR_RETRY; |
| conn_param.flow_control = 0; |
| |
| ret = rdma_accept(t->cm_id, &conn_param); |
| if (ret) { |
| pr_err("error at rdma_accept: %d\n", ret); |
| return ret; |
| } |
| return 0; |
| } |
| |
| static int smb_direct_prepare_negotiation(struct smb_direct_transport *t) |
| { |
| int ret; |
| struct smb_direct_recvmsg *recvmsg; |
| |
| recvmsg = get_free_recvmsg(t); |
| if (!recvmsg) |
| return -ENOMEM; |
| recvmsg->type = SMB_DIRECT_MSG_NEGOTIATE_REQ; |
| |
| ret = smb_direct_post_recv(t, recvmsg); |
| if (ret) { |
| pr_err("Can't post recv: %d\n", ret); |
| goto out_err; |
| } |
| |
| t->negotiation_requested = false; |
| ret = smb_direct_accept_client(t); |
| if (ret) { |
| pr_err("Can't accept client\n"); |
| goto out_err; |
| } |
| |
| smb_direct_post_recv_credits(&t->post_recv_credits_work.work); |
| return 0; |
| out_err: |
| put_recvmsg(t, recvmsg); |
| return ret; |
| } |
| |
| static unsigned int smb_direct_get_max_fr_pages(struct smb_direct_transport *t) |
| { |
| return min_t(unsigned int, |
| t->cm_id->device->attrs.max_fast_reg_page_list_len, |
| 256); |
| } |
| |
| static int smb_direct_init_params(struct smb_direct_transport *t, |
| struct ib_qp_cap *cap) |
| { |
| struct ib_device *device = t->cm_id->device; |
| int max_send_sges, max_rw_wrs, max_send_wrs; |
| unsigned int max_sge_per_wr, wrs_per_credit; |
| |
| /* need 3 more sge. because a SMB_DIRECT header, SMB2 header, |
| * SMB2 response could be mapped. |
| */ |
| t->max_send_size = smb_direct_max_send_size; |
| max_send_sges = DIV_ROUND_UP(t->max_send_size, PAGE_SIZE) + 3; |
| if (max_send_sges > SMB_DIRECT_MAX_SEND_SGES) { |
| pr_err("max_send_size %d is too large\n", t->max_send_size); |
| return -EINVAL; |
| } |
| |
| /* Calculate the number of work requests for RDMA R/W. |
| * The maximum number of pages which can be registered |
| * with one Memory region can be transferred with one |
| * R/W credit. And at least 4 work requests for each credit |
| * are needed for MR registration, RDMA R/W, local & remote |
| * MR invalidation. |
| */ |
| t->max_rdma_rw_size = smb_direct_max_read_write_size; |
| t->pages_per_rw_credit = smb_direct_get_max_fr_pages(t); |
| t->max_rw_credits = DIV_ROUND_UP(t->max_rdma_rw_size, |
| (t->pages_per_rw_credit - 1) * |
| PAGE_SIZE); |
| |
| max_sge_per_wr = min_t(unsigned int, device->attrs.max_send_sge, |
| device->attrs.max_sge_rd); |
| max_sge_per_wr = max_t(unsigned int, max_sge_per_wr, |
| max_send_sges); |
| wrs_per_credit = max_t(unsigned int, 4, |
| DIV_ROUND_UP(t->pages_per_rw_credit, |
| max_sge_per_wr) + 1); |
| max_rw_wrs = t->max_rw_credits * wrs_per_credit; |
| |
| max_send_wrs = smb_direct_send_credit_target + max_rw_wrs; |
| if (max_send_wrs > device->attrs.max_cqe || |
| max_send_wrs > device->attrs.max_qp_wr) { |
| pr_err("consider lowering send_credit_target = %d\n", |
| smb_direct_send_credit_target); |
| pr_err("Possible CQE overrun, device reporting max_cqe %d max_qp_wr %d\n", |
| device->attrs.max_cqe, device->attrs.max_qp_wr); |
| return -EINVAL; |
| } |
| |
| if (smb_direct_receive_credit_max > device->attrs.max_cqe || |
| smb_direct_receive_credit_max > device->attrs.max_qp_wr) { |
| pr_err("consider lowering receive_credit_max = %d\n", |
| smb_direct_receive_credit_max); |
| pr_err("Possible CQE overrun, device reporting max_cpe %d max_qp_wr %d\n", |
| device->attrs.max_cqe, device->attrs.max_qp_wr); |
| return -EINVAL; |
| } |
| |
| if (device->attrs.max_recv_sge < SMB_DIRECT_MAX_RECV_SGES) { |
| pr_err("warning: device max_recv_sge = %d too small\n", |
| device->attrs.max_recv_sge); |
| return -EINVAL; |
| } |
| |
| t->recv_credits = 0; |
| t->count_avail_recvmsg = 0; |
| |
| t->recv_credit_max = smb_direct_receive_credit_max; |
| t->recv_credit_target = 10; |
| t->new_recv_credits = 0; |
| |
| t->send_credit_target = smb_direct_send_credit_target; |
| atomic_set(&t->send_credits, 0); |
| atomic_set(&t->rw_credits, t->max_rw_credits); |
| |
| t->max_send_size = smb_direct_max_send_size; |
| t->max_recv_size = smb_direct_max_receive_size; |
| t->max_fragmented_recv_size = smb_direct_max_fragmented_recv_size; |
| |
| cap->max_send_wr = max_send_wrs; |
| cap->max_recv_wr = t->recv_credit_max; |
| cap->max_send_sge = max_sge_per_wr; |
| cap->max_recv_sge = SMB_DIRECT_MAX_RECV_SGES; |
| cap->max_inline_data = 0; |
| cap->max_rdma_ctxs = t->max_rw_credits; |
| return 0; |
| } |
| |
| static void smb_direct_destroy_pools(struct smb_direct_transport *t) |
| { |
| struct smb_direct_recvmsg *recvmsg; |
| |
| while ((recvmsg = get_free_recvmsg(t))) |
| mempool_free(recvmsg, t->recvmsg_mempool); |
| while ((recvmsg = get_empty_recvmsg(t))) |
| mempool_free(recvmsg, t->recvmsg_mempool); |
| |
| mempool_destroy(t->recvmsg_mempool); |
| t->recvmsg_mempool = NULL; |
| |
| kmem_cache_destroy(t->recvmsg_cache); |
| t->recvmsg_cache = NULL; |
| |
| mempool_destroy(t->sendmsg_mempool); |
| t->sendmsg_mempool = NULL; |
| |
| kmem_cache_destroy(t->sendmsg_cache); |
| t->sendmsg_cache = NULL; |
| } |
| |
| static int smb_direct_create_pools(struct smb_direct_transport *t) |
| { |
| char name[80]; |
| int i; |
| struct smb_direct_recvmsg *recvmsg; |
| |
| snprintf(name, sizeof(name), "smb_direct_rqst_pool_%p", t); |
| t->sendmsg_cache = kmem_cache_create(name, |
| sizeof(struct smb_direct_sendmsg) + |
| sizeof(struct smb_direct_negotiate_resp), |
| 0, SLAB_HWCACHE_ALIGN, NULL); |
| if (!t->sendmsg_cache) |
| return -ENOMEM; |
| |
| t->sendmsg_mempool = mempool_create(t->send_credit_target, |
| mempool_alloc_slab, mempool_free_slab, |
| t->sendmsg_cache); |
| if (!t->sendmsg_mempool) |
| goto err; |
| |
| snprintf(name, sizeof(name), "smb_direct_resp_%p", t); |
| t->recvmsg_cache = kmem_cache_create(name, |
| sizeof(struct smb_direct_recvmsg) + |
| t->max_recv_size, |
| 0, SLAB_HWCACHE_ALIGN, NULL); |
| if (!t->recvmsg_cache) |
| goto err; |
| |
| t->recvmsg_mempool = |
| mempool_create(t->recv_credit_max, mempool_alloc_slab, |
| mempool_free_slab, t->recvmsg_cache); |
| if (!t->recvmsg_mempool) |
| goto err; |
| |
| INIT_LIST_HEAD(&t->recvmsg_queue); |
| |
| for (i = 0; i < t->recv_credit_max; i++) { |
| recvmsg = mempool_alloc(t->recvmsg_mempool, GFP_KERNEL); |
| if (!recvmsg) |
| goto err; |
| recvmsg->transport = t; |
| list_add(&recvmsg->list, &t->recvmsg_queue); |
| } |
| t->count_avail_recvmsg = t->recv_credit_max; |
| |
| return 0; |
| err: |
| smb_direct_destroy_pools(t); |
| return -ENOMEM; |
| } |
| |
| static int smb_direct_create_qpair(struct smb_direct_transport *t, |
| struct ib_qp_cap *cap) |
| { |
| int ret; |
| struct ib_qp_init_attr qp_attr; |
| int pages_per_rw; |
| |
| t->pd = ib_alloc_pd(t->cm_id->device, 0); |
| if (IS_ERR(t->pd)) { |
| pr_err("Can't create RDMA PD\n"); |
| ret = PTR_ERR(t->pd); |
| t->pd = NULL; |
| return ret; |
| } |
| |
| t->send_cq = ib_alloc_cq(t->cm_id->device, t, |
| smb_direct_send_credit_target + cap->max_rdma_ctxs, |
| 0, IB_POLL_WORKQUEUE); |
| if (IS_ERR(t->send_cq)) { |
| pr_err("Can't create RDMA send CQ\n"); |
| ret = PTR_ERR(t->send_cq); |
| t->send_cq = NULL; |
| goto err; |
| } |
| |
| t->recv_cq = ib_alloc_cq(t->cm_id->device, t, |
| t->recv_credit_max, 0, IB_POLL_WORKQUEUE); |
| if (IS_ERR(t->recv_cq)) { |
| pr_err("Can't create RDMA recv CQ\n"); |
| ret = PTR_ERR(t->recv_cq); |
| t->recv_cq = NULL; |
| goto err; |
| } |
| |
| memset(&qp_attr, 0, sizeof(qp_attr)); |
| qp_attr.event_handler = smb_direct_qpair_handler; |
| qp_attr.qp_context = t; |
| qp_attr.cap = *cap; |
| qp_attr.sq_sig_type = IB_SIGNAL_REQ_WR; |
| qp_attr.qp_type = IB_QPT_RC; |
| qp_attr.send_cq = t->send_cq; |
| qp_attr.recv_cq = t->recv_cq; |
| qp_attr.port_num = ~0; |
| |
| ret = rdma_create_qp(t->cm_id, t->pd, &qp_attr); |
| if (ret) { |
| pr_err("Can't create RDMA QP: %d\n", ret); |
| goto err; |
| } |
| |
| t->qp = t->cm_id->qp; |
| t->cm_id->event_handler = smb_direct_cm_handler; |
| |
| pages_per_rw = DIV_ROUND_UP(t->max_rdma_rw_size, PAGE_SIZE) + 1; |
| if (pages_per_rw > t->cm_id->device->attrs.max_sgl_rd) { |
| ret = ib_mr_pool_init(t->qp, &t->qp->rdma_mrs, |
| t->max_rw_credits, IB_MR_TYPE_MEM_REG, |
| t->pages_per_rw_credit, 0); |
| if (ret) { |
| pr_err("failed to init mr pool count %d pages %d\n", |
| t->max_rw_credits, t->pages_per_rw_credit); |
| goto err; |
| } |
| } |
| |
| return 0; |
| err: |
| if (t->qp) { |
| ib_destroy_qp(t->qp); |
| t->qp = NULL; |
| } |
| if (t->recv_cq) { |
| ib_destroy_cq(t->recv_cq); |
| t->recv_cq = NULL; |
| } |
| if (t->send_cq) { |
| ib_destroy_cq(t->send_cq); |
| t->send_cq = NULL; |
| } |
| if (t->pd) { |
| ib_dealloc_pd(t->pd); |
| t->pd = NULL; |
| } |
| return ret; |
| } |
| |
| static int smb_direct_prepare(struct ksmbd_transport *t) |
| { |
| struct smb_direct_transport *st = smb_trans_direct_transfort(t); |
| struct smb_direct_recvmsg *recvmsg; |
| struct smb_direct_negotiate_req *req; |
| int ret; |
| |
| ksmbd_debug(RDMA, "Waiting for SMB_DIRECT negotiate request\n"); |
| ret = wait_event_interruptible_timeout(st->wait_status, |
| st->negotiation_requested || |
| st->status == SMB_DIRECT_CS_DISCONNECTED, |
| SMB_DIRECT_NEGOTIATE_TIMEOUT * HZ); |
| if (ret <= 0 || st->status == SMB_DIRECT_CS_DISCONNECTED) |
| return ret < 0 ? ret : -ETIMEDOUT; |
| |
| recvmsg = get_first_reassembly(st); |
| if (!recvmsg) |
| return -ECONNABORTED; |
| |
| ret = smb_direct_check_recvmsg(recvmsg); |
| if (ret == -ECONNABORTED) |
| goto out; |
| |
| req = (struct smb_direct_negotiate_req *)recvmsg->packet; |
| st->max_recv_size = min_t(int, st->max_recv_size, |
| le32_to_cpu(req->preferred_send_size)); |
| st->max_send_size = min_t(int, st->max_send_size, |
| le32_to_cpu(req->max_receive_size)); |
| st->max_fragmented_send_size = |
| le32_to_cpu(req->max_fragmented_size); |
| st->max_fragmented_recv_size = |
| (st->recv_credit_max * st->max_recv_size) / 2; |
| |
| ret = smb_direct_send_negotiate_response(st, ret); |
| out: |
| spin_lock_irq(&st->reassembly_queue_lock); |
| st->reassembly_queue_length--; |
| list_del(&recvmsg->list); |
| spin_unlock_irq(&st->reassembly_queue_lock); |
| put_recvmsg(st, recvmsg); |
| |
| return ret; |
| } |
| |
| static int smb_direct_connect(struct smb_direct_transport *st) |
| { |
| int ret; |
| struct ib_qp_cap qp_cap; |
| |
| ret = smb_direct_init_params(st, &qp_cap); |
| if (ret) { |
| pr_err("Can't configure RDMA parameters\n"); |
| return ret; |
| } |
| |
| ret = smb_direct_create_pools(st); |
| if (ret) { |
| pr_err("Can't init RDMA pool: %d\n", ret); |
| return ret; |
| } |
| |
| ret = smb_direct_create_qpair(st, &qp_cap); |
| if (ret) { |
| pr_err("Can't accept RDMA client: %d\n", ret); |
| return ret; |
| } |
| |
| ret = smb_direct_prepare_negotiation(st); |
| if (ret) { |
| pr_err("Can't negotiate: %d\n", ret); |
| return ret; |
| } |
| return 0; |
| } |
| |
| static bool rdma_frwr_is_supported(struct ib_device_attr *attrs) |
| { |
| if (!(attrs->device_cap_flags & IB_DEVICE_MEM_MGT_EXTENSIONS)) |
| return false; |
| if (attrs->max_fast_reg_page_list_len == 0) |
| return false; |
| return true; |
| } |
| |
| static int smb_direct_handle_connect_request(struct rdma_cm_id *new_cm_id) |
| { |
| struct smb_direct_transport *t; |
| int ret; |
| |
| if (!rdma_frwr_is_supported(&new_cm_id->device->attrs)) { |
| ksmbd_debug(RDMA, |
| "Fast Registration Work Requests is not supported. device capabilities=%llx\n", |
| new_cm_id->device->attrs.device_cap_flags); |
| return -EPROTONOSUPPORT; |
| } |
| |
| t = alloc_transport(new_cm_id); |
| if (!t) |
| return -ENOMEM; |
| |
| ret = smb_direct_connect(t); |
| if (ret) |
| goto out_err; |
| |
| KSMBD_TRANS(t)->handler = kthread_run(ksmbd_conn_handler_loop, |
| KSMBD_TRANS(t)->conn, "ksmbd:r%u", |
| smb_direct_port); |
| if (IS_ERR(KSMBD_TRANS(t)->handler)) { |
| ret = PTR_ERR(KSMBD_TRANS(t)->handler); |
| pr_err("Can't start thread\n"); |
| goto out_err; |
| } |
| |
| return 0; |
| out_err: |
| free_transport(t); |
| return ret; |
| } |
| |
| static int smb_direct_listen_handler(struct rdma_cm_id *cm_id, |
| struct rdma_cm_event *event) |
| { |
| switch (event->event) { |
| case RDMA_CM_EVENT_CONNECT_REQUEST: { |
| int ret = smb_direct_handle_connect_request(cm_id); |
| |
| if (ret) { |
| pr_err("Can't create transport: %d\n", ret); |
| return ret; |
| } |
| |
| ksmbd_debug(RDMA, "Received connection request. cm_id=%p\n", |
| cm_id); |
| break; |
| } |
| default: |
| pr_err("Unexpected listen event. cm_id=%p, event=%s (%d)\n", |
| cm_id, rdma_event_msg(event->event), event->event); |
| break; |
| } |
| return 0; |
| } |
| |
| static int smb_direct_listen(int port) |
| { |
| int ret; |
| struct rdma_cm_id *cm_id; |
| struct sockaddr_in sin = { |
| .sin_family = AF_INET, |
| .sin_addr.s_addr = htonl(INADDR_ANY), |
| .sin_port = htons(port), |
| }; |
| |
| cm_id = rdma_create_id(&init_net, smb_direct_listen_handler, |
| &smb_direct_listener, RDMA_PS_TCP, IB_QPT_RC); |
| if (IS_ERR(cm_id)) { |
| pr_err("Can't create cm id: %ld\n", PTR_ERR(cm_id)); |
| return PTR_ERR(cm_id); |
| } |
| |
| ret = rdma_bind_addr(cm_id, (struct sockaddr *)&sin); |
| if (ret) { |
| pr_err("Can't bind: %d\n", ret); |
| goto err; |
| } |
| |
| smb_direct_listener.cm_id = cm_id; |
| |
| ret = rdma_listen(cm_id, 10); |
| if (ret) { |
| pr_err("Can't listen: %d\n", ret); |
| goto err; |
| } |
| return 0; |
| err: |
| smb_direct_listener.cm_id = NULL; |
| rdma_destroy_id(cm_id); |
| return ret; |
| } |
| |
| static int smb_direct_ib_client_add(struct ib_device *ib_dev) |
| { |
| struct smb_direct_device *smb_dev; |
| |
| /* Set 5445 port if device type is iWARP(No IB) */ |
| if (ib_dev->node_type != RDMA_NODE_IB_CA) |
| smb_direct_port = SMB_DIRECT_PORT_IWARP; |
| |
| if (!ib_dev->ops.get_netdev || |
| !rdma_frwr_is_supported(&ib_dev->attrs)) |
| return 0; |
| |
| smb_dev = kzalloc(sizeof(*smb_dev), GFP_KERNEL); |
| if (!smb_dev) |
| return -ENOMEM; |
| smb_dev->ib_dev = ib_dev; |
| |
| write_lock(&smb_direct_device_lock); |
| list_add(&smb_dev->list, &smb_direct_device_list); |
| write_unlock(&smb_direct_device_lock); |
| |
| ksmbd_debug(RDMA, "ib device added: name %s\n", ib_dev->name); |
| return 0; |
| } |
| |
| static void smb_direct_ib_client_remove(struct ib_device *ib_dev, |
| void *client_data) |
| { |
| struct smb_direct_device *smb_dev, *tmp; |
| |
| write_lock(&smb_direct_device_lock); |
| list_for_each_entry_safe(smb_dev, tmp, &smb_direct_device_list, list) { |
| if (smb_dev->ib_dev == ib_dev) { |
| list_del(&smb_dev->list); |
| kfree(smb_dev); |
| break; |
| } |
| } |
| write_unlock(&smb_direct_device_lock); |
| } |
| |
| static struct ib_client smb_direct_ib_client = { |
| .name = "ksmbd_smb_direct_ib", |
| .add = smb_direct_ib_client_add, |
| .remove = smb_direct_ib_client_remove, |
| }; |
| |
| int ksmbd_rdma_init(void) |
| { |
| int ret; |
| |
| smb_direct_listener.cm_id = NULL; |
| |
| ret = ib_register_client(&smb_direct_ib_client); |
| if (ret) { |
| pr_err("failed to ib_register_client\n"); |
| return ret; |
| } |
| |
| /* When a client is running out of send credits, the credits are |
| * granted by the server's sending a packet using this queue. |
| * This avoids the situation that a clients cannot send packets |
| * for lack of credits |
| */ |
| smb_direct_wq = alloc_workqueue("ksmbd-smb_direct-wq", |
| WQ_HIGHPRI | WQ_MEM_RECLAIM, 0); |
| if (!smb_direct_wq) |
| return -ENOMEM; |
| |
| ret = smb_direct_listen(smb_direct_port); |
| if (ret) { |
| destroy_workqueue(smb_direct_wq); |
| smb_direct_wq = NULL; |
| pr_err("Can't listen: %d\n", ret); |
| return ret; |
| } |
| |
| ksmbd_debug(RDMA, "init RDMA listener. cm_id=%p\n", |
| smb_direct_listener.cm_id); |
| return 0; |
| } |
| |
| void ksmbd_rdma_destroy(void) |
| { |
| if (!smb_direct_listener.cm_id) |
| return; |
| |
| ib_unregister_client(&smb_direct_ib_client); |
| rdma_destroy_id(smb_direct_listener.cm_id); |
| |
| smb_direct_listener.cm_id = NULL; |
| |
| if (smb_direct_wq) { |
| destroy_workqueue(smb_direct_wq); |
| smb_direct_wq = NULL; |
| } |
| } |
| |
| bool ksmbd_rdma_capable_netdev(struct net_device *netdev) |
| { |
| struct smb_direct_device *smb_dev; |
| int i; |
| bool rdma_capable = false; |
| |
| read_lock(&smb_direct_device_lock); |
| list_for_each_entry(smb_dev, &smb_direct_device_list, list) { |
| for (i = 0; i < smb_dev->ib_dev->phys_port_cnt; i++) { |
| struct net_device *ndev; |
| |
| ndev = smb_dev->ib_dev->ops.get_netdev(smb_dev->ib_dev, |
| i + 1); |
| if (!ndev) |
| continue; |
| |
| if (ndev == netdev) { |
| dev_put(ndev); |
| rdma_capable = true; |
| goto out; |
| } |
| dev_put(ndev); |
| } |
| } |
| out: |
| read_unlock(&smb_direct_device_lock); |
| |
| if (rdma_capable == false) { |
| struct ib_device *ibdev; |
| |
| ibdev = ib_device_get_by_netdev(netdev, RDMA_DRIVER_UNKNOWN); |
| if (ibdev) { |
| if (rdma_frwr_is_supported(&ibdev->attrs)) |
| rdma_capable = true; |
| ib_device_put(ibdev); |
| } |
| } |
| |
| return rdma_capable; |
| } |
| |
| static struct ksmbd_transport_ops ksmbd_smb_direct_transport_ops = { |
| .prepare = smb_direct_prepare, |
| .disconnect = smb_direct_disconnect, |
| .shutdown = smb_direct_shutdown, |
| .writev = smb_direct_writev, |
| .read = smb_direct_read, |
| .rdma_read = smb_direct_rdma_read, |
| .rdma_write = smb_direct_rdma_write, |
| }; |