138c8a9a5SSteve French // SPDX-License-Identifier: GPL-2.0-or-later
238c8a9a5SSteve French /*
338c8a9a5SSteve French * Copyright (C) 2017, Microsoft Corporation.
438c8a9a5SSteve French *
538c8a9a5SSteve French * Author(s): Long Li <longli@microsoft.com>
638c8a9a5SSteve French */
738c8a9a5SSteve French #include <linux/module.h>
838c8a9a5SSteve French #include <linux/highmem.h>
938c8a9a5SSteve French #include "smbdirect.h"
1038c8a9a5SSteve French #include "cifs_debug.h"
1138c8a9a5SSteve French #include "cifsproto.h"
1238c8a9a5SSteve French #include "smb2proto.h"
1338c8a9a5SSteve French
1438c8a9a5SSteve French static struct smbd_response *get_empty_queue_buffer(
1538c8a9a5SSteve French struct smbd_connection *info);
1638c8a9a5SSteve French static struct smbd_response *get_receive_buffer(
1738c8a9a5SSteve French struct smbd_connection *info);
1838c8a9a5SSteve French static void put_receive_buffer(
1938c8a9a5SSteve French struct smbd_connection *info,
2038c8a9a5SSteve French struct smbd_response *response);
2138c8a9a5SSteve French static int allocate_receive_buffers(struct smbd_connection *info, int num_buf);
2238c8a9a5SSteve French static void destroy_receive_buffers(struct smbd_connection *info);
2338c8a9a5SSteve French
2438c8a9a5SSteve French static void put_empty_packet(
2538c8a9a5SSteve French struct smbd_connection *info, struct smbd_response *response);
2638c8a9a5SSteve French static void enqueue_reassembly(
2738c8a9a5SSteve French struct smbd_connection *info,
2838c8a9a5SSteve French struct smbd_response *response, int data_length);
2938c8a9a5SSteve French static struct smbd_response *_get_first_reassembly(
3038c8a9a5SSteve French struct smbd_connection *info);
3138c8a9a5SSteve French
3238c8a9a5SSteve French static int smbd_post_recv(
3338c8a9a5SSteve French struct smbd_connection *info,
3438c8a9a5SSteve French struct smbd_response *response);
3538c8a9a5SSteve French
3638c8a9a5SSteve French static int smbd_post_send_empty(struct smbd_connection *info);
3738c8a9a5SSteve French
3838c8a9a5SSteve French static void destroy_mr_list(struct smbd_connection *info);
3938c8a9a5SSteve French static int allocate_mr_list(struct smbd_connection *info);
4038c8a9a5SSteve French
4138c8a9a5SSteve French struct smb_extract_to_rdma {
4238c8a9a5SSteve French struct ib_sge *sge;
4338c8a9a5SSteve French unsigned int nr_sge;
4438c8a9a5SSteve French unsigned int max_sge;
4538c8a9a5SSteve French struct ib_device *device;
4638c8a9a5SSteve French u32 local_dma_lkey;
4738c8a9a5SSteve French enum dma_data_direction direction;
4838c8a9a5SSteve French };
4938c8a9a5SSteve French static ssize_t smb_extract_iter_to_rdma(struct iov_iter *iter, size_t len,
5038c8a9a5SSteve French struct smb_extract_to_rdma *rdma);
5138c8a9a5SSteve French
5238c8a9a5SSteve French /* SMBD version number */
5338c8a9a5SSteve French #define SMBD_V1 0x0100
5438c8a9a5SSteve French
5538c8a9a5SSteve French /* Port numbers for SMBD transport */
5638c8a9a5SSteve French #define SMB_PORT 445
5738c8a9a5SSteve French #define SMBD_PORT 5445
5838c8a9a5SSteve French
5938c8a9a5SSteve French /* Address lookup and resolve timeout in ms */
6038c8a9a5SSteve French #define RDMA_RESOLVE_TIMEOUT 5000
6138c8a9a5SSteve French
6238c8a9a5SSteve French /* SMBD negotiation timeout in seconds */
6338c8a9a5SSteve French #define SMBD_NEGOTIATE_TIMEOUT 120
6438c8a9a5SSteve French
6538c8a9a5SSteve French /* SMBD minimum receive size and fragmented sized defined in [MS-SMBD] */
6638c8a9a5SSteve French #define SMBD_MIN_RECEIVE_SIZE 128
6738c8a9a5SSteve French #define SMBD_MIN_FRAGMENTED_SIZE 131072
6838c8a9a5SSteve French
6938c8a9a5SSteve French /*
7038c8a9a5SSteve French * Default maximum number of RDMA read/write outstanding on this connection
7138c8a9a5SSteve French * This value is possibly decreased during QP creation on hardware limit
7238c8a9a5SSteve French */
7338c8a9a5SSteve French #define SMBD_CM_RESPONDER_RESOURCES 32
7438c8a9a5SSteve French
7538c8a9a5SSteve French /* Maximum number of retries on data transfer operations */
7638c8a9a5SSteve French #define SMBD_CM_RETRY 6
7738c8a9a5SSteve French /* No need to retry on Receiver Not Ready since SMBD manages credits */
7838c8a9a5SSteve French #define SMBD_CM_RNR_RETRY 0
7938c8a9a5SSteve French
8038c8a9a5SSteve French /*
8138c8a9a5SSteve French * User configurable initial values per SMBD transport connection
8238c8a9a5SSteve French * as defined in [MS-SMBD] 3.1.1.1
8338c8a9a5SSteve French * Those may change after a SMBD negotiation
8438c8a9a5SSteve French */
8538c8a9a5SSteve French /* The local peer's maximum number of credits to grant to the peer */
8638c8a9a5SSteve French int smbd_receive_credit_max = 255;
8738c8a9a5SSteve French
8838c8a9a5SSteve French /* The remote peer's credit request of local peer */
8938c8a9a5SSteve French int smbd_send_credit_target = 255;
9038c8a9a5SSteve French
9138c8a9a5SSteve French /* The maximum single message size can be sent to remote peer */
9238c8a9a5SSteve French int smbd_max_send_size = 1364;
9338c8a9a5SSteve French
9438c8a9a5SSteve French /* The maximum fragmented upper-layer payload receive size supported */
9538c8a9a5SSteve French int smbd_max_fragmented_recv_size = 1024 * 1024;
9638c8a9a5SSteve French
9738c8a9a5SSteve French /* The maximum single-message size which can be received */
9838c8a9a5SSteve French int smbd_max_receive_size = 1364;
9938c8a9a5SSteve French
10038c8a9a5SSteve French /* The timeout to initiate send of a keepalive message on idle */
10138c8a9a5SSteve French int smbd_keep_alive_interval = 120;
10238c8a9a5SSteve French
10338c8a9a5SSteve French /*
10438c8a9a5SSteve French * User configurable initial values for RDMA transport
10538c8a9a5SSteve French * The actual values used may be lower and are limited to hardware capabilities
10638c8a9a5SSteve French */
10738c8a9a5SSteve French /* Default maximum number of pages in a single RDMA write/read */
10838c8a9a5SSteve French int smbd_max_frmr_depth = 2048;
10938c8a9a5SSteve French
11038c8a9a5SSteve French /* If payload is less than this byte, use RDMA send/recv not read/write */
11138c8a9a5SSteve French int rdma_readwrite_threshold = 4096;
11238c8a9a5SSteve French
11338c8a9a5SSteve French /* Transport logging functions
11438c8a9a5SSteve French * Logging are defined as classes. They can be OR'ed to define the actual
11538c8a9a5SSteve French * logging level via module parameter smbd_logging_class
11638c8a9a5SSteve French * e.g. cifs.smbd_logging_class=0xa0 will log all log_rdma_recv() and
11738c8a9a5SSteve French * log_rdma_event()
11838c8a9a5SSteve French */
11938c8a9a5SSteve French #define LOG_OUTGOING 0x1
12038c8a9a5SSteve French #define LOG_INCOMING 0x2
12138c8a9a5SSteve French #define LOG_READ 0x4
12238c8a9a5SSteve French #define LOG_WRITE 0x8
12338c8a9a5SSteve French #define LOG_RDMA_SEND 0x10
12438c8a9a5SSteve French #define LOG_RDMA_RECV 0x20
12538c8a9a5SSteve French #define LOG_KEEP_ALIVE 0x40
12638c8a9a5SSteve French #define LOG_RDMA_EVENT 0x80
12738c8a9a5SSteve French #define LOG_RDMA_MR 0x100
12838c8a9a5SSteve French static unsigned int smbd_logging_class;
12938c8a9a5SSteve French module_param(smbd_logging_class, uint, 0644);
13038c8a9a5SSteve French MODULE_PARM_DESC(smbd_logging_class,
13138c8a9a5SSteve French "Logging class for SMBD transport 0x0 to 0x100");
13238c8a9a5SSteve French
13338c8a9a5SSteve French #define ERR 0x0
13438c8a9a5SSteve French #define INFO 0x1
13538c8a9a5SSteve French static unsigned int smbd_logging_level = ERR;
13638c8a9a5SSteve French module_param(smbd_logging_level, uint, 0644);
13738c8a9a5SSteve French MODULE_PARM_DESC(smbd_logging_level,
13838c8a9a5SSteve French "Logging level for SMBD transport, 0 (default): error, 1: info");
13938c8a9a5SSteve French
14038c8a9a5SSteve French #define log_rdma(level, class, fmt, args...) \
14138c8a9a5SSteve French do { \
14238c8a9a5SSteve French if (level <= smbd_logging_level || class & smbd_logging_class) \
14338c8a9a5SSteve French cifs_dbg(VFS, "%s:%d " fmt, __func__, __LINE__, ##args);\
14438c8a9a5SSteve French } while (0)
14538c8a9a5SSteve French
14638c8a9a5SSteve French #define log_outgoing(level, fmt, args...) \
14738c8a9a5SSteve French log_rdma(level, LOG_OUTGOING, fmt, ##args)
14838c8a9a5SSteve French #define log_incoming(level, fmt, args...) \
14938c8a9a5SSteve French log_rdma(level, LOG_INCOMING, fmt, ##args)
15038c8a9a5SSteve French #define log_read(level, fmt, args...) log_rdma(level, LOG_READ, fmt, ##args)
15138c8a9a5SSteve French #define log_write(level, fmt, args...) log_rdma(level, LOG_WRITE, fmt, ##args)
15238c8a9a5SSteve French #define log_rdma_send(level, fmt, args...) \
15338c8a9a5SSteve French log_rdma(level, LOG_RDMA_SEND, fmt, ##args)
15438c8a9a5SSteve French #define log_rdma_recv(level, fmt, args...) \
15538c8a9a5SSteve French log_rdma(level, LOG_RDMA_RECV, fmt, ##args)
15638c8a9a5SSteve French #define log_keep_alive(level, fmt, args...) \
15738c8a9a5SSteve French log_rdma(level, LOG_KEEP_ALIVE, fmt, ##args)
15838c8a9a5SSteve French #define log_rdma_event(level, fmt, args...) \
15938c8a9a5SSteve French log_rdma(level, LOG_RDMA_EVENT, fmt, ##args)
16038c8a9a5SSteve French #define log_rdma_mr(level, fmt, args...) \
16138c8a9a5SSteve French log_rdma(level, LOG_RDMA_MR, fmt, ##args)
16238c8a9a5SSteve French
smbd_disconnect_rdma_work(struct work_struct * work)16338c8a9a5SSteve French static void smbd_disconnect_rdma_work(struct work_struct *work)
16438c8a9a5SSteve French {
16538c8a9a5SSteve French struct smbd_connection *info =
16638c8a9a5SSteve French container_of(work, struct smbd_connection, disconnect_work);
16738c8a9a5SSteve French
16838c8a9a5SSteve French if (info->transport_status == SMBD_CONNECTED) {
16938c8a9a5SSteve French info->transport_status = SMBD_DISCONNECTING;
17038c8a9a5SSteve French rdma_disconnect(info->id);
17138c8a9a5SSteve French }
17238c8a9a5SSteve French }
17338c8a9a5SSteve French
smbd_disconnect_rdma_connection(struct smbd_connection * info)17438c8a9a5SSteve French static void smbd_disconnect_rdma_connection(struct smbd_connection *info)
17538c8a9a5SSteve French {
17638c8a9a5SSteve French queue_work(info->workqueue, &info->disconnect_work);
17738c8a9a5SSteve French }
17838c8a9a5SSteve French
17938c8a9a5SSteve French /* Upcall from RDMA CM */
smbd_conn_upcall(struct rdma_cm_id * id,struct rdma_cm_event * event)18038c8a9a5SSteve French static int smbd_conn_upcall(
18138c8a9a5SSteve French struct rdma_cm_id *id, struct rdma_cm_event *event)
18238c8a9a5SSteve French {
18338c8a9a5SSteve French struct smbd_connection *info = id->context;
18438c8a9a5SSteve French
18538c8a9a5SSteve French log_rdma_event(INFO, "event=%d status=%d\n",
18638c8a9a5SSteve French event->event, event->status);
18738c8a9a5SSteve French
18838c8a9a5SSteve French switch (event->event) {
18938c8a9a5SSteve French case RDMA_CM_EVENT_ADDR_RESOLVED:
19038c8a9a5SSteve French case RDMA_CM_EVENT_ROUTE_RESOLVED:
19138c8a9a5SSteve French info->ri_rc = 0;
19238c8a9a5SSteve French complete(&info->ri_done);
19338c8a9a5SSteve French break;
19438c8a9a5SSteve French
19538c8a9a5SSteve French case RDMA_CM_EVENT_ADDR_ERROR:
19638c8a9a5SSteve French info->ri_rc = -EHOSTUNREACH;
19738c8a9a5SSteve French complete(&info->ri_done);
19838c8a9a5SSteve French break;
19938c8a9a5SSteve French
20038c8a9a5SSteve French case RDMA_CM_EVENT_ROUTE_ERROR:
20138c8a9a5SSteve French info->ri_rc = -ENETUNREACH;
20238c8a9a5SSteve French complete(&info->ri_done);
20338c8a9a5SSteve French break;
20438c8a9a5SSteve French
20538c8a9a5SSteve French case RDMA_CM_EVENT_ESTABLISHED:
20638c8a9a5SSteve French log_rdma_event(INFO, "connected event=%d\n", event->event);
20738c8a9a5SSteve French info->transport_status = SMBD_CONNECTED;
20838c8a9a5SSteve French wake_up_interruptible(&info->conn_wait);
20938c8a9a5SSteve French break;
21038c8a9a5SSteve French
21138c8a9a5SSteve French case RDMA_CM_EVENT_CONNECT_ERROR:
21238c8a9a5SSteve French case RDMA_CM_EVENT_UNREACHABLE:
21338c8a9a5SSteve French case RDMA_CM_EVENT_REJECTED:
21438c8a9a5SSteve French log_rdma_event(INFO, "connecting failed event=%d\n", event->event);
21538c8a9a5SSteve French info->transport_status = SMBD_DISCONNECTED;
21638c8a9a5SSteve French wake_up_interruptible(&info->conn_wait);
21738c8a9a5SSteve French break;
21838c8a9a5SSteve French
21938c8a9a5SSteve French case RDMA_CM_EVENT_DEVICE_REMOVAL:
22038c8a9a5SSteve French case RDMA_CM_EVENT_DISCONNECTED:
22138c8a9a5SSteve French /* This happenes when we fail the negotiation */
22238c8a9a5SSteve French if (info->transport_status == SMBD_NEGOTIATE_FAILED) {
22338c8a9a5SSteve French info->transport_status = SMBD_DISCONNECTED;
22438c8a9a5SSteve French wake_up(&info->conn_wait);
22538c8a9a5SSteve French break;
22638c8a9a5SSteve French }
22738c8a9a5SSteve French
22838c8a9a5SSteve French info->transport_status = SMBD_DISCONNECTED;
22938c8a9a5SSteve French wake_up_interruptible(&info->disconn_wait);
23038c8a9a5SSteve French wake_up_interruptible(&info->wait_reassembly_queue);
23138c8a9a5SSteve French wake_up_interruptible_all(&info->wait_send_queue);
23238c8a9a5SSteve French break;
23338c8a9a5SSteve French
23438c8a9a5SSteve French default:
23538c8a9a5SSteve French break;
23638c8a9a5SSteve French }
23738c8a9a5SSteve French
23838c8a9a5SSteve French return 0;
23938c8a9a5SSteve French }
24038c8a9a5SSteve French
24138c8a9a5SSteve French /* Upcall from RDMA QP */
24238c8a9a5SSteve French static void
smbd_qp_async_error_upcall(struct ib_event * event,void * context)24338c8a9a5SSteve French smbd_qp_async_error_upcall(struct ib_event *event, void *context)
24438c8a9a5SSteve French {
24538c8a9a5SSteve French struct smbd_connection *info = context;
24638c8a9a5SSteve French
24738c8a9a5SSteve French log_rdma_event(ERR, "%s on device %s info %p\n",
24838c8a9a5SSteve French ib_event_msg(event->event), event->device->name, info);
24938c8a9a5SSteve French
25038c8a9a5SSteve French switch (event->event) {
25138c8a9a5SSteve French case IB_EVENT_CQ_ERR:
25238c8a9a5SSteve French case IB_EVENT_QP_FATAL:
25338c8a9a5SSteve French smbd_disconnect_rdma_connection(info);
25438c8a9a5SSteve French break;
25538c8a9a5SSteve French
25638c8a9a5SSteve French default:
25738c8a9a5SSteve French break;
25838c8a9a5SSteve French }
25938c8a9a5SSteve French }
26038c8a9a5SSteve French
smbd_request_payload(struct smbd_request * request)26138c8a9a5SSteve French static inline void *smbd_request_payload(struct smbd_request *request)
26238c8a9a5SSteve French {
26338c8a9a5SSteve French return (void *)request->packet;
26438c8a9a5SSteve French }
26538c8a9a5SSteve French
smbd_response_payload(struct smbd_response * response)26638c8a9a5SSteve French static inline void *smbd_response_payload(struct smbd_response *response)
26738c8a9a5SSteve French {
26838c8a9a5SSteve French return (void *)response->packet;
26938c8a9a5SSteve French }
27038c8a9a5SSteve French
27138c8a9a5SSteve French /* Called when a RDMA send is done */
send_done(struct ib_cq * cq,struct ib_wc * wc)27238c8a9a5SSteve French static void send_done(struct ib_cq *cq, struct ib_wc *wc)
27338c8a9a5SSteve French {
27438c8a9a5SSteve French int i;
27538c8a9a5SSteve French struct smbd_request *request =
27638c8a9a5SSteve French container_of(wc->wr_cqe, struct smbd_request, cqe);
27738c8a9a5SSteve French
27838c8a9a5SSteve French log_rdma_send(INFO, "smbd_request 0x%p completed wc->status=%d\n",
27938c8a9a5SSteve French request, wc->status);
28038c8a9a5SSteve French
28138c8a9a5SSteve French if (wc->status != IB_WC_SUCCESS || wc->opcode != IB_WC_SEND) {
28238c8a9a5SSteve French log_rdma_send(ERR, "wc->status=%d wc->opcode=%d\n",
28338c8a9a5SSteve French wc->status, wc->opcode);
28438c8a9a5SSteve French smbd_disconnect_rdma_connection(request->info);
28538c8a9a5SSteve French }
28638c8a9a5SSteve French
28738c8a9a5SSteve French for (i = 0; i < request->num_sge; i++)
28838c8a9a5SSteve French ib_dma_unmap_single(request->info->id->device,
28938c8a9a5SSteve French request->sge[i].addr,
29038c8a9a5SSteve French request->sge[i].length,
29138c8a9a5SSteve French DMA_TO_DEVICE);
29238c8a9a5SSteve French
29338c8a9a5SSteve French if (atomic_dec_and_test(&request->info->send_pending))
29438c8a9a5SSteve French wake_up(&request->info->wait_send_pending);
29538c8a9a5SSteve French
29638c8a9a5SSteve French wake_up(&request->info->wait_post_send);
29738c8a9a5SSteve French
29838c8a9a5SSteve French mempool_free(request, request->info->request_mempool);
29938c8a9a5SSteve French }
30038c8a9a5SSteve French
dump_smbd_negotiate_resp(struct smbd_negotiate_resp * resp)30138c8a9a5SSteve French static void dump_smbd_negotiate_resp(struct smbd_negotiate_resp *resp)
30238c8a9a5SSteve French {
30338c8a9a5SSteve French log_rdma_event(INFO, "resp message min_version %u max_version %u negotiated_version %u credits_requested %u credits_granted %u status %u max_readwrite_size %u preferred_send_size %u max_receive_size %u max_fragmented_size %u\n",
30438c8a9a5SSteve French resp->min_version, resp->max_version,
30538c8a9a5SSteve French resp->negotiated_version, resp->credits_requested,
30638c8a9a5SSteve French resp->credits_granted, resp->status,
30738c8a9a5SSteve French resp->max_readwrite_size, resp->preferred_send_size,
30838c8a9a5SSteve French resp->max_receive_size, resp->max_fragmented_size);
30938c8a9a5SSteve French }
31038c8a9a5SSteve French
31138c8a9a5SSteve French /*
31238c8a9a5SSteve French * Process a negotiation response message, according to [MS-SMBD]3.1.5.7
31338c8a9a5SSteve French * response, packet_length: the negotiation response message
31438c8a9a5SSteve French * return value: true if negotiation is a success, false if failed
31538c8a9a5SSteve French */
process_negotiation_response(struct smbd_response * response,int packet_length)31638c8a9a5SSteve French static bool process_negotiation_response(
31738c8a9a5SSteve French struct smbd_response *response, int packet_length)
31838c8a9a5SSteve French {
31938c8a9a5SSteve French struct smbd_connection *info = response->info;
32038c8a9a5SSteve French struct smbd_negotiate_resp *packet = smbd_response_payload(response);
32138c8a9a5SSteve French
32238c8a9a5SSteve French if (packet_length < sizeof(struct smbd_negotiate_resp)) {
32338c8a9a5SSteve French log_rdma_event(ERR,
32438c8a9a5SSteve French "error: packet_length=%d\n", packet_length);
32538c8a9a5SSteve French return false;
32638c8a9a5SSteve French }
32738c8a9a5SSteve French
32838c8a9a5SSteve French if (le16_to_cpu(packet->negotiated_version) != SMBD_V1) {
32938c8a9a5SSteve French log_rdma_event(ERR, "error: negotiated_version=%x\n",
33038c8a9a5SSteve French le16_to_cpu(packet->negotiated_version));
33138c8a9a5SSteve French return false;
33238c8a9a5SSteve French }
33338c8a9a5SSteve French info->protocol = le16_to_cpu(packet->negotiated_version);
33438c8a9a5SSteve French
33538c8a9a5SSteve French if (packet->credits_requested == 0) {
33638c8a9a5SSteve French log_rdma_event(ERR, "error: credits_requested==0\n");
33738c8a9a5SSteve French return false;
33838c8a9a5SSteve French }
33938c8a9a5SSteve French info->receive_credit_target = le16_to_cpu(packet->credits_requested);
34038c8a9a5SSteve French
34138c8a9a5SSteve French if (packet->credits_granted == 0) {
34238c8a9a5SSteve French log_rdma_event(ERR, "error: credits_granted==0\n");
34338c8a9a5SSteve French return false;
34438c8a9a5SSteve French }
34538c8a9a5SSteve French atomic_set(&info->send_credits, le16_to_cpu(packet->credits_granted));
34638c8a9a5SSteve French
34738c8a9a5SSteve French atomic_set(&info->receive_credits, 0);
34838c8a9a5SSteve French
34938c8a9a5SSteve French if (le32_to_cpu(packet->preferred_send_size) > info->max_receive_size) {
35038c8a9a5SSteve French log_rdma_event(ERR, "error: preferred_send_size=%d\n",
35138c8a9a5SSteve French le32_to_cpu(packet->preferred_send_size));
35238c8a9a5SSteve French return false;
35338c8a9a5SSteve French }
35438c8a9a5SSteve French info->max_receive_size = le32_to_cpu(packet->preferred_send_size);
35538c8a9a5SSteve French
35638c8a9a5SSteve French if (le32_to_cpu(packet->max_receive_size) < SMBD_MIN_RECEIVE_SIZE) {
35738c8a9a5SSteve French log_rdma_event(ERR, "error: max_receive_size=%d\n",
35838c8a9a5SSteve French le32_to_cpu(packet->max_receive_size));
35938c8a9a5SSteve French return false;
36038c8a9a5SSteve French }
36138c8a9a5SSteve French info->max_send_size = min_t(int, info->max_send_size,
36238c8a9a5SSteve French le32_to_cpu(packet->max_receive_size));
36338c8a9a5SSteve French
36438c8a9a5SSteve French if (le32_to_cpu(packet->max_fragmented_size) <
36538c8a9a5SSteve French SMBD_MIN_FRAGMENTED_SIZE) {
36638c8a9a5SSteve French log_rdma_event(ERR, "error: max_fragmented_size=%d\n",
36738c8a9a5SSteve French le32_to_cpu(packet->max_fragmented_size));
36838c8a9a5SSteve French return false;
36938c8a9a5SSteve French }
37038c8a9a5SSteve French info->max_fragmented_send_size =
37138c8a9a5SSteve French le32_to_cpu(packet->max_fragmented_size);
37238c8a9a5SSteve French info->rdma_readwrite_threshold =
37338c8a9a5SSteve French rdma_readwrite_threshold > info->max_fragmented_send_size ?
37438c8a9a5SSteve French info->max_fragmented_send_size :
37538c8a9a5SSteve French rdma_readwrite_threshold;
37638c8a9a5SSteve French
37738c8a9a5SSteve French
37838c8a9a5SSteve French info->max_readwrite_size = min_t(u32,
37938c8a9a5SSteve French le32_to_cpu(packet->max_readwrite_size),
38038c8a9a5SSteve French info->max_frmr_depth * PAGE_SIZE);
38138c8a9a5SSteve French info->max_frmr_depth = info->max_readwrite_size / PAGE_SIZE;
38238c8a9a5SSteve French
38338c8a9a5SSteve French return true;
38438c8a9a5SSteve French }
38538c8a9a5SSteve French
smbd_post_send_credits(struct work_struct * work)38638c8a9a5SSteve French static void smbd_post_send_credits(struct work_struct *work)
38738c8a9a5SSteve French {
38838c8a9a5SSteve French int ret = 0;
38938c8a9a5SSteve French int use_receive_queue = 1;
39038c8a9a5SSteve French int rc;
39138c8a9a5SSteve French struct smbd_response *response;
39238c8a9a5SSteve French struct smbd_connection *info =
39338c8a9a5SSteve French container_of(work, struct smbd_connection,
39438c8a9a5SSteve French post_send_credits_work);
39538c8a9a5SSteve French
39638c8a9a5SSteve French if (info->transport_status != SMBD_CONNECTED) {
39738c8a9a5SSteve French wake_up(&info->wait_receive_queues);
39838c8a9a5SSteve French return;
39938c8a9a5SSteve French }
40038c8a9a5SSteve French
40138c8a9a5SSteve French if (info->receive_credit_target >
40238c8a9a5SSteve French atomic_read(&info->receive_credits)) {
40338c8a9a5SSteve French while (true) {
40438c8a9a5SSteve French if (use_receive_queue)
40538c8a9a5SSteve French response = get_receive_buffer(info);
40638c8a9a5SSteve French else
40738c8a9a5SSteve French response = get_empty_queue_buffer(info);
40838c8a9a5SSteve French if (!response) {
40938c8a9a5SSteve French /* now switch to emtpy packet queue */
41038c8a9a5SSteve French if (use_receive_queue) {
41138c8a9a5SSteve French use_receive_queue = 0;
41238c8a9a5SSteve French continue;
41338c8a9a5SSteve French } else
41438c8a9a5SSteve French break;
41538c8a9a5SSteve French }
41638c8a9a5SSteve French
41738c8a9a5SSteve French response->type = SMBD_TRANSFER_DATA;
41838c8a9a5SSteve French response->first_segment = false;
41938c8a9a5SSteve French rc = smbd_post_recv(info, response);
42038c8a9a5SSteve French if (rc) {
42138c8a9a5SSteve French log_rdma_recv(ERR,
42238c8a9a5SSteve French "post_recv failed rc=%d\n", rc);
42338c8a9a5SSteve French put_receive_buffer(info, response);
42438c8a9a5SSteve French break;
42538c8a9a5SSteve French }
42638c8a9a5SSteve French
42738c8a9a5SSteve French ret++;
42838c8a9a5SSteve French }
42938c8a9a5SSteve French }
43038c8a9a5SSteve French
43138c8a9a5SSteve French spin_lock(&info->lock_new_credits_offered);
43238c8a9a5SSteve French info->new_credits_offered += ret;
43338c8a9a5SSteve French spin_unlock(&info->lock_new_credits_offered);
43438c8a9a5SSteve French
43538c8a9a5SSteve French /* Promptly send an immediate packet as defined in [MS-SMBD] 3.1.1.1 */
43638c8a9a5SSteve French info->send_immediate = true;
43738c8a9a5SSteve French if (atomic_read(&info->receive_credits) <
43838c8a9a5SSteve French info->receive_credit_target - 1) {
43938c8a9a5SSteve French if (info->keep_alive_requested == KEEP_ALIVE_PENDING ||
44038c8a9a5SSteve French info->send_immediate) {
44138c8a9a5SSteve French log_keep_alive(INFO, "send an empty message\n");
44238c8a9a5SSteve French smbd_post_send_empty(info);
44338c8a9a5SSteve French }
44438c8a9a5SSteve French }
44538c8a9a5SSteve French }
44638c8a9a5SSteve French
44738c8a9a5SSteve French /* Called from softirq, when recv is done */
recv_done(struct ib_cq * cq,struct ib_wc * wc)44838c8a9a5SSteve French static void recv_done(struct ib_cq *cq, struct ib_wc *wc)
44938c8a9a5SSteve French {
45038c8a9a5SSteve French struct smbd_data_transfer *data_transfer;
45138c8a9a5SSteve French struct smbd_response *response =
45238c8a9a5SSteve French container_of(wc->wr_cqe, struct smbd_response, cqe);
45338c8a9a5SSteve French struct smbd_connection *info = response->info;
45438c8a9a5SSteve French int data_length = 0;
45538c8a9a5SSteve French
45638c8a9a5SSteve French log_rdma_recv(INFO, "response=0x%p type=%d wc status=%d wc opcode %d byte_len=%d pkey_index=%u\n",
45738c8a9a5SSteve French response, response->type, wc->status, wc->opcode,
45838c8a9a5SSteve French wc->byte_len, wc->pkey_index);
45938c8a9a5SSteve French
46038c8a9a5SSteve French if (wc->status != IB_WC_SUCCESS || wc->opcode != IB_WC_RECV) {
46138c8a9a5SSteve French log_rdma_recv(INFO, "wc->status=%d opcode=%d\n",
46238c8a9a5SSteve French wc->status, wc->opcode);
46338c8a9a5SSteve French smbd_disconnect_rdma_connection(info);
46438c8a9a5SSteve French goto error;
46538c8a9a5SSteve French }
46638c8a9a5SSteve French
46738c8a9a5SSteve French ib_dma_sync_single_for_cpu(
46838c8a9a5SSteve French wc->qp->device,
46938c8a9a5SSteve French response->sge.addr,
47038c8a9a5SSteve French response->sge.length,
47138c8a9a5SSteve French DMA_FROM_DEVICE);
47238c8a9a5SSteve French
47338c8a9a5SSteve French switch (response->type) {
47438c8a9a5SSteve French /* SMBD negotiation response */
47538c8a9a5SSteve French case SMBD_NEGOTIATE_RESP:
47638c8a9a5SSteve French dump_smbd_negotiate_resp(smbd_response_payload(response));
47738c8a9a5SSteve French info->full_packet_received = true;
47838c8a9a5SSteve French info->negotiate_done =
47938c8a9a5SSteve French process_negotiation_response(response, wc->byte_len);
48038c8a9a5SSteve French complete(&info->negotiate_completion);
48138c8a9a5SSteve French break;
48238c8a9a5SSteve French
48338c8a9a5SSteve French /* SMBD data transfer packet */
48438c8a9a5SSteve French case SMBD_TRANSFER_DATA:
48538c8a9a5SSteve French data_transfer = smbd_response_payload(response);
48638c8a9a5SSteve French data_length = le32_to_cpu(data_transfer->data_length);
48738c8a9a5SSteve French
48838c8a9a5SSteve French /*
48938c8a9a5SSteve French * If this is a packet with data playload place the data in
49038c8a9a5SSteve French * reassembly queue and wake up the reading thread
49138c8a9a5SSteve French */
49238c8a9a5SSteve French if (data_length) {
49338c8a9a5SSteve French if (info->full_packet_received)
49438c8a9a5SSteve French response->first_segment = true;
49538c8a9a5SSteve French
49638c8a9a5SSteve French if (le32_to_cpu(data_transfer->remaining_data_length))
49738c8a9a5SSteve French info->full_packet_received = false;
49838c8a9a5SSteve French else
49938c8a9a5SSteve French info->full_packet_received = true;
50038c8a9a5SSteve French
50138c8a9a5SSteve French enqueue_reassembly(
50238c8a9a5SSteve French info,
50338c8a9a5SSteve French response,
50438c8a9a5SSteve French data_length);
50538c8a9a5SSteve French } else
50638c8a9a5SSteve French put_empty_packet(info, response);
50738c8a9a5SSteve French
50838c8a9a5SSteve French if (data_length)
50938c8a9a5SSteve French wake_up_interruptible(&info->wait_reassembly_queue);
51038c8a9a5SSteve French
51138c8a9a5SSteve French atomic_dec(&info->receive_credits);
51238c8a9a5SSteve French info->receive_credit_target =
51338c8a9a5SSteve French le16_to_cpu(data_transfer->credits_requested);
51438c8a9a5SSteve French if (le16_to_cpu(data_transfer->credits_granted)) {
51538c8a9a5SSteve French atomic_add(le16_to_cpu(data_transfer->credits_granted),
51638c8a9a5SSteve French &info->send_credits);
51738c8a9a5SSteve French /*
51838c8a9a5SSteve French * We have new send credits granted from remote peer
51938c8a9a5SSteve French * If any sender is waiting for credits, unblock it
52038c8a9a5SSteve French */
52138c8a9a5SSteve French wake_up_interruptible(&info->wait_send_queue);
52238c8a9a5SSteve French }
52338c8a9a5SSteve French
52438c8a9a5SSteve French log_incoming(INFO, "data flags %d data_offset %d data_length %d remaining_data_length %d\n",
52538c8a9a5SSteve French le16_to_cpu(data_transfer->flags),
52638c8a9a5SSteve French le32_to_cpu(data_transfer->data_offset),
52738c8a9a5SSteve French le32_to_cpu(data_transfer->data_length),
52838c8a9a5SSteve French le32_to_cpu(data_transfer->remaining_data_length));
52938c8a9a5SSteve French
53038c8a9a5SSteve French /* Send a KEEP_ALIVE response right away if requested */
53138c8a9a5SSteve French info->keep_alive_requested = KEEP_ALIVE_NONE;
53238c8a9a5SSteve French if (le16_to_cpu(data_transfer->flags) &
53338c8a9a5SSteve French SMB_DIRECT_RESPONSE_REQUESTED) {
53438c8a9a5SSteve French info->keep_alive_requested = KEEP_ALIVE_PENDING;
53538c8a9a5SSteve French }
53638c8a9a5SSteve French
53738c8a9a5SSteve French return;
53838c8a9a5SSteve French
53938c8a9a5SSteve French default:
54038c8a9a5SSteve French log_rdma_recv(ERR,
54138c8a9a5SSteve French "unexpected response type=%d\n", response->type);
54238c8a9a5SSteve French }
54338c8a9a5SSteve French
54438c8a9a5SSteve French error:
54538c8a9a5SSteve French put_receive_buffer(info, response);
54638c8a9a5SSteve French }
54738c8a9a5SSteve French
smbd_create_id(struct smbd_connection * info,struct sockaddr * dstaddr,int port)54838c8a9a5SSteve French static struct rdma_cm_id *smbd_create_id(
54938c8a9a5SSteve French struct smbd_connection *info,
55038c8a9a5SSteve French struct sockaddr *dstaddr, int port)
55138c8a9a5SSteve French {
55238c8a9a5SSteve French struct rdma_cm_id *id;
55338c8a9a5SSteve French int rc;
55438c8a9a5SSteve French __be16 *sport;
55538c8a9a5SSteve French
55638c8a9a5SSteve French id = rdma_create_id(&init_net, smbd_conn_upcall, info,
55738c8a9a5SSteve French RDMA_PS_TCP, IB_QPT_RC);
55838c8a9a5SSteve French if (IS_ERR(id)) {
55938c8a9a5SSteve French rc = PTR_ERR(id);
56038c8a9a5SSteve French log_rdma_event(ERR, "rdma_create_id() failed %i\n", rc);
56138c8a9a5SSteve French return id;
56238c8a9a5SSteve French }
56338c8a9a5SSteve French
56438c8a9a5SSteve French if (dstaddr->sa_family == AF_INET6)
56538c8a9a5SSteve French sport = &((struct sockaddr_in6 *)dstaddr)->sin6_port;
56638c8a9a5SSteve French else
56738c8a9a5SSteve French sport = &((struct sockaddr_in *)dstaddr)->sin_port;
56838c8a9a5SSteve French
56938c8a9a5SSteve French *sport = htons(port);
57038c8a9a5SSteve French
57138c8a9a5SSteve French init_completion(&info->ri_done);
57238c8a9a5SSteve French info->ri_rc = -ETIMEDOUT;
57338c8a9a5SSteve French
57438c8a9a5SSteve French rc = rdma_resolve_addr(id, NULL, (struct sockaddr *)dstaddr,
57538c8a9a5SSteve French RDMA_RESOLVE_TIMEOUT);
57638c8a9a5SSteve French if (rc) {
57738c8a9a5SSteve French log_rdma_event(ERR, "rdma_resolve_addr() failed %i\n", rc);
57838c8a9a5SSteve French goto out;
57938c8a9a5SSteve French }
58038c8a9a5SSteve French rc = wait_for_completion_interruptible_timeout(
58138c8a9a5SSteve French &info->ri_done, msecs_to_jiffies(RDMA_RESOLVE_TIMEOUT));
58238c8a9a5SSteve French /* e.g. if interrupted returns -ERESTARTSYS */
58338c8a9a5SSteve French if (rc < 0) {
58438c8a9a5SSteve French log_rdma_event(ERR, "rdma_resolve_addr timeout rc: %i\n", rc);
58538c8a9a5SSteve French goto out;
58638c8a9a5SSteve French }
58738c8a9a5SSteve French rc = info->ri_rc;
58838c8a9a5SSteve French if (rc) {
58938c8a9a5SSteve French log_rdma_event(ERR, "rdma_resolve_addr() completed %i\n", rc);
59038c8a9a5SSteve French goto out;
59138c8a9a5SSteve French }
59238c8a9a5SSteve French
59338c8a9a5SSteve French info->ri_rc = -ETIMEDOUT;
59438c8a9a5SSteve French rc = rdma_resolve_route(id, RDMA_RESOLVE_TIMEOUT);
59538c8a9a5SSteve French if (rc) {
59638c8a9a5SSteve French log_rdma_event(ERR, "rdma_resolve_route() failed %i\n", rc);
59738c8a9a5SSteve French goto out;
59838c8a9a5SSteve French }
59938c8a9a5SSteve French rc = wait_for_completion_interruptible_timeout(
60038c8a9a5SSteve French &info->ri_done, msecs_to_jiffies(RDMA_RESOLVE_TIMEOUT));
60138c8a9a5SSteve French /* e.g. if interrupted returns -ERESTARTSYS */
60238c8a9a5SSteve French if (rc < 0) {
60338c8a9a5SSteve French log_rdma_event(ERR, "rdma_resolve_addr timeout rc: %i\n", rc);
60438c8a9a5SSteve French goto out;
60538c8a9a5SSteve French }
60638c8a9a5SSteve French rc = info->ri_rc;
60738c8a9a5SSteve French if (rc) {
60838c8a9a5SSteve French log_rdma_event(ERR, "rdma_resolve_route() completed %i\n", rc);
60938c8a9a5SSteve French goto out;
61038c8a9a5SSteve French }
61138c8a9a5SSteve French
61238c8a9a5SSteve French return id;
61338c8a9a5SSteve French
61438c8a9a5SSteve French out:
61538c8a9a5SSteve French rdma_destroy_id(id);
61638c8a9a5SSteve French return ERR_PTR(rc);
61738c8a9a5SSteve French }
61838c8a9a5SSteve French
61938c8a9a5SSteve French /*
62038c8a9a5SSteve French * Test if FRWR (Fast Registration Work Requests) is supported on the device
62138c8a9a5SSteve French * This implementation requries FRWR on RDMA read/write
62238c8a9a5SSteve French * return value: true if it is supported
62338c8a9a5SSteve French */
frwr_is_supported(struct ib_device_attr * attrs)62438c8a9a5SSteve French static bool frwr_is_supported(struct ib_device_attr *attrs)
62538c8a9a5SSteve French {
62638c8a9a5SSteve French if (!(attrs->device_cap_flags & IB_DEVICE_MEM_MGT_EXTENSIONS))
62738c8a9a5SSteve French return false;
62838c8a9a5SSteve French if (attrs->max_fast_reg_page_list_len == 0)
62938c8a9a5SSteve French return false;
63038c8a9a5SSteve French return true;
63138c8a9a5SSteve French }
63238c8a9a5SSteve French
smbd_ia_open(struct smbd_connection * info,struct sockaddr * dstaddr,int port)63338c8a9a5SSteve French static int smbd_ia_open(
63438c8a9a5SSteve French struct smbd_connection *info,
63538c8a9a5SSteve French struct sockaddr *dstaddr, int port)
63638c8a9a5SSteve French {
63738c8a9a5SSteve French int rc;
63838c8a9a5SSteve French
63938c8a9a5SSteve French info->id = smbd_create_id(info, dstaddr, port);
64038c8a9a5SSteve French if (IS_ERR(info->id)) {
64138c8a9a5SSteve French rc = PTR_ERR(info->id);
64238c8a9a5SSteve French goto out1;
64338c8a9a5SSteve French }
64438c8a9a5SSteve French
64538c8a9a5SSteve French if (!frwr_is_supported(&info->id->device->attrs)) {
64638c8a9a5SSteve French log_rdma_event(ERR, "Fast Registration Work Requests (FRWR) is not supported\n");
64738c8a9a5SSteve French log_rdma_event(ERR, "Device capability flags = %llx max_fast_reg_page_list_len = %u\n",
64838c8a9a5SSteve French info->id->device->attrs.device_cap_flags,
64938c8a9a5SSteve French info->id->device->attrs.max_fast_reg_page_list_len);
65038c8a9a5SSteve French rc = -EPROTONOSUPPORT;
65138c8a9a5SSteve French goto out2;
65238c8a9a5SSteve French }
65338c8a9a5SSteve French info->max_frmr_depth = min_t(int,
65438c8a9a5SSteve French smbd_max_frmr_depth,
65538c8a9a5SSteve French info->id->device->attrs.max_fast_reg_page_list_len);
65638c8a9a5SSteve French info->mr_type = IB_MR_TYPE_MEM_REG;
65738c8a9a5SSteve French if (info->id->device->attrs.kernel_cap_flags & IBK_SG_GAPS_REG)
65838c8a9a5SSteve French info->mr_type = IB_MR_TYPE_SG_GAPS;
65938c8a9a5SSteve French
66038c8a9a5SSteve French info->pd = ib_alloc_pd(info->id->device, 0);
66138c8a9a5SSteve French if (IS_ERR(info->pd)) {
66238c8a9a5SSteve French rc = PTR_ERR(info->pd);
66338c8a9a5SSteve French log_rdma_event(ERR, "ib_alloc_pd() returned %d\n", rc);
66438c8a9a5SSteve French goto out2;
66538c8a9a5SSteve French }
66638c8a9a5SSteve French
66738c8a9a5SSteve French return 0;
66838c8a9a5SSteve French
66938c8a9a5SSteve French out2:
67038c8a9a5SSteve French rdma_destroy_id(info->id);
67138c8a9a5SSteve French info->id = NULL;
67238c8a9a5SSteve French
67338c8a9a5SSteve French out1:
67438c8a9a5SSteve French return rc;
67538c8a9a5SSteve French }
67638c8a9a5SSteve French
67738c8a9a5SSteve French /*
67838c8a9a5SSteve French * Send a negotiation request message to the peer
67938c8a9a5SSteve French * The negotiation procedure is in [MS-SMBD] 3.1.5.2 and 3.1.5.3
68038c8a9a5SSteve French * After negotiation, the transport is connected and ready for
68138c8a9a5SSteve French * carrying upper layer SMB payload
68238c8a9a5SSteve French */
smbd_post_send_negotiate_req(struct smbd_connection * info)68338c8a9a5SSteve French static int smbd_post_send_negotiate_req(struct smbd_connection *info)
68438c8a9a5SSteve French {
68538c8a9a5SSteve French struct ib_send_wr send_wr;
68638c8a9a5SSteve French int rc = -ENOMEM;
68738c8a9a5SSteve French struct smbd_request *request;
68838c8a9a5SSteve French struct smbd_negotiate_req *packet;
68938c8a9a5SSteve French
69038c8a9a5SSteve French request = mempool_alloc(info->request_mempool, GFP_KERNEL);
69138c8a9a5SSteve French if (!request)
69238c8a9a5SSteve French return rc;
69338c8a9a5SSteve French
69438c8a9a5SSteve French request->info = info;
69538c8a9a5SSteve French
69638c8a9a5SSteve French packet = smbd_request_payload(request);
69738c8a9a5SSteve French packet->min_version = cpu_to_le16(SMBD_V1);
69838c8a9a5SSteve French packet->max_version = cpu_to_le16(SMBD_V1);
69938c8a9a5SSteve French packet->reserved = 0;
70038c8a9a5SSteve French packet->credits_requested = cpu_to_le16(info->send_credit_target);
70138c8a9a5SSteve French packet->preferred_send_size = cpu_to_le32(info->max_send_size);
70238c8a9a5SSteve French packet->max_receive_size = cpu_to_le32(info->max_receive_size);
70338c8a9a5SSteve French packet->max_fragmented_size =
70438c8a9a5SSteve French cpu_to_le32(info->max_fragmented_recv_size);
70538c8a9a5SSteve French
70638c8a9a5SSteve French request->num_sge = 1;
70738c8a9a5SSteve French request->sge[0].addr = ib_dma_map_single(
70838c8a9a5SSteve French info->id->device, (void *)packet,
70938c8a9a5SSteve French sizeof(*packet), DMA_TO_DEVICE);
71038c8a9a5SSteve French if (ib_dma_mapping_error(info->id->device, request->sge[0].addr)) {
71138c8a9a5SSteve French rc = -EIO;
71238c8a9a5SSteve French goto dma_mapping_failed;
71338c8a9a5SSteve French }
71438c8a9a5SSteve French
71538c8a9a5SSteve French request->sge[0].length = sizeof(*packet);
71638c8a9a5SSteve French request->sge[0].lkey = info->pd->local_dma_lkey;
71738c8a9a5SSteve French
71838c8a9a5SSteve French ib_dma_sync_single_for_device(
71938c8a9a5SSteve French info->id->device, request->sge[0].addr,
72038c8a9a5SSteve French request->sge[0].length, DMA_TO_DEVICE);
72138c8a9a5SSteve French
72238c8a9a5SSteve French request->cqe.done = send_done;
72338c8a9a5SSteve French
72438c8a9a5SSteve French send_wr.next = NULL;
72538c8a9a5SSteve French send_wr.wr_cqe = &request->cqe;
72638c8a9a5SSteve French send_wr.sg_list = request->sge;
72738c8a9a5SSteve French send_wr.num_sge = request->num_sge;
72838c8a9a5SSteve French send_wr.opcode = IB_WR_SEND;
72938c8a9a5SSteve French send_wr.send_flags = IB_SEND_SIGNALED;
73038c8a9a5SSteve French
73138c8a9a5SSteve French log_rdma_send(INFO, "sge addr=0x%llx length=%u lkey=0x%x\n",
73238c8a9a5SSteve French request->sge[0].addr,
73338c8a9a5SSteve French request->sge[0].length, request->sge[0].lkey);
73438c8a9a5SSteve French
73538c8a9a5SSteve French atomic_inc(&info->send_pending);
73638c8a9a5SSteve French rc = ib_post_send(info->id->qp, &send_wr, NULL);
73738c8a9a5SSteve French if (!rc)
73838c8a9a5SSteve French return 0;
73938c8a9a5SSteve French
74038c8a9a5SSteve French /* if we reach here, post send failed */
74138c8a9a5SSteve French log_rdma_send(ERR, "ib_post_send failed rc=%d\n", rc);
74238c8a9a5SSteve French atomic_dec(&info->send_pending);
74338c8a9a5SSteve French ib_dma_unmap_single(info->id->device, request->sge[0].addr,
74438c8a9a5SSteve French request->sge[0].length, DMA_TO_DEVICE);
74538c8a9a5SSteve French
74638c8a9a5SSteve French smbd_disconnect_rdma_connection(info);
74738c8a9a5SSteve French
74838c8a9a5SSteve French dma_mapping_failed:
74938c8a9a5SSteve French mempool_free(request, info->request_mempool);
75038c8a9a5SSteve French return rc;
75138c8a9a5SSteve French }
75238c8a9a5SSteve French
75338c8a9a5SSteve French /*
75438c8a9a5SSteve French * Extend the credits to remote peer
75538c8a9a5SSteve French * This implements [MS-SMBD] 3.1.5.9
75638c8a9a5SSteve French * The idea is that we should extend credits to remote peer as quickly as
75738c8a9a5SSteve French * it's allowed, to maintain data flow. We allocate as much receive
75838c8a9a5SSteve French * buffer as possible, and extend the receive credits to remote peer
75938c8a9a5SSteve French * return value: the new credtis being granted.
76038c8a9a5SSteve French */
manage_credits_prior_sending(struct smbd_connection * info)76138c8a9a5SSteve French static int manage_credits_prior_sending(struct smbd_connection *info)
76238c8a9a5SSteve French {
76338c8a9a5SSteve French int new_credits;
76438c8a9a5SSteve French
76538c8a9a5SSteve French spin_lock(&info->lock_new_credits_offered);
76638c8a9a5SSteve French new_credits = info->new_credits_offered;
76738c8a9a5SSteve French info->new_credits_offered = 0;
76838c8a9a5SSteve French spin_unlock(&info->lock_new_credits_offered);
76938c8a9a5SSteve French
77038c8a9a5SSteve French return new_credits;
77138c8a9a5SSteve French }
77238c8a9a5SSteve French
77338c8a9a5SSteve French /*
77438c8a9a5SSteve French * Check if we need to send a KEEP_ALIVE message
77538c8a9a5SSteve French * The idle connection timer triggers a KEEP_ALIVE message when expires
77638c8a9a5SSteve French * SMB_DIRECT_RESPONSE_REQUESTED is set in the message flag to have peer send
77738c8a9a5SSteve French * back a response.
77838c8a9a5SSteve French * return value:
77938c8a9a5SSteve French * 1 if SMB_DIRECT_RESPONSE_REQUESTED needs to be set
78038c8a9a5SSteve French * 0: otherwise
78138c8a9a5SSteve French */
manage_keep_alive_before_sending(struct smbd_connection * info)78238c8a9a5SSteve French static int manage_keep_alive_before_sending(struct smbd_connection *info)
78338c8a9a5SSteve French {
78438c8a9a5SSteve French if (info->keep_alive_requested == KEEP_ALIVE_PENDING) {
78538c8a9a5SSteve French info->keep_alive_requested = KEEP_ALIVE_SENT;
78638c8a9a5SSteve French return 1;
78738c8a9a5SSteve French }
78838c8a9a5SSteve French return 0;
78938c8a9a5SSteve French }
79038c8a9a5SSteve French
79138c8a9a5SSteve French /* Post the send request */
smbd_post_send(struct smbd_connection * info,struct smbd_request * request)79238c8a9a5SSteve French static int smbd_post_send(struct smbd_connection *info,
79338c8a9a5SSteve French struct smbd_request *request)
79438c8a9a5SSteve French {
79538c8a9a5SSteve French struct ib_send_wr send_wr;
79638c8a9a5SSteve French int rc, i;
79738c8a9a5SSteve French
79838c8a9a5SSteve French for (i = 0; i < request->num_sge; i++) {
79938c8a9a5SSteve French log_rdma_send(INFO,
80038c8a9a5SSteve French "rdma_request sge[%d] addr=0x%llx length=%u\n",
80138c8a9a5SSteve French i, request->sge[i].addr, request->sge[i].length);
80238c8a9a5SSteve French ib_dma_sync_single_for_device(
80338c8a9a5SSteve French info->id->device,
80438c8a9a5SSteve French request->sge[i].addr,
80538c8a9a5SSteve French request->sge[i].length,
80638c8a9a5SSteve French DMA_TO_DEVICE);
80738c8a9a5SSteve French }
80838c8a9a5SSteve French
80938c8a9a5SSteve French request->cqe.done = send_done;
81038c8a9a5SSteve French
81138c8a9a5SSteve French send_wr.next = NULL;
81238c8a9a5SSteve French send_wr.wr_cqe = &request->cqe;
81338c8a9a5SSteve French send_wr.sg_list = request->sge;
81438c8a9a5SSteve French send_wr.num_sge = request->num_sge;
81538c8a9a5SSteve French send_wr.opcode = IB_WR_SEND;
81638c8a9a5SSteve French send_wr.send_flags = IB_SEND_SIGNALED;
81738c8a9a5SSteve French
81838c8a9a5SSteve French rc = ib_post_send(info->id->qp, &send_wr, NULL);
81938c8a9a5SSteve French if (rc) {
82038c8a9a5SSteve French log_rdma_send(ERR, "ib_post_send failed rc=%d\n", rc);
82138c8a9a5SSteve French smbd_disconnect_rdma_connection(info);
82238c8a9a5SSteve French rc = -EAGAIN;
82338c8a9a5SSteve French } else
82438c8a9a5SSteve French /* Reset timer for idle connection after packet is sent */
82538c8a9a5SSteve French mod_delayed_work(info->workqueue, &info->idle_timer_work,
82638c8a9a5SSteve French info->keep_alive_interval*HZ);
82738c8a9a5SSteve French
82838c8a9a5SSteve French return rc;
82938c8a9a5SSteve French }
83038c8a9a5SSteve French
smbd_post_send_iter(struct smbd_connection * info,struct iov_iter * iter,int * _remaining_data_length)83138c8a9a5SSteve French static int smbd_post_send_iter(struct smbd_connection *info,
83238c8a9a5SSteve French struct iov_iter *iter,
83338c8a9a5SSteve French int *_remaining_data_length)
83438c8a9a5SSteve French {
83538c8a9a5SSteve French int i, rc;
83638c8a9a5SSteve French int header_length;
83738c8a9a5SSteve French int data_length;
83838c8a9a5SSteve French struct smbd_request *request;
83938c8a9a5SSteve French struct smbd_data_transfer *packet;
84038c8a9a5SSteve French int new_credits = 0;
84138c8a9a5SSteve French
84238c8a9a5SSteve French wait_credit:
84338c8a9a5SSteve French /* Wait for send credits. A SMBD packet needs one credit */
84438c8a9a5SSteve French rc = wait_event_interruptible(info->wait_send_queue,
84538c8a9a5SSteve French atomic_read(&info->send_credits) > 0 ||
84638c8a9a5SSteve French info->transport_status != SMBD_CONNECTED);
84738c8a9a5SSteve French if (rc)
84838c8a9a5SSteve French goto err_wait_credit;
84938c8a9a5SSteve French
85038c8a9a5SSteve French if (info->transport_status != SMBD_CONNECTED) {
85138c8a9a5SSteve French log_outgoing(ERR, "disconnected not sending on wait_credit\n");
85238c8a9a5SSteve French rc = -EAGAIN;
85338c8a9a5SSteve French goto err_wait_credit;
85438c8a9a5SSteve French }
85538c8a9a5SSteve French if (unlikely(atomic_dec_return(&info->send_credits) < 0)) {
85638c8a9a5SSteve French atomic_inc(&info->send_credits);
85738c8a9a5SSteve French goto wait_credit;
85838c8a9a5SSteve French }
85938c8a9a5SSteve French
86038c8a9a5SSteve French wait_send_queue:
86138c8a9a5SSteve French wait_event(info->wait_post_send,
86238c8a9a5SSteve French atomic_read(&info->send_pending) < info->send_credit_target ||
86338c8a9a5SSteve French info->transport_status != SMBD_CONNECTED);
86438c8a9a5SSteve French
86538c8a9a5SSteve French if (info->transport_status != SMBD_CONNECTED) {
86638c8a9a5SSteve French log_outgoing(ERR, "disconnected not sending on wait_send_queue\n");
86738c8a9a5SSteve French rc = -EAGAIN;
86838c8a9a5SSteve French goto err_wait_send_queue;
86938c8a9a5SSteve French }
87038c8a9a5SSteve French
87138c8a9a5SSteve French if (unlikely(atomic_inc_return(&info->send_pending) >
87238c8a9a5SSteve French info->send_credit_target)) {
87338c8a9a5SSteve French atomic_dec(&info->send_pending);
87438c8a9a5SSteve French goto wait_send_queue;
87538c8a9a5SSteve French }
87638c8a9a5SSteve French
87738c8a9a5SSteve French request = mempool_alloc(info->request_mempool, GFP_KERNEL);
87838c8a9a5SSteve French if (!request) {
87938c8a9a5SSteve French rc = -ENOMEM;
88038c8a9a5SSteve French goto err_alloc;
88138c8a9a5SSteve French }
88238c8a9a5SSteve French
88338c8a9a5SSteve French request->info = info;
88438c8a9a5SSteve French memset(request->sge, 0, sizeof(request->sge));
88538c8a9a5SSteve French
88638c8a9a5SSteve French /* Fill in the data payload to find out how much data we can add */
88738c8a9a5SSteve French if (iter) {
88838c8a9a5SSteve French struct smb_extract_to_rdma extract = {
88938c8a9a5SSteve French .nr_sge = 1,
89038c8a9a5SSteve French .max_sge = SMBDIRECT_MAX_SEND_SGE,
89138c8a9a5SSteve French .sge = request->sge,
89238c8a9a5SSteve French .device = info->id->device,
89338c8a9a5SSteve French .local_dma_lkey = info->pd->local_dma_lkey,
89438c8a9a5SSteve French .direction = DMA_TO_DEVICE,
89538c8a9a5SSteve French };
89638c8a9a5SSteve French
89738c8a9a5SSteve French rc = smb_extract_iter_to_rdma(iter, *_remaining_data_length,
89838c8a9a5SSteve French &extract);
89938c8a9a5SSteve French if (rc < 0)
90038c8a9a5SSteve French goto err_dma;
90138c8a9a5SSteve French data_length = rc;
90238c8a9a5SSteve French request->num_sge = extract.nr_sge;
90338c8a9a5SSteve French *_remaining_data_length -= data_length;
90438c8a9a5SSteve French } else {
90538c8a9a5SSteve French data_length = 0;
90638c8a9a5SSteve French request->num_sge = 1;
90738c8a9a5SSteve French }
90838c8a9a5SSteve French
90938c8a9a5SSteve French /* Fill in the packet header */
91038c8a9a5SSteve French packet = smbd_request_payload(request);
91138c8a9a5SSteve French packet->credits_requested = cpu_to_le16(info->send_credit_target);
91238c8a9a5SSteve French
91338c8a9a5SSteve French new_credits = manage_credits_prior_sending(info);
91438c8a9a5SSteve French atomic_add(new_credits, &info->receive_credits);
91538c8a9a5SSteve French packet->credits_granted = cpu_to_le16(new_credits);
91638c8a9a5SSteve French
91738c8a9a5SSteve French info->send_immediate = false;
91838c8a9a5SSteve French
91938c8a9a5SSteve French packet->flags = 0;
92038c8a9a5SSteve French if (manage_keep_alive_before_sending(info))
92138c8a9a5SSteve French packet->flags |= cpu_to_le16(SMB_DIRECT_RESPONSE_REQUESTED);
92238c8a9a5SSteve French
92338c8a9a5SSteve French packet->reserved = 0;
92438c8a9a5SSteve French if (!data_length)
92538c8a9a5SSteve French packet->data_offset = 0;
92638c8a9a5SSteve French else
92738c8a9a5SSteve French packet->data_offset = cpu_to_le32(24);
92838c8a9a5SSteve French packet->data_length = cpu_to_le32(data_length);
92938c8a9a5SSteve French packet->remaining_data_length = cpu_to_le32(*_remaining_data_length);
93038c8a9a5SSteve French packet->padding = 0;
93138c8a9a5SSteve French
93238c8a9a5SSteve French log_outgoing(INFO, "credits_requested=%d credits_granted=%d data_offset=%d data_length=%d remaining_data_length=%d\n",
93338c8a9a5SSteve French le16_to_cpu(packet->credits_requested),
93438c8a9a5SSteve French le16_to_cpu(packet->credits_granted),
93538c8a9a5SSteve French le32_to_cpu(packet->data_offset),
93638c8a9a5SSteve French le32_to_cpu(packet->data_length),
93738c8a9a5SSteve French le32_to_cpu(packet->remaining_data_length));
93838c8a9a5SSteve French
93938c8a9a5SSteve French /* Map the packet to DMA */
94038c8a9a5SSteve French header_length = sizeof(struct smbd_data_transfer);
94138c8a9a5SSteve French /* If this is a packet without payload, don't send padding */
94238c8a9a5SSteve French if (!data_length)
94338c8a9a5SSteve French header_length = offsetof(struct smbd_data_transfer, padding);
94438c8a9a5SSteve French
94538c8a9a5SSteve French request->sge[0].addr = ib_dma_map_single(info->id->device,
94638c8a9a5SSteve French (void *)packet,
94738c8a9a5SSteve French header_length,
94838c8a9a5SSteve French DMA_TO_DEVICE);
94938c8a9a5SSteve French if (ib_dma_mapping_error(info->id->device, request->sge[0].addr)) {
95038c8a9a5SSteve French rc = -EIO;
95138c8a9a5SSteve French request->sge[0].addr = 0;
95238c8a9a5SSteve French goto err_dma;
95338c8a9a5SSteve French }
95438c8a9a5SSteve French
95538c8a9a5SSteve French request->sge[0].length = header_length;
95638c8a9a5SSteve French request->sge[0].lkey = info->pd->local_dma_lkey;
95738c8a9a5SSteve French
95838c8a9a5SSteve French rc = smbd_post_send(info, request);
95938c8a9a5SSteve French if (!rc)
96038c8a9a5SSteve French return 0;
96138c8a9a5SSteve French
96238c8a9a5SSteve French err_dma:
96338c8a9a5SSteve French for (i = 0; i < request->num_sge; i++)
96438c8a9a5SSteve French if (request->sge[i].addr)
96538c8a9a5SSteve French ib_dma_unmap_single(info->id->device,
96638c8a9a5SSteve French request->sge[i].addr,
96738c8a9a5SSteve French request->sge[i].length,
96838c8a9a5SSteve French DMA_TO_DEVICE);
96938c8a9a5SSteve French mempool_free(request, info->request_mempool);
97038c8a9a5SSteve French
97138c8a9a5SSteve French /* roll back receive credits and credits to be offered */
97238c8a9a5SSteve French spin_lock(&info->lock_new_credits_offered);
97338c8a9a5SSteve French info->new_credits_offered += new_credits;
97438c8a9a5SSteve French spin_unlock(&info->lock_new_credits_offered);
97538c8a9a5SSteve French atomic_sub(new_credits, &info->receive_credits);
97638c8a9a5SSteve French
97738c8a9a5SSteve French err_alloc:
97838c8a9a5SSteve French if (atomic_dec_and_test(&info->send_pending))
97938c8a9a5SSteve French wake_up(&info->wait_send_pending);
98038c8a9a5SSteve French
98138c8a9a5SSteve French err_wait_send_queue:
98238c8a9a5SSteve French /* roll back send credits and pending */
98338c8a9a5SSteve French atomic_inc(&info->send_credits);
98438c8a9a5SSteve French
98538c8a9a5SSteve French err_wait_credit:
98638c8a9a5SSteve French return rc;
98738c8a9a5SSteve French }
98838c8a9a5SSteve French
98938c8a9a5SSteve French /*
99038c8a9a5SSteve French * Send an empty message
99138c8a9a5SSteve French * Empty message is used to extend credits to peer to for keep live
99238c8a9a5SSteve French * while there is no upper layer payload to send at the time
99338c8a9a5SSteve French */
smbd_post_send_empty(struct smbd_connection * info)99438c8a9a5SSteve French static int smbd_post_send_empty(struct smbd_connection *info)
99538c8a9a5SSteve French {
99638c8a9a5SSteve French int remaining_data_length = 0;
99738c8a9a5SSteve French
99838c8a9a5SSteve French info->count_send_empty++;
99938c8a9a5SSteve French return smbd_post_send_iter(info, NULL, &remaining_data_length);
100038c8a9a5SSteve French }
100138c8a9a5SSteve French
100238c8a9a5SSteve French /*
100338c8a9a5SSteve French * Post a receive request to the transport
100438c8a9a5SSteve French * The remote peer can only send data when a receive request is posted
100538c8a9a5SSteve French * The interaction is controlled by send/receive credit system
100638c8a9a5SSteve French */
smbd_post_recv(struct smbd_connection * info,struct smbd_response * response)100738c8a9a5SSteve French static int smbd_post_recv(
100838c8a9a5SSteve French struct smbd_connection *info, struct smbd_response *response)
100938c8a9a5SSteve French {
101038c8a9a5SSteve French struct ib_recv_wr recv_wr;
101138c8a9a5SSteve French int rc = -EIO;
101238c8a9a5SSteve French
101338c8a9a5SSteve French response->sge.addr = ib_dma_map_single(
101438c8a9a5SSteve French info->id->device, response->packet,
101538c8a9a5SSteve French info->max_receive_size, DMA_FROM_DEVICE);
101638c8a9a5SSteve French if (ib_dma_mapping_error(info->id->device, response->sge.addr))
101738c8a9a5SSteve French return rc;
101838c8a9a5SSteve French
101938c8a9a5SSteve French response->sge.length = info->max_receive_size;
102038c8a9a5SSteve French response->sge.lkey = info->pd->local_dma_lkey;
102138c8a9a5SSteve French
102238c8a9a5SSteve French response->cqe.done = recv_done;
102338c8a9a5SSteve French
102438c8a9a5SSteve French recv_wr.wr_cqe = &response->cqe;
102538c8a9a5SSteve French recv_wr.next = NULL;
102638c8a9a5SSteve French recv_wr.sg_list = &response->sge;
102738c8a9a5SSteve French recv_wr.num_sge = 1;
102838c8a9a5SSteve French
102938c8a9a5SSteve French rc = ib_post_recv(info->id->qp, &recv_wr, NULL);
103038c8a9a5SSteve French if (rc) {
103138c8a9a5SSteve French ib_dma_unmap_single(info->id->device, response->sge.addr,
103238c8a9a5SSteve French response->sge.length, DMA_FROM_DEVICE);
103338c8a9a5SSteve French smbd_disconnect_rdma_connection(info);
103438c8a9a5SSteve French log_rdma_recv(ERR, "ib_post_recv failed rc=%d\n", rc);
103538c8a9a5SSteve French }
103638c8a9a5SSteve French
103738c8a9a5SSteve French return rc;
103838c8a9a5SSteve French }
103938c8a9a5SSteve French
104038c8a9a5SSteve French /* Perform SMBD negotiate according to [MS-SMBD] 3.1.5.2 */
smbd_negotiate(struct smbd_connection * info)104138c8a9a5SSteve French static int smbd_negotiate(struct smbd_connection *info)
104238c8a9a5SSteve French {
104338c8a9a5SSteve French int rc;
104438c8a9a5SSteve French struct smbd_response *response = get_receive_buffer(info);
104538c8a9a5SSteve French
104638c8a9a5SSteve French response->type = SMBD_NEGOTIATE_RESP;
104738c8a9a5SSteve French rc = smbd_post_recv(info, response);
104838c8a9a5SSteve French log_rdma_event(INFO, "smbd_post_recv rc=%d iov.addr=0x%llx iov.length=%u iov.lkey=0x%x\n",
104938c8a9a5SSteve French rc, response->sge.addr,
105038c8a9a5SSteve French response->sge.length, response->sge.lkey);
105138c8a9a5SSteve French if (rc)
105238c8a9a5SSteve French return rc;
105338c8a9a5SSteve French
105438c8a9a5SSteve French init_completion(&info->negotiate_completion);
105538c8a9a5SSteve French info->negotiate_done = false;
105638c8a9a5SSteve French rc = smbd_post_send_negotiate_req(info);
105738c8a9a5SSteve French if (rc)
105838c8a9a5SSteve French return rc;
105938c8a9a5SSteve French
106038c8a9a5SSteve French rc = wait_for_completion_interruptible_timeout(
106138c8a9a5SSteve French &info->negotiate_completion, SMBD_NEGOTIATE_TIMEOUT * HZ);
106238c8a9a5SSteve French log_rdma_event(INFO, "wait_for_completion_timeout rc=%d\n", rc);
106338c8a9a5SSteve French
106438c8a9a5SSteve French if (info->negotiate_done)
106538c8a9a5SSteve French return 0;
106638c8a9a5SSteve French
106738c8a9a5SSteve French if (rc == 0)
106838c8a9a5SSteve French rc = -ETIMEDOUT;
106938c8a9a5SSteve French else if (rc == -ERESTARTSYS)
107038c8a9a5SSteve French rc = -EINTR;
107138c8a9a5SSteve French else
107238c8a9a5SSteve French rc = -ENOTCONN;
107338c8a9a5SSteve French
107438c8a9a5SSteve French return rc;
107538c8a9a5SSteve French }
107638c8a9a5SSteve French
put_empty_packet(struct smbd_connection * info,struct smbd_response * response)107738c8a9a5SSteve French static void put_empty_packet(
107838c8a9a5SSteve French struct smbd_connection *info, struct smbd_response *response)
107938c8a9a5SSteve French {
108038c8a9a5SSteve French spin_lock(&info->empty_packet_queue_lock);
108138c8a9a5SSteve French list_add_tail(&response->list, &info->empty_packet_queue);
108238c8a9a5SSteve French info->count_empty_packet_queue++;
108338c8a9a5SSteve French spin_unlock(&info->empty_packet_queue_lock);
108438c8a9a5SSteve French
108538c8a9a5SSteve French queue_work(info->workqueue, &info->post_send_credits_work);
108638c8a9a5SSteve French }
108738c8a9a5SSteve French
108838c8a9a5SSteve French /*
108938c8a9a5SSteve French * Implement Connection.FragmentReassemblyBuffer defined in [MS-SMBD] 3.1.1.1
109038c8a9a5SSteve French * This is a queue for reassembling upper layer payload and present to upper
109138c8a9a5SSteve French * layer. All the inncoming payload go to the reassembly queue, regardless of
109238c8a9a5SSteve French * if reassembly is required. The uuper layer code reads from the queue for all
109338c8a9a5SSteve French * incoming payloads.
109438c8a9a5SSteve French * Put a received packet to the reassembly queue
109538c8a9a5SSteve French * response: the packet received
109638c8a9a5SSteve French * data_length: the size of payload in this packet
109738c8a9a5SSteve French */
enqueue_reassembly(struct smbd_connection * info,struct smbd_response * response,int data_length)109838c8a9a5SSteve French static void enqueue_reassembly(
109938c8a9a5SSteve French struct smbd_connection *info,
110038c8a9a5SSteve French struct smbd_response *response,
110138c8a9a5SSteve French int data_length)
110238c8a9a5SSteve French {
110338c8a9a5SSteve French spin_lock(&info->reassembly_queue_lock);
110438c8a9a5SSteve French list_add_tail(&response->list, &info->reassembly_queue);
110538c8a9a5SSteve French info->reassembly_queue_length++;
110638c8a9a5SSteve French /*
110738c8a9a5SSteve French * Make sure reassembly_data_length is updated after list and
110838c8a9a5SSteve French * reassembly_queue_length are updated. On the dequeue side
110938c8a9a5SSteve French * reassembly_data_length is checked without a lock to determine
111038c8a9a5SSteve French * if reassembly_queue_length and list is up to date
111138c8a9a5SSteve French */
111238c8a9a5SSteve French virt_wmb();
111338c8a9a5SSteve French info->reassembly_data_length += data_length;
111438c8a9a5SSteve French spin_unlock(&info->reassembly_queue_lock);
111538c8a9a5SSteve French info->count_reassembly_queue++;
111638c8a9a5SSteve French info->count_enqueue_reassembly_queue++;
111738c8a9a5SSteve French }
111838c8a9a5SSteve French
111938c8a9a5SSteve French /*
112038c8a9a5SSteve French * Get the first entry at the front of reassembly queue
112138c8a9a5SSteve French * Caller is responsible for locking
112238c8a9a5SSteve French * return value: the first entry if any, NULL if queue is empty
112338c8a9a5SSteve French */
_get_first_reassembly(struct smbd_connection * info)112438c8a9a5SSteve French static struct smbd_response *_get_first_reassembly(struct smbd_connection *info)
112538c8a9a5SSteve French {
112638c8a9a5SSteve French struct smbd_response *ret = NULL;
112738c8a9a5SSteve French
112838c8a9a5SSteve French if (!list_empty(&info->reassembly_queue)) {
112938c8a9a5SSteve French ret = list_first_entry(
113038c8a9a5SSteve French &info->reassembly_queue,
113138c8a9a5SSteve French struct smbd_response, list);
113238c8a9a5SSteve French }
113338c8a9a5SSteve French return ret;
113438c8a9a5SSteve French }
113538c8a9a5SSteve French
get_empty_queue_buffer(struct smbd_connection * info)113638c8a9a5SSteve French static struct smbd_response *get_empty_queue_buffer(
113738c8a9a5SSteve French struct smbd_connection *info)
113838c8a9a5SSteve French {
113938c8a9a5SSteve French struct smbd_response *ret = NULL;
114038c8a9a5SSteve French unsigned long flags;
114138c8a9a5SSteve French
114238c8a9a5SSteve French spin_lock_irqsave(&info->empty_packet_queue_lock, flags);
114338c8a9a5SSteve French if (!list_empty(&info->empty_packet_queue)) {
114438c8a9a5SSteve French ret = list_first_entry(
114538c8a9a5SSteve French &info->empty_packet_queue,
114638c8a9a5SSteve French struct smbd_response, list);
114738c8a9a5SSteve French list_del(&ret->list);
114838c8a9a5SSteve French info->count_empty_packet_queue--;
114938c8a9a5SSteve French }
115038c8a9a5SSteve French spin_unlock_irqrestore(&info->empty_packet_queue_lock, flags);
115138c8a9a5SSteve French
115238c8a9a5SSteve French return ret;
115338c8a9a5SSteve French }
115438c8a9a5SSteve French
115538c8a9a5SSteve French /*
115638c8a9a5SSteve French * Get a receive buffer
115738c8a9a5SSteve French * For each remote send, we need to post a receive. The receive buffers are
115838c8a9a5SSteve French * pre-allocated in advance.
115938c8a9a5SSteve French * return value: the receive buffer, NULL if none is available
116038c8a9a5SSteve French */
get_receive_buffer(struct smbd_connection * info)116138c8a9a5SSteve French static struct smbd_response *get_receive_buffer(struct smbd_connection *info)
116238c8a9a5SSteve French {
116338c8a9a5SSteve French struct smbd_response *ret = NULL;
116438c8a9a5SSteve French unsigned long flags;
116538c8a9a5SSteve French
116638c8a9a5SSteve French spin_lock_irqsave(&info->receive_queue_lock, flags);
116738c8a9a5SSteve French if (!list_empty(&info->receive_queue)) {
116838c8a9a5SSteve French ret = list_first_entry(
116938c8a9a5SSteve French &info->receive_queue,
117038c8a9a5SSteve French struct smbd_response, list);
117138c8a9a5SSteve French list_del(&ret->list);
117238c8a9a5SSteve French info->count_receive_queue--;
117338c8a9a5SSteve French info->count_get_receive_buffer++;
117438c8a9a5SSteve French }
117538c8a9a5SSteve French spin_unlock_irqrestore(&info->receive_queue_lock, flags);
117638c8a9a5SSteve French
117738c8a9a5SSteve French return ret;
117838c8a9a5SSteve French }
117938c8a9a5SSteve French
118038c8a9a5SSteve French /*
118138c8a9a5SSteve French * Return a receive buffer
118238c8a9a5SSteve French * Upon returning of a receive buffer, we can post new receive and extend
118338c8a9a5SSteve French * more receive credits to remote peer. This is done immediately after a
118438c8a9a5SSteve French * receive buffer is returned.
118538c8a9a5SSteve French */
put_receive_buffer(struct smbd_connection * info,struct smbd_response * response)118638c8a9a5SSteve French static void put_receive_buffer(
118738c8a9a5SSteve French struct smbd_connection *info, struct smbd_response *response)
118838c8a9a5SSteve French {
118938c8a9a5SSteve French unsigned long flags;
119038c8a9a5SSteve French
119138c8a9a5SSteve French ib_dma_unmap_single(info->id->device, response->sge.addr,
119238c8a9a5SSteve French response->sge.length, DMA_FROM_DEVICE);
119338c8a9a5SSteve French
119438c8a9a5SSteve French spin_lock_irqsave(&info->receive_queue_lock, flags);
119538c8a9a5SSteve French list_add_tail(&response->list, &info->receive_queue);
119638c8a9a5SSteve French info->count_receive_queue++;
119738c8a9a5SSteve French info->count_put_receive_buffer++;
119838c8a9a5SSteve French spin_unlock_irqrestore(&info->receive_queue_lock, flags);
119938c8a9a5SSteve French
120038c8a9a5SSteve French queue_work(info->workqueue, &info->post_send_credits_work);
120138c8a9a5SSteve French }
120238c8a9a5SSteve French
120338c8a9a5SSteve French /* Preallocate all receive buffer on transport establishment */
allocate_receive_buffers(struct smbd_connection * info,int num_buf)120438c8a9a5SSteve French static int allocate_receive_buffers(struct smbd_connection *info, int num_buf)
120538c8a9a5SSteve French {
120638c8a9a5SSteve French int i;
120738c8a9a5SSteve French struct smbd_response *response;
120838c8a9a5SSteve French
120938c8a9a5SSteve French INIT_LIST_HEAD(&info->reassembly_queue);
121038c8a9a5SSteve French spin_lock_init(&info->reassembly_queue_lock);
121138c8a9a5SSteve French info->reassembly_data_length = 0;
121238c8a9a5SSteve French info->reassembly_queue_length = 0;
121338c8a9a5SSteve French
121438c8a9a5SSteve French INIT_LIST_HEAD(&info->receive_queue);
121538c8a9a5SSteve French spin_lock_init(&info->receive_queue_lock);
121638c8a9a5SSteve French info->count_receive_queue = 0;
121738c8a9a5SSteve French
121838c8a9a5SSteve French INIT_LIST_HEAD(&info->empty_packet_queue);
121938c8a9a5SSteve French spin_lock_init(&info->empty_packet_queue_lock);
122038c8a9a5SSteve French info->count_empty_packet_queue = 0;
122138c8a9a5SSteve French
122238c8a9a5SSteve French init_waitqueue_head(&info->wait_receive_queues);
122338c8a9a5SSteve French
122438c8a9a5SSteve French for (i = 0; i < num_buf; i++) {
122538c8a9a5SSteve French response = mempool_alloc(info->response_mempool, GFP_KERNEL);
122638c8a9a5SSteve French if (!response)
122738c8a9a5SSteve French goto allocate_failed;
122838c8a9a5SSteve French
122938c8a9a5SSteve French response->info = info;
123038c8a9a5SSteve French list_add_tail(&response->list, &info->receive_queue);
123138c8a9a5SSteve French info->count_receive_queue++;
123238c8a9a5SSteve French }
123338c8a9a5SSteve French
123438c8a9a5SSteve French return 0;
123538c8a9a5SSteve French
123638c8a9a5SSteve French allocate_failed:
123738c8a9a5SSteve French while (!list_empty(&info->receive_queue)) {
123838c8a9a5SSteve French response = list_first_entry(
123938c8a9a5SSteve French &info->receive_queue,
124038c8a9a5SSteve French struct smbd_response, list);
124138c8a9a5SSteve French list_del(&response->list);
124238c8a9a5SSteve French info->count_receive_queue--;
124338c8a9a5SSteve French
124438c8a9a5SSteve French mempool_free(response, info->response_mempool);
124538c8a9a5SSteve French }
124638c8a9a5SSteve French return -ENOMEM;
124738c8a9a5SSteve French }
124838c8a9a5SSteve French
destroy_receive_buffers(struct smbd_connection * info)124938c8a9a5SSteve French static void destroy_receive_buffers(struct smbd_connection *info)
125038c8a9a5SSteve French {
125138c8a9a5SSteve French struct smbd_response *response;
125238c8a9a5SSteve French
125338c8a9a5SSteve French while ((response = get_receive_buffer(info)))
125438c8a9a5SSteve French mempool_free(response, info->response_mempool);
125538c8a9a5SSteve French
125638c8a9a5SSteve French while ((response = get_empty_queue_buffer(info)))
125738c8a9a5SSteve French mempool_free(response, info->response_mempool);
125838c8a9a5SSteve French }
125938c8a9a5SSteve French
126038c8a9a5SSteve French /* Implement idle connection timer [MS-SMBD] 3.1.6.2 */
idle_connection_timer(struct work_struct * work)126138c8a9a5SSteve French static void idle_connection_timer(struct work_struct *work)
126238c8a9a5SSteve French {
126338c8a9a5SSteve French struct smbd_connection *info = container_of(
126438c8a9a5SSteve French work, struct smbd_connection,
126538c8a9a5SSteve French idle_timer_work.work);
126638c8a9a5SSteve French
126738c8a9a5SSteve French if (info->keep_alive_requested != KEEP_ALIVE_NONE) {
126838c8a9a5SSteve French log_keep_alive(ERR,
126938c8a9a5SSteve French "error status info->keep_alive_requested=%d\n",
127038c8a9a5SSteve French info->keep_alive_requested);
127138c8a9a5SSteve French smbd_disconnect_rdma_connection(info);
127238c8a9a5SSteve French return;
127338c8a9a5SSteve French }
127438c8a9a5SSteve French
127538c8a9a5SSteve French log_keep_alive(INFO, "about to send an empty idle message\n");
127638c8a9a5SSteve French smbd_post_send_empty(info);
127738c8a9a5SSteve French
127838c8a9a5SSteve French /* Setup the next idle timeout work */
127938c8a9a5SSteve French queue_delayed_work(info->workqueue, &info->idle_timer_work,
128038c8a9a5SSteve French info->keep_alive_interval*HZ);
128138c8a9a5SSteve French }
128238c8a9a5SSteve French
128338c8a9a5SSteve French /*
128438c8a9a5SSteve French * Destroy the transport and related RDMA and memory resources
128538c8a9a5SSteve French * Need to go through all the pending counters and make sure on one is using
128638c8a9a5SSteve French * the transport while it is destroyed
128738c8a9a5SSteve French */
smbd_destroy(struct TCP_Server_Info * server)128838c8a9a5SSteve French void smbd_destroy(struct TCP_Server_Info *server)
128938c8a9a5SSteve French {
129038c8a9a5SSteve French struct smbd_connection *info = server->smbd_conn;
129138c8a9a5SSteve French struct smbd_response *response;
129238c8a9a5SSteve French unsigned long flags;
129338c8a9a5SSteve French
129438c8a9a5SSteve French if (!info) {
129538c8a9a5SSteve French log_rdma_event(INFO, "rdma session already destroyed\n");
129638c8a9a5SSteve French return;
129738c8a9a5SSteve French }
129838c8a9a5SSteve French
129938c8a9a5SSteve French log_rdma_event(INFO, "destroying rdma session\n");
130038c8a9a5SSteve French if (info->transport_status != SMBD_DISCONNECTED) {
130138c8a9a5SSteve French rdma_disconnect(server->smbd_conn->id);
130238c8a9a5SSteve French log_rdma_event(INFO, "wait for transport being disconnected\n");
130338c8a9a5SSteve French wait_event_interruptible(
130438c8a9a5SSteve French info->disconn_wait,
130538c8a9a5SSteve French info->transport_status == SMBD_DISCONNECTED);
130638c8a9a5SSteve French }
130738c8a9a5SSteve French
130838c8a9a5SSteve French log_rdma_event(INFO, "destroying qp\n");
130938c8a9a5SSteve French ib_drain_qp(info->id->qp);
131038c8a9a5SSteve French rdma_destroy_qp(info->id);
131138c8a9a5SSteve French
131238c8a9a5SSteve French log_rdma_event(INFO, "cancelling idle timer\n");
131338c8a9a5SSteve French cancel_delayed_work_sync(&info->idle_timer_work);
131438c8a9a5SSteve French
131538c8a9a5SSteve French log_rdma_event(INFO, "wait for all send posted to IB to finish\n");
131638c8a9a5SSteve French wait_event(info->wait_send_pending,
131738c8a9a5SSteve French atomic_read(&info->send_pending) == 0);
131838c8a9a5SSteve French
131938c8a9a5SSteve French /* It's not possible for upper layer to get to reassembly */
132038c8a9a5SSteve French log_rdma_event(INFO, "drain the reassembly queue\n");
132138c8a9a5SSteve French do {
132238c8a9a5SSteve French spin_lock_irqsave(&info->reassembly_queue_lock, flags);
132338c8a9a5SSteve French response = _get_first_reassembly(info);
132438c8a9a5SSteve French if (response) {
132538c8a9a5SSteve French list_del(&response->list);
132638c8a9a5SSteve French spin_unlock_irqrestore(
132738c8a9a5SSteve French &info->reassembly_queue_lock, flags);
132838c8a9a5SSteve French put_receive_buffer(info, response);
132938c8a9a5SSteve French } else
133038c8a9a5SSteve French spin_unlock_irqrestore(
133138c8a9a5SSteve French &info->reassembly_queue_lock, flags);
133238c8a9a5SSteve French } while (response);
133338c8a9a5SSteve French info->reassembly_data_length = 0;
133438c8a9a5SSteve French
133538c8a9a5SSteve French log_rdma_event(INFO, "free receive buffers\n");
133638c8a9a5SSteve French wait_event(info->wait_receive_queues,
133738c8a9a5SSteve French info->count_receive_queue + info->count_empty_packet_queue
133838c8a9a5SSteve French == info->receive_credit_max);
133938c8a9a5SSteve French destroy_receive_buffers(info);
134038c8a9a5SSteve French
134138c8a9a5SSteve French /*
134238c8a9a5SSteve French * For performance reasons, memory registration and deregistration
134338c8a9a5SSteve French * are not locked by srv_mutex. It is possible some processes are
134438c8a9a5SSteve French * blocked on transport srv_mutex while holding memory registration.
134538c8a9a5SSteve French * Release the transport srv_mutex to allow them to hit the failure
134638c8a9a5SSteve French * path when sending data, and then release memory registartions.
134738c8a9a5SSteve French */
134838c8a9a5SSteve French log_rdma_event(INFO, "freeing mr list\n");
134938c8a9a5SSteve French wake_up_interruptible_all(&info->wait_mr);
135038c8a9a5SSteve French while (atomic_read(&info->mr_used_count)) {
135138c8a9a5SSteve French cifs_server_unlock(server);
135238c8a9a5SSteve French msleep(1000);
135338c8a9a5SSteve French cifs_server_lock(server);
135438c8a9a5SSteve French }
135538c8a9a5SSteve French destroy_mr_list(info);
135638c8a9a5SSteve French
135738c8a9a5SSteve French ib_free_cq(info->send_cq);
135838c8a9a5SSteve French ib_free_cq(info->recv_cq);
135938c8a9a5SSteve French ib_dealloc_pd(info->pd);
136038c8a9a5SSteve French rdma_destroy_id(info->id);
136138c8a9a5SSteve French
136238c8a9a5SSteve French /* free mempools */
136338c8a9a5SSteve French mempool_destroy(info->request_mempool);
136438c8a9a5SSteve French kmem_cache_destroy(info->request_cache);
136538c8a9a5SSteve French
136638c8a9a5SSteve French mempool_destroy(info->response_mempool);
136738c8a9a5SSteve French kmem_cache_destroy(info->response_cache);
136838c8a9a5SSteve French
136938c8a9a5SSteve French info->transport_status = SMBD_DESTROYED;
137038c8a9a5SSteve French
137138c8a9a5SSteve French destroy_workqueue(info->workqueue);
137238c8a9a5SSteve French log_rdma_event(INFO, "rdma session destroyed\n");
137338c8a9a5SSteve French kfree(info);
137438c8a9a5SSteve French server->smbd_conn = NULL;
137538c8a9a5SSteve French }
137638c8a9a5SSteve French
137738c8a9a5SSteve French /*
137838c8a9a5SSteve French * Reconnect this SMBD connection, called from upper layer
137938c8a9a5SSteve French * return value: 0 on success, or actual error code
138038c8a9a5SSteve French */
smbd_reconnect(struct TCP_Server_Info * server)138138c8a9a5SSteve French int smbd_reconnect(struct TCP_Server_Info *server)
138238c8a9a5SSteve French {
138338c8a9a5SSteve French log_rdma_event(INFO, "reconnecting rdma session\n");
138438c8a9a5SSteve French
138538c8a9a5SSteve French if (!server->smbd_conn) {
138638c8a9a5SSteve French log_rdma_event(INFO, "rdma session already destroyed\n");
138738c8a9a5SSteve French goto create_conn;
138838c8a9a5SSteve French }
138938c8a9a5SSteve French
139038c8a9a5SSteve French /*
139138c8a9a5SSteve French * This is possible if transport is disconnected and we haven't received
139238c8a9a5SSteve French * notification from RDMA, but upper layer has detected timeout
139338c8a9a5SSteve French */
139438c8a9a5SSteve French if (server->smbd_conn->transport_status == SMBD_CONNECTED) {
139538c8a9a5SSteve French log_rdma_event(INFO, "disconnecting transport\n");
139638c8a9a5SSteve French smbd_destroy(server);
139738c8a9a5SSteve French }
139838c8a9a5SSteve French
139938c8a9a5SSteve French create_conn:
140038c8a9a5SSteve French log_rdma_event(INFO, "creating rdma session\n");
140138c8a9a5SSteve French server->smbd_conn = smbd_get_connection(
140238c8a9a5SSteve French server, (struct sockaddr *) &server->dstaddr);
140338c8a9a5SSteve French
1404e3603ccfSSteve French if (server->smbd_conn) {
140538c8a9a5SSteve French cifs_dbg(VFS, "RDMA transport re-established\n");
1406e3603ccfSSteve French trace_smb3_smbd_connect_done(server->hostname, server->conn_id, &server->dstaddr);
1407e3603ccfSSteve French return 0;
1408e3603ccfSSteve French }
1409e3603ccfSSteve French trace_smb3_smbd_connect_err(server->hostname, server->conn_id, &server->dstaddr);
1410e3603ccfSSteve French return -ENOENT;
141138c8a9a5SSteve French }
141238c8a9a5SSteve French
destroy_caches_and_workqueue(struct smbd_connection * info)141338c8a9a5SSteve French static void destroy_caches_and_workqueue(struct smbd_connection *info)
141438c8a9a5SSteve French {
141538c8a9a5SSteve French destroy_receive_buffers(info);
141638c8a9a5SSteve French destroy_workqueue(info->workqueue);
141738c8a9a5SSteve French mempool_destroy(info->response_mempool);
141838c8a9a5SSteve French kmem_cache_destroy(info->response_cache);
141938c8a9a5SSteve French mempool_destroy(info->request_mempool);
142038c8a9a5SSteve French kmem_cache_destroy(info->request_cache);
142138c8a9a5SSteve French }
142238c8a9a5SSteve French
142338c8a9a5SSteve French #define MAX_NAME_LEN 80
allocate_caches_and_workqueue(struct smbd_connection * info)142438c8a9a5SSteve French static int allocate_caches_and_workqueue(struct smbd_connection *info)
142538c8a9a5SSteve French {
142638c8a9a5SSteve French char name[MAX_NAME_LEN];
142738c8a9a5SSteve French int rc;
142838c8a9a5SSteve French
142938c8a9a5SSteve French scnprintf(name, MAX_NAME_LEN, "smbd_request_%p", info);
143038c8a9a5SSteve French info->request_cache =
143138c8a9a5SSteve French kmem_cache_create(
143238c8a9a5SSteve French name,
143338c8a9a5SSteve French sizeof(struct smbd_request) +
143438c8a9a5SSteve French sizeof(struct smbd_data_transfer),
143538c8a9a5SSteve French 0, SLAB_HWCACHE_ALIGN, NULL);
143638c8a9a5SSteve French if (!info->request_cache)
143738c8a9a5SSteve French return -ENOMEM;
143838c8a9a5SSteve French
143938c8a9a5SSteve French info->request_mempool =
144038c8a9a5SSteve French mempool_create(info->send_credit_target, mempool_alloc_slab,
144138c8a9a5SSteve French mempool_free_slab, info->request_cache);
144238c8a9a5SSteve French if (!info->request_mempool)
144338c8a9a5SSteve French goto out1;
144438c8a9a5SSteve French
144538c8a9a5SSteve French scnprintf(name, MAX_NAME_LEN, "smbd_response_%p", info);
144638c8a9a5SSteve French info->response_cache =
144738c8a9a5SSteve French kmem_cache_create(
144838c8a9a5SSteve French name,
144938c8a9a5SSteve French sizeof(struct smbd_response) +
145038c8a9a5SSteve French info->max_receive_size,
145138c8a9a5SSteve French 0, SLAB_HWCACHE_ALIGN, NULL);
145238c8a9a5SSteve French if (!info->response_cache)
145338c8a9a5SSteve French goto out2;
145438c8a9a5SSteve French
145538c8a9a5SSteve French info->response_mempool =
145638c8a9a5SSteve French mempool_create(info->receive_credit_max, mempool_alloc_slab,
145738c8a9a5SSteve French mempool_free_slab, info->response_cache);
145838c8a9a5SSteve French if (!info->response_mempool)
145938c8a9a5SSteve French goto out3;
146038c8a9a5SSteve French
146138c8a9a5SSteve French scnprintf(name, MAX_NAME_LEN, "smbd_%p", info);
146238c8a9a5SSteve French info->workqueue = create_workqueue(name);
146338c8a9a5SSteve French if (!info->workqueue)
146438c8a9a5SSteve French goto out4;
146538c8a9a5SSteve French
146638c8a9a5SSteve French rc = allocate_receive_buffers(info, info->receive_credit_max);
146738c8a9a5SSteve French if (rc) {
146838c8a9a5SSteve French log_rdma_event(ERR, "failed to allocate receive buffers\n");
146938c8a9a5SSteve French goto out5;
147038c8a9a5SSteve French }
147138c8a9a5SSteve French
147238c8a9a5SSteve French return 0;
147338c8a9a5SSteve French
147438c8a9a5SSteve French out5:
147538c8a9a5SSteve French destroy_workqueue(info->workqueue);
147638c8a9a5SSteve French out4:
147738c8a9a5SSteve French mempool_destroy(info->response_mempool);
147838c8a9a5SSteve French out3:
147938c8a9a5SSteve French kmem_cache_destroy(info->response_cache);
148038c8a9a5SSteve French out2:
148138c8a9a5SSteve French mempool_destroy(info->request_mempool);
148238c8a9a5SSteve French out1:
148338c8a9a5SSteve French kmem_cache_destroy(info->request_cache);
148438c8a9a5SSteve French return -ENOMEM;
148538c8a9a5SSteve French }
148638c8a9a5SSteve French
148738c8a9a5SSteve French /* Create a SMBD connection, called by upper layer */
_smbd_get_connection(struct TCP_Server_Info * server,struct sockaddr * dstaddr,int port)148838c8a9a5SSteve French static struct smbd_connection *_smbd_get_connection(
148938c8a9a5SSteve French struct TCP_Server_Info *server, struct sockaddr *dstaddr, int port)
149038c8a9a5SSteve French {
149138c8a9a5SSteve French int rc;
149238c8a9a5SSteve French struct smbd_connection *info;
149338c8a9a5SSteve French struct rdma_conn_param conn_param;
149438c8a9a5SSteve French struct ib_qp_init_attr qp_attr;
149538c8a9a5SSteve French struct sockaddr_in *addr_in = (struct sockaddr_in *) dstaddr;
149638c8a9a5SSteve French struct ib_port_immutable port_immutable;
149738c8a9a5SSteve French u32 ird_ord_hdr[2];
149838c8a9a5SSteve French
149938c8a9a5SSteve French info = kzalloc(sizeof(struct smbd_connection), GFP_KERNEL);
150038c8a9a5SSteve French if (!info)
150138c8a9a5SSteve French return NULL;
150238c8a9a5SSteve French
150338c8a9a5SSteve French info->transport_status = SMBD_CONNECTING;
150438c8a9a5SSteve French rc = smbd_ia_open(info, dstaddr, port);
150538c8a9a5SSteve French if (rc) {
150638c8a9a5SSteve French log_rdma_event(INFO, "smbd_ia_open rc=%d\n", rc);
150738c8a9a5SSteve French goto create_id_failed;
150838c8a9a5SSteve French }
150938c8a9a5SSteve French
151038c8a9a5SSteve French if (smbd_send_credit_target > info->id->device->attrs.max_cqe ||
151138c8a9a5SSteve French smbd_send_credit_target > info->id->device->attrs.max_qp_wr) {
151238c8a9a5SSteve French log_rdma_event(ERR, "consider lowering send_credit_target = %d. Possible CQE overrun, device reporting max_cqe %d max_qp_wr %d\n",
151338c8a9a5SSteve French smbd_send_credit_target,
151438c8a9a5SSteve French info->id->device->attrs.max_cqe,
151538c8a9a5SSteve French info->id->device->attrs.max_qp_wr);
151638c8a9a5SSteve French goto config_failed;
151738c8a9a5SSteve French }
151838c8a9a5SSteve French
151938c8a9a5SSteve French if (smbd_receive_credit_max > info->id->device->attrs.max_cqe ||
152038c8a9a5SSteve French smbd_receive_credit_max > info->id->device->attrs.max_qp_wr) {
152138c8a9a5SSteve French log_rdma_event(ERR, "consider lowering receive_credit_max = %d. Possible CQE overrun, device reporting max_cqe %d max_qp_wr %d\n",
152238c8a9a5SSteve French smbd_receive_credit_max,
152338c8a9a5SSteve French info->id->device->attrs.max_cqe,
152438c8a9a5SSteve French info->id->device->attrs.max_qp_wr);
152538c8a9a5SSteve French goto config_failed;
152638c8a9a5SSteve French }
152738c8a9a5SSteve French
152838c8a9a5SSteve French info->receive_credit_max = smbd_receive_credit_max;
152938c8a9a5SSteve French info->send_credit_target = smbd_send_credit_target;
153038c8a9a5SSteve French info->max_send_size = smbd_max_send_size;
153138c8a9a5SSteve French info->max_fragmented_recv_size = smbd_max_fragmented_recv_size;
153238c8a9a5SSteve French info->max_receive_size = smbd_max_receive_size;
153338c8a9a5SSteve French info->keep_alive_interval = smbd_keep_alive_interval;
153438c8a9a5SSteve French
153538c8a9a5SSteve French if (info->id->device->attrs.max_send_sge < SMBDIRECT_MAX_SEND_SGE ||
153638c8a9a5SSteve French info->id->device->attrs.max_recv_sge < SMBDIRECT_MAX_RECV_SGE) {
153738c8a9a5SSteve French log_rdma_event(ERR,
153838c8a9a5SSteve French "device %.*s max_send_sge/max_recv_sge = %d/%d too small\n",
153938c8a9a5SSteve French IB_DEVICE_NAME_MAX,
154038c8a9a5SSteve French info->id->device->name,
154138c8a9a5SSteve French info->id->device->attrs.max_send_sge,
154238c8a9a5SSteve French info->id->device->attrs.max_recv_sge);
154338c8a9a5SSteve French goto config_failed;
154438c8a9a5SSteve French }
154538c8a9a5SSteve French
154638c8a9a5SSteve French info->send_cq = NULL;
154738c8a9a5SSteve French info->recv_cq = NULL;
154838c8a9a5SSteve French info->send_cq =
154938c8a9a5SSteve French ib_alloc_cq_any(info->id->device, info,
155038c8a9a5SSteve French info->send_credit_target, IB_POLL_SOFTIRQ);
155138c8a9a5SSteve French if (IS_ERR(info->send_cq)) {
155238c8a9a5SSteve French info->send_cq = NULL;
155338c8a9a5SSteve French goto alloc_cq_failed;
155438c8a9a5SSteve French }
155538c8a9a5SSteve French
155638c8a9a5SSteve French info->recv_cq =
155738c8a9a5SSteve French ib_alloc_cq_any(info->id->device, info,
155838c8a9a5SSteve French info->receive_credit_max, IB_POLL_SOFTIRQ);
155938c8a9a5SSteve French if (IS_ERR(info->recv_cq)) {
156038c8a9a5SSteve French info->recv_cq = NULL;
156138c8a9a5SSteve French goto alloc_cq_failed;
156238c8a9a5SSteve French }
156338c8a9a5SSteve French
156438c8a9a5SSteve French memset(&qp_attr, 0, sizeof(qp_attr));
156538c8a9a5SSteve French qp_attr.event_handler = smbd_qp_async_error_upcall;
156638c8a9a5SSteve French qp_attr.qp_context = info;
156738c8a9a5SSteve French qp_attr.cap.max_send_wr = info->send_credit_target;
156838c8a9a5SSteve French qp_attr.cap.max_recv_wr = info->receive_credit_max;
156938c8a9a5SSteve French qp_attr.cap.max_send_sge = SMBDIRECT_MAX_SEND_SGE;
157038c8a9a5SSteve French qp_attr.cap.max_recv_sge = SMBDIRECT_MAX_RECV_SGE;
157138c8a9a5SSteve French qp_attr.cap.max_inline_data = 0;
157238c8a9a5SSteve French qp_attr.sq_sig_type = IB_SIGNAL_REQ_WR;
157338c8a9a5SSteve French qp_attr.qp_type = IB_QPT_RC;
157438c8a9a5SSteve French qp_attr.send_cq = info->send_cq;
157538c8a9a5SSteve French qp_attr.recv_cq = info->recv_cq;
157638c8a9a5SSteve French qp_attr.port_num = ~0;
157738c8a9a5SSteve French
157838c8a9a5SSteve French rc = rdma_create_qp(info->id, info->pd, &qp_attr);
157938c8a9a5SSteve French if (rc) {
158038c8a9a5SSteve French log_rdma_event(ERR, "rdma_create_qp failed %i\n", rc);
158138c8a9a5SSteve French goto create_qp_failed;
158238c8a9a5SSteve French }
158338c8a9a5SSteve French
158438c8a9a5SSteve French memset(&conn_param, 0, sizeof(conn_param));
158538c8a9a5SSteve French conn_param.initiator_depth = 0;
158638c8a9a5SSteve French
158738c8a9a5SSteve French conn_param.responder_resources =
158838c8a9a5SSteve French info->id->device->attrs.max_qp_rd_atom
158938c8a9a5SSteve French < SMBD_CM_RESPONDER_RESOURCES ?
159038c8a9a5SSteve French info->id->device->attrs.max_qp_rd_atom :
159138c8a9a5SSteve French SMBD_CM_RESPONDER_RESOURCES;
159238c8a9a5SSteve French info->responder_resources = conn_param.responder_resources;
159338c8a9a5SSteve French log_rdma_mr(INFO, "responder_resources=%d\n",
159438c8a9a5SSteve French info->responder_resources);
159538c8a9a5SSteve French
159638c8a9a5SSteve French /* Need to send IRD/ORD in private data for iWARP */
159738c8a9a5SSteve French info->id->device->ops.get_port_immutable(
159838c8a9a5SSteve French info->id->device, info->id->port_num, &port_immutable);
159938c8a9a5SSteve French if (port_immutable.core_cap_flags & RDMA_CORE_PORT_IWARP) {
160038c8a9a5SSteve French ird_ord_hdr[0] = info->responder_resources;
160138c8a9a5SSteve French ird_ord_hdr[1] = 1;
160238c8a9a5SSteve French conn_param.private_data = ird_ord_hdr;
160338c8a9a5SSteve French conn_param.private_data_len = sizeof(ird_ord_hdr);
160438c8a9a5SSteve French } else {
160538c8a9a5SSteve French conn_param.private_data = NULL;
160638c8a9a5SSteve French conn_param.private_data_len = 0;
160738c8a9a5SSteve French }
160838c8a9a5SSteve French
160938c8a9a5SSteve French conn_param.retry_count = SMBD_CM_RETRY;
161038c8a9a5SSteve French conn_param.rnr_retry_count = SMBD_CM_RNR_RETRY;
161138c8a9a5SSteve French conn_param.flow_control = 0;
161238c8a9a5SSteve French
161338c8a9a5SSteve French log_rdma_event(INFO, "connecting to IP %pI4 port %d\n",
161438c8a9a5SSteve French &addr_in->sin_addr, port);
161538c8a9a5SSteve French
161638c8a9a5SSteve French init_waitqueue_head(&info->conn_wait);
161738c8a9a5SSteve French init_waitqueue_head(&info->disconn_wait);
161838c8a9a5SSteve French init_waitqueue_head(&info->wait_reassembly_queue);
161938c8a9a5SSteve French rc = rdma_connect(info->id, &conn_param);
162038c8a9a5SSteve French if (rc) {
162138c8a9a5SSteve French log_rdma_event(ERR, "rdma_connect() failed with %i\n", rc);
162238c8a9a5SSteve French goto rdma_connect_failed;
162338c8a9a5SSteve French }
162438c8a9a5SSteve French
162538c8a9a5SSteve French wait_event_interruptible(
162638c8a9a5SSteve French info->conn_wait, info->transport_status != SMBD_CONNECTING);
162738c8a9a5SSteve French
162838c8a9a5SSteve French if (info->transport_status != SMBD_CONNECTED) {
162938c8a9a5SSteve French log_rdma_event(ERR, "rdma_connect failed port=%d\n", port);
163038c8a9a5SSteve French goto rdma_connect_failed;
163138c8a9a5SSteve French }
163238c8a9a5SSteve French
163338c8a9a5SSteve French log_rdma_event(INFO, "rdma_connect connected\n");
163438c8a9a5SSteve French
163538c8a9a5SSteve French rc = allocate_caches_and_workqueue(info);
163638c8a9a5SSteve French if (rc) {
163738c8a9a5SSteve French log_rdma_event(ERR, "cache allocation failed\n");
163838c8a9a5SSteve French goto allocate_cache_failed;
163938c8a9a5SSteve French }
164038c8a9a5SSteve French
164138c8a9a5SSteve French init_waitqueue_head(&info->wait_send_queue);
164238c8a9a5SSteve French INIT_DELAYED_WORK(&info->idle_timer_work, idle_connection_timer);
164338c8a9a5SSteve French queue_delayed_work(info->workqueue, &info->idle_timer_work,
164438c8a9a5SSteve French info->keep_alive_interval*HZ);
164538c8a9a5SSteve French
164638c8a9a5SSteve French init_waitqueue_head(&info->wait_send_pending);
164738c8a9a5SSteve French atomic_set(&info->send_pending, 0);
164838c8a9a5SSteve French
164938c8a9a5SSteve French init_waitqueue_head(&info->wait_post_send);
165038c8a9a5SSteve French
165138c8a9a5SSteve French INIT_WORK(&info->disconnect_work, smbd_disconnect_rdma_work);
165238c8a9a5SSteve French INIT_WORK(&info->post_send_credits_work, smbd_post_send_credits);
165338c8a9a5SSteve French info->new_credits_offered = 0;
165438c8a9a5SSteve French spin_lock_init(&info->lock_new_credits_offered);
165538c8a9a5SSteve French
165638c8a9a5SSteve French rc = smbd_negotiate(info);
165738c8a9a5SSteve French if (rc) {
165838c8a9a5SSteve French log_rdma_event(ERR, "smbd_negotiate rc=%d\n", rc);
165938c8a9a5SSteve French goto negotiation_failed;
166038c8a9a5SSteve French }
166138c8a9a5SSteve French
166238c8a9a5SSteve French rc = allocate_mr_list(info);
166338c8a9a5SSteve French if (rc) {
166438c8a9a5SSteve French log_rdma_mr(ERR, "memory registration allocation failed\n");
166538c8a9a5SSteve French goto allocate_mr_failed;
166638c8a9a5SSteve French }
166738c8a9a5SSteve French
166838c8a9a5SSteve French return info;
166938c8a9a5SSteve French
167038c8a9a5SSteve French allocate_mr_failed:
167138c8a9a5SSteve French /* At this point, need to a full transport shutdown */
167238c8a9a5SSteve French server->smbd_conn = info;
167338c8a9a5SSteve French smbd_destroy(server);
167438c8a9a5SSteve French return NULL;
167538c8a9a5SSteve French
167638c8a9a5SSteve French negotiation_failed:
167738c8a9a5SSteve French cancel_delayed_work_sync(&info->idle_timer_work);
167838c8a9a5SSteve French destroy_caches_and_workqueue(info);
167938c8a9a5SSteve French info->transport_status = SMBD_NEGOTIATE_FAILED;
168038c8a9a5SSteve French init_waitqueue_head(&info->conn_wait);
168138c8a9a5SSteve French rdma_disconnect(info->id);
168238c8a9a5SSteve French wait_event(info->conn_wait,
168338c8a9a5SSteve French info->transport_status == SMBD_DISCONNECTED);
168438c8a9a5SSteve French
168538c8a9a5SSteve French allocate_cache_failed:
168638c8a9a5SSteve French rdma_connect_failed:
168738c8a9a5SSteve French rdma_destroy_qp(info->id);
168838c8a9a5SSteve French
168938c8a9a5SSteve French create_qp_failed:
169038c8a9a5SSteve French alloc_cq_failed:
169138c8a9a5SSteve French if (info->send_cq)
169238c8a9a5SSteve French ib_free_cq(info->send_cq);
169338c8a9a5SSteve French if (info->recv_cq)
169438c8a9a5SSteve French ib_free_cq(info->recv_cq);
169538c8a9a5SSteve French
169638c8a9a5SSteve French config_failed:
169738c8a9a5SSteve French ib_dealloc_pd(info->pd);
169838c8a9a5SSteve French rdma_destroy_id(info->id);
169938c8a9a5SSteve French
170038c8a9a5SSteve French create_id_failed:
170138c8a9a5SSteve French kfree(info);
170238c8a9a5SSteve French return NULL;
170338c8a9a5SSteve French }
170438c8a9a5SSteve French
smbd_get_connection(struct TCP_Server_Info * server,struct sockaddr * dstaddr)170538c8a9a5SSteve French struct smbd_connection *smbd_get_connection(
170638c8a9a5SSteve French struct TCP_Server_Info *server, struct sockaddr *dstaddr)
170738c8a9a5SSteve French {
170838c8a9a5SSteve French struct smbd_connection *ret;
170938c8a9a5SSteve French int port = SMBD_PORT;
171038c8a9a5SSteve French
171138c8a9a5SSteve French try_again:
171238c8a9a5SSteve French ret = _smbd_get_connection(server, dstaddr, port);
171338c8a9a5SSteve French
171438c8a9a5SSteve French /* Try SMB_PORT if SMBD_PORT doesn't work */
171538c8a9a5SSteve French if (!ret && port == SMBD_PORT) {
171638c8a9a5SSteve French port = SMB_PORT;
171738c8a9a5SSteve French goto try_again;
171838c8a9a5SSteve French }
171938c8a9a5SSteve French return ret;
172038c8a9a5SSteve French }
172138c8a9a5SSteve French
172238c8a9a5SSteve French /*
172338c8a9a5SSteve French * Receive data from receive reassembly queue
172438c8a9a5SSteve French * All the incoming data packets are placed in reassembly queue
172538c8a9a5SSteve French * buf: the buffer to read data into
172638c8a9a5SSteve French * size: the length of data to read
172738c8a9a5SSteve French * return value: actual data read
172838c8a9a5SSteve French * Note: this implementation copies the data from reassebmly queue to receive
172938c8a9a5SSteve French * buffers used by upper layer. This is not the optimal code path. A better way
173038c8a9a5SSteve French * to do it is to not have upper layer allocate its receive buffers but rather
173138c8a9a5SSteve French * borrow the buffer from reassembly queue, and return it after data is
173238c8a9a5SSteve French * consumed. But this will require more changes to upper layer code, and also
173338c8a9a5SSteve French * need to consider packet boundaries while they still being reassembled.
173438c8a9a5SSteve French */
smbd_recv_buf(struct smbd_connection * info,char * buf,unsigned int size)173538c8a9a5SSteve French static int smbd_recv_buf(struct smbd_connection *info, char *buf,
173638c8a9a5SSteve French unsigned int size)
173738c8a9a5SSteve French {
173838c8a9a5SSteve French struct smbd_response *response;
173938c8a9a5SSteve French struct smbd_data_transfer *data_transfer;
174038c8a9a5SSteve French int to_copy, to_read, data_read, offset;
174138c8a9a5SSteve French u32 data_length, remaining_data_length, data_offset;
174238c8a9a5SSteve French int rc;
174338c8a9a5SSteve French
174438c8a9a5SSteve French again:
174538c8a9a5SSteve French /*
174638c8a9a5SSteve French * No need to hold the reassembly queue lock all the time as we are
174738c8a9a5SSteve French * the only one reading from the front of the queue. The transport
174838c8a9a5SSteve French * may add more entries to the back of the queue at the same time
174938c8a9a5SSteve French */
175038c8a9a5SSteve French log_read(INFO, "size=%d info->reassembly_data_length=%d\n", size,
175138c8a9a5SSteve French info->reassembly_data_length);
175238c8a9a5SSteve French if (info->reassembly_data_length >= size) {
175338c8a9a5SSteve French int queue_length;
175438c8a9a5SSteve French int queue_removed = 0;
175538c8a9a5SSteve French
175638c8a9a5SSteve French /*
175738c8a9a5SSteve French * Need to make sure reassembly_data_length is read before
175838c8a9a5SSteve French * reading reassembly_queue_length and calling
175938c8a9a5SSteve French * _get_first_reassembly. This call is lock free
176038c8a9a5SSteve French * as we never read at the end of the queue which are being
176138c8a9a5SSteve French * updated in SOFTIRQ as more data is received
176238c8a9a5SSteve French */
176338c8a9a5SSteve French virt_rmb();
176438c8a9a5SSteve French queue_length = info->reassembly_queue_length;
176538c8a9a5SSteve French data_read = 0;
176638c8a9a5SSteve French to_read = size;
176738c8a9a5SSteve French offset = info->first_entry_offset;
176838c8a9a5SSteve French while (data_read < size) {
176938c8a9a5SSteve French response = _get_first_reassembly(info);
177038c8a9a5SSteve French data_transfer = smbd_response_payload(response);
177138c8a9a5SSteve French data_length = le32_to_cpu(data_transfer->data_length);
177238c8a9a5SSteve French remaining_data_length =
177338c8a9a5SSteve French le32_to_cpu(
177438c8a9a5SSteve French data_transfer->remaining_data_length);
177538c8a9a5SSteve French data_offset = le32_to_cpu(data_transfer->data_offset);
177638c8a9a5SSteve French
177738c8a9a5SSteve French /*
177838c8a9a5SSteve French * The upper layer expects RFC1002 length at the
177938c8a9a5SSteve French * beginning of the payload. Return it to indicate
178038c8a9a5SSteve French * the total length of the packet. This minimize the
178138c8a9a5SSteve French * change to upper layer packet processing logic. This
178238c8a9a5SSteve French * will be eventually remove when an intermediate
178338c8a9a5SSteve French * transport layer is added
178438c8a9a5SSteve French */
178538c8a9a5SSteve French if (response->first_segment && size == 4) {
178638c8a9a5SSteve French unsigned int rfc1002_len =
178738c8a9a5SSteve French data_length + remaining_data_length;
178838c8a9a5SSteve French *((__be32 *)buf) = cpu_to_be32(rfc1002_len);
178938c8a9a5SSteve French data_read = 4;
179038c8a9a5SSteve French response->first_segment = false;
179138c8a9a5SSteve French log_read(INFO, "returning rfc1002 length %d\n",
179238c8a9a5SSteve French rfc1002_len);
179338c8a9a5SSteve French goto read_rfc1002_done;
179438c8a9a5SSteve French }
179538c8a9a5SSteve French
179638c8a9a5SSteve French to_copy = min_t(int, data_length - offset, to_read);
179738c8a9a5SSteve French memcpy(
179838c8a9a5SSteve French buf + data_read,
179938c8a9a5SSteve French (char *)data_transfer + data_offset + offset,
180038c8a9a5SSteve French to_copy);
180138c8a9a5SSteve French
180238c8a9a5SSteve French /* move on to the next buffer? */
180338c8a9a5SSteve French if (to_copy == data_length - offset) {
180438c8a9a5SSteve French queue_length--;
180538c8a9a5SSteve French /*
180638c8a9a5SSteve French * No need to lock if we are not at the
180738c8a9a5SSteve French * end of the queue
180838c8a9a5SSteve French */
180938c8a9a5SSteve French if (queue_length)
181038c8a9a5SSteve French list_del(&response->list);
181138c8a9a5SSteve French else {
181238c8a9a5SSteve French spin_lock_irq(
181338c8a9a5SSteve French &info->reassembly_queue_lock);
181438c8a9a5SSteve French list_del(&response->list);
181538c8a9a5SSteve French spin_unlock_irq(
181638c8a9a5SSteve French &info->reassembly_queue_lock);
181738c8a9a5SSteve French }
181838c8a9a5SSteve French queue_removed++;
181938c8a9a5SSteve French info->count_reassembly_queue--;
182038c8a9a5SSteve French info->count_dequeue_reassembly_queue++;
182138c8a9a5SSteve French put_receive_buffer(info, response);
182238c8a9a5SSteve French offset = 0;
182338c8a9a5SSteve French log_read(INFO, "put_receive_buffer offset=0\n");
182438c8a9a5SSteve French } else
182538c8a9a5SSteve French offset += to_copy;
182638c8a9a5SSteve French
182738c8a9a5SSteve French to_read -= to_copy;
182838c8a9a5SSteve French data_read += to_copy;
182938c8a9a5SSteve French
183038c8a9a5SSteve French log_read(INFO, "_get_first_reassembly memcpy %d bytes data_transfer_length-offset=%d after that to_read=%d data_read=%d offset=%d\n",
183138c8a9a5SSteve French to_copy, data_length - offset,
183238c8a9a5SSteve French to_read, data_read, offset);
183338c8a9a5SSteve French }
183438c8a9a5SSteve French
183538c8a9a5SSteve French spin_lock_irq(&info->reassembly_queue_lock);
183638c8a9a5SSteve French info->reassembly_data_length -= data_read;
183738c8a9a5SSteve French info->reassembly_queue_length -= queue_removed;
183838c8a9a5SSteve French spin_unlock_irq(&info->reassembly_queue_lock);
183938c8a9a5SSteve French
184038c8a9a5SSteve French info->first_entry_offset = offset;
184138c8a9a5SSteve French log_read(INFO, "returning to thread data_read=%d reassembly_data_length=%d first_entry_offset=%d\n",
184238c8a9a5SSteve French data_read, info->reassembly_data_length,
184338c8a9a5SSteve French info->first_entry_offset);
184438c8a9a5SSteve French read_rfc1002_done:
184538c8a9a5SSteve French return data_read;
184638c8a9a5SSteve French }
184738c8a9a5SSteve French
184838c8a9a5SSteve French log_read(INFO, "wait_event on more data\n");
184938c8a9a5SSteve French rc = wait_event_interruptible(
185038c8a9a5SSteve French info->wait_reassembly_queue,
185138c8a9a5SSteve French info->reassembly_data_length >= size ||
185238c8a9a5SSteve French info->transport_status != SMBD_CONNECTED);
185338c8a9a5SSteve French /* Don't return any data if interrupted */
185438c8a9a5SSteve French if (rc)
185538c8a9a5SSteve French return rc;
185638c8a9a5SSteve French
185738c8a9a5SSteve French if (info->transport_status != SMBD_CONNECTED) {
185838c8a9a5SSteve French log_read(ERR, "disconnected\n");
185938c8a9a5SSteve French return -ECONNABORTED;
186038c8a9a5SSteve French }
186138c8a9a5SSteve French
186238c8a9a5SSteve French goto again;
186338c8a9a5SSteve French }
186438c8a9a5SSteve French
186538c8a9a5SSteve French /*
186638c8a9a5SSteve French * Receive a page from receive reassembly queue
186738c8a9a5SSteve French * page: the page to read data into
186838c8a9a5SSteve French * to_read: the length of data to read
186938c8a9a5SSteve French * return value: actual data read
187038c8a9a5SSteve French */
smbd_recv_page(struct smbd_connection * info,struct page * page,unsigned int page_offset,unsigned int to_read)187138c8a9a5SSteve French static int smbd_recv_page(struct smbd_connection *info,
187238c8a9a5SSteve French struct page *page, unsigned int page_offset,
187338c8a9a5SSteve French unsigned int to_read)
187438c8a9a5SSteve French {
187538c8a9a5SSteve French int ret;
187638c8a9a5SSteve French char *to_address;
187738c8a9a5SSteve French void *page_address;
187838c8a9a5SSteve French
187938c8a9a5SSteve French /* make sure we have the page ready for read */
188038c8a9a5SSteve French ret = wait_event_interruptible(
188138c8a9a5SSteve French info->wait_reassembly_queue,
188238c8a9a5SSteve French info->reassembly_data_length >= to_read ||
188338c8a9a5SSteve French info->transport_status != SMBD_CONNECTED);
188438c8a9a5SSteve French if (ret)
188538c8a9a5SSteve French return ret;
188638c8a9a5SSteve French
188738c8a9a5SSteve French /* now we can read from reassembly queue and not sleep */
188838c8a9a5SSteve French page_address = kmap_atomic(page);
188938c8a9a5SSteve French to_address = (char *) page_address + page_offset;
189038c8a9a5SSteve French
189138c8a9a5SSteve French log_read(INFO, "reading from page=%p address=%p to_read=%d\n",
189238c8a9a5SSteve French page, to_address, to_read);
189338c8a9a5SSteve French
189438c8a9a5SSteve French ret = smbd_recv_buf(info, to_address, to_read);
189538c8a9a5SSteve French kunmap_atomic(page_address);
189638c8a9a5SSteve French
189738c8a9a5SSteve French return ret;
189838c8a9a5SSteve French }
189938c8a9a5SSteve French
190038c8a9a5SSteve French /*
190138c8a9a5SSteve French * Receive data from transport
190238c8a9a5SSteve French * msg: a msghdr point to the buffer, can be ITER_KVEC or ITER_BVEC
190338c8a9a5SSteve French * return: total bytes read, or 0. SMB Direct will not do partial read.
190438c8a9a5SSteve French */
smbd_recv(struct smbd_connection * info,struct msghdr * msg)190538c8a9a5SSteve French int smbd_recv(struct smbd_connection *info, struct msghdr *msg)
190638c8a9a5SSteve French {
190738c8a9a5SSteve French char *buf;
190838c8a9a5SSteve French struct page *page;
190938c8a9a5SSteve French unsigned int to_read, page_offset;
191038c8a9a5SSteve French int rc;
191138c8a9a5SSteve French
191238c8a9a5SSteve French if (iov_iter_rw(&msg->msg_iter) == WRITE) {
191338c8a9a5SSteve French /* It's a bug in upper layer to get there */
191438c8a9a5SSteve French cifs_dbg(VFS, "Invalid msg iter dir %u\n",
191538c8a9a5SSteve French iov_iter_rw(&msg->msg_iter));
191638c8a9a5SSteve French rc = -EINVAL;
191738c8a9a5SSteve French goto out;
191838c8a9a5SSteve French }
191938c8a9a5SSteve French
192038c8a9a5SSteve French switch (iov_iter_type(&msg->msg_iter)) {
192138c8a9a5SSteve French case ITER_KVEC:
192238c8a9a5SSteve French buf = msg->msg_iter.kvec->iov_base;
192338c8a9a5SSteve French to_read = msg->msg_iter.kvec->iov_len;
192438c8a9a5SSteve French rc = smbd_recv_buf(info, buf, to_read);
192538c8a9a5SSteve French break;
192638c8a9a5SSteve French
192738c8a9a5SSteve French case ITER_BVEC:
192838c8a9a5SSteve French page = msg->msg_iter.bvec->bv_page;
192938c8a9a5SSteve French page_offset = msg->msg_iter.bvec->bv_offset;
193038c8a9a5SSteve French to_read = msg->msg_iter.bvec->bv_len;
193138c8a9a5SSteve French rc = smbd_recv_page(info, page, page_offset, to_read);
193238c8a9a5SSteve French break;
193338c8a9a5SSteve French
193438c8a9a5SSteve French default:
193538c8a9a5SSteve French /* It's a bug in upper layer to get there */
193638c8a9a5SSteve French cifs_dbg(VFS, "Invalid msg type %d\n",
193738c8a9a5SSteve French iov_iter_type(&msg->msg_iter));
193838c8a9a5SSteve French rc = -EINVAL;
193938c8a9a5SSteve French }
194038c8a9a5SSteve French
194138c8a9a5SSteve French out:
194238c8a9a5SSteve French /* SMBDirect will read it all or nothing */
194338c8a9a5SSteve French if (rc > 0)
194438c8a9a5SSteve French msg->msg_iter.count = 0;
194538c8a9a5SSteve French return rc;
194638c8a9a5SSteve French }
194738c8a9a5SSteve French
194838c8a9a5SSteve French /*
194938c8a9a5SSteve French * Send data to transport
195038c8a9a5SSteve French * Each rqst is transported as a SMBDirect payload
195138c8a9a5SSteve French * rqst: the data to write
195238c8a9a5SSteve French * return value: 0 if successfully write, otherwise error code
195338c8a9a5SSteve French */
smbd_send(struct TCP_Server_Info * server,int num_rqst,struct smb_rqst * rqst_array)195438c8a9a5SSteve French int smbd_send(struct TCP_Server_Info *server,
195538c8a9a5SSteve French int num_rqst, struct smb_rqst *rqst_array)
195638c8a9a5SSteve French {
195738c8a9a5SSteve French struct smbd_connection *info = server->smbd_conn;
195838c8a9a5SSteve French struct smb_rqst *rqst;
195938c8a9a5SSteve French struct iov_iter iter;
196038c8a9a5SSteve French unsigned int remaining_data_length, klen;
196138c8a9a5SSteve French int rc, i, rqst_idx;
196238c8a9a5SSteve French
196338c8a9a5SSteve French if (info->transport_status != SMBD_CONNECTED)
196438c8a9a5SSteve French return -EAGAIN;
196538c8a9a5SSteve French
196638c8a9a5SSteve French /*
196738c8a9a5SSteve French * Add in the page array if there is one. The caller needs to set
196838c8a9a5SSteve French * rq_tailsz to PAGE_SIZE when the buffer has multiple pages and
196938c8a9a5SSteve French * ends at page boundary
197038c8a9a5SSteve French */
197138c8a9a5SSteve French remaining_data_length = 0;
197238c8a9a5SSteve French for (i = 0; i < num_rqst; i++)
197338c8a9a5SSteve French remaining_data_length += smb_rqst_len(server, &rqst_array[i]);
197438c8a9a5SSteve French
197538c8a9a5SSteve French if (unlikely(remaining_data_length > info->max_fragmented_send_size)) {
197638c8a9a5SSteve French /* assertion: payload never exceeds negotiated maximum */
197738c8a9a5SSteve French log_write(ERR, "payload size %d > max size %d\n",
197838c8a9a5SSteve French remaining_data_length, info->max_fragmented_send_size);
197938c8a9a5SSteve French return -EINVAL;
198038c8a9a5SSteve French }
198138c8a9a5SSteve French
198238c8a9a5SSteve French log_write(INFO, "num_rqst=%d total length=%u\n",
198338c8a9a5SSteve French num_rqst, remaining_data_length);
198438c8a9a5SSteve French
198538c8a9a5SSteve French rqst_idx = 0;
198638c8a9a5SSteve French do {
198738c8a9a5SSteve French rqst = &rqst_array[rqst_idx];
198838c8a9a5SSteve French
198938c8a9a5SSteve French cifs_dbg(FYI, "Sending smb (RDMA): idx=%d smb_len=%lu\n",
199038c8a9a5SSteve French rqst_idx, smb_rqst_len(server, rqst));
199138c8a9a5SSteve French for (i = 0; i < rqst->rq_nvec; i++)
199238c8a9a5SSteve French dump_smb(rqst->rq_iov[i].iov_base, rqst->rq_iov[i].iov_len);
199338c8a9a5SSteve French
199438c8a9a5SSteve French log_write(INFO, "RDMA-WR[%u] nvec=%d len=%u iter=%zu rqlen=%lu\n",
199538c8a9a5SSteve French rqst_idx, rqst->rq_nvec, remaining_data_length,
199638c8a9a5SSteve French iov_iter_count(&rqst->rq_iter), smb_rqst_len(server, rqst));
199738c8a9a5SSteve French
199838c8a9a5SSteve French /* Send the metadata pages. */
199938c8a9a5SSteve French klen = 0;
200038c8a9a5SSteve French for (i = 0; i < rqst->rq_nvec; i++)
200138c8a9a5SSteve French klen += rqst->rq_iov[i].iov_len;
200238c8a9a5SSteve French iov_iter_kvec(&iter, ITER_SOURCE, rqst->rq_iov, rqst->rq_nvec, klen);
200338c8a9a5SSteve French
200438c8a9a5SSteve French rc = smbd_post_send_iter(info, &iter, &remaining_data_length);
200538c8a9a5SSteve French if (rc < 0)
200638c8a9a5SSteve French break;
200738c8a9a5SSteve French
200838c8a9a5SSteve French if (iov_iter_count(&rqst->rq_iter) > 0) {
200938c8a9a5SSteve French /* And then the data pages if there are any */
201038c8a9a5SSteve French rc = smbd_post_send_iter(info, &rqst->rq_iter,
201138c8a9a5SSteve French &remaining_data_length);
201238c8a9a5SSteve French if (rc < 0)
201338c8a9a5SSteve French break;
201438c8a9a5SSteve French }
201538c8a9a5SSteve French
201638c8a9a5SSteve French } while (++rqst_idx < num_rqst);
201738c8a9a5SSteve French
201838c8a9a5SSteve French /*
201938c8a9a5SSteve French * As an optimization, we don't wait for individual I/O to finish
202038c8a9a5SSteve French * before sending the next one.
202138c8a9a5SSteve French * Send them all and wait for pending send count to get to 0
202238c8a9a5SSteve French * that means all the I/Os have been out and we are good to return
202338c8a9a5SSteve French */
202438c8a9a5SSteve French
202538c8a9a5SSteve French wait_event(info->wait_send_pending,
202638c8a9a5SSteve French atomic_read(&info->send_pending) == 0);
202738c8a9a5SSteve French
202838c8a9a5SSteve French return rc;
202938c8a9a5SSteve French }
203038c8a9a5SSteve French
register_mr_done(struct ib_cq * cq,struct ib_wc * wc)203138c8a9a5SSteve French static void register_mr_done(struct ib_cq *cq, struct ib_wc *wc)
203238c8a9a5SSteve French {
203338c8a9a5SSteve French struct smbd_mr *mr;
203438c8a9a5SSteve French struct ib_cqe *cqe;
203538c8a9a5SSteve French
203638c8a9a5SSteve French if (wc->status) {
203738c8a9a5SSteve French log_rdma_mr(ERR, "status=%d\n", wc->status);
203838c8a9a5SSteve French cqe = wc->wr_cqe;
203938c8a9a5SSteve French mr = container_of(cqe, struct smbd_mr, cqe);
204038c8a9a5SSteve French smbd_disconnect_rdma_connection(mr->conn);
204138c8a9a5SSteve French }
204238c8a9a5SSteve French }
204338c8a9a5SSteve French
204438c8a9a5SSteve French /*
204538c8a9a5SSteve French * The work queue function that recovers MRs
204638c8a9a5SSteve French * We need to call ib_dereg_mr() and ib_alloc_mr() before this MR can be used
204738c8a9a5SSteve French * again. Both calls are slow, so finish them in a workqueue. This will not
204838c8a9a5SSteve French * block I/O path.
204938c8a9a5SSteve French * There is one workqueue that recovers MRs, there is no need to lock as the
205038c8a9a5SSteve French * I/O requests calling smbd_register_mr will never update the links in the
205138c8a9a5SSteve French * mr_list.
205238c8a9a5SSteve French */
smbd_mr_recovery_work(struct work_struct * work)205338c8a9a5SSteve French static void smbd_mr_recovery_work(struct work_struct *work)
205438c8a9a5SSteve French {
205538c8a9a5SSteve French struct smbd_connection *info =
205638c8a9a5SSteve French container_of(work, struct smbd_connection, mr_recovery_work);
205738c8a9a5SSteve French struct smbd_mr *smbdirect_mr;
205838c8a9a5SSteve French int rc;
205938c8a9a5SSteve French
206038c8a9a5SSteve French list_for_each_entry(smbdirect_mr, &info->mr_list, list) {
206138c8a9a5SSteve French if (smbdirect_mr->state == MR_ERROR) {
206238c8a9a5SSteve French
206338c8a9a5SSteve French /* recover this MR entry */
206438c8a9a5SSteve French rc = ib_dereg_mr(smbdirect_mr->mr);
206538c8a9a5SSteve French if (rc) {
206638c8a9a5SSteve French log_rdma_mr(ERR,
206738c8a9a5SSteve French "ib_dereg_mr failed rc=%x\n",
206838c8a9a5SSteve French rc);
206938c8a9a5SSteve French smbd_disconnect_rdma_connection(info);
207038c8a9a5SSteve French continue;
207138c8a9a5SSteve French }
207238c8a9a5SSteve French
207338c8a9a5SSteve French smbdirect_mr->mr = ib_alloc_mr(
207438c8a9a5SSteve French info->pd, info->mr_type,
207538c8a9a5SSteve French info->max_frmr_depth);
207638c8a9a5SSteve French if (IS_ERR(smbdirect_mr->mr)) {
207738c8a9a5SSteve French log_rdma_mr(ERR, "ib_alloc_mr failed mr_type=%x max_frmr_depth=%x\n",
207838c8a9a5SSteve French info->mr_type,
207938c8a9a5SSteve French info->max_frmr_depth);
208038c8a9a5SSteve French smbd_disconnect_rdma_connection(info);
208138c8a9a5SSteve French continue;
208238c8a9a5SSteve French }
208338c8a9a5SSteve French } else
208438c8a9a5SSteve French /* This MR is being used, don't recover it */
208538c8a9a5SSteve French continue;
208638c8a9a5SSteve French
208738c8a9a5SSteve French smbdirect_mr->state = MR_READY;
208838c8a9a5SSteve French
208938c8a9a5SSteve French /* smbdirect_mr->state is updated by this function
209038c8a9a5SSteve French * and is read and updated by I/O issuing CPUs trying
209138c8a9a5SSteve French * to get a MR, the call to atomic_inc_return
209238c8a9a5SSteve French * implicates a memory barrier and guarantees this
209338c8a9a5SSteve French * value is updated before waking up any calls to
209438c8a9a5SSteve French * get_mr() from the I/O issuing CPUs
209538c8a9a5SSteve French */
209638c8a9a5SSteve French if (atomic_inc_return(&info->mr_ready_count) == 1)
209738c8a9a5SSteve French wake_up_interruptible(&info->wait_mr);
209838c8a9a5SSteve French }
209938c8a9a5SSteve French }
210038c8a9a5SSteve French
destroy_mr_list(struct smbd_connection * info)210138c8a9a5SSteve French static void destroy_mr_list(struct smbd_connection *info)
210238c8a9a5SSteve French {
210338c8a9a5SSteve French struct smbd_mr *mr, *tmp;
210438c8a9a5SSteve French
210538c8a9a5SSteve French cancel_work_sync(&info->mr_recovery_work);
210638c8a9a5SSteve French list_for_each_entry_safe(mr, tmp, &info->mr_list, list) {
210738c8a9a5SSteve French if (mr->state == MR_INVALIDATED)
210838c8a9a5SSteve French ib_dma_unmap_sg(info->id->device, mr->sgt.sgl,
210938c8a9a5SSteve French mr->sgt.nents, mr->dir);
211038c8a9a5SSteve French ib_dereg_mr(mr->mr);
211138c8a9a5SSteve French kfree(mr->sgt.sgl);
211238c8a9a5SSteve French kfree(mr);
211338c8a9a5SSteve French }
211438c8a9a5SSteve French }
211538c8a9a5SSteve French
211638c8a9a5SSteve French /*
211738c8a9a5SSteve French * Allocate MRs used for RDMA read/write
211838c8a9a5SSteve French * The number of MRs will not exceed hardware capability in responder_resources
211938c8a9a5SSteve French * All MRs are kept in mr_list. The MR can be recovered after it's used
212038c8a9a5SSteve French * Recovery is done in smbd_mr_recovery_work. The content of list entry changes
212138c8a9a5SSteve French * as MRs are used and recovered for I/O, but the list links will not change
212238c8a9a5SSteve French */
allocate_mr_list(struct smbd_connection * info)212338c8a9a5SSteve French static int allocate_mr_list(struct smbd_connection *info)
212438c8a9a5SSteve French {
212538c8a9a5SSteve French int i;
212638c8a9a5SSteve French struct smbd_mr *smbdirect_mr, *tmp;
212738c8a9a5SSteve French
212838c8a9a5SSteve French INIT_LIST_HEAD(&info->mr_list);
212938c8a9a5SSteve French init_waitqueue_head(&info->wait_mr);
213038c8a9a5SSteve French spin_lock_init(&info->mr_list_lock);
213138c8a9a5SSteve French atomic_set(&info->mr_ready_count, 0);
213238c8a9a5SSteve French atomic_set(&info->mr_used_count, 0);
213338c8a9a5SSteve French init_waitqueue_head(&info->wait_for_mr_cleanup);
213438c8a9a5SSteve French INIT_WORK(&info->mr_recovery_work, smbd_mr_recovery_work);
213538c8a9a5SSteve French /* Allocate more MRs (2x) than hardware responder_resources */
213638c8a9a5SSteve French for (i = 0; i < info->responder_resources * 2; i++) {
213738c8a9a5SSteve French smbdirect_mr = kzalloc(sizeof(*smbdirect_mr), GFP_KERNEL);
213838c8a9a5SSteve French if (!smbdirect_mr)
2139*88e07de2SMarkus Elfring goto cleanup_entries;
214038c8a9a5SSteve French smbdirect_mr->mr = ib_alloc_mr(info->pd, info->mr_type,
214138c8a9a5SSteve French info->max_frmr_depth);
214238c8a9a5SSteve French if (IS_ERR(smbdirect_mr->mr)) {
214338c8a9a5SSteve French log_rdma_mr(ERR, "ib_alloc_mr failed mr_type=%x max_frmr_depth=%x\n",
214438c8a9a5SSteve French info->mr_type, info->max_frmr_depth);
214538c8a9a5SSteve French goto out;
214638c8a9a5SSteve French }
214738c8a9a5SSteve French smbdirect_mr->sgt.sgl = kcalloc(info->max_frmr_depth,
214838c8a9a5SSteve French sizeof(struct scatterlist),
214938c8a9a5SSteve French GFP_KERNEL);
215038c8a9a5SSteve French if (!smbdirect_mr->sgt.sgl) {
215138c8a9a5SSteve French log_rdma_mr(ERR, "failed to allocate sgl\n");
215238c8a9a5SSteve French ib_dereg_mr(smbdirect_mr->mr);
215338c8a9a5SSteve French goto out;
215438c8a9a5SSteve French }
215538c8a9a5SSteve French smbdirect_mr->state = MR_READY;
215638c8a9a5SSteve French smbdirect_mr->conn = info;
215738c8a9a5SSteve French
215838c8a9a5SSteve French list_add_tail(&smbdirect_mr->list, &info->mr_list);
215938c8a9a5SSteve French atomic_inc(&info->mr_ready_count);
216038c8a9a5SSteve French }
216138c8a9a5SSteve French return 0;
216238c8a9a5SSteve French
216338c8a9a5SSteve French out:
216438c8a9a5SSteve French kfree(smbdirect_mr);
2165*88e07de2SMarkus Elfring cleanup_entries:
216638c8a9a5SSteve French list_for_each_entry_safe(smbdirect_mr, tmp, &info->mr_list, list) {
216738c8a9a5SSteve French list_del(&smbdirect_mr->list);
216838c8a9a5SSteve French ib_dereg_mr(smbdirect_mr->mr);
216938c8a9a5SSteve French kfree(smbdirect_mr->sgt.sgl);
217038c8a9a5SSteve French kfree(smbdirect_mr);
217138c8a9a5SSteve French }
217238c8a9a5SSteve French return -ENOMEM;
217338c8a9a5SSteve French }
217438c8a9a5SSteve French
217538c8a9a5SSteve French /*
217638c8a9a5SSteve French * Get a MR from mr_list. This function waits until there is at least one
217738c8a9a5SSteve French * MR available in the list. It may access the list while the
217838c8a9a5SSteve French * smbd_mr_recovery_work is recovering the MR list. This doesn't need a lock
217938c8a9a5SSteve French * as they never modify the same places. However, there may be several CPUs
218038c8a9a5SSteve French * issueing I/O trying to get MR at the same time, mr_list_lock is used to
218138c8a9a5SSteve French * protect this situation.
218238c8a9a5SSteve French */
get_mr(struct smbd_connection * info)218338c8a9a5SSteve French static struct smbd_mr *get_mr(struct smbd_connection *info)
218438c8a9a5SSteve French {
218538c8a9a5SSteve French struct smbd_mr *ret;
218638c8a9a5SSteve French int rc;
218738c8a9a5SSteve French again:
218838c8a9a5SSteve French rc = wait_event_interruptible(info->wait_mr,
218938c8a9a5SSteve French atomic_read(&info->mr_ready_count) ||
219038c8a9a5SSteve French info->transport_status != SMBD_CONNECTED);
219138c8a9a5SSteve French if (rc) {
219238c8a9a5SSteve French log_rdma_mr(ERR, "wait_event_interruptible rc=%x\n", rc);
219338c8a9a5SSteve French return NULL;
219438c8a9a5SSteve French }
219538c8a9a5SSteve French
219638c8a9a5SSteve French if (info->transport_status != SMBD_CONNECTED) {
219738c8a9a5SSteve French log_rdma_mr(ERR, "info->transport_status=%x\n",
219838c8a9a5SSteve French info->transport_status);
219938c8a9a5SSteve French return NULL;
220038c8a9a5SSteve French }
220138c8a9a5SSteve French
220238c8a9a5SSteve French spin_lock(&info->mr_list_lock);
220338c8a9a5SSteve French list_for_each_entry(ret, &info->mr_list, list) {
220438c8a9a5SSteve French if (ret->state == MR_READY) {
220538c8a9a5SSteve French ret->state = MR_REGISTERED;
220638c8a9a5SSteve French spin_unlock(&info->mr_list_lock);
220738c8a9a5SSteve French atomic_dec(&info->mr_ready_count);
220838c8a9a5SSteve French atomic_inc(&info->mr_used_count);
220938c8a9a5SSteve French return ret;
221038c8a9a5SSteve French }
221138c8a9a5SSteve French }
221238c8a9a5SSteve French
221338c8a9a5SSteve French spin_unlock(&info->mr_list_lock);
221438c8a9a5SSteve French /*
221538c8a9a5SSteve French * It is possible that we could fail to get MR because other processes may
221638c8a9a5SSteve French * try to acquire a MR at the same time. If this is the case, retry it.
221738c8a9a5SSteve French */
221838c8a9a5SSteve French goto again;
221938c8a9a5SSteve French }
222038c8a9a5SSteve French
222138c8a9a5SSteve French /*
222238c8a9a5SSteve French * Transcribe the pages from an iterator into an MR scatterlist.
222338c8a9a5SSteve French */
smbd_iter_to_mr(struct smbd_connection * info,struct iov_iter * iter,struct sg_table * sgt,unsigned int max_sg)222438c8a9a5SSteve French static int smbd_iter_to_mr(struct smbd_connection *info,
222538c8a9a5SSteve French struct iov_iter *iter,
222638c8a9a5SSteve French struct sg_table *sgt,
222738c8a9a5SSteve French unsigned int max_sg)
222838c8a9a5SSteve French {
222938c8a9a5SSteve French int ret;
223038c8a9a5SSteve French
223138c8a9a5SSteve French memset(sgt->sgl, 0, max_sg * sizeof(struct scatterlist));
223238c8a9a5SSteve French
22330d7aeb68SDavid Howells ret = extract_iter_to_sg(iter, iov_iter_count(iter), sgt, max_sg, 0);
223438c8a9a5SSteve French WARN_ON(ret < 0);
223538c8a9a5SSteve French if (sgt->nents > 0)
223638c8a9a5SSteve French sg_mark_end(&sgt->sgl[sgt->nents - 1]);
223738c8a9a5SSteve French return ret;
223838c8a9a5SSteve French }
223938c8a9a5SSteve French
224038c8a9a5SSteve French /*
224138c8a9a5SSteve French * Register memory for RDMA read/write
224238c8a9a5SSteve French * iter: the buffer to register memory with
224338c8a9a5SSteve French * writing: true if this is a RDMA write (SMB read), false for RDMA read
224438c8a9a5SSteve French * need_invalidate: true if this MR needs to be locally invalidated after I/O
224538c8a9a5SSteve French * return value: the MR registered, NULL if failed.
224638c8a9a5SSteve French */
smbd_register_mr(struct smbd_connection * info,struct iov_iter * iter,bool writing,bool need_invalidate)224738c8a9a5SSteve French struct smbd_mr *smbd_register_mr(struct smbd_connection *info,
224838c8a9a5SSteve French struct iov_iter *iter,
224938c8a9a5SSteve French bool writing, bool need_invalidate)
225038c8a9a5SSteve French {
225138c8a9a5SSteve French struct smbd_mr *smbdirect_mr;
225238c8a9a5SSteve French int rc, num_pages;
225338c8a9a5SSteve French enum dma_data_direction dir;
225438c8a9a5SSteve French struct ib_reg_wr *reg_wr;
225538c8a9a5SSteve French
225638c8a9a5SSteve French num_pages = iov_iter_npages(iter, info->max_frmr_depth + 1);
225738c8a9a5SSteve French if (num_pages > info->max_frmr_depth) {
225838c8a9a5SSteve French log_rdma_mr(ERR, "num_pages=%d max_frmr_depth=%d\n",
225938c8a9a5SSteve French num_pages, info->max_frmr_depth);
226038c8a9a5SSteve French WARN_ON_ONCE(1);
226138c8a9a5SSteve French return NULL;
226238c8a9a5SSteve French }
226338c8a9a5SSteve French
226438c8a9a5SSteve French smbdirect_mr = get_mr(info);
226538c8a9a5SSteve French if (!smbdirect_mr) {
226638c8a9a5SSteve French log_rdma_mr(ERR, "get_mr returning NULL\n");
226738c8a9a5SSteve French return NULL;
226838c8a9a5SSteve French }
226938c8a9a5SSteve French
227038c8a9a5SSteve French dir = writing ? DMA_FROM_DEVICE : DMA_TO_DEVICE;
227138c8a9a5SSteve French smbdirect_mr->dir = dir;
227238c8a9a5SSteve French smbdirect_mr->need_invalidate = need_invalidate;
227338c8a9a5SSteve French smbdirect_mr->sgt.nents = 0;
227438c8a9a5SSteve French smbdirect_mr->sgt.orig_nents = 0;
227538c8a9a5SSteve French
227638c8a9a5SSteve French log_rdma_mr(INFO, "num_pages=0x%x count=0x%zx depth=%u\n",
227738c8a9a5SSteve French num_pages, iov_iter_count(iter), info->max_frmr_depth);
227838c8a9a5SSteve French smbd_iter_to_mr(info, iter, &smbdirect_mr->sgt, info->max_frmr_depth);
227938c8a9a5SSteve French
228038c8a9a5SSteve French rc = ib_dma_map_sg(info->id->device, smbdirect_mr->sgt.sgl,
228138c8a9a5SSteve French smbdirect_mr->sgt.nents, dir);
228238c8a9a5SSteve French if (!rc) {
228338c8a9a5SSteve French log_rdma_mr(ERR, "ib_dma_map_sg num_pages=%x dir=%x rc=%x\n",
228438c8a9a5SSteve French num_pages, dir, rc);
228538c8a9a5SSteve French goto dma_map_error;
228638c8a9a5SSteve French }
228738c8a9a5SSteve French
228838c8a9a5SSteve French rc = ib_map_mr_sg(smbdirect_mr->mr, smbdirect_mr->sgt.sgl,
228938c8a9a5SSteve French smbdirect_mr->sgt.nents, NULL, PAGE_SIZE);
229038c8a9a5SSteve French if (rc != smbdirect_mr->sgt.nents) {
229138c8a9a5SSteve French log_rdma_mr(ERR,
229238c8a9a5SSteve French "ib_map_mr_sg failed rc = %d nents = %x\n",
229338c8a9a5SSteve French rc, smbdirect_mr->sgt.nents);
229438c8a9a5SSteve French goto map_mr_error;
229538c8a9a5SSteve French }
229638c8a9a5SSteve French
229738c8a9a5SSteve French ib_update_fast_reg_key(smbdirect_mr->mr,
229838c8a9a5SSteve French ib_inc_rkey(smbdirect_mr->mr->rkey));
229938c8a9a5SSteve French reg_wr = &smbdirect_mr->wr;
230038c8a9a5SSteve French reg_wr->wr.opcode = IB_WR_REG_MR;
230138c8a9a5SSteve French smbdirect_mr->cqe.done = register_mr_done;
230238c8a9a5SSteve French reg_wr->wr.wr_cqe = &smbdirect_mr->cqe;
230338c8a9a5SSteve French reg_wr->wr.num_sge = 0;
230438c8a9a5SSteve French reg_wr->wr.send_flags = IB_SEND_SIGNALED;
230538c8a9a5SSteve French reg_wr->mr = smbdirect_mr->mr;
230638c8a9a5SSteve French reg_wr->key = smbdirect_mr->mr->rkey;
230738c8a9a5SSteve French reg_wr->access = writing ?
230838c8a9a5SSteve French IB_ACCESS_REMOTE_WRITE | IB_ACCESS_LOCAL_WRITE :
230938c8a9a5SSteve French IB_ACCESS_REMOTE_READ;
231038c8a9a5SSteve French
231138c8a9a5SSteve French /*
231238c8a9a5SSteve French * There is no need for waiting for complemtion on ib_post_send
231338c8a9a5SSteve French * on IB_WR_REG_MR. Hardware enforces a barrier and order of execution
231438c8a9a5SSteve French * on the next ib_post_send when we actaully send I/O to remote peer
231538c8a9a5SSteve French */
231638c8a9a5SSteve French rc = ib_post_send(info->id->qp, ®_wr->wr, NULL);
231738c8a9a5SSteve French if (!rc)
231838c8a9a5SSteve French return smbdirect_mr;
231938c8a9a5SSteve French
232038c8a9a5SSteve French log_rdma_mr(ERR, "ib_post_send failed rc=%x reg_wr->key=%x\n",
232138c8a9a5SSteve French rc, reg_wr->key);
232238c8a9a5SSteve French
232338c8a9a5SSteve French /* If all failed, attempt to recover this MR by setting it MR_ERROR*/
232438c8a9a5SSteve French map_mr_error:
232538c8a9a5SSteve French ib_dma_unmap_sg(info->id->device, smbdirect_mr->sgt.sgl,
232638c8a9a5SSteve French smbdirect_mr->sgt.nents, smbdirect_mr->dir);
232738c8a9a5SSteve French
232838c8a9a5SSteve French dma_map_error:
232938c8a9a5SSteve French smbdirect_mr->state = MR_ERROR;
233038c8a9a5SSteve French if (atomic_dec_and_test(&info->mr_used_count))
233138c8a9a5SSteve French wake_up(&info->wait_for_mr_cleanup);
233238c8a9a5SSteve French
233338c8a9a5SSteve French smbd_disconnect_rdma_connection(info);
233438c8a9a5SSteve French
233538c8a9a5SSteve French return NULL;
233638c8a9a5SSteve French }
233738c8a9a5SSteve French
local_inv_done(struct ib_cq * cq,struct ib_wc * wc)233838c8a9a5SSteve French static void local_inv_done(struct ib_cq *cq, struct ib_wc *wc)
233938c8a9a5SSteve French {
234038c8a9a5SSteve French struct smbd_mr *smbdirect_mr;
234138c8a9a5SSteve French struct ib_cqe *cqe;
234238c8a9a5SSteve French
234338c8a9a5SSteve French cqe = wc->wr_cqe;
234438c8a9a5SSteve French smbdirect_mr = container_of(cqe, struct smbd_mr, cqe);
234538c8a9a5SSteve French smbdirect_mr->state = MR_INVALIDATED;
234638c8a9a5SSteve French if (wc->status != IB_WC_SUCCESS) {
234738c8a9a5SSteve French log_rdma_mr(ERR, "invalidate failed status=%x\n", wc->status);
234838c8a9a5SSteve French smbdirect_mr->state = MR_ERROR;
234938c8a9a5SSteve French }
235038c8a9a5SSteve French complete(&smbdirect_mr->invalidate_done);
235138c8a9a5SSteve French }
235238c8a9a5SSteve French
235338c8a9a5SSteve French /*
235438c8a9a5SSteve French * Deregister a MR after I/O is done
235538c8a9a5SSteve French * This function may wait if remote invalidation is not used
235638c8a9a5SSteve French * and we have to locally invalidate the buffer to prevent data is being
235738c8a9a5SSteve French * modified by remote peer after upper layer consumes it
235838c8a9a5SSteve French */
smbd_deregister_mr(struct smbd_mr * smbdirect_mr)235938c8a9a5SSteve French int smbd_deregister_mr(struct smbd_mr *smbdirect_mr)
236038c8a9a5SSteve French {
236138c8a9a5SSteve French struct ib_send_wr *wr;
236238c8a9a5SSteve French struct smbd_connection *info = smbdirect_mr->conn;
236338c8a9a5SSteve French int rc = 0;
236438c8a9a5SSteve French
236538c8a9a5SSteve French if (smbdirect_mr->need_invalidate) {
236638c8a9a5SSteve French /* Need to finish local invalidation before returning */
236738c8a9a5SSteve French wr = &smbdirect_mr->inv_wr;
236838c8a9a5SSteve French wr->opcode = IB_WR_LOCAL_INV;
236938c8a9a5SSteve French smbdirect_mr->cqe.done = local_inv_done;
237038c8a9a5SSteve French wr->wr_cqe = &smbdirect_mr->cqe;
237138c8a9a5SSteve French wr->num_sge = 0;
237238c8a9a5SSteve French wr->ex.invalidate_rkey = smbdirect_mr->mr->rkey;
237338c8a9a5SSteve French wr->send_flags = IB_SEND_SIGNALED;
237438c8a9a5SSteve French
237538c8a9a5SSteve French init_completion(&smbdirect_mr->invalidate_done);
237638c8a9a5SSteve French rc = ib_post_send(info->id->qp, wr, NULL);
237738c8a9a5SSteve French if (rc) {
237838c8a9a5SSteve French log_rdma_mr(ERR, "ib_post_send failed rc=%x\n", rc);
237938c8a9a5SSteve French smbd_disconnect_rdma_connection(info);
238038c8a9a5SSteve French goto done;
238138c8a9a5SSteve French }
238238c8a9a5SSteve French wait_for_completion(&smbdirect_mr->invalidate_done);
238338c8a9a5SSteve French smbdirect_mr->need_invalidate = false;
238438c8a9a5SSteve French } else
238538c8a9a5SSteve French /*
238638c8a9a5SSteve French * For remote invalidation, just set it to MR_INVALIDATED
238738c8a9a5SSteve French * and defer to mr_recovery_work to recover the MR for next use
238838c8a9a5SSteve French */
238938c8a9a5SSteve French smbdirect_mr->state = MR_INVALIDATED;
239038c8a9a5SSteve French
239138c8a9a5SSteve French if (smbdirect_mr->state == MR_INVALIDATED) {
239238c8a9a5SSteve French ib_dma_unmap_sg(
239338c8a9a5SSteve French info->id->device, smbdirect_mr->sgt.sgl,
239438c8a9a5SSteve French smbdirect_mr->sgt.nents,
239538c8a9a5SSteve French smbdirect_mr->dir);
239638c8a9a5SSteve French smbdirect_mr->state = MR_READY;
239738c8a9a5SSteve French if (atomic_inc_return(&info->mr_ready_count) == 1)
239838c8a9a5SSteve French wake_up_interruptible(&info->wait_mr);
239938c8a9a5SSteve French } else
240038c8a9a5SSteve French /*
240138c8a9a5SSteve French * Schedule the work to do MR recovery for future I/Os MR
240238c8a9a5SSteve French * recovery is slow and don't want it to block current I/O
240338c8a9a5SSteve French */
240438c8a9a5SSteve French queue_work(info->workqueue, &info->mr_recovery_work);
240538c8a9a5SSteve French
240638c8a9a5SSteve French done:
240738c8a9a5SSteve French if (atomic_dec_and_test(&info->mr_used_count))
240838c8a9a5SSteve French wake_up(&info->wait_for_mr_cleanup);
240938c8a9a5SSteve French
241038c8a9a5SSteve French return rc;
241138c8a9a5SSteve French }
241238c8a9a5SSteve French
smb_set_sge(struct smb_extract_to_rdma * rdma,struct page * lowest_page,size_t off,size_t len)241338c8a9a5SSteve French static bool smb_set_sge(struct smb_extract_to_rdma *rdma,
241438c8a9a5SSteve French struct page *lowest_page, size_t off, size_t len)
241538c8a9a5SSteve French {
241638c8a9a5SSteve French struct ib_sge *sge = &rdma->sge[rdma->nr_sge];
241738c8a9a5SSteve French u64 addr;
241838c8a9a5SSteve French
241938c8a9a5SSteve French addr = ib_dma_map_page(rdma->device, lowest_page,
242038c8a9a5SSteve French off, len, rdma->direction);
242138c8a9a5SSteve French if (ib_dma_mapping_error(rdma->device, addr))
242238c8a9a5SSteve French return false;
242338c8a9a5SSteve French
242438c8a9a5SSteve French sge->addr = addr;
242538c8a9a5SSteve French sge->length = len;
242638c8a9a5SSteve French sge->lkey = rdma->local_dma_lkey;
242738c8a9a5SSteve French rdma->nr_sge++;
242838c8a9a5SSteve French return true;
242938c8a9a5SSteve French }
243038c8a9a5SSteve French
243138c8a9a5SSteve French /*
243238c8a9a5SSteve French * Extract page fragments from a BVEC-class iterator and add them to an RDMA
243338c8a9a5SSteve French * element list. The pages are not pinned.
243438c8a9a5SSteve French */
smb_extract_bvec_to_rdma(struct iov_iter * iter,struct smb_extract_to_rdma * rdma,ssize_t maxsize)243538c8a9a5SSteve French static ssize_t smb_extract_bvec_to_rdma(struct iov_iter *iter,
243638c8a9a5SSteve French struct smb_extract_to_rdma *rdma,
243738c8a9a5SSteve French ssize_t maxsize)
243838c8a9a5SSteve French {
243938c8a9a5SSteve French const struct bio_vec *bv = iter->bvec;
244038c8a9a5SSteve French unsigned long start = iter->iov_offset;
244138c8a9a5SSteve French unsigned int i;
244238c8a9a5SSteve French ssize_t ret = 0;
244338c8a9a5SSteve French
244438c8a9a5SSteve French for (i = 0; i < iter->nr_segs; i++) {
244538c8a9a5SSteve French size_t off, len;
244638c8a9a5SSteve French
244738c8a9a5SSteve French len = bv[i].bv_len;
244838c8a9a5SSteve French if (start >= len) {
244938c8a9a5SSteve French start -= len;
245038c8a9a5SSteve French continue;
245138c8a9a5SSteve French }
245238c8a9a5SSteve French
245338c8a9a5SSteve French len = min_t(size_t, maxsize, len - start);
245438c8a9a5SSteve French off = bv[i].bv_offset + start;
245538c8a9a5SSteve French
245638c8a9a5SSteve French if (!smb_set_sge(rdma, bv[i].bv_page, off, len))
245738c8a9a5SSteve French return -EIO;
245838c8a9a5SSteve French
245938c8a9a5SSteve French ret += len;
246038c8a9a5SSteve French maxsize -= len;
246138c8a9a5SSteve French if (rdma->nr_sge >= rdma->max_sge || maxsize <= 0)
246238c8a9a5SSteve French break;
246338c8a9a5SSteve French start = 0;
246438c8a9a5SSteve French }
246538c8a9a5SSteve French
246638c8a9a5SSteve French return ret;
246738c8a9a5SSteve French }
246838c8a9a5SSteve French
246938c8a9a5SSteve French /*
247038c8a9a5SSteve French * Extract fragments from a KVEC-class iterator and add them to an RDMA list.
247138c8a9a5SSteve French * This can deal with vmalloc'd buffers as well as kmalloc'd or static buffers.
247238c8a9a5SSteve French * The pages are not pinned.
247338c8a9a5SSteve French */
smb_extract_kvec_to_rdma(struct iov_iter * iter,struct smb_extract_to_rdma * rdma,ssize_t maxsize)247438c8a9a5SSteve French static ssize_t smb_extract_kvec_to_rdma(struct iov_iter *iter,
247538c8a9a5SSteve French struct smb_extract_to_rdma *rdma,
247638c8a9a5SSteve French ssize_t maxsize)
247738c8a9a5SSteve French {
247838c8a9a5SSteve French const struct kvec *kv = iter->kvec;
247938c8a9a5SSteve French unsigned long start = iter->iov_offset;
248038c8a9a5SSteve French unsigned int i;
248138c8a9a5SSteve French ssize_t ret = 0;
248238c8a9a5SSteve French
248338c8a9a5SSteve French for (i = 0; i < iter->nr_segs; i++) {
248438c8a9a5SSteve French struct page *page;
248538c8a9a5SSteve French unsigned long kaddr;
248638c8a9a5SSteve French size_t off, len, seg;
248738c8a9a5SSteve French
248838c8a9a5SSteve French len = kv[i].iov_len;
248938c8a9a5SSteve French if (start >= len) {
249038c8a9a5SSteve French start -= len;
249138c8a9a5SSteve French continue;
249238c8a9a5SSteve French }
249338c8a9a5SSteve French
249438c8a9a5SSteve French kaddr = (unsigned long)kv[i].iov_base + start;
249538c8a9a5SSteve French off = kaddr & ~PAGE_MASK;
249638c8a9a5SSteve French len = min_t(size_t, maxsize, len - start);
249738c8a9a5SSteve French kaddr &= PAGE_MASK;
249838c8a9a5SSteve French
249938c8a9a5SSteve French maxsize -= len;
250038c8a9a5SSteve French do {
250138c8a9a5SSteve French seg = min_t(size_t, len, PAGE_SIZE - off);
250238c8a9a5SSteve French
250338c8a9a5SSteve French if (is_vmalloc_or_module_addr((void *)kaddr))
250438c8a9a5SSteve French page = vmalloc_to_page((void *)kaddr);
250538c8a9a5SSteve French else
25063b1ddbb6SArnd Bergmann page = virt_to_page((void *)kaddr);
250738c8a9a5SSteve French
250838c8a9a5SSteve French if (!smb_set_sge(rdma, page, off, seg))
250938c8a9a5SSteve French return -EIO;
251038c8a9a5SSteve French
251138c8a9a5SSteve French ret += seg;
251238c8a9a5SSteve French len -= seg;
251338c8a9a5SSteve French kaddr += PAGE_SIZE;
251438c8a9a5SSteve French off = 0;
251538c8a9a5SSteve French } while (len > 0 && rdma->nr_sge < rdma->max_sge);
251638c8a9a5SSteve French
251738c8a9a5SSteve French if (rdma->nr_sge >= rdma->max_sge || maxsize <= 0)
251838c8a9a5SSteve French break;
251938c8a9a5SSteve French start = 0;
252038c8a9a5SSteve French }
252138c8a9a5SSteve French
252238c8a9a5SSteve French return ret;
252338c8a9a5SSteve French }
252438c8a9a5SSteve French
252538c8a9a5SSteve French /*
252638c8a9a5SSteve French * Extract folio fragments from an XARRAY-class iterator and add them to an
252738c8a9a5SSteve French * RDMA list. The folios are not pinned.
252838c8a9a5SSteve French */
smb_extract_xarray_to_rdma(struct iov_iter * iter,struct smb_extract_to_rdma * rdma,ssize_t maxsize)252938c8a9a5SSteve French static ssize_t smb_extract_xarray_to_rdma(struct iov_iter *iter,
253038c8a9a5SSteve French struct smb_extract_to_rdma *rdma,
253138c8a9a5SSteve French ssize_t maxsize)
253238c8a9a5SSteve French {
253338c8a9a5SSteve French struct xarray *xa = iter->xarray;
253438c8a9a5SSteve French struct folio *folio;
253538c8a9a5SSteve French loff_t start = iter->xarray_start + iter->iov_offset;
253638c8a9a5SSteve French pgoff_t index = start / PAGE_SIZE;
253738c8a9a5SSteve French ssize_t ret = 0;
253838c8a9a5SSteve French size_t off, len;
253938c8a9a5SSteve French XA_STATE(xas, xa, index);
254038c8a9a5SSteve French
254138c8a9a5SSteve French rcu_read_lock();
254238c8a9a5SSteve French
254338c8a9a5SSteve French xas_for_each(&xas, folio, ULONG_MAX) {
254438c8a9a5SSteve French if (xas_retry(&xas, folio))
254538c8a9a5SSteve French continue;
254638c8a9a5SSteve French if (WARN_ON(xa_is_value(folio)))
254738c8a9a5SSteve French break;
254838c8a9a5SSteve French if (WARN_ON(folio_test_hugetlb(folio)))
254938c8a9a5SSteve French break;
255038c8a9a5SSteve French
255138c8a9a5SSteve French off = offset_in_folio(folio, start);
255238c8a9a5SSteve French len = min_t(size_t, maxsize, folio_size(folio) - off);
255338c8a9a5SSteve French
255438c8a9a5SSteve French if (!smb_set_sge(rdma, folio_page(folio, 0), off, len)) {
255538c8a9a5SSteve French rcu_read_unlock();
255638c8a9a5SSteve French return -EIO;
255738c8a9a5SSteve French }
255838c8a9a5SSteve French
255938c8a9a5SSteve French maxsize -= len;
256038c8a9a5SSteve French ret += len;
256138c8a9a5SSteve French if (rdma->nr_sge >= rdma->max_sge || maxsize <= 0)
256238c8a9a5SSteve French break;
256338c8a9a5SSteve French }
256438c8a9a5SSteve French
256538c8a9a5SSteve French rcu_read_unlock();
256638c8a9a5SSteve French return ret;
256738c8a9a5SSteve French }
256838c8a9a5SSteve French
256938c8a9a5SSteve French /*
257038c8a9a5SSteve French * Extract page fragments from up to the given amount of the source iterator
257138c8a9a5SSteve French * and build up an RDMA list that refers to all of those bits. The RDMA list
257238c8a9a5SSteve French * is appended to, up to the maximum number of elements set in the parameter
257338c8a9a5SSteve French * block.
257438c8a9a5SSteve French *
257538c8a9a5SSteve French * The extracted page fragments are not pinned or ref'd in any way; if an
257638c8a9a5SSteve French * IOVEC/UBUF-type iterator is to be used, it should be converted to a
257738c8a9a5SSteve French * BVEC-type iterator and the pages pinned, ref'd or otherwise held in some
257838c8a9a5SSteve French * way.
257938c8a9a5SSteve French */
smb_extract_iter_to_rdma(struct iov_iter * iter,size_t len,struct smb_extract_to_rdma * rdma)258038c8a9a5SSteve French static ssize_t smb_extract_iter_to_rdma(struct iov_iter *iter, size_t len,
258138c8a9a5SSteve French struct smb_extract_to_rdma *rdma)
258238c8a9a5SSteve French {
258338c8a9a5SSteve French ssize_t ret;
258438c8a9a5SSteve French int before = rdma->nr_sge;
258538c8a9a5SSteve French
258638c8a9a5SSteve French switch (iov_iter_type(iter)) {
258738c8a9a5SSteve French case ITER_BVEC:
258838c8a9a5SSteve French ret = smb_extract_bvec_to_rdma(iter, rdma, len);
258938c8a9a5SSteve French break;
259038c8a9a5SSteve French case ITER_KVEC:
259138c8a9a5SSteve French ret = smb_extract_kvec_to_rdma(iter, rdma, len);
259238c8a9a5SSteve French break;
259338c8a9a5SSteve French case ITER_XARRAY:
259438c8a9a5SSteve French ret = smb_extract_xarray_to_rdma(iter, rdma, len);
259538c8a9a5SSteve French break;
259638c8a9a5SSteve French default:
259738c8a9a5SSteve French WARN_ON_ONCE(1);
259838c8a9a5SSteve French return -EIO;
259938c8a9a5SSteve French }
260038c8a9a5SSteve French
260138c8a9a5SSteve French if (ret > 0) {
260238c8a9a5SSteve French iov_iter_advance(iter, ret);
260338c8a9a5SSteve French } else if (ret < 0) {
260438c8a9a5SSteve French while (rdma->nr_sge > before) {
260538c8a9a5SSteve French struct ib_sge *sge = &rdma->sge[rdma->nr_sge--];
260638c8a9a5SSteve French
260738c8a9a5SSteve French ib_dma_unmap_single(rdma->device, sge->addr, sge->length,
260838c8a9a5SSteve French rdma->direction);
260938c8a9a5SSteve French sge->addr = 0;
261038c8a9a5SSteve French }
261138c8a9a5SSteve French }
261238c8a9a5SSteve French
261338c8a9a5SSteve French return ret;
261438c8a9a5SSteve French }
2615