162306a36Sopenharmony_ci/*
262306a36Sopenharmony_ci * Copyright (c) 2006 - 2009 Mellanox Technology Inc.  All rights reserved.
362306a36Sopenharmony_ci * Copyright (C) 2008 - 2011 Bart Van Assche <bvanassche@acm.org>.
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * This software is available to you under a choice of one of two
662306a36Sopenharmony_ci * licenses.  You may choose to be licensed under the terms of the GNU
762306a36Sopenharmony_ci * General Public License (GPL) Version 2, available from the file
862306a36Sopenharmony_ci * COPYING in the main directory of this source tree, or the
962306a36Sopenharmony_ci * OpenIB.org BSD license below:
1062306a36Sopenharmony_ci *
1162306a36Sopenharmony_ci *     Redistribution and use in source and binary forms, with or
1262306a36Sopenharmony_ci *     without modification, are permitted provided that the following
1362306a36Sopenharmony_ci *     conditions are met:
1462306a36Sopenharmony_ci *
1562306a36Sopenharmony_ci *      - Redistributions of source code must retain the above
1662306a36Sopenharmony_ci *        copyright notice, this list of conditions and the following
1762306a36Sopenharmony_ci *        disclaimer.
1862306a36Sopenharmony_ci *
1962306a36Sopenharmony_ci *      - Redistributions in binary form must reproduce the above
2062306a36Sopenharmony_ci *        copyright notice, this list of conditions and the following
2162306a36Sopenharmony_ci *        disclaimer in the documentation and/or other materials
2262306a36Sopenharmony_ci *        provided with the distribution.
2362306a36Sopenharmony_ci *
2462306a36Sopenharmony_ci * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
2562306a36Sopenharmony_ci * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
2662306a36Sopenharmony_ci * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
2762306a36Sopenharmony_ci * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
2862306a36Sopenharmony_ci * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
2962306a36Sopenharmony_ci * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
3062306a36Sopenharmony_ci * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
3162306a36Sopenharmony_ci * SOFTWARE.
3262306a36Sopenharmony_ci *
3362306a36Sopenharmony_ci */
3462306a36Sopenharmony_ci
3562306a36Sopenharmony_ci#include <linux/module.h>
3662306a36Sopenharmony_ci#include <linux/init.h>
3762306a36Sopenharmony_ci#include <linux/slab.h>
3862306a36Sopenharmony_ci#include <linux/err.h>
3962306a36Sopenharmony_ci#include <linux/ctype.h>
4062306a36Sopenharmony_ci#include <linux/kthread.h>
4162306a36Sopenharmony_ci#include <linux/string.h>
4262306a36Sopenharmony_ci#include <linux/delay.h>
4362306a36Sopenharmony_ci#include <linux/atomic.h>
4462306a36Sopenharmony_ci#include <linux/inet.h>
4562306a36Sopenharmony_ci#include <rdma/ib_cache.h>
4662306a36Sopenharmony_ci#include <scsi/scsi_proto.h>
4762306a36Sopenharmony_ci#include <scsi/scsi_tcq.h>
4862306a36Sopenharmony_ci#include <target/target_core_base.h>
4962306a36Sopenharmony_ci#include <target/target_core_fabric.h>
5062306a36Sopenharmony_ci#include "ib_srpt.h"
5162306a36Sopenharmony_ci
5262306a36Sopenharmony_ci/* Name of this kernel module. */
5362306a36Sopenharmony_ci#define DRV_NAME		"ib_srpt"
5462306a36Sopenharmony_ci
5562306a36Sopenharmony_ci#define SRPT_ID_STRING	"Linux SRP target"
5662306a36Sopenharmony_ci
5762306a36Sopenharmony_ci#undef pr_fmt
5862306a36Sopenharmony_ci#define pr_fmt(fmt) DRV_NAME " " fmt
5962306a36Sopenharmony_ci
6062306a36Sopenharmony_ciMODULE_AUTHOR("Vu Pham and Bart Van Assche");
6162306a36Sopenharmony_ciMODULE_DESCRIPTION("SCSI RDMA Protocol target driver");
6262306a36Sopenharmony_ciMODULE_LICENSE("Dual BSD/GPL");
6362306a36Sopenharmony_ci
6462306a36Sopenharmony_ci/*
6562306a36Sopenharmony_ci * Global Variables
6662306a36Sopenharmony_ci */
6762306a36Sopenharmony_ci
6862306a36Sopenharmony_cistatic u64 srpt_service_guid;
6962306a36Sopenharmony_cistatic DEFINE_SPINLOCK(srpt_dev_lock);	/* Protects srpt_dev_list. */
7062306a36Sopenharmony_cistatic LIST_HEAD(srpt_dev_list);	/* List of srpt_device structures. */
7162306a36Sopenharmony_ci
7262306a36Sopenharmony_cistatic unsigned srp_max_req_size = DEFAULT_MAX_REQ_SIZE;
7362306a36Sopenharmony_cimodule_param(srp_max_req_size, int, 0444);
7462306a36Sopenharmony_ciMODULE_PARM_DESC(srp_max_req_size,
7562306a36Sopenharmony_ci		 "Maximum size of SRP request messages in bytes.");
7662306a36Sopenharmony_ci
7762306a36Sopenharmony_cistatic int srpt_srq_size = DEFAULT_SRPT_SRQ_SIZE;
7862306a36Sopenharmony_cimodule_param(srpt_srq_size, int, 0444);
7962306a36Sopenharmony_ciMODULE_PARM_DESC(srpt_srq_size,
8062306a36Sopenharmony_ci		 "Shared receive queue (SRQ) size.");
8162306a36Sopenharmony_ci
8262306a36Sopenharmony_cistatic int srpt_set_u64_x(const char *buffer, const struct kernel_param *kp)
8362306a36Sopenharmony_ci{
8462306a36Sopenharmony_ci	return kstrtou64(buffer, 16, (u64 *)kp->arg);
8562306a36Sopenharmony_ci}
8662306a36Sopenharmony_cistatic int srpt_get_u64_x(char *buffer, const struct kernel_param *kp)
8762306a36Sopenharmony_ci{
8862306a36Sopenharmony_ci	return sprintf(buffer, "0x%016llx\n", *(u64 *)kp->arg);
8962306a36Sopenharmony_ci}
9062306a36Sopenharmony_cimodule_param_call(srpt_service_guid, srpt_set_u64_x, srpt_get_u64_x,
9162306a36Sopenharmony_ci		  &srpt_service_guid, 0444);
9262306a36Sopenharmony_ciMODULE_PARM_DESC(srpt_service_guid,
9362306a36Sopenharmony_ci		 "Using this value for ioc_guid, id_ext, and cm_listen_id instead of using the node_guid of the first HCA.");
9462306a36Sopenharmony_ci
9562306a36Sopenharmony_cistatic struct ib_client srpt_client;
9662306a36Sopenharmony_ci/* Protects both rdma_cm_port and rdma_cm_id. */
9762306a36Sopenharmony_cistatic DEFINE_MUTEX(rdma_cm_mutex);
9862306a36Sopenharmony_ci/* Port number RDMA/CM will bind to. */
9962306a36Sopenharmony_cistatic u16 rdma_cm_port;
10062306a36Sopenharmony_cistatic struct rdma_cm_id *rdma_cm_id;
10162306a36Sopenharmony_cistatic void srpt_release_cmd(struct se_cmd *se_cmd);
10262306a36Sopenharmony_cistatic void srpt_free_ch(struct kref *kref);
10362306a36Sopenharmony_cistatic int srpt_queue_status(struct se_cmd *cmd);
10462306a36Sopenharmony_cistatic void srpt_recv_done(struct ib_cq *cq, struct ib_wc *wc);
10562306a36Sopenharmony_cistatic void srpt_send_done(struct ib_cq *cq, struct ib_wc *wc);
10662306a36Sopenharmony_cistatic void srpt_process_wait_list(struct srpt_rdma_ch *ch);
10762306a36Sopenharmony_ci
10862306a36Sopenharmony_ci/*
10962306a36Sopenharmony_ci * The only allowed channel state changes are those that change the channel
11062306a36Sopenharmony_ci * state into a state with a higher numerical value. Hence the new > prev test.
11162306a36Sopenharmony_ci */
11262306a36Sopenharmony_cistatic bool srpt_set_ch_state(struct srpt_rdma_ch *ch, enum rdma_ch_state new)
11362306a36Sopenharmony_ci{
11462306a36Sopenharmony_ci	unsigned long flags;
11562306a36Sopenharmony_ci	enum rdma_ch_state prev;
11662306a36Sopenharmony_ci	bool changed = false;
11762306a36Sopenharmony_ci
11862306a36Sopenharmony_ci	spin_lock_irqsave(&ch->spinlock, flags);
11962306a36Sopenharmony_ci	prev = ch->state;
12062306a36Sopenharmony_ci	if (new > prev) {
12162306a36Sopenharmony_ci		ch->state = new;
12262306a36Sopenharmony_ci		changed = true;
12362306a36Sopenharmony_ci	}
12462306a36Sopenharmony_ci	spin_unlock_irqrestore(&ch->spinlock, flags);
12562306a36Sopenharmony_ci
12662306a36Sopenharmony_ci	return changed;
12762306a36Sopenharmony_ci}
12862306a36Sopenharmony_ci
12962306a36Sopenharmony_ci/**
13062306a36Sopenharmony_ci * srpt_event_handler - asynchronous IB event callback function
13162306a36Sopenharmony_ci * @handler: IB event handler registered by ib_register_event_handler().
13262306a36Sopenharmony_ci * @event: Description of the event that occurred.
13362306a36Sopenharmony_ci *
13462306a36Sopenharmony_ci * Callback function called by the InfiniBand core when an asynchronous IB
13562306a36Sopenharmony_ci * event occurs. This callback may occur in interrupt context. See also
13662306a36Sopenharmony_ci * section 11.5.2, Set Asynchronous Event Handler in the InfiniBand
13762306a36Sopenharmony_ci * Architecture Specification.
13862306a36Sopenharmony_ci */
13962306a36Sopenharmony_cistatic void srpt_event_handler(struct ib_event_handler *handler,
14062306a36Sopenharmony_ci			       struct ib_event *event)
14162306a36Sopenharmony_ci{
14262306a36Sopenharmony_ci	struct srpt_device *sdev =
14362306a36Sopenharmony_ci		container_of(handler, struct srpt_device, event_handler);
14462306a36Sopenharmony_ci	struct srpt_port *sport;
14562306a36Sopenharmony_ci	u8 port_num;
14662306a36Sopenharmony_ci
14762306a36Sopenharmony_ci	pr_debug("ASYNC event= %d on device= %s\n", event->event,
14862306a36Sopenharmony_ci		 dev_name(&sdev->device->dev));
14962306a36Sopenharmony_ci
15062306a36Sopenharmony_ci	switch (event->event) {
15162306a36Sopenharmony_ci	case IB_EVENT_PORT_ERR:
15262306a36Sopenharmony_ci		port_num = event->element.port_num - 1;
15362306a36Sopenharmony_ci		if (port_num < sdev->device->phys_port_cnt) {
15462306a36Sopenharmony_ci			sport = &sdev->port[port_num];
15562306a36Sopenharmony_ci			sport->lid = 0;
15662306a36Sopenharmony_ci			sport->sm_lid = 0;
15762306a36Sopenharmony_ci		} else {
15862306a36Sopenharmony_ci			WARN(true, "event %d: port_num %d out of range 1..%d\n",
15962306a36Sopenharmony_ci			     event->event, port_num + 1,
16062306a36Sopenharmony_ci			     sdev->device->phys_port_cnt);
16162306a36Sopenharmony_ci		}
16262306a36Sopenharmony_ci		break;
16362306a36Sopenharmony_ci	case IB_EVENT_PORT_ACTIVE:
16462306a36Sopenharmony_ci	case IB_EVENT_LID_CHANGE:
16562306a36Sopenharmony_ci	case IB_EVENT_PKEY_CHANGE:
16662306a36Sopenharmony_ci	case IB_EVENT_SM_CHANGE:
16762306a36Sopenharmony_ci	case IB_EVENT_CLIENT_REREGISTER:
16862306a36Sopenharmony_ci	case IB_EVENT_GID_CHANGE:
16962306a36Sopenharmony_ci		/* Refresh port data asynchronously. */
17062306a36Sopenharmony_ci		port_num = event->element.port_num - 1;
17162306a36Sopenharmony_ci		if (port_num < sdev->device->phys_port_cnt) {
17262306a36Sopenharmony_ci			sport = &sdev->port[port_num];
17362306a36Sopenharmony_ci			if (!sport->lid && !sport->sm_lid)
17462306a36Sopenharmony_ci				schedule_work(&sport->work);
17562306a36Sopenharmony_ci		} else {
17662306a36Sopenharmony_ci			WARN(true, "event %d: port_num %d out of range 1..%d\n",
17762306a36Sopenharmony_ci			     event->event, port_num + 1,
17862306a36Sopenharmony_ci			     sdev->device->phys_port_cnt);
17962306a36Sopenharmony_ci		}
18062306a36Sopenharmony_ci		break;
18162306a36Sopenharmony_ci	default:
18262306a36Sopenharmony_ci		pr_err("received unrecognized IB event %d\n", event->event);
18362306a36Sopenharmony_ci		break;
18462306a36Sopenharmony_ci	}
18562306a36Sopenharmony_ci}
18662306a36Sopenharmony_ci
18762306a36Sopenharmony_ci/**
18862306a36Sopenharmony_ci * srpt_srq_event - SRQ event callback function
18962306a36Sopenharmony_ci * @event: Description of the event that occurred.
19062306a36Sopenharmony_ci * @ctx: Context pointer specified at SRQ creation time.
19162306a36Sopenharmony_ci */
19262306a36Sopenharmony_cistatic void srpt_srq_event(struct ib_event *event, void *ctx)
19362306a36Sopenharmony_ci{
19462306a36Sopenharmony_ci	pr_debug("SRQ event %d\n", event->event);
19562306a36Sopenharmony_ci}
19662306a36Sopenharmony_ci
19762306a36Sopenharmony_cistatic const char *get_ch_state_name(enum rdma_ch_state s)
19862306a36Sopenharmony_ci{
19962306a36Sopenharmony_ci	switch (s) {
20062306a36Sopenharmony_ci	case CH_CONNECTING:
20162306a36Sopenharmony_ci		return "connecting";
20262306a36Sopenharmony_ci	case CH_LIVE:
20362306a36Sopenharmony_ci		return "live";
20462306a36Sopenharmony_ci	case CH_DISCONNECTING:
20562306a36Sopenharmony_ci		return "disconnecting";
20662306a36Sopenharmony_ci	case CH_DRAINING:
20762306a36Sopenharmony_ci		return "draining";
20862306a36Sopenharmony_ci	case CH_DISCONNECTED:
20962306a36Sopenharmony_ci		return "disconnected";
21062306a36Sopenharmony_ci	}
21162306a36Sopenharmony_ci	return "???";
21262306a36Sopenharmony_ci}
21362306a36Sopenharmony_ci
21462306a36Sopenharmony_ci/**
21562306a36Sopenharmony_ci * srpt_qp_event - QP event callback function
21662306a36Sopenharmony_ci * @event: Description of the event that occurred.
21762306a36Sopenharmony_ci * @ptr: SRPT RDMA channel.
21862306a36Sopenharmony_ci */
21962306a36Sopenharmony_cistatic void srpt_qp_event(struct ib_event *event, void *ptr)
22062306a36Sopenharmony_ci{
22162306a36Sopenharmony_ci	struct srpt_rdma_ch *ch = ptr;
22262306a36Sopenharmony_ci
22362306a36Sopenharmony_ci	pr_debug("QP event %d on ch=%p sess_name=%s-%d state=%s\n",
22462306a36Sopenharmony_ci		 event->event, ch, ch->sess_name, ch->qp->qp_num,
22562306a36Sopenharmony_ci		 get_ch_state_name(ch->state));
22662306a36Sopenharmony_ci
22762306a36Sopenharmony_ci	switch (event->event) {
22862306a36Sopenharmony_ci	case IB_EVENT_COMM_EST:
22962306a36Sopenharmony_ci		if (ch->using_rdma_cm)
23062306a36Sopenharmony_ci			rdma_notify(ch->rdma_cm.cm_id, event->event);
23162306a36Sopenharmony_ci		else
23262306a36Sopenharmony_ci			ib_cm_notify(ch->ib_cm.cm_id, event->event);
23362306a36Sopenharmony_ci		break;
23462306a36Sopenharmony_ci	case IB_EVENT_QP_LAST_WQE_REACHED:
23562306a36Sopenharmony_ci		pr_debug("%s-%d, state %s: received Last WQE event.\n",
23662306a36Sopenharmony_ci			 ch->sess_name, ch->qp->qp_num,
23762306a36Sopenharmony_ci			 get_ch_state_name(ch->state));
23862306a36Sopenharmony_ci		break;
23962306a36Sopenharmony_ci	default:
24062306a36Sopenharmony_ci		pr_err("received unrecognized IB QP event %d\n", event->event);
24162306a36Sopenharmony_ci		break;
24262306a36Sopenharmony_ci	}
24362306a36Sopenharmony_ci}
24462306a36Sopenharmony_ci
24562306a36Sopenharmony_ci/**
24662306a36Sopenharmony_ci * srpt_set_ioc - initialize a IOUnitInfo structure
24762306a36Sopenharmony_ci * @c_list: controller list.
24862306a36Sopenharmony_ci * @slot: one-based slot number.
24962306a36Sopenharmony_ci * @value: four-bit value.
25062306a36Sopenharmony_ci *
25162306a36Sopenharmony_ci * Copies the lowest four bits of value in element slot of the array of four
25262306a36Sopenharmony_ci * bit elements called c_list (controller list). The index slot is one-based.
25362306a36Sopenharmony_ci */
25462306a36Sopenharmony_cistatic void srpt_set_ioc(u8 *c_list, u32 slot, u8 value)
25562306a36Sopenharmony_ci{
25662306a36Sopenharmony_ci	u16 id;
25762306a36Sopenharmony_ci	u8 tmp;
25862306a36Sopenharmony_ci
25962306a36Sopenharmony_ci	id = (slot - 1) / 2;
26062306a36Sopenharmony_ci	if (slot & 0x1) {
26162306a36Sopenharmony_ci		tmp = c_list[id] & 0xf;
26262306a36Sopenharmony_ci		c_list[id] = (value << 4) | tmp;
26362306a36Sopenharmony_ci	} else {
26462306a36Sopenharmony_ci		tmp = c_list[id] & 0xf0;
26562306a36Sopenharmony_ci		c_list[id] = (value & 0xf) | tmp;
26662306a36Sopenharmony_ci	}
26762306a36Sopenharmony_ci}
26862306a36Sopenharmony_ci
26962306a36Sopenharmony_ci/**
27062306a36Sopenharmony_ci * srpt_get_class_port_info - copy ClassPortInfo to a management datagram
27162306a36Sopenharmony_ci * @mad: Datagram that will be sent as response to DM_ATTR_CLASS_PORT_INFO.
27262306a36Sopenharmony_ci *
27362306a36Sopenharmony_ci * See also section 16.3.3.1 ClassPortInfo in the InfiniBand Architecture
27462306a36Sopenharmony_ci * Specification.
27562306a36Sopenharmony_ci */
27662306a36Sopenharmony_cistatic void srpt_get_class_port_info(struct ib_dm_mad *mad)
27762306a36Sopenharmony_ci{
27862306a36Sopenharmony_ci	struct ib_class_port_info *cif;
27962306a36Sopenharmony_ci
28062306a36Sopenharmony_ci	cif = (struct ib_class_port_info *)mad->data;
28162306a36Sopenharmony_ci	memset(cif, 0, sizeof(*cif));
28262306a36Sopenharmony_ci	cif->base_version = 1;
28362306a36Sopenharmony_ci	cif->class_version = 1;
28462306a36Sopenharmony_ci
28562306a36Sopenharmony_ci	ib_set_cpi_resp_time(cif, 20);
28662306a36Sopenharmony_ci	mad->mad_hdr.status = 0;
28762306a36Sopenharmony_ci}
28862306a36Sopenharmony_ci
28962306a36Sopenharmony_ci/**
29062306a36Sopenharmony_ci * srpt_get_iou - write IOUnitInfo to a management datagram
29162306a36Sopenharmony_ci * @mad: Datagram that will be sent as response to DM_ATTR_IOU_INFO.
29262306a36Sopenharmony_ci *
29362306a36Sopenharmony_ci * See also section 16.3.3.3 IOUnitInfo in the InfiniBand Architecture
29462306a36Sopenharmony_ci * Specification. See also section B.7, table B.6 in the SRP r16a document.
29562306a36Sopenharmony_ci */
29662306a36Sopenharmony_cistatic void srpt_get_iou(struct ib_dm_mad *mad)
29762306a36Sopenharmony_ci{
29862306a36Sopenharmony_ci	struct ib_dm_iou_info *ioui;
29962306a36Sopenharmony_ci	u8 slot;
30062306a36Sopenharmony_ci	int i;
30162306a36Sopenharmony_ci
30262306a36Sopenharmony_ci	ioui = (struct ib_dm_iou_info *)mad->data;
30362306a36Sopenharmony_ci	ioui->change_id = cpu_to_be16(1);
30462306a36Sopenharmony_ci	ioui->max_controllers = 16;
30562306a36Sopenharmony_ci
30662306a36Sopenharmony_ci	/* set present for slot 1 and empty for the rest */
30762306a36Sopenharmony_ci	srpt_set_ioc(ioui->controller_list, 1, 1);
30862306a36Sopenharmony_ci	for (i = 1, slot = 2; i < 16; i++, slot++)
30962306a36Sopenharmony_ci		srpt_set_ioc(ioui->controller_list, slot, 0);
31062306a36Sopenharmony_ci
31162306a36Sopenharmony_ci	mad->mad_hdr.status = 0;
31262306a36Sopenharmony_ci}
31362306a36Sopenharmony_ci
31462306a36Sopenharmony_ci/**
31562306a36Sopenharmony_ci * srpt_get_ioc - write IOControllerprofile to a management datagram
31662306a36Sopenharmony_ci * @sport: HCA port through which the MAD has been received.
31762306a36Sopenharmony_ci * @slot: Slot number specified in DM_ATTR_IOC_PROFILE query.
31862306a36Sopenharmony_ci * @mad: Datagram that will be sent as response to DM_ATTR_IOC_PROFILE.
31962306a36Sopenharmony_ci *
32062306a36Sopenharmony_ci * See also section 16.3.3.4 IOControllerProfile in the InfiniBand
32162306a36Sopenharmony_ci * Architecture Specification. See also section B.7, table B.7 in the SRP
32262306a36Sopenharmony_ci * r16a document.
32362306a36Sopenharmony_ci */
32462306a36Sopenharmony_cistatic void srpt_get_ioc(struct srpt_port *sport, u32 slot,
32562306a36Sopenharmony_ci			 struct ib_dm_mad *mad)
32662306a36Sopenharmony_ci{
32762306a36Sopenharmony_ci	struct srpt_device *sdev = sport->sdev;
32862306a36Sopenharmony_ci	struct ib_dm_ioc_profile *iocp;
32962306a36Sopenharmony_ci	int send_queue_depth;
33062306a36Sopenharmony_ci
33162306a36Sopenharmony_ci	iocp = (struct ib_dm_ioc_profile *)mad->data;
33262306a36Sopenharmony_ci
33362306a36Sopenharmony_ci	if (!slot || slot > 16) {
33462306a36Sopenharmony_ci		mad->mad_hdr.status
33562306a36Sopenharmony_ci			= cpu_to_be16(DM_MAD_STATUS_INVALID_FIELD);
33662306a36Sopenharmony_ci		return;
33762306a36Sopenharmony_ci	}
33862306a36Sopenharmony_ci
33962306a36Sopenharmony_ci	if (slot > 2) {
34062306a36Sopenharmony_ci		mad->mad_hdr.status
34162306a36Sopenharmony_ci			= cpu_to_be16(DM_MAD_STATUS_NO_IOC);
34262306a36Sopenharmony_ci		return;
34362306a36Sopenharmony_ci	}
34462306a36Sopenharmony_ci
34562306a36Sopenharmony_ci	if (sdev->use_srq)
34662306a36Sopenharmony_ci		send_queue_depth = sdev->srq_size;
34762306a36Sopenharmony_ci	else
34862306a36Sopenharmony_ci		send_queue_depth = min(MAX_SRPT_RQ_SIZE,
34962306a36Sopenharmony_ci				       sdev->device->attrs.max_qp_wr);
35062306a36Sopenharmony_ci
35162306a36Sopenharmony_ci	memset(iocp, 0, sizeof(*iocp));
35262306a36Sopenharmony_ci	strcpy(iocp->id_string, SRPT_ID_STRING);
35362306a36Sopenharmony_ci	iocp->guid = cpu_to_be64(srpt_service_guid);
35462306a36Sopenharmony_ci	iocp->vendor_id = cpu_to_be32(sdev->device->attrs.vendor_id);
35562306a36Sopenharmony_ci	iocp->device_id = cpu_to_be32(sdev->device->attrs.vendor_part_id);
35662306a36Sopenharmony_ci	iocp->device_version = cpu_to_be16(sdev->device->attrs.hw_ver);
35762306a36Sopenharmony_ci	iocp->subsys_vendor_id = cpu_to_be32(sdev->device->attrs.vendor_id);
35862306a36Sopenharmony_ci	iocp->subsys_device_id = 0x0;
35962306a36Sopenharmony_ci	iocp->io_class = cpu_to_be16(SRP_REV16A_IB_IO_CLASS);
36062306a36Sopenharmony_ci	iocp->io_subclass = cpu_to_be16(SRP_IO_SUBCLASS);
36162306a36Sopenharmony_ci	iocp->protocol = cpu_to_be16(SRP_PROTOCOL);
36262306a36Sopenharmony_ci	iocp->protocol_version = cpu_to_be16(SRP_PROTOCOL_VERSION);
36362306a36Sopenharmony_ci	iocp->send_queue_depth = cpu_to_be16(send_queue_depth);
36462306a36Sopenharmony_ci	iocp->rdma_read_depth = 4;
36562306a36Sopenharmony_ci	iocp->send_size = cpu_to_be32(srp_max_req_size);
36662306a36Sopenharmony_ci	iocp->rdma_size = cpu_to_be32(min(sport->port_attrib.srp_max_rdma_size,
36762306a36Sopenharmony_ci					  1U << 24));
36862306a36Sopenharmony_ci	iocp->num_svc_entries = 1;
36962306a36Sopenharmony_ci	iocp->op_cap_mask = SRP_SEND_TO_IOC | SRP_SEND_FROM_IOC |
37062306a36Sopenharmony_ci		SRP_RDMA_READ_FROM_IOC | SRP_RDMA_WRITE_FROM_IOC;
37162306a36Sopenharmony_ci
37262306a36Sopenharmony_ci	mad->mad_hdr.status = 0;
37362306a36Sopenharmony_ci}
37462306a36Sopenharmony_ci
37562306a36Sopenharmony_ci/**
37662306a36Sopenharmony_ci * srpt_get_svc_entries - write ServiceEntries to a management datagram
37762306a36Sopenharmony_ci * @ioc_guid: I/O controller GUID to use in reply.
37862306a36Sopenharmony_ci * @slot: I/O controller number.
37962306a36Sopenharmony_ci * @hi: End of the range of service entries to be specified in the reply.
38062306a36Sopenharmony_ci * @lo: Start of the range of service entries to be specified in the reply..
38162306a36Sopenharmony_ci * @mad: Datagram that will be sent as response to DM_ATTR_SVC_ENTRIES.
38262306a36Sopenharmony_ci *
38362306a36Sopenharmony_ci * See also section 16.3.3.5 ServiceEntries in the InfiniBand Architecture
38462306a36Sopenharmony_ci * Specification. See also section B.7, table B.8 in the SRP r16a document.
38562306a36Sopenharmony_ci */
38662306a36Sopenharmony_cistatic void srpt_get_svc_entries(u64 ioc_guid,
38762306a36Sopenharmony_ci				 u16 slot, u8 hi, u8 lo, struct ib_dm_mad *mad)
38862306a36Sopenharmony_ci{
38962306a36Sopenharmony_ci	struct ib_dm_svc_entries *svc_entries;
39062306a36Sopenharmony_ci
39162306a36Sopenharmony_ci	WARN_ON(!ioc_guid);
39262306a36Sopenharmony_ci
39362306a36Sopenharmony_ci	if (!slot || slot > 16) {
39462306a36Sopenharmony_ci		mad->mad_hdr.status
39562306a36Sopenharmony_ci			= cpu_to_be16(DM_MAD_STATUS_INVALID_FIELD);
39662306a36Sopenharmony_ci		return;
39762306a36Sopenharmony_ci	}
39862306a36Sopenharmony_ci
39962306a36Sopenharmony_ci	if (slot > 2 || lo > hi || hi > 1) {
40062306a36Sopenharmony_ci		mad->mad_hdr.status
40162306a36Sopenharmony_ci			= cpu_to_be16(DM_MAD_STATUS_NO_IOC);
40262306a36Sopenharmony_ci		return;
40362306a36Sopenharmony_ci	}
40462306a36Sopenharmony_ci
40562306a36Sopenharmony_ci	svc_entries = (struct ib_dm_svc_entries *)mad->data;
40662306a36Sopenharmony_ci	memset(svc_entries, 0, sizeof(*svc_entries));
40762306a36Sopenharmony_ci	svc_entries->service_entries[0].id = cpu_to_be64(ioc_guid);
40862306a36Sopenharmony_ci	snprintf(svc_entries->service_entries[0].name,
40962306a36Sopenharmony_ci		 sizeof(svc_entries->service_entries[0].name),
41062306a36Sopenharmony_ci		 "%s%016llx",
41162306a36Sopenharmony_ci		 SRP_SERVICE_NAME_PREFIX,
41262306a36Sopenharmony_ci		 ioc_guid);
41362306a36Sopenharmony_ci
41462306a36Sopenharmony_ci	mad->mad_hdr.status = 0;
41562306a36Sopenharmony_ci}
41662306a36Sopenharmony_ci
41762306a36Sopenharmony_ci/**
41862306a36Sopenharmony_ci * srpt_mgmt_method_get - process a received management datagram
41962306a36Sopenharmony_ci * @sp:      HCA port through which the MAD has been received.
42062306a36Sopenharmony_ci * @rq_mad:  received MAD.
42162306a36Sopenharmony_ci * @rsp_mad: response MAD.
42262306a36Sopenharmony_ci */
42362306a36Sopenharmony_cistatic void srpt_mgmt_method_get(struct srpt_port *sp, struct ib_mad *rq_mad,
42462306a36Sopenharmony_ci				 struct ib_dm_mad *rsp_mad)
42562306a36Sopenharmony_ci{
42662306a36Sopenharmony_ci	u16 attr_id;
42762306a36Sopenharmony_ci	u32 slot;
42862306a36Sopenharmony_ci	u8 hi, lo;
42962306a36Sopenharmony_ci
43062306a36Sopenharmony_ci	attr_id = be16_to_cpu(rq_mad->mad_hdr.attr_id);
43162306a36Sopenharmony_ci	switch (attr_id) {
43262306a36Sopenharmony_ci	case DM_ATTR_CLASS_PORT_INFO:
43362306a36Sopenharmony_ci		srpt_get_class_port_info(rsp_mad);
43462306a36Sopenharmony_ci		break;
43562306a36Sopenharmony_ci	case DM_ATTR_IOU_INFO:
43662306a36Sopenharmony_ci		srpt_get_iou(rsp_mad);
43762306a36Sopenharmony_ci		break;
43862306a36Sopenharmony_ci	case DM_ATTR_IOC_PROFILE:
43962306a36Sopenharmony_ci		slot = be32_to_cpu(rq_mad->mad_hdr.attr_mod);
44062306a36Sopenharmony_ci		srpt_get_ioc(sp, slot, rsp_mad);
44162306a36Sopenharmony_ci		break;
44262306a36Sopenharmony_ci	case DM_ATTR_SVC_ENTRIES:
44362306a36Sopenharmony_ci		slot = be32_to_cpu(rq_mad->mad_hdr.attr_mod);
44462306a36Sopenharmony_ci		hi = (u8) ((slot >> 8) & 0xff);
44562306a36Sopenharmony_ci		lo = (u8) (slot & 0xff);
44662306a36Sopenharmony_ci		slot = (u16) ((slot >> 16) & 0xffff);
44762306a36Sopenharmony_ci		srpt_get_svc_entries(srpt_service_guid,
44862306a36Sopenharmony_ci				     slot, hi, lo, rsp_mad);
44962306a36Sopenharmony_ci		break;
45062306a36Sopenharmony_ci	default:
45162306a36Sopenharmony_ci		rsp_mad->mad_hdr.status =
45262306a36Sopenharmony_ci		    cpu_to_be16(DM_MAD_STATUS_UNSUP_METHOD_ATTR);
45362306a36Sopenharmony_ci		break;
45462306a36Sopenharmony_ci	}
45562306a36Sopenharmony_ci}
45662306a36Sopenharmony_ci
45762306a36Sopenharmony_ci/**
45862306a36Sopenharmony_ci * srpt_mad_send_handler - MAD send completion callback
45962306a36Sopenharmony_ci * @mad_agent: Return value of ib_register_mad_agent().
46062306a36Sopenharmony_ci * @mad_wc: Work completion reporting that the MAD has been sent.
46162306a36Sopenharmony_ci */
46262306a36Sopenharmony_cistatic void srpt_mad_send_handler(struct ib_mad_agent *mad_agent,
46362306a36Sopenharmony_ci				  struct ib_mad_send_wc *mad_wc)
46462306a36Sopenharmony_ci{
46562306a36Sopenharmony_ci	rdma_destroy_ah(mad_wc->send_buf->ah, RDMA_DESTROY_AH_SLEEPABLE);
46662306a36Sopenharmony_ci	ib_free_send_mad(mad_wc->send_buf);
46762306a36Sopenharmony_ci}
46862306a36Sopenharmony_ci
46962306a36Sopenharmony_ci/**
47062306a36Sopenharmony_ci * srpt_mad_recv_handler - MAD reception callback function
47162306a36Sopenharmony_ci * @mad_agent: Return value of ib_register_mad_agent().
47262306a36Sopenharmony_ci * @send_buf: Not used.
47362306a36Sopenharmony_ci * @mad_wc: Work completion reporting that a MAD has been received.
47462306a36Sopenharmony_ci */
47562306a36Sopenharmony_cistatic void srpt_mad_recv_handler(struct ib_mad_agent *mad_agent,
47662306a36Sopenharmony_ci				  struct ib_mad_send_buf *send_buf,
47762306a36Sopenharmony_ci				  struct ib_mad_recv_wc *mad_wc)
47862306a36Sopenharmony_ci{
47962306a36Sopenharmony_ci	struct srpt_port *sport = (struct srpt_port *)mad_agent->context;
48062306a36Sopenharmony_ci	struct ib_ah *ah;
48162306a36Sopenharmony_ci	struct ib_mad_send_buf *rsp;
48262306a36Sopenharmony_ci	struct ib_dm_mad *dm_mad;
48362306a36Sopenharmony_ci
48462306a36Sopenharmony_ci	if (!mad_wc || !mad_wc->recv_buf.mad)
48562306a36Sopenharmony_ci		return;
48662306a36Sopenharmony_ci
48762306a36Sopenharmony_ci	ah = ib_create_ah_from_wc(mad_agent->qp->pd, mad_wc->wc,
48862306a36Sopenharmony_ci				  mad_wc->recv_buf.grh, mad_agent->port_num);
48962306a36Sopenharmony_ci	if (IS_ERR(ah))
49062306a36Sopenharmony_ci		goto err;
49162306a36Sopenharmony_ci
49262306a36Sopenharmony_ci	BUILD_BUG_ON(offsetof(struct ib_dm_mad, data) != IB_MGMT_DEVICE_HDR);
49362306a36Sopenharmony_ci
49462306a36Sopenharmony_ci	rsp = ib_create_send_mad(mad_agent, mad_wc->wc->src_qp,
49562306a36Sopenharmony_ci				 mad_wc->wc->pkey_index, 0,
49662306a36Sopenharmony_ci				 IB_MGMT_DEVICE_HDR, IB_MGMT_DEVICE_DATA,
49762306a36Sopenharmony_ci				 GFP_KERNEL,
49862306a36Sopenharmony_ci				 IB_MGMT_BASE_VERSION);
49962306a36Sopenharmony_ci	if (IS_ERR(rsp))
50062306a36Sopenharmony_ci		goto err_rsp;
50162306a36Sopenharmony_ci
50262306a36Sopenharmony_ci	rsp->ah = ah;
50362306a36Sopenharmony_ci
50462306a36Sopenharmony_ci	dm_mad = rsp->mad;
50562306a36Sopenharmony_ci	memcpy(dm_mad, mad_wc->recv_buf.mad, sizeof(*dm_mad));
50662306a36Sopenharmony_ci	dm_mad->mad_hdr.method = IB_MGMT_METHOD_GET_RESP;
50762306a36Sopenharmony_ci	dm_mad->mad_hdr.status = 0;
50862306a36Sopenharmony_ci
50962306a36Sopenharmony_ci	switch (mad_wc->recv_buf.mad->mad_hdr.method) {
51062306a36Sopenharmony_ci	case IB_MGMT_METHOD_GET:
51162306a36Sopenharmony_ci		srpt_mgmt_method_get(sport, mad_wc->recv_buf.mad, dm_mad);
51262306a36Sopenharmony_ci		break;
51362306a36Sopenharmony_ci	case IB_MGMT_METHOD_SET:
51462306a36Sopenharmony_ci		dm_mad->mad_hdr.status =
51562306a36Sopenharmony_ci		    cpu_to_be16(DM_MAD_STATUS_UNSUP_METHOD_ATTR);
51662306a36Sopenharmony_ci		break;
51762306a36Sopenharmony_ci	default:
51862306a36Sopenharmony_ci		dm_mad->mad_hdr.status =
51962306a36Sopenharmony_ci		    cpu_to_be16(DM_MAD_STATUS_UNSUP_METHOD);
52062306a36Sopenharmony_ci		break;
52162306a36Sopenharmony_ci	}
52262306a36Sopenharmony_ci
52362306a36Sopenharmony_ci	if (!ib_post_send_mad(rsp, NULL)) {
52462306a36Sopenharmony_ci		ib_free_recv_mad(mad_wc);
52562306a36Sopenharmony_ci		/* will destroy_ah & free_send_mad in send completion */
52662306a36Sopenharmony_ci		return;
52762306a36Sopenharmony_ci	}
52862306a36Sopenharmony_ci
52962306a36Sopenharmony_ci	ib_free_send_mad(rsp);
53062306a36Sopenharmony_ci
53162306a36Sopenharmony_cierr_rsp:
53262306a36Sopenharmony_ci	rdma_destroy_ah(ah, RDMA_DESTROY_AH_SLEEPABLE);
53362306a36Sopenharmony_cierr:
53462306a36Sopenharmony_ci	ib_free_recv_mad(mad_wc);
53562306a36Sopenharmony_ci}
53662306a36Sopenharmony_ci
53762306a36Sopenharmony_cistatic int srpt_format_guid(char *buf, unsigned int size, const __be64 *guid)
53862306a36Sopenharmony_ci{
53962306a36Sopenharmony_ci	const __be16 *g = (const __be16 *)guid;
54062306a36Sopenharmony_ci
54162306a36Sopenharmony_ci	return snprintf(buf, size, "%04x:%04x:%04x:%04x",
54262306a36Sopenharmony_ci			be16_to_cpu(g[0]), be16_to_cpu(g[1]),
54362306a36Sopenharmony_ci			be16_to_cpu(g[2]), be16_to_cpu(g[3]));
54462306a36Sopenharmony_ci}
54562306a36Sopenharmony_ci
54662306a36Sopenharmony_ci/**
54762306a36Sopenharmony_ci * srpt_refresh_port - configure a HCA port
54862306a36Sopenharmony_ci * @sport: SRPT HCA port.
54962306a36Sopenharmony_ci *
55062306a36Sopenharmony_ci * Enable InfiniBand management datagram processing, update the cached sm_lid,
55162306a36Sopenharmony_ci * lid and gid values, and register a callback function for processing MADs
55262306a36Sopenharmony_ci * on the specified port.
55362306a36Sopenharmony_ci *
55462306a36Sopenharmony_ci * Note: It is safe to call this function more than once for the same port.
55562306a36Sopenharmony_ci */
55662306a36Sopenharmony_cistatic int srpt_refresh_port(struct srpt_port *sport)
55762306a36Sopenharmony_ci{
55862306a36Sopenharmony_ci	struct ib_mad_agent *mad_agent;
55962306a36Sopenharmony_ci	struct ib_mad_reg_req reg_req;
56062306a36Sopenharmony_ci	struct ib_port_modify port_modify;
56162306a36Sopenharmony_ci	struct ib_port_attr port_attr;
56262306a36Sopenharmony_ci	int ret;
56362306a36Sopenharmony_ci
56462306a36Sopenharmony_ci	ret = ib_query_port(sport->sdev->device, sport->port, &port_attr);
56562306a36Sopenharmony_ci	if (ret)
56662306a36Sopenharmony_ci		return ret;
56762306a36Sopenharmony_ci
56862306a36Sopenharmony_ci	sport->sm_lid = port_attr.sm_lid;
56962306a36Sopenharmony_ci	sport->lid = port_attr.lid;
57062306a36Sopenharmony_ci
57162306a36Sopenharmony_ci	ret = rdma_query_gid(sport->sdev->device, sport->port, 0, &sport->gid);
57262306a36Sopenharmony_ci	if (ret)
57362306a36Sopenharmony_ci		return ret;
57462306a36Sopenharmony_ci
57562306a36Sopenharmony_ci	srpt_format_guid(sport->guid_name, ARRAY_SIZE(sport->guid_name),
57662306a36Sopenharmony_ci			 &sport->gid.global.interface_id);
57762306a36Sopenharmony_ci	snprintf(sport->gid_name, ARRAY_SIZE(sport->gid_name),
57862306a36Sopenharmony_ci		 "0x%016llx%016llx",
57962306a36Sopenharmony_ci		 be64_to_cpu(sport->gid.global.subnet_prefix),
58062306a36Sopenharmony_ci		 be64_to_cpu(sport->gid.global.interface_id));
58162306a36Sopenharmony_ci
58262306a36Sopenharmony_ci	if (rdma_protocol_iwarp(sport->sdev->device, sport->port))
58362306a36Sopenharmony_ci		return 0;
58462306a36Sopenharmony_ci
58562306a36Sopenharmony_ci	memset(&port_modify, 0, sizeof(port_modify));
58662306a36Sopenharmony_ci	port_modify.set_port_cap_mask = IB_PORT_DEVICE_MGMT_SUP;
58762306a36Sopenharmony_ci	port_modify.clr_port_cap_mask = 0;
58862306a36Sopenharmony_ci
58962306a36Sopenharmony_ci	ret = ib_modify_port(sport->sdev->device, sport->port, 0, &port_modify);
59062306a36Sopenharmony_ci	if (ret) {
59162306a36Sopenharmony_ci		pr_warn("%s-%d: enabling device management failed (%d). Note: this is expected if SR-IOV is enabled.\n",
59262306a36Sopenharmony_ci			dev_name(&sport->sdev->device->dev), sport->port, ret);
59362306a36Sopenharmony_ci		return 0;
59462306a36Sopenharmony_ci	}
59562306a36Sopenharmony_ci
59662306a36Sopenharmony_ci	if (!sport->mad_agent) {
59762306a36Sopenharmony_ci		memset(&reg_req, 0, sizeof(reg_req));
59862306a36Sopenharmony_ci		reg_req.mgmt_class = IB_MGMT_CLASS_DEVICE_MGMT;
59962306a36Sopenharmony_ci		reg_req.mgmt_class_version = IB_MGMT_BASE_VERSION;
60062306a36Sopenharmony_ci		set_bit(IB_MGMT_METHOD_GET, reg_req.method_mask);
60162306a36Sopenharmony_ci		set_bit(IB_MGMT_METHOD_SET, reg_req.method_mask);
60262306a36Sopenharmony_ci
60362306a36Sopenharmony_ci		mad_agent = ib_register_mad_agent(sport->sdev->device,
60462306a36Sopenharmony_ci						  sport->port,
60562306a36Sopenharmony_ci						  IB_QPT_GSI,
60662306a36Sopenharmony_ci						  &reg_req, 0,
60762306a36Sopenharmony_ci						  srpt_mad_send_handler,
60862306a36Sopenharmony_ci						  srpt_mad_recv_handler,
60962306a36Sopenharmony_ci						  sport, 0);
61062306a36Sopenharmony_ci		if (IS_ERR(mad_agent)) {
61162306a36Sopenharmony_ci			pr_err("%s-%d: MAD agent registration failed (%ld). Note: this is expected if SR-IOV is enabled.\n",
61262306a36Sopenharmony_ci			       dev_name(&sport->sdev->device->dev), sport->port,
61362306a36Sopenharmony_ci			       PTR_ERR(mad_agent));
61462306a36Sopenharmony_ci			sport->mad_agent = NULL;
61562306a36Sopenharmony_ci			memset(&port_modify, 0, sizeof(port_modify));
61662306a36Sopenharmony_ci			port_modify.clr_port_cap_mask = IB_PORT_DEVICE_MGMT_SUP;
61762306a36Sopenharmony_ci			ib_modify_port(sport->sdev->device, sport->port, 0,
61862306a36Sopenharmony_ci				       &port_modify);
61962306a36Sopenharmony_ci			return 0;
62062306a36Sopenharmony_ci		}
62162306a36Sopenharmony_ci
62262306a36Sopenharmony_ci		sport->mad_agent = mad_agent;
62362306a36Sopenharmony_ci	}
62462306a36Sopenharmony_ci
62562306a36Sopenharmony_ci	return 0;
62662306a36Sopenharmony_ci}
62762306a36Sopenharmony_ci
62862306a36Sopenharmony_ci/**
62962306a36Sopenharmony_ci * srpt_unregister_mad_agent - unregister MAD callback functions
63062306a36Sopenharmony_ci * @sdev: SRPT HCA pointer.
63162306a36Sopenharmony_ci * @port_cnt: number of ports with registered MAD
63262306a36Sopenharmony_ci *
63362306a36Sopenharmony_ci * Note: It is safe to call this function more than once for the same device.
63462306a36Sopenharmony_ci */
63562306a36Sopenharmony_cistatic void srpt_unregister_mad_agent(struct srpt_device *sdev, int port_cnt)
63662306a36Sopenharmony_ci{
63762306a36Sopenharmony_ci	struct ib_port_modify port_modify = {
63862306a36Sopenharmony_ci		.clr_port_cap_mask = IB_PORT_DEVICE_MGMT_SUP,
63962306a36Sopenharmony_ci	};
64062306a36Sopenharmony_ci	struct srpt_port *sport;
64162306a36Sopenharmony_ci	int i;
64262306a36Sopenharmony_ci
64362306a36Sopenharmony_ci	for (i = 1; i <= port_cnt; i++) {
64462306a36Sopenharmony_ci		sport = &sdev->port[i - 1];
64562306a36Sopenharmony_ci		WARN_ON(sport->port != i);
64662306a36Sopenharmony_ci		if (sport->mad_agent) {
64762306a36Sopenharmony_ci			ib_modify_port(sdev->device, i, 0, &port_modify);
64862306a36Sopenharmony_ci			ib_unregister_mad_agent(sport->mad_agent);
64962306a36Sopenharmony_ci			sport->mad_agent = NULL;
65062306a36Sopenharmony_ci		}
65162306a36Sopenharmony_ci	}
65262306a36Sopenharmony_ci}
65362306a36Sopenharmony_ci
65462306a36Sopenharmony_ci/**
65562306a36Sopenharmony_ci * srpt_alloc_ioctx - allocate a SRPT I/O context structure
65662306a36Sopenharmony_ci * @sdev: SRPT HCA pointer.
65762306a36Sopenharmony_ci * @ioctx_size: I/O context size.
65862306a36Sopenharmony_ci * @buf_cache: I/O buffer cache.
65962306a36Sopenharmony_ci * @dir: DMA data direction.
66062306a36Sopenharmony_ci */
66162306a36Sopenharmony_cistatic struct srpt_ioctx *srpt_alloc_ioctx(struct srpt_device *sdev,
66262306a36Sopenharmony_ci					   int ioctx_size,
66362306a36Sopenharmony_ci					   struct kmem_cache *buf_cache,
66462306a36Sopenharmony_ci					   enum dma_data_direction dir)
66562306a36Sopenharmony_ci{
66662306a36Sopenharmony_ci	struct srpt_ioctx *ioctx;
66762306a36Sopenharmony_ci
66862306a36Sopenharmony_ci	ioctx = kzalloc(ioctx_size, GFP_KERNEL);
66962306a36Sopenharmony_ci	if (!ioctx)
67062306a36Sopenharmony_ci		goto err;
67162306a36Sopenharmony_ci
67262306a36Sopenharmony_ci	ioctx->buf = kmem_cache_alloc(buf_cache, GFP_KERNEL);
67362306a36Sopenharmony_ci	if (!ioctx->buf)
67462306a36Sopenharmony_ci		goto err_free_ioctx;
67562306a36Sopenharmony_ci
67662306a36Sopenharmony_ci	ioctx->dma = ib_dma_map_single(sdev->device, ioctx->buf,
67762306a36Sopenharmony_ci				       kmem_cache_size(buf_cache), dir);
67862306a36Sopenharmony_ci	if (ib_dma_mapping_error(sdev->device, ioctx->dma))
67962306a36Sopenharmony_ci		goto err_free_buf;
68062306a36Sopenharmony_ci
68162306a36Sopenharmony_ci	return ioctx;
68262306a36Sopenharmony_ci
68362306a36Sopenharmony_cierr_free_buf:
68462306a36Sopenharmony_ci	kmem_cache_free(buf_cache, ioctx->buf);
68562306a36Sopenharmony_cierr_free_ioctx:
68662306a36Sopenharmony_ci	kfree(ioctx);
68762306a36Sopenharmony_cierr:
68862306a36Sopenharmony_ci	return NULL;
68962306a36Sopenharmony_ci}
69062306a36Sopenharmony_ci
69162306a36Sopenharmony_ci/**
69262306a36Sopenharmony_ci * srpt_free_ioctx - free a SRPT I/O context structure
69362306a36Sopenharmony_ci * @sdev: SRPT HCA pointer.
69462306a36Sopenharmony_ci * @ioctx: I/O context pointer.
69562306a36Sopenharmony_ci * @buf_cache: I/O buffer cache.
69662306a36Sopenharmony_ci * @dir: DMA data direction.
69762306a36Sopenharmony_ci */
69862306a36Sopenharmony_cistatic void srpt_free_ioctx(struct srpt_device *sdev, struct srpt_ioctx *ioctx,
69962306a36Sopenharmony_ci			    struct kmem_cache *buf_cache,
70062306a36Sopenharmony_ci			    enum dma_data_direction dir)
70162306a36Sopenharmony_ci{
70262306a36Sopenharmony_ci	if (!ioctx)
70362306a36Sopenharmony_ci		return;
70462306a36Sopenharmony_ci
70562306a36Sopenharmony_ci	ib_dma_unmap_single(sdev->device, ioctx->dma,
70662306a36Sopenharmony_ci			    kmem_cache_size(buf_cache), dir);
70762306a36Sopenharmony_ci	kmem_cache_free(buf_cache, ioctx->buf);
70862306a36Sopenharmony_ci	kfree(ioctx);
70962306a36Sopenharmony_ci}
71062306a36Sopenharmony_ci
71162306a36Sopenharmony_ci/**
71262306a36Sopenharmony_ci * srpt_alloc_ioctx_ring - allocate a ring of SRPT I/O context structures
71362306a36Sopenharmony_ci * @sdev:       Device to allocate the I/O context ring for.
71462306a36Sopenharmony_ci * @ring_size:  Number of elements in the I/O context ring.
71562306a36Sopenharmony_ci * @ioctx_size: I/O context size.
71662306a36Sopenharmony_ci * @buf_cache:  I/O buffer cache.
71762306a36Sopenharmony_ci * @alignment_offset: Offset in each ring buffer at which the SRP information
71862306a36Sopenharmony_ci *		unit starts.
71962306a36Sopenharmony_ci * @dir:        DMA data direction.
72062306a36Sopenharmony_ci */
72162306a36Sopenharmony_cistatic struct srpt_ioctx **srpt_alloc_ioctx_ring(struct srpt_device *sdev,
72262306a36Sopenharmony_ci				int ring_size, int ioctx_size,
72362306a36Sopenharmony_ci				struct kmem_cache *buf_cache,
72462306a36Sopenharmony_ci				int alignment_offset,
72562306a36Sopenharmony_ci				enum dma_data_direction dir)
72662306a36Sopenharmony_ci{
72762306a36Sopenharmony_ci	struct srpt_ioctx **ring;
72862306a36Sopenharmony_ci	int i;
72962306a36Sopenharmony_ci
73062306a36Sopenharmony_ci	WARN_ON(ioctx_size != sizeof(struct srpt_recv_ioctx) &&
73162306a36Sopenharmony_ci		ioctx_size != sizeof(struct srpt_send_ioctx));
73262306a36Sopenharmony_ci
73362306a36Sopenharmony_ci	ring = kvmalloc_array(ring_size, sizeof(ring[0]), GFP_KERNEL);
73462306a36Sopenharmony_ci	if (!ring)
73562306a36Sopenharmony_ci		goto out;
73662306a36Sopenharmony_ci	for (i = 0; i < ring_size; ++i) {
73762306a36Sopenharmony_ci		ring[i] = srpt_alloc_ioctx(sdev, ioctx_size, buf_cache, dir);
73862306a36Sopenharmony_ci		if (!ring[i])
73962306a36Sopenharmony_ci			goto err;
74062306a36Sopenharmony_ci		ring[i]->index = i;
74162306a36Sopenharmony_ci		ring[i]->offset = alignment_offset;
74262306a36Sopenharmony_ci	}
74362306a36Sopenharmony_ci	goto out;
74462306a36Sopenharmony_ci
74562306a36Sopenharmony_cierr:
74662306a36Sopenharmony_ci	while (--i >= 0)
74762306a36Sopenharmony_ci		srpt_free_ioctx(sdev, ring[i], buf_cache, dir);
74862306a36Sopenharmony_ci	kvfree(ring);
74962306a36Sopenharmony_ci	ring = NULL;
75062306a36Sopenharmony_ciout:
75162306a36Sopenharmony_ci	return ring;
75262306a36Sopenharmony_ci}
75362306a36Sopenharmony_ci
75462306a36Sopenharmony_ci/**
75562306a36Sopenharmony_ci * srpt_free_ioctx_ring - free the ring of SRPT I/O context structures
75662306a36Sopenharmony_ci * @ioctx_ring: I/O context ring to be freed.
75762306a36Sopenharmony_ci * @sdev: SRPT HCA pointer.
75862306a36Sopenharmony_ci * @ring_size: Number of ring elements.
75962306a36Sopenharmony_ci * @buf_cache: I/O buffer cache.
76062306a36Sopenharmony_ci * @dir: DMA data direction.
76162306a36Sopenharmony_ci */
76262306a36Sopenharmony_cistatic void srpt_free_ioctx_ring(struct srpt_ioctx **ioctx_ring,
76362306a36Sopenharmony_ci				 struct srpt_device *sdev, int ring_size,
76462306a36Sopenharmony_ci				 struct kmem_cache *buf_cache,
76562306a36Sopenharmony_ci				 enum dma_data_direction dir)
76662306a36Sopenharmony_ci{
76762306a36Sopenharmony_ci	int i;
76862306a36Sopenharmony_ci
76962306a36Sopenharmony_ci	if (!ioctx_ring)
77062306a36Sopenharmony_ci		return;
77162306a36Sopenharmony_ci
77262306a36Sopenharmony_ci	for (i = 0; i < ring_size; ++i)
77362306a36Sopenharmony_ci		srpt_free_ioctx(sdev, ioctx_ring[i], buf_cache, dir);
77462306a36Sopenharmony_ci	kvfree(ioctx_ring);
77562306a36Sopenharmony_ci}
77662306a36Sopenharmony_ci
77762306a36Sopenharmony_ci/**
77862306a36Sopenharmony_ci * srpt_set_cmd_state - set the state of a SCSI command
77962306a36Sopenharmony_ci * @ioctx: Send I/O context.
78062306a36Sopenharmony_ci * @new: New I/O context state.
78162306a36Sopenharmony_ci *
78262306a36Sopenharmony_ci * Does not modify the state of aborted commands. Returns the previous command
78362306a36Sopenharmony_ci * state.
78462306a36Sopenharmony_ci */
78562306a36Sopenharmony_cistatic enum srpt_command_state srpt_set_cmd_state(struct srpt_send_ioctx *ioctx,
78662306a36Sopenharmony_ci						  enum srpt_command_state new)
78762306a36Sopenharmony_ci{
78862306a36Sopenharmony_ci	enum srpt_command_state previous;
78962306a36Sopenharmony_ci
79062306a36Sopenharmony_ci	previous = ioctx->state;
79162306a36Sopenharmony_ci	if (previous != SRPT_STATE_DONE)
79262306a36Sopenharmony_ci		ioctx->state = new;
79362306a36Sopenharmony_ci
79462306a36Sopenharmony_ci	return previous;
79562306a36Sopenharmony_ci}
79662306a36Sopenharmony_ci
79762306a36Sopenharmony_ci/**
79862306a36Sopenharmony_ci * srpt_test_and_set_cmd_state - test and set the state of a command
79962306a36Sopenharmony_ci * @ioctx: Send I/O context.
80062306a36Sopenharmony_ci * @old: Current I/O context state.
80162306a36Sopenharmony_ci * @new: New I/O context state.
80262306a36Sopenharmony_ci *
80362306a36Sopenharmony_ci * Returns true if and only if the previous command state was equal to 'old'.
80462306a36Sopenharmony_ci */
80562306a36Sopenharmony_cistatic bool srpt_test_and_set_cmd_state(struct srpt_send_ioctx *ioctx,
80662306a36Sopenharmony_ci					enum srpt_command_state old,
80762306a36Sopenharmony_ci					enum srpt_command_state new)
80862306a36Sopenharmony_ci{
80962306a36Sopenharmony_ci	enum srpt_command_state previous;
81062306a36Sopenharmony_ci
81162306a36Sopenharmony_ci	WARN_ON(!ioctx);
81262306a36Sopenharmony_ci	WARN_ON(old == SRPT_STATE_DONE);
81362306a36Sopenharmony_ci	WARN_ON(new == SRPT_STATE_NEW);
81462306a36Sopenharmony_ci
81562306a36Sopenharmony_ci	previous = ioctx->state;
81662306a36Sopenharmony_ci	if (previous == old)
81762306a36Sopenharmony_ci		ioctx->state = new;
81862306a36Sopenharmony_ci
81962306a36Sopenharmony_ci	return previous == old;
82062306a36Sopenharmony_ci}
82162306a36Sopenharmony_ci
82262306a36Sopenharmony_ci/**
82362306a36Sopenharmony_ci * srpt_post_recv - post an IB receive request
82462306a36Sopenharmony_ci * @sdev: SRPT HCA pointer.
82562306a36Sopenharmony_ci * @ch: SRPT RDMA channel.
82662306a36Sopenharmony_ci * @ioctx: Receive I/O context pointer.
82762306a36Sopenharmony_ci */
82862306a36Sopenharmony_cistatic int srpt_post_recv(struct srpt_device *sdev, struct srpt_rdma_ch *ch,
82962306a36Sopenharmony_ci			  struct srpt_recv_ioctx *ioctx)
83062306a36Sopenharmony_ci{
83162306a36Sopenharmony_ci	struct ib_sge list;
83262306a36Sopenharmony_ci	struct ib_recv_wr wr;
83362306a36Sopenharmony_ci
83462306a36Sopenharmony_ci	BUG_ON(!sdev);
83562306a36Sopenharmony_ci	list.addr = ioctx->ioctx.dma + ioctx->ioctx.offset;
83662306a36Sopenharmony_ci	list.length = srp_max_req_size;
83762306a36Sopenharmony_ci	list.lkey = sdev->lkey;
83862306a36Sopenharmony_ci
83962306a36Sopenharmony_ci	ioctx->ioctx.cqe.done = srpt_recv_done;
84062306a36Sopenharmony_ci	wr.wr_cqe = &ioctx->ioctx.cqe;
84162306a36Sopenharmony_ci	wr.next = NULL;
84262306a36Sopenharmony_ci	wr.sg_list = &list;
84362306a36Sopenharmony_ci	wr.num_sge = 1;
84462306a36Sopenharmony_ci
84562306a36Sopenharmony_ci	if (sdev->use_srq)
84662306a36Sopenharmony_ci		return ib_post_srq_recv(sdev->srq, &wr, NULL);
84762306a36Sopenharmony_ci	else
84862306a36Sopenharmony_ci		return ib_post_recv(ch->qp, &wr, NULL);
84962306a36Sopenharmony_ci}
85062306a36Sopenharmony_ci
85162306a36Sopenharmony_ci/**
85262306a36Sopenharmony_ci * srpt_zerolength_write - perform a zero-length RDMA write
85362306a36Sopenharmony_ci * @ch: SRPT RDMA channel.
85462306a36Sopenharmony_ci *
85562306a36Sopenharmony_ci * A quote from the InfiniBand specification: C9-88: For an HCA responder
85662306a36Sopenharmony_ci * using Reliable Connection service, for each zero-length RDMA READ or WRITE
85762306a36Sopenharmony_ci * request, the R_Key shall not be validated, even if the request includes
85862306a36Sopenharmony_ci * Immediate data.
85962306a36Sopenharmony_ci */
86062306a36Sopenharmony_cistatic int srpt_zerolength_write(struct srpt_rdma_ch *ch)
86162306a36Sopenharmony_ci{
86262306a36Sopenharmony_ci	struct ib_rdma_wr wr = {
86362306a36Sopenharmony_ci		.wr = {
86462306a36Sopenharmony_ci			.next		= NULL,
86562306a36Sopenharmony_ci			{ .wr_cqe	= &ch->zw_cqe, },
86662306a36Sopenharmony_ci			.opcode		= IB_WR_RDMA_WRITE,
86762306a36Sopenharmony_ci			.send_flags	= IB_SEND_SIGNALED,
86862306a36Sopenharmony_ci		}
86962306a36Sopenharmony_ci	};
87062306a36Sopenharmony_ci
87162306a36Sopenharmony_ci	pr_debug("%s-%d: queued zerolength write\n", ch->sess_name,
87262306a36Sopenharmony_ci		 ch->qp->qp_num);
87362306a36Sopenharmony_ci
87462306a36Sopenharmony_ci	return ib_post_send(ch->qp, &wr.wr, NULL);
87562306a36Sopenharmony_ci}
87662306a36Sopenharmony_ci
87762306a36Sopenharmony_cistatic void srpt_zerolength_write_done(struct ib_cq *cq, struct ib_wc *wc)
87862306a36Sopenharmony_ci{
87962306a36Sopenharmony_ci	struct srpt_rdma_ch *ch = wc->qp->qp_context;
88062306a36Sopenharmony_ci
88162306a36Sopenharmony_ci	pr_debug("%s-%d wc->status %d\n", ch->sess_name, ch->qp->qp_num,
88262306a36Sopenharmony_ci		 wc->status);
88362306a36Sopenharmony_ci
88462306a36Sopenharmony_ci	if (wc->status == IB_WC_SUCCESS) {
88562306a36Sopenharmony_ci		srpt_process_wait_list(ch);
88662306a36Sopenharmony_ci	} else {
88762306a36Sopenharmony_ci		if (srpt_set_ch_state(ch, CH_DISCONNECTED))
88862306a36Sopenharmony_ci			schedule_work(&ch->release_work);
88962306a36Sopenharmony_ci		else
89062306a36Sopenharmony_ci			pr_debug("%s-%d: already disconnected.\n",
89162306a36Sopenharmony_ci				 ch->sess_name, ch->qp->qp_num);
89262306a36Sopenharmony_ci	}
89362306a36Sopenharmony_ci}
89462306a36Sopenharmony_ci
89562306a36Sopenharmony_cistatic int srpt_alloc_rw_ctxs(struct srpt_send_ioctx *ioctx,
89662306a36Sopenharmony_ci		struct srp_direct_buf *db, int nbufs, struct scatterlist **sg,
89762306a36Sopenharmony_ci		unsigned *sg_cnt)
89862306a36Sopenharmony_ci{
89962306a36Sopenharmony_ci	enum dma_data_direction dir = target_reverse_dma_direction(&ioctx->cmd);
90062306a36Sopenharmony_ci	struct srpt_rdma_ch *ch = ioctx->ch;
90162306a36Sopenharmony_ci	struct scatterlist *prev = NULL;
90262306a36Sopenharmony_ci	unsigned prev_nents;
90362306a36Sopenharmony_ci	int ret, i;
90462306a36Sopenharmony_ci
90562306a36Sopenharmony_ci	if (nbufs == 1) {
90662306a36Sopenharmony_ci		ioctx->rw_ctxs = &ioctx->s_rw_ctx;
90762306a36Sopenharmony_ci	} else {
90862306a36Sopenharmony_ci		ioctx->rw_ctxs = kmalloc_array(nbufs, sizeof(*ioctx->rw_ctxs),
90962306a36Sopenharmony_ci			GFP_KERNEL);
91062306a36Sopenharmony_ci		if (!ioctx->rw_ctxs)
91162306a36Sopenharmony_ci			return -ENOMEM;
91262306a36Sopenharmony_ci	}
91362306a36Sopenharmony_ci
91462306a36Sopenharmony_ci	for (i = ioctx->n_rw_ctx; i < nbufs; i++, db++) {
91562306a36Sopenharmony_ci		struct srpt_rw_ctx *ctx = &ioctx->rw_ctxs[i];
91662306a36Sopenharmony_ci		u64 remote_addr = be64_to_cpu(db->va);
91762306a36Sopenharmony_ci		u32 size = be32_to_cpu(db->len);
91862306a36Sopenharmony_ci		u32 rkey = be32_to_cpu(db->key);
91962306a36Sopenharmony_ci
92062306a36Sopenharmony_ci		ret = target_alloc_sgl(&ctx->sg, &ctx->nents, size, false,
92162306a36Sopenharmony_ci				i < nbufs - 1);
92262306a36Sopenharmony_ci		if (ret)
92362306a36Sopenharmony_ci			goto unwind;
92462306a36Sopenharmony_ci
92562306a36Sopenharmony_ci		ret = rdma_rw_ctx_init(&ctx->rw, ch->qp, ch->sport->port,
92662306a36Sopenharmony_ci				ctx->sg, ctx->nents, 0, remote_addr, rkey, dir);
92762306a36Sopenharmony_ci		if (ret < 0) {
92862306a36Sopenharmony_ci			target_free_sgl(ctx->sg, ctx->nents);
92962306a36Sopenharmony_ci			goto unwind;
93062306a36Sopenharmony_ci		}
93162306a36Sopenharmony_ci
93262306a36Sopenharmony_ci		ioctx->n_rdma += ret;
93362306a36Sopenharmony_ci		ioctx->n_rw_ctx++;
93462306a36Sopenharmony_ci
93562306a36Sopenharmony_ci		if (prev) {
93662306a36Sopenharmony_ci			sg_unmark_end(&prev[prev_nents - 1]);
93762306a36Sopenharmony_ci			sg_chain(prev, prev_nents + 1, ctx->sg);
93862306a36Sopenharmony_ci		} else {
93962306a36Sopenharmony_ci			*sg = ctx->sg;
94062306a36Sopenharmony_ci		}
94162306a36Sopenharmony_ci
94262306a36Sopenharmony_ci		prev = ctx->sg;
94362306a36Sopenharmony_ci		prev_nents = ctx->nents;
94462306a36Sopenharmony_ci
94562306a36Sopenharmony_ci		*sg_cnt += ctx->nents;
94662306a36Sopenharmony_ci	}
94762306a36Sopenharmony_ci
94862306a36Sopenharmony_ci	return 0;
94962306a36Sopenharmony_ci
95062306a36Sopenharmony_ciunwind:
95162306a36Sopenharmony_ci	while (--i >= 0) {
95262306a36Sopenharmony_ci		struct srpt_rw_ctx *ctx = &ioctx->rw_ctxs[i];
95362306a36Sopenharmony_ci
95462306a36Sopenharmony_ci		rdma_rw_ctx_destroy(&ctx->rw, ch->qp, ch->sport->port,
95562306a36Sopenharmony_ci				ctx->sg, ctx->nents, dir);
95662306a36Sopenharmony_ci		target_free_sgl(ctx->sg, ctx->nents);
95762306a36Sopenharmony_ci	}
95862306a36Sopenharmony_ci	if (ioctx->rw_ctxs != &ioctx->s_rw_ctx)
95962306a36Sopenharmony_ci		kfree(ioctx->rw_ctxs);
96062306a36Sopenharmony_ci	return ret;
96162306a36Sopenharmony_ci}
96262306a36Sopenharmony_ci
96362306a36Sopenharmony_cistatic void srpt_free_rw_ctxs(struct srpt_rdma_ch *ch,
96462306a36Sopenharmony_ci				    struct srpt_send_ioctx *ioctx)
96562306a36Sopenharmony_ci{
96662306a36Sopenharmony_ci	enum dma_data_direction dir = target_reverse_dma_direction(&ioctx->cmd);
96762306a36Sopenharmony_ci	int i;
96862306a36Sopenharmony_ci
96962306a36Sopenharmony_ci	for (i = 0; i < ioctx->n_rw_ctx; i++) {
97062306a36Sopenharmony_ci		struct srpt_rw_ctx *ctx = &ioctx->rw_ctxs[i];
97162306a36Sopenharmony_ci
97262306a36Sopenharmony_ci		rdma_rw_ctx_destroy(&ctx->rw, ch->qp, ch->sport->port,
97362306a36Sopenharmony_ci				ctx->sg, ctx->nents, dir);
97462306a36Sopenharmony_ci		target_free_sgl(ctx->sg, ctx->nents);
97562306a36Sopenharmony_ci	}
97662306a36Sopenharmony_ci
97762306a36Sopenharmony_ci	if (ioctx->rw_ctxs != &ioctx->s_rw_ctx)
97862306a36Sopenharmony_ci		kfree(ioctx->rw_ctxs);
97962306a36Sopenharmony_ci}
98062306a36Sopenharmony_ci
98162306a36Sopenharmony_cistatic inline void *srpt_get_desc_buf(struct srp_cmd *srp_cmd)
98262306a36Sopenharmony_ci{
98362306a36Sopenharmony_ci	/*
98462306a36Sopenharmony_ci	 * The pointer computations below will only be compiled correctly
98562306a36Sopenharmony_ci	 * if srp_cmd::add_data is declared as s8*, u8*, s8[] or u8[], so check
98662306a36Sopenharmony_ci	 * whether srp_cmd::add_data has been declared as a byte pointer.
98762306a36Sopenharmony_ci	 */
98862306a36Sopenharmony_ci	BUILD_BUG_ON(!__same_type(srp_cmd->add_data[0], (s8)0) &&
98962306a36Sopenharmony_ci		     !__same_type(srp_cmd->add_data[0], (u8)0));
99062306a36Sopenharmony_ci
99162306a36Sopenharmony_ci	/*
99262306a36Sopenharmony_ci	 * According to the SRP spec, the lower two bits of the 'ADDITIONAL
99362306a36Sopenharmony_ci	 * CDB LENGTH' field are reserved and the size in bytes of this field
99462306a36Sopenharmony_ci	 * is four times the value specified in bits 3..7. Hence the "& ~3".
99562306a36Sopenharmony_ci	 */
99662306a36Sopenharmony_ci	return srp_cmd->add_data + (srp_cmd->add_cdb_len & ~3);
99762306a36Sopenharmony_ci}
99862306a36Sopenharmony_ci
99962306a36Sopenharmony_ci/**
100062306a36Sopenharmony_ci * srpt_get_desc_tbl - parse the data descriptors of a SRP_CMD request
100162306a36Sopenharmony_ci * @recv_ioctx: I/O context associated with the received command @srp_cmd.
100262306a36Sopenharmony_ci * @ioctx: I/O context that will be used for responding to the initiator.
100362306a36Sopenharmony_ci * @srp_cmd: Pointer to the SRP_CMD request data.
100462306a36Sopenharmony_ci * @dir: Pointer to the variable to which the transfer direction will be
100562306a36Sopenharmony_ci *   written.
100662306a36Sopenharmony_ci * @sg: [out] scatterlist for the parsed SRP_CMD.
100762306a36Sopenharmony_ci * @sg_cnt: [out] length of @sg.
100862306a36Sopenharmony_ci * @data_len: Pointer to the variable to which the total data length of all
100962306a36Sopenharmony_ci *   descriptors in the SRP_CMD request will be written.
101062306a36Sopenharmony_ci * @imm_data_offset: [in] Offset in SRP_CMD requests at which immediate data
101162306a36Sopenharmony_ci *   starts.
101262306a36Sopenharmony_ci *
101362306a36Sopenharmony_ci * This function initializes ioctx->nrbuf and ioctx->r_bufs.
101462306a36Sopenharmony_ci *
101562306a36Sopenharmony_ci * Returns -EINVAL when the SRP_CMD request contains inconsistent descriptors;
101662306a36Sopenharmony_ci * -ENOMEM when memory allocation fails and zero upon success.
101762306a36Sopenharmony_ci */
101862306a36Sopenharmony_cistatic int srpt_get_desc_tbl(struct srpt_recv_ioctx *recv_ioctx,
101962306a36Sopenharmony_ci		struct srpt_send_ioctx *ioctx,
102062306a36Sopenharmony_ci		struct srp_cmd *srp_cmd, enum dma_data_direction *dir,
102162306a36Sopenharmony_ci		struct scatterlist **sg, unsigned int *sg_cnt, u64 *data_len,
102262306a36Sopenharmony_ci		u16 imm_data_offset)
102362306a36Sopenharmony_ci{
102462306a36Sopenharmony_ci	BUG_ON(!dir);
102562306a36Sopenharmony_ci	BUG_ON(!data_len);
102662306a36Sopenharmony_ci
102762306a36Sopenharmony_ci	/*
102862306a36Sopenharmony_ci	 * The lower four bits of the buffer format field contain the DATA-IN
102962306a36Sopenharmony_ci	 * buffer descriptor format, and the highest four bits contain the
103062306a36Sopenharmony_ci	 * DATA-OUT buffer descriptor format.
103162306a36Sopenharmony_ci	 */
103262306a36Sopenharmony_ci	if (srp_cmd->buf_fmt & 0xf)
103362306a36Sopenharmony_ci		/* DATA-IN: transfer data from target to initiator (read). */
103462306a36Sopenharmony_ci		*dir = DMA_FROM_DEVICE;
103562306a36Sopenharmony_ci	else if (srp_cmd->buf_fmt >> 4)
103662306a36Sopenharmony_ci		/* DATA-OUT: transfer data from initiator to target (write). */
103762306a36Sopenharmony_ci		*dir = DMA_TO_DEVICE;
103862306a36Sopenharmony_ci	else
103962306a36Sopenharmony_ci		*dir = DMA_NONE;
104062306a36Sopenharmony_ci
104162306a36Sopenharmony_ci	/* initialize data_direction early as srpt_alloc_rw_ctxs needs it */
104262306a36Sopenharmony_ci	ioctx->cmd.data_direction = *dir;
104362306a36Sopenharmony_ci
104462306a36Sopenharmony_ci	if (((srp_cmd->buf_fmt & 0xf) == SRP_DATA_DESC_DIRECT) ||
104562306a36Sopenharmony_ci	    ((srp_cmd->buf_fmt >> 4) == SRP_DATA_DESC_DIRECT)) {
104662306a36Sopenharmony_ci		struct srp_direct_buf *db = srpt_get_desc_buf(srp_cmd);
104762306a36Sopenharmony_ci
104862306a36Sopenharmony_ci		*data_len = be32_to_cpu(db->len);
104962306a36Sopenharmony_ci		return srpt_alloc_rw_ctxs(ioctx, db, 1, sg, sg_cnt);
105062306a36Sopenharmony_ci	} else if (((srp_cmd->buf_fmt & 0xf) == SRP_DATA_DESC_INDIRECT) ||
105162306a36Sopenharmony_ci		   ((srp_cmd->buf_fmt >> 4) == SRP_DATA_DESC_INDIRECT)) {
105262306a36Sopenharmony_ci		struct srp_indirect_buf *idb = srpt_get_desc_buf(srp_cmd);
105362306a36Sopenharmony_ci		int nbufs = be32_to_cpu(idb->table_desc.len) /
105462306a36Sopenharmony_ci				sizeof(struct srp_direct_buf);
105562306a36Sopenharmony_ci
105662306a36Sopenharmony_ci		if (nbufs >
105762306a36Sopenharmony_ci		    (srp_cmd->data_out_desc_cnt + srp_cmd->data_in_desc_cnt)) {
105862306a36Sopenharmony_ci			pr_err("received unsupported SRP_CMD request type (%u out + %u in != %u / %zu)\n",
105962306a36Sopenharmony_ci			       srp_cmd->data_out_desc_cnt,
106062306a36Sopenharmony_ci			       srp_cmd->data_in_desc_cnt,
106162306a36Sopenharmony_ci			       be32_to_cpu(idb->table_desc.len),
106262306a36Sopenharmony_ci			       sizeof(struct srp_direct_buf));
106362306a36Sopenharmony_ci			return -EINVAL;
106462306a36Sopenharmony_ci		}
106562306a36Sopenharmony_ci
106662306a36Sopenharmony_ci		*data_len = be32_to_cpu(idb->len);
106762306a36Sopenharmony_ci		return srpt_alloc_rw_ctxs(ioctx, idb->desc_list, nbufs,
106862306a36Sopenharmony_ci				sg, sg_cnt);
106962306a36Sopenharmony_ci	} else if ((srp_cmd->buf_fmt >> 4) == SRP_DATA_DESC_IMM) {
107062306a36Sopenharmony_ci		struct srp_imm_buf *imm_buf = srpt_get_desc_buf(srp_cmd);
107162306a36Sopenharmony_ci		void *data = (void *)srp_cmd + imm_data_offset;
107262306a36Sopenharmony_ci		uint32_t len = be32_to_cpu(imm_buf->len);
107362306a36Sopenharmony_ci		uint32_t req_size = imm_data_offset + len;
107462306a36Sopenharmony_ci
107562306a36Sopenharmony_ci		if (req_size > srp_max_req_size) {
107662306a36Sopenharmony_ci			pr_err("Immediate data (length %d + %d) exceeds request size %d\n",
107762306a36Sopenharmony_ci			       imm_data_offset, len, srp_max_req_size);
107862306a36Sopenharmony_ci			return -EINVAL;
107962306a36Sopenharmony_ci		}
108062306a36Sopenharmony_ci		if (recv_ioctx->byte_len < req_size) {
108162306a36Sopenharmony_ci			pr_err("Received too few data - %d < %d\n",
108262306a36Sopenharmony_ci			       recv_ioctx->byte_len, req_size);
108362306a36Sopenharmony_ci			return -EIO;
108462306a36Sopenharmony_ci		}
108562306a36Sopenharmony_ci		/*
108662306a36Sopenharmony_ci		 * The immediate data buffer descriptor must occur before the
108762306a36Sopenharmony_ci		 * immediate data itself.
108862306a36Sopenharmony_ci		 */
108962306a36Sopenharmony_ci		if ((void *)(imm_buf + 1) > (void *)data) {
109062306a36Sopenharmony_ci			pr_err("Received invalid write request\n");
109162306a36Sopenharmony_ci			return -EINVAL;
109262306a36Sopenharmony_ci		}
109362306a36Sopenharmony_ci		*data_len = len;
109462306a36Sopenharmony_ci		ioctx->recv_ioctx = recv_ioctx;
109562306a36Sopenharmony_ci		if ((uintptr_t)data & 511) {
109662306a36Sopenharmony_ci			pr_warn_once("Internal error - the receive buffers are not aligned properly.\n");
109762306a36Sopenharmony_ci			return -EINVAL;
109862306a36Sopenharmony_ci		}
109962306a36Sopenharmony_ci		sg_init_one(&ioctx->imm_sg, data, len);
110062306a36Sopenharmony_ci		*sg = &ioctx->imm_sg;
110162306a36Sopenharmony_ci		*sg_cnt = 1;
110262306a36Sopenharmony_ci		return 0;
110362306a36Sopenharmony_ci	} else {
110462306a36Sopenharmony_ci		*data_len = 0;
110562306a36Sopenharmony_ci		return 0;
110662306a36Sopenharmony_ci	}
110762306a36Sopenharmony_ci}
110862306a36Sopenharmony_ci
110962306a36Sopenharmony_ci/**
111062306a36Sopenharmony_ci * srpt_init_ch_qp - initialize queue pair attributes
111162306a36Sopenharmony_ci * @ch: SRPT RDMA channel.
111262306a36Sopenharmony_ci * @qp: Queue pair pointer.
111362306a36Sopenharmony_ci *
111462306a36Sopenharmony_ci * Initialized the attributes of queue pair 'qp' by allowing local write,
111562306a36Sopenharmony_ci * remote read and remote write. Also transitions 'qp' to state IB_QPS_INIT.
111662306a36Sopenharmony_ci */
111762306a36Sopenharmony_cistatic int srpt_init_ch_qp(struct srpt_rdma_ch *ch, struct ib_qp *qp)
111862306a36Sopenharmony_ci{
111962306a36Sopenharmony_ci	struct ib_qp_attr *attr;
112062306a36Sopenharmony_ci	int ret;
112162306a36Sopenharmony_ci
112262306a36Sopenharmony_ci	WARN_ON_ONCE(ch->using_rdma_cm);
112362306a36Sopenharmony_ci
112462306a36Sopenharmony_ci	attr = kzalloc(sizeof(*attr), GFP_KERNEL);
112562306a36Sopenharmony_ci	if (!attr)
112662306a36Sopenharmony_ci		return -ENOMEM;
112762306a36Sopenharmony_ci
112862306a36Sopenharmony_ci	attr->qp_state = IB_QPS_INIT;
112962306a36Sopenharmony_ci	attr->qp_access_flags = IB_ACCESS_LOCAL_WRITE;
113062306a36Sopenharmony_ci	attr->port_num = ch->sport->port;
113162306a36Sopenharmony_ci
113262306a36Sopenharmony_ci	ret = ib_find_cached_pkey(ch->sport->sdev->device, ch->sport->port,
113362306a36Sopenharmony_ci				  ch->pkey, &attr->pkey_index);
113462306a36Sopenharmony_ci	if (ret < 0)
113562306a36Sopenharmony_ci		pr_err("Translating pkey %#x failed (%d) - using index 0\n",
113662306a36Sopenharmony_ci		       ch->pkey, ret);
113762306a36Sopenharmony_ci
113862306a36Sopenharmony_ci	ret = ib_modify_qp(qp, attr,
113962306a36Sopenharmony_ci			   IB_QP_STATE | IB_QP_ACCESS_FLAGS | IB_QP_PORT |
114062306a36Sopenharmony_ci			   IB_QP_PKEY_INDEX);
114162306a36Sopenharmony_ci
114262306a36Sopenharmony_ci	kfree(attr);
114362306a36Sopenharmony_ci	return ret;
114462306a36Sopenharmony_ci}
114562306a36Sopenharmony_ci
114662306a36Sopenharmony_ci/**
114762306a36Sopenharmony_ci * srpt_ch_qp_rtr - change the state of a channel to 'ready to receive' (RTR)
114862306a36Sopenharmony_ci * @ch: channel of the queue pair.
114962306a36Sopenharmony_ci * @qp: queue pair to change the state of.
115062306a36Sopenharmony_ci *
115162306a36Sopenharmony_ci * Returns zero upon success and a negative value upon failure.
115262306a36Sopenharmony_ci *
115362306a36Sopenharmony_ci * Note: currently a struct ib_qp_attr takes 136 bytes on a 64-bit system.
115462306a36Sopenharmony_ci * If this structure ever becomes larger, it might be necessary to allocate
115562306a36Sopenharmony_ci * it dynamically instead of on the stack.
115662306a36Sopenharmony_ci */
115762306a36Sopenharmony_cistatic int srpt_ch_qp_rtr(struct srpt_rdma_ch *ch, struct ib_qp *qp)
115862306a36Sopenharmony_ci{
115962306a36Sopenharmony_ci	struct ib_qp_attr qp_attr;
116062306a36Sopenharmony_ci	int attr_mask;
116162306a36Sopenharmony_ci	int ret;
116262306a36Sopenharmony_ci
116362306a36Sopenharmony_ci	WARN_ON_ONCE(ch->using_rdma_cm);
116462306a36Sopenharmony_ci
116562306a36Sopenharmony_ci	qp_attr.qp_state = IB_QPS_RTR;
116662306a36Sopenharmony_ci	ret = ib_cm_init_qp_attr(ch->ib_cm.cm_id, &qp_attr, &attr_mask);
116762306a36Sopenharmony_ci	if (ret)
116862306a36Sopenharmony_ci		goto out;
116962306a36Sopenharmony_ci
117062306a36Sopenharmony_ci	qp_attr.max_dest_rd_atomic = 4;
117162306a36Sopenharmony_ci
117262306a36Sopenharmony_ci	ret = ib_modify_qp(qp, &qp_attr, attr_mask);
117362306a36Sopenharmony_ci
117462306a36Sopenharmony_ciout:
117562306a36Sopenharmony_ci	return ret;
117662306a36Sopenharmony_ci}
117762306a36Sopenharmony_ci
117862306a36Sopenharmony_ci/**
117962306a36Sopenharmony_ci * srpt_ch_qp_rts - change the state of a channel to 'ready to send' (RTS)
118062306a36Sopenharmony_ci * @ch: channel of the queue pair.
118162306a36Sopenharmony_ci * @qp: queue pair to change the state of.
118262306a36Sopenharmony_ci *
118362306a36Sopenharmony_ci * Returns zero upon success and a negative value upon failure.
118462306a36Sopenharmony_ci *
118562306a36Sopenharmony_ci * Note: currently a struct ib_qp_attr takes 136 bytes on a 64-bit system.
118662306a36Sopenharmony_ci * If this structure ever becomes larger, it might be necessary to allocate
118762306a36Sopenharmony_ci * it dynamically instead of on the stack.
118862306a36Sopenharmony_ci */
118962306a36Sopenharmony_cistatic int srpt_ch_qp_rts(struct srpt_rdma_ch *ch, struct ib_qp *qp)
119062306a36Sopenharmony_ci{
119162306a36Sopenharmony_ci	struct ib_qp_attr qp_attr;
119262306a36Sopenharmony_ci	int attr_mask;
119362306a36Sopenharmony_ci	int ret;
119462306a36Sopenharmony_ci
119562306a36Sopenharmony_ci	qp_attr.qp_state = IB_QPS_RTS;
119662306a36Sopenharmony_ci	ret = ib_cm_init_qp_attr(ch->ib_cm.cm_id, &qp_attr, &attr_mask);
119762306a36Sopenharmony_ci	if (ret)
119862306a36Sopenharmony_ci		goto out;
119962306a36Sopenharmony_ci
120062306a36Sopenharmony_ci	qp_attr.max_rd_atomic = 4;
120162306a36Sopenharmony_ci
120262306a36Sopenharmony_ci	ret = ib_modify_qp(qp, &qp_attr, attr_mask);
120362306a36Sopenharmony_ci
120462306a36Sopenharmony_ciout:
120562306a36Sopenharmony_ci	return ret;
120662306a36Sopenharmony_ci}
120762306a36Sopenharmony_ci
120862306a36Sopenharmony_ci/**
120962306a36Sopenharmony_ci * srpt_ch_qp_err - set the channel queue pair state to 'error'
121062306a36Sopenharmony_ci * @ch: SRPT RDMA channel.
121162306a36Sopenharmony_ci */
121262306a36Sopenharmony_cistatic int srpt_ch_qp_err(struct srpt_rdma_ch *ch)
121362306a36Sopenharmony_ci{
121462306a36Sopenharmony_ci	struct ib_qp_attr qp_attr;
121562306a36Sopenharmony_ci
121662306a36Sopenharmony_ci	qp_attr.qp_state = IB_QPS_ERR;
121762306a36Sopenharmony_ci	return ib_modify_qp(ch->qp, &qp_attr, IB_QP_STATE);
121862306a36Sopenharmony_ci}
121962306a36Sopenharmony_ci
122062306a36Sopenharmony_ci/**
122162306a36Sopenharmony_ci * srpt_get_send_ioctx - obtain an I/O context for sending to the initiator
122262306a36Sopenharmony_ci * @ch: SRPT RDMA channel.
122362306a36Sopenharmony_ci */
122462306a36Sopenharmony_cistatic struct srpt_send_ioctx *srpt_get_send_ioctx(struct srpt_rdma_ch *ch)
122562306a36Sopenharmony_ci{
122662306a36Sopenharmony_ci	struct srpt_send_ioctx *ioctx;
122762306a36Sopenharmony_ci	int tag, cpu;
122862306a36Sopenharmony_ci
122962306a36Sopenharmony_ci	BUG_ON(!ch);
123062306a36Sopenharmony_ci
123162306a36Sopenharmony_ci	tag = sbitmap_queue_get(&ch->sess->sess_tag_pool, &cpu);
123262306a36Sopenharmony_ci	if (tag < 0)
123362306a36Sopenharmony_ci		return NULL;
123462306a36Sopenharmony_ci
123562306a36Sopenharmony_ci	ioctx = ch->ioctx_ring[tag];
123662306a36Sopenharmony_ci	BUG_ON(ioctx->ch != ch);
123762306a36Sopenharmony_ci	ioctx->state = SRPT_STATE_NEW;
123862306a36Sopenharmony_ci	WARN_ON_ONCE(ioctx->recv_ioctx);
123962306a36Sopenharmony_ci	ioctx->n_rdma = 0;
124062306a36Sopenharmony_ci	ioctx->n_rw_ctx = 0;
124162306a36Sopenharmony_ci	ioctx->queue_status_only = false;
124262306a36Sopenharmony_ci	/*
124362306a36Sopenharmony_ci	 * transport_init_se_cmd() does not initialize all fields, so do it
124462306a36Sopenharmony_ci	 * here.
124562306a36Sopenharmony_ci	 */
124662306a36Sopenharmony_ci	memset(&ioctx->cmd, 0, sizeof(ioctx->cmd));
124762306a36Sopenharmony_ci	memset(&ioctx->sense_data, 0, sizeof(ioctx->sense_data));
124862306a36Sopenharmony_ci	ioctx->cmd.map_tag = tag;
124962306a36Sopenharmony_ci	ioctx->cmd.map_cpu = cpu;
125062306a36Sopenharmony_ci
125162306a36Sopenharmony_ci	return ioctx;
125262306a36Sopenharmony_ci}
125362306a36Sopenharmony_ci
125462306a36Sopenharmony_ci/**
125562306a36Sopenharmony_ci * srpt_abort_cmd - abort a SCSI command
125662306a36Sopenharmony_ci * @ioctx:   I/O context associated with the SCSI command.
125762306a36Sopenharmony_ci */
125862306a36Sopenharmony_cistatic int srpt_abort_cmd(struct srpt_send_ioctx *ioctx)
125962306a36Sopenharmony_ci{
126062306a36Sopenharmony_ci	enum srpt_command_state state;
126162306a36Sopenharmony_ci
126262306a36Sopenharmony_ci	BUG_ON(!ioctx);
126362306a36Sopenharmony_ci
126462306a36Sopenharmony_ci	/*
126562306a36Sopenharmony_ci	 * If the command is in a state where the target core is waiting for
126662306a36Sopenharmony_ci	 * the ib_srpt driver, change the state to the next state.
126762306a36Sopenharmony_ci	 */
126862306a36Sopenharmony_ci
126962306a36Sopenharmony_ci	state = ioctx->state;
127062306a36Sopenharmony_ci	switch (state) {
127162306a36Sopenharmony_ci	case SRPT_STATE_NEED_DATA:
127262306a36Sopenharmony_ci		ioctx->state = SRPT_STATE_DATA_IN;
127362306a36Sopenharmony_ci		break;
127462306a36Sopenharmony_ci	case SRPT_STATE_CMD_RSP_SENT:
127562306a36Sopenharmony_ci	case SRPT_STATE_MGMT_RSP_SENT:
127662306a36Sopenharmony_ci		ioctx->state = SRPT_STATE_DONE;
127762306a36Sopenharmony_ci		break;
127862306a36Sopenharmony_ci	default:
127962306a36Sopenharmony_ci		WARN_ONCE(true, "%s: unexpected I/O context state %d\n",
128062306a36Sopenharmony_ci			  __func__, state);
128162306a36Sopenharmony_ci		break;
128262306a36Sopenharmony_ci	}
128362306a36Sopenharmony_ci
128462306a36Sopenharmony_ci	pr_debug("Aborting cmd with state %d -> %d and tag %lld\n", state,
128562306a36Sopenharmony_ci		 ioctx->state, ioctx->cmd.tag);
128662306a36Sopenharmony_ci
128762306a36Sopenharmony_ci	switch (state) {
128862306a36Sopenharmony_ci	case SRPT_STATE_NEW:
128962306a36Sopenharmony_ci	case SRPT_STATE_DATA_IN:
129062306a36Sopenharmony_ci	case SRPT_STATE_MGMT:
129162306a36Sopenharmony_ci	case SRPT_STATE_DONE:
129262306a36Sopenharmony_ci		/*
129362306a36Sopenharmony_ci		 * Do nothing - defer abort processing until
129462306a36Sopenharmony_ci		 * srpt_queue_response() is invoked.
129562306a36Sopenharmony_ci		 */
129662306a36Sopenharmony_ci		break;
129762306a36Sopenharmony_ci	case SRPT_STATE_NEED_DATA:
129862306a36Sopenharmony_ci		pr_debug("tag %#llx: RDMA read error\n", ioctx->cmd.tag);
129962306a36Sopenharmony_ci		transport_generic_request_failure(&ioctx->cmd,
130062306a36Sopenharmony_ci					TCM_CHECK_CONDITION_ABORT_CMD);
130162306a36Sopenharmony_ci		break;
130262306a36Sopenharmony_ci	case SRPT_STATE_CMD_RSP_SENT:
130362306a36Sopenharmony_ci		/*
130462306a36Sopenharmony_ci		 * SRP_RSP sending failed or the SRP_RSP send completion has
130562306a36Sopenharmony_ci		 * not been received in time.
130662306a36Sopenharmony_ci		 */
130762306a36Sopenharmony_ci		transport_generic_free_cmd(&ioctx->cmd, 0);
130862306a36Sopenharmony_ci		break;
130962306a36Sopenharmony_ci	case SRPT_STATE_MGMT_RSP_SENT:
131062306a36Sopenharmony_ci		transport_generic_free_cmd(&ioctx->cmd, 0);
131162306a36Sopenharmony_ci		break;
131262306a36Sopenharmony_ci	default:
131362306a36Sopenharmony_ci		WARN(1, "Unexpected command state (%d)", state);
131462306a36Sopenharmony_ci		break;
131562306a36Sopenharmony_ci	}
131662306a36Sopenharmony_ci
131762306a36Sopenharmony_ci	return state;
131862306a36Sopenharmony_ci}
131962306a36Sopenharmony_ci
132062306a36Sopenharmony_ci/**
132162306a36Sopenharmony_ci * srpt_rdma_read_done - RDMA read completion callback
132262306a36Sopenharmony_ci * @cq: Completion queue.
132362306a36Sopenharmony_ci * @wc: Work completion.
132462306a36Sopenharmony_ci *
132562306a36Sopenharmony_ci * XXX: what is now target_execute_cmd used to be asynchronous, and unmapping
132662306a36Sopenharmony_ci * the data that has been transferred via IB RDMA had to be postponed until the
132762306a36Sopenharmony_ci * check_stop_free() callback.  None of this is necessary anymore and needs to
132862306a36Sopenharmony_ci * be cleaned up.
132962306a36Sopenharmony_ci */
133062306a36Sopenharmony_cistatic void srpt_rdma_read_done(struct ib_cq *cq, struct ib_wc *wc)
133162306a36Sopenharmony_ci{
133262306a36Sopenharmony_ci	struct srpt_rdma_ch *ch = wc->qp->qp_context;
133362306a36Sopenharmony_ci	struct srpt_send_ioctx *ioctx =
133462306a36Sopenharmony_ci		container_of(wc->wr_cqe, struct srpt_send_ioctx, rdma_cqe);
133562306a36Sopenharmony_ci
133662306a36Sopenharmony_ci	WARN_ON(ioctx->n_rdma <= 0);
133762306a36Sopenharmony_ci	atomic_add(ioctx->n_rdma, &ch->sq_wr_avail);
133862306a36Sopenharmony_ci	ioctx->n_rdma = 0;
133962306a36Sopenharmony_ci
134062306a36Sopenharmony_ci	if (unlikely(wc->status != IB_WC_SUCCESS)) {
134162306a36Sopenharmony_ci		pr_info("RDMA_READ for ioctx 0x%p failed with status %d\n",
134262306a36Sopenharmony_ci			ioctx, wc->status);
134362306a36Sopenharmony_ci		srpt_abort_cmd(ioctx);
134462306a36Sopenharmony_ci		return;
134562306a36Sopenharmony_ci	}
134662306a36Sopenharmony_ci
134762306a36Sopenharmony_ci	if (srpt_test_and_set_cmd_state(ioctx, SRPT_STATE_NEED_DATA,
134862306a36Sopenharmony_ci					SRPT_STATE_DATA_IN))
134962306a36Sopenharmony_ci		target_execute_cmd(&ioctx->cmd);
135062306a36Sopenharmony_ci	else
135162306a36Sopenharmony_ci		pr_err("%s[%d]: wrong state = %d\n", __func__,
135262306a36Sopenharmony_ci		       __LINE__, ioctx->state);
135362306a36Sopenharmony_ci}
135462306a36Sopenharmony_ci
135562306a36Sopenharmony_ci/**
135662306a36Sopenharmony_ci * srpt_build_cmd_rsp - build a SRP_RSP response
135762306a36Sopenharmony_ci * @ch: RDMA channel through which the request has been received.
135862306a36Sopenharmony_ci * @ioctx: I/O context associated with the SRP_CMD request. The response will
135962306a36Sopenharmony_ci *   be built in the buffer ioctx->buf points at and hence this function will
136062306a36Sopenharmony_ci *   overwrite the request data.
136162306a36Sopenharmony_ci * @tag: tag of the request for which this response is being generated.
136262306a36Sopenharmony_ci * @status: value for the STATUS field of the SRP_RSP information unit.
136362306a36Sopenharmony_ci *
136462306a36Sopenharmony_ci * Returns the size in bytes of the SRP_RSP response.
136562306a36Sopenharmony_ci *
136662306a36Sopenharmony_ci * An SRP_RSP response contains a SCSI status or service response. See also
136762306a36Sopenharmony_ci * section 6.9 in the SRP r16a document for the format of an SRP_RSP
136862306a36Sopenharmony_ci * response. See also SPC-2 for more information about sense data.
136962306a36Sopenharmony_ci */
137062306a36Sopenharmony_cistatic int srpt_build_cmd_rsp(struct srpt_rdma_ch *ch,
137162306a36Sopenharmony_ci			      struct srpt_send_ioctx *ioctx, u64 tag,
137262306a36Sopenharmony_ci			      int status)
137362306a36Sopenharmony_ci{
137462306a36Sopenharmony_ci	struct se_cmd *cmd = &ioctx->cmd;
137562306a36Sopenharmony_ci	struct srp_rsp *srp_rsp;
137662306a36Sopenharmony_ci	const u8 *sense_data;
137762306a36Sopenharmony_ci	int sense_data_len, max_sense_len;
137862306a36Sopenharmony_ci	u32 resid = cmd->residual_count;
137962306a36Sopenharmony_ci
138062306a36Sopenharmony_ci	/*
138162306a36Sopenharmony_ci	 * The lowest bit of all SAM-3 status codes is zero (see also
138262306a36Sopenharmony_ci	 * paragraph 5.3 in SAM-3).
138362306a36Sopenharmony_ci	 */
138462306a36Sopenharmony_ci	WARN_ON(status & 1);
138562306a36Sopenharmony_ci
138662306a36Sopenharmony_ci	srp_rsp = ioctx->ioctx.buf;
138762306a36Sopenharmony_ci	BUG_ON(!srp_rsp);
138862306a36Sopenharmony_ci
138962306a36Sopenharmony_ci	sense_data = ioctx->sense_data;
139062306a36Sopenharmony_ci	sense_data_len = ioctx->cmd.scsi_sense_length;
139162306a36Sopenharmony_ci	WARN_ON(sense_data_len > sizeof(ioctx->sense_data));
139262306a36Sopenharmony_ci
139362306a36Sopenharmony_ci	memset(srp_rsp, 0, sizeof(*srp_rsp));
139462306a36Sopenharmony_ci	srp_rsp->opcode = SRP_RSP;
139562306a36Sopenharmony_ci	srp_rsp->req_lim_delta =
139662306a36Sopenharmony_ci		cpu_to_be32(1 + atomic_xchg(&ch->req_lim_delta, 0));
139762306a36Sopenharmony_ci	srp_rsp->tag = tag;
139862306a36Sopenharmony_ci	srp_rsp->status = status;
139962306a36Sopenharmony_ci
140062306a36Sopenharmony_ci	if (cmd->se_cmd_flags & SCF_UNDERFLOW_BIT) {
140162306a36Sopenharmony_ci		if (cmd->data_direction == DMA_TO_DEVICE) {
140262306a36Sopenharmony_ci			/* residual data from an underflow write */
140362306a36Sopenharmony_ci			srp_rsp->flags = SRP_RSP_FLAG_DOUNDER;
140462306a36Sopenharmony_ci			srp_rsp->data_out_res_cnt = cpu_to_be32(resid);
140562306a36Sopenharmony_ci		} else if (cmd->data_direction == DMA_FROM_DEVICE) {
140662306a36Sopenharmony_ci			/* residual data from an underflow read */
140762306a36Sopenharmony_ci			srp_rsp->flags = SRP_RSP_FLAG_DIUNDER;
140862306a36Sopenharmony_ci			srp_rsp->data_in_res_cnt = cpu_to_be32(resid);
140962306a36Sopenharmony_ci		}
141062306a36Sopenharmony_ci	} else if (cmd->se_cmd_flags & SCF_OVERFLOW_BIT) {
141162306a36Sopenharmony_ci		if (cmd->data_direction == DMA_TO_DEVICE) {
141262306a36Sopenharmony_ci			/* residual data from an overflow write */
141362306a36Sopenharmony_ci			srp_rsp->flags = SRP_RSP_FLAG_DOOVER;
141462306a36Sopenharmony_ci			srp_rsp->data_out_res_cnt = cpu_to_be32(resid);
141562306a36Sopenharmony_ci		} else if (cmd->data_direction == DMA_FROM_DEVICE) {
141662306a36Sopenharmony_ci			/* residual data from an overflow read */
141762306a36Sopenharmony_ci			srp_rsp->flags = SRP_RSP_FLAG_DIOVER;
141862306a36Sopenharmony_ci			srp_rsp->data_in_res_cnt = cpu_to_be32(resid);
141962306a36Sopenharmony_ci		}
142062306a36Sopenharmony_ci	}
142162306a36Sopenharmony_ci
142262306a36Sopenharmony_ci	if (sense_data_len) {
142362306a36Sopenharmony_ci		BUILD_BUG_ON(MIN_MAX_RSP_SIZE <= sizeof(*srp_rsp));
142462306a36Sopenharmony_ci		max_sense_len = ch->max_ti_iu_len - sizeof(*srp_rsp);
142562306a36Sopenharmony_ci		if (sense_data_len > max_sense_len) {
142662306a36Sopenharmony_ci			pr_warn("truncated sense data from %d to %d bytes\n",
142762306a36Sopenharmony_ci				sense_data_len, max_sense_len);
142862306a36Sopenharmony_ci			sense_data_len = max_sense_len;
142962306a36Sopenharmony_ci		}
143062306a36Sopenharmony_ci
143162306a36Sopenharmony_ci		srp_rsp->flags |= SRP_RSP_FLAG_SNSVALID;
143262306a36Sopenharmony_ci		srp_rsp->sense_data_len = cpu_to_be32(sense_data_len);
143362306a36Sopenharmony_ci		memcpy(srp_rsp->data, sense_data, sense_data_len);
143462306a36Sopenharmony_ci	}
143562306a36Sopenharmony_ci
143662306a36Sopenharmony_ci	return sizeof(*srp_rsp) + sense_data_len;
143762306a36Sopenharmony_ci}
143862306a36Sopenharmony_ci
143962306a36Sopenharmony_ci/**
144062306a36Sopenharmony_ci * srpt_build_tskmgmt_rsp - build a task management response
144162306a36Sopenharmony_ci * @ch:       RDMA channel through which the request has been received.
144262306a36Sopenharmony_ci * @ioctx:    I/O context in which the SRP_RSP response will be built.
144362306a36Sopenharmony_ci * @rsp_code: RSP_CODE that will be stored in the response.
144462306a36Sopenharmony_ci * @tag:      Tag of the request for which this response is being generated.
144562306a36Sopenharmony_ci *
144662306a36Sopenharmony_ci * Returns the size in bytes of the SRP_RSP response.
144762306a36Sopenharmony_ci *
144862306a36Sopenharmony_ci * An SRP_RSP response contains a SCSI status or service response. See also
144962306a36Sopenharmony_ci * section 6.9 in the SRP r16a document for the format of an SRP_RSP
145062306a36Sopenharmony_ci * response.
145162306a36Sopenharmony_ci */
145262306a36Sopenharmony_cistatic int srpt_build_tskmgmt_rsp(struct srpt_rdma_ch *ch,
145362306a36Sopenharmony_ci				  struct srpt_send_ioctx *ioctx,
145462306a36Sopenharmony_ci				  u8 rsp_code, u64 tag)
145562306a36Sopenharmony_ci{
145662306a36Sopenharmony_ci	struct srp_rsp *srp_rsp;
145762306a36Sopenharmony_ci	int resp_data_len;
145862306a36Sopenharmony_ci	int resp_len;
145962306a36Sopenharmony_ci
146062306a36Sopenharmony_ci	resp_data_len = 4;
146162306a36Sopenharmony_ci	resp_len = sizeof(*srp_rsp) + resp_data_len;
146262306a36Sopenharmony_ci
146362306a36Sopenharmony_ci	srp_rsp = ioctx->ioctx.buf;
146462306a36Sopenharmony_ci	BUG_ON(!srp_rsp);
146562306a36Sopenharmony_ci	memset(srp_rsp, 0, sizeof(*srp_rsp));
146662306a36Sopenharmony_ci
146762306a36Sopenharmony_ci	srp_rsp->opcode = SRP_RSP;
146862306a36Sopenharmony_ci	srp_rsp->req_lim_delta =
146962306a36Sopenharmony_ci		cpu_to_be32(1 + atomic_xchg(&ch->req_lim_delta, 0));
147062306a36Sopenharmony_ci	srp_rsp->tag = tag;
147162306a36Sopenharmony_ci
147262306a36Sopenharmony_ci	srp_rsp->flags |= SRP_RSP_FLAG_RSPVALID;
147362306a36Sopenharmony_ci	srp_rsp->resp_data_len = cpu_to_be32(resp_data_len);
147462306a36Sopenharmony_ci	srp_rsp->data[3] = rsp_code;
147562306a36Sopenharmony_ci
147662306a36Sopenharmony_ci	return resp_len;
147762306a36Sopenharmony_ci}
147862306a36Sopenharmony_ci
147962306a36Sopenharmony_cistatic int srpt_check_stop_free(struct se_cmd *cmd)
148062306a36Sopenharmony_ci{
148162306a36Sopenharmony_ci	struct srpt_send_ioctx *ioctx = container_of(cmd,
148262306a36Sopenharmony_ci				struct srpt_send_ioctx, cmd);
148362306a36Sopenharmony_ci
148462306a36Sopenharmony_ci	return target_put_sess_cmd(&ioctx->cmd);
148562306a36Sopenharmony_ci}
148662306a36Sopenharmony_ci
148762306a36Sopenharmony_ci/**
148862306a36Sopenharmony_ci * srpt_handle_cmd - process a SRP_CMD information unit
148962306a36Sopenharmony_ci * @ch: SRPT RDMA channel.
149062306a36Sopenharmony_ci * @recv_ioctx: Receive I/O context.
149162306a36Sopenharmony_ci * @send_ioctx: Send I/O context.
149262306a36Sopenharmony_ci */
149362306a36Sopenharmony_cistatic void srpt_handle_cmd(struct srpt_rdma_ch *ch,
149462306a36Sopenharmony_ci			    struct srpt_recv_ioctx *recv_ioctx,
149562306a36Sopenharmony_ci			    struct srpt_send_ioctx *send_ioctx)
149662306a36Sopenharmony_ci{
149762306a36Sopenharmony_ci	struct se_cmd *cmd;
149862306a36Sopenharmony_ci	struct srp_cmd *srp_cmd;
149962306a36Sopenharmony_ci	struct scatterlist *sg = NULL;
150062306a36Sopenharmony_ci	unsigned sg_cnt = 0;
150162306a36Sopenharmony_ci	u64 data_len;
150262306a36Sopenharmony_ci	enum dma_data_direction dir;
150362306a36Sopenharmony_ci	int rc;
150462306a36Sopenharmony_ci
150562306a36Sopenharmony_ci	BUG_ON(!send_ioctx);
150662306a36Sopenharmony_ci
150762306a36Sopenharmony_ci	srp_cmd = recv_ioctx->ioctx.buf + recv_ioctx->ioctx.offset;
150862306a36Sopenharmony_ci	cmd = &send_ioctx->cmd;
150962306a36Sopenharmony_ci	cmd->tag = srp_cmd->tag;
151062306a36Sopenharmony_ci
151162306a36Sopenharmony_ci	switch (srp_cmd->task_attr) {
151262306a36Sopenharmony_ci	case SRP_CMD_SIMPLE_Q:
151362306a36Sopenharmony_ci		cmd->sam_task_attr = TCM_SIMPLE_TAG;
151462306a36Sopenharmony_ci		break;
151562306a36Sopenharmony_ci	case SRP_CMD_ORDERED_Q:
151662306a36Sopenharmony_ci	default:
151762306a36Sopenharmony_ci		cmd->sam_task_attr = TCM_ORDERED_TAG;
151862306a36Sopenharmony_ci		break;
151962306a36Sopenharmony_ci	case SRP_CMD_HEAD_OF_Q:
152062306a36Sopenharmony_ci		cmd->sam_task_attr = TCM_HEAD_TAG;
152162306a36Sopenharmony_ci		break;
152262306a36Sopenharmony_ci	case SRP_CMD_ACA:
152362306a36Sopenharmony_ci		cmd->sam_task_attr = TCM_ACA_TAG;
152462306a36Sopenharmony_ci		break;
152562306a36Sopenharmony_ci	}
152662306a36Sopenharmony_ci
152762306a36Sopenharmony_ci	rc = srpt_get_desc_tbl(recv_ioctx, send_ioctx, srp_cmd, &dir,
152862306a36Sopenharmony_ci			       &sg, &sg_cnt, &data_len, ch->imm_data_offset);
152962306a36Sopenharmony_ci	if (rc) {
153062306a36Sopenharmony_ci		if (rc != -EAGAIN) {
153162306a36Sopenharmony_ci			pr_err("0x%llx: parsing SRP descriptor table failed.\n",
153262306a36Sopenharmony_ci			       srp_cmd->tag);
153362306a36Sopenharmony_ci		}
153462306a36Sopenharmony_ci		goto busy;
153562306a36Sopenharmony_ci	}
153662306a36Sopenharmony_ci
153762306a36Sopenharmony_ci	rc = target_init_cmd(cmd, ch->sess, &send_ioctx->sense_data[0],
153862306a36Sopenharmony_ci			     scsilun_to_int(&srp_cmd->lun), data_len,
153962306a36Sopenharmony_ci			     TCM_SIMPLE_TAG, dir, TARGET_SCF_ACK_KREF);
154062306a36Sopenharmony_ci	if (rc != 0) {
154162306a36Sopenharmony_ci		pr_debug("target_submit_cmd() returned %d for tag %#llx\n", rc,
154262306a36Sopenharmony_ci			 srp_cmd->tag);
154362306a36Sopenharmony_ci		goto busy;
154462306a36Sopenharmony_ci	}
154562306a36Sopenharmony_ci
154662306a36Sopenharmony_ci	if (target_submit_prep(cmd, srp_cmd->cdb, sg, sg_cnt, NULL, 0, NULL, 0,
154762306a36Sopenharmony_ci			       GFP_KERNEL))
154862306a36Sopenharmony_ci		return;
154962306a36Sopenharmony_ci
155062306a36Sopenharmony_ci	target_submit(cmd);
155162306a36Sopenharmony_ci	return;
155262306a36Sopenharmony_ci
155362306a36Sopenharmony_cibusy:
155462306a36Sopenharmony_ci	target_send_busy(cmd);
155562306a36Sopenharmony_ci}
155662306a36Sopenharmony_ci
155762306a36Sopenharmony_cistatic int srp_tmr_to_tcm(int fn)
155862306a36Sopenharmony_ci{
155962306a36Sopenharmony_ci	switch (fn) {
156062306a36Sopenharmony_ci	case SRP_TSK_ABORT_TASK:
156162306a36Sopenharmony_ci		return TMR_ABORT_TASK;
156262306a36Sopenharmony_ci	case SRP_TSK_ABORT_TASK_SET:
156362306a36Sopenharmony_ci		return TMR_ABORT_TASK_SET;
156462306a36Sopenharmony_ci	case SRP_TSK_CLEAR_TASK_SET:
156562306a36Sopenharmony_ci		return TMR_CLEAR_TASK_SET;
156662306a36Sopenharmony_ci	case SRP_TSK_LUN_RESET:
156762306a36Sopenharmony_ci		return TMR_LUN_RESET;
156862306a36Sopenharmony_ci	case SRP_TSK_CLEAR_ACA:
156962306a36Sopenharmony_ci		return TMR_CLEAR_ACA;
157062306a36Sopenharmony_ci	default:
157162306a36Sopenharmony_ci		return -1;
157262306a36Sopenharmony_ci	}
157362306a36Sopenharmony_ci}
157462306a36Sopenharmony_ci
157562306a36Sopenharmony_ci/**
157662306a36Sopenharmony_ci * srpt_handle_tsk_mgmt - process a SRP_TSK_MGMT information unit
157762306a36Sopenharmony_ci * @ch: SRPT RDMA channel.
157862306a36Sopenharmony_ci * @recv_ioctx: Receive I/O context.
157962306a36Sopenharmony_ci * @send_ioctx: Send I/O context.
158062306a36Sopenharmony_ci *
158162306a36Sopenharmony_ci * Returns 0 if and only if the request will be processed by the target core.
158262306a36Sopenharmony_ci *
158362306a36Sopenharmony_ci * For more information about SRP_TSK_MGMT information units, see also section
158462306a36Sopenharmony_ci * 6.7 in the SRP r16a document.
158562306a36Sopenharmony_ci */
158662306a36Sopenharmony_cistatic void srpt_handle_tsk_mgmt(struct srpt_rdma_ch *ch,
158762306a36Sopenharmony_ci				 struct srpt_recv_ioctx *recv_ioctx,
158862306a36Sopenharmony_ci				 struct srpt_send_ioctx *send_ioctx)
158962306a36Sopenharmony_ci{
159062306a36Sopenharmony_ci	struct srp_tsk_mgmt *srp_tsk;
159162306a36Sopenharmony_ci	struct se_cmd *cmd;
159262306a36Sopenharmony_ci	struct se_session *sess = ch->sess;
159362306a36Sopenharmony_ci	int tcm_tmr;
159462306a36Sopenharmony_ci	int rc;
159562306a36Sopenharmony_ci
159662306a36Sopenharmony_ci	BUG_ON(!send_ioctx);
159762306a36Sopenharmony_ci
159862306a36Sopenharmony_ci	srp_tsk = recv_ioctx->ioctx.buf + recv_ioctx->ioctx.offset;
159962306a36Sopenharmony_ci	cmd = &send_ioctx->cmd;
160062306a36Sopenharmony_ci
160162306a36Sopenharmony_ci	pr_debug("recv tsk_mgmt fn %d for task_tag %lld and cmd tag %lld ch %p sess %p\n",
160262306a36Sopenharmony_ci		 srp_tsk->tsk_mgmt_func, srp_tsk->task_tag, srp_tsk->tag, ch,
160362306a36Sopenharmony_ci		 ch->sess);
160462306a36Sopenharmony_ci
160562306a36Sopenharmony_ci	srpt_set_cmd_state(send_ioctx, SRPT_STATE_MGMT);
160662306a36Sopenharmony_ci	send_ioctx->cmd.tag = srp_tsk->tag;
160762306a36Sopenharmony_ci	tcm_tmr = srp_tmr_to_tcm(srp_tsk->tsk_mgmt_func);
160862306a36Sopenharmony_ci	rc = target_submit_tmr(&send_ioctx->cmd, sess, NULL,
160962306a36Sopenharmony_ci			       scsilun_to_int(&srp_tsk->lun), srp_tsk, tcm_tmr,
161062306a36Sopenharmony_ci			       GFP_KERNEL, srp_tsk->task_tag,
161162306a36Sopenharmony_ci			       TARGET_SCF_ACK_KREF);
161262306a36Sopenharmony_ci	if (rc != 0) {
161362306a36Sopenharmony_ci		send_ioctx->cmd.se_tmr_req->response = TMR_FUNCTION_REJECTED;
161462306a36Sopenharmony_ci		cmd->se_tfo->queue_tm_rsp(cmd);
161562306a36Sopenharmony_ci	}
161662306a36Sopenharmony_ci	return;
161762306a36Sopenharmony_ci}
161862306a36Sopenharmony_ci
161962306a36Sopenharmony_ci/**
162062306a36Sopenharmony_ci * srpt_handle_new_iu - process a newly received information unit
162162306a36Sopenharmony_ci * @ch:    RDMA channel through which the information unit has been received.
162262306a36Sopenharmony_ci * @recv_ioctx: Receive I/O context associated with the information unit.
162362306a36Sopenharmony_ci */
162462306a36Sopenharmony_cistatic bool
162562306a36Sopenharmony_cisrpt_handle_new_iu(struct srpt_rdma_ch *ch, struct srpt_recv_ioctx *recv_ioctx)
162662306a36Sopenharmony_ci{
162762306a36Sopenharmony_ci	struct srpt_send_ioctx *send_ioctx = NULL;
162862306a36Sopenharmony_ci	struct srp_cmd *srp_cmd;
162962306a36Sopenharmony_ci	bool res = false;
163062306a36Sopenharmony_ci	u8 opcode;
163162306a36Sopenharmony_ci
163262306a36Sopenharmony_ci	BUG_ON(!ch);
163362306a36Sopenharmony_ci	BUG_ON(!recv_ioctx);
163462306a36Sopenharmony_ci
163562306a36Sopenharmony_ci	if (unlikely(ch->state == CH_CONNECTING))
163662306a36Sopenharmony_ci		goto push;
163762306a36Sopenharmony_ci
163862306a36Sopenharmony_ci	ib_dma_sync_single_for_cpu(ch->sport->sdev->device,
163962306a36Sopenharmony_ci				   recv_ioctx->ioctx.dma,
164062306a36Sopenharmony_ci				   recv_ioctx->ioctx.offset + srp_max_req_size,
164162306a36Sopenharmony_ci				   DMA_FROM_DEVICE);
164262306a36Sopenharmony_ci
164362306a36Sopenharmony_ci	srp_cmd = recv_ioctx->ioctx.buf + recv_ioctx->ioctx.offset;
164462306a36Sopenharmony_ci	opcode = srp_cmd->opcode;
164562306a36Sopenharmony_ci	if (opcode == SRP_CMD || opcode == SRP_TSK_MGMT) {
164662306a36Sopenharmony_ci		send_ioctx = srpt_get_send_ioctx(ch);
164762306a36Sopenharmony_ci		if (unlikely(!send_ioctx))
164862306a36Sopenharmony_ci			goto push;
164962306a36Sopenharmony_ci	}
165062306a36Sopenharmony_ci
165162306a36Sopenharmony_ci	if (!list_empty(&recv_ioctx->wait_list)) {
165262306a36Sopenharmony_ci		WARN_ON_ONCE(!ch->processing_wait_list);
165362306a36Sopenharmony_ci		list_del_init(&recv_ioctx->wait_list);
165462306a36Sopenharmony_ci	}
165562306a36Sopenharmony_ci
165662306a36Sopenharmony_ci	switch (opcode) {
165762306a36Sopenharmony_ci	case SRP_CMD:
165862306a36Sopenharmony_ci		srpt_handle_cmd(ch, recv_ioctx, send_ioctx);
165962306a36Sopenharmony_ci		break;
166062306a36Sopenharmony_ci	case SRP_TSK_MGMT:
166162306a36Sopenharmony_ci		srpt_handle_tsk_mgmt(ch, recv_ioctx, send_ioctx);
166262306a36Sopenharmony_ci		break;
166362306a36Sopenharmony_ci	case SRP_I_LOGOUT:
166462306a36Sopenharmony_ci		pr_err("Not yet implemented: SRP_I_LOGOUT\n");
166562306a36Sopenharmony_ci		break;
166662306a36Sopenharmony_ci	case SRP_CRED_RSP:
166762306a36Sopenharmony_ci		pr_debug("received SRP_CRED_RSP\n");
166862306a36Sopenharmony_ci		break;
166962306a36Sopenharmony_ci	case SRP_AER_RSP:
167062306a36Sopenharmony_ci		pr_debug("received SRP_AER_RSP\n");
167162306a36Sopenharmony_ci		break;
167262306a36Sopenharmony_ci	case SRP_RSP:
167362306a36Sopenharmony_ci		pr_err("Received SRP_RSP\n");
167462306a36Sopenharmony_ci		break;
167562306a36Sopenharmony_ci	default:
167662306a36Sopenharmony_ci		pr_err("received IU with unknown opcode 0x%x\n", opcode);
167762306a36Sopenharmony_ci		break;
167862306a36Sopenharmony_ci	}
167962306a36Sopenharmony_ci
168062306a36Sopenharmony_ci	if (!send_ioctx || !send_ioctx->recv_ioctx)
168162306a36Sopenharmony_ci		srpt_post_recv(ch->sport->sdev, ch, recv_ioctx);
168262306a36Sopenharmony_ci	res = true;
168362306a36Sopenharmony_ci
168462306a36Sopenharmony_ciout:
168562306a36Sopenharmony_ci	return res;
168662306a36Sopenharmony_ci
168762306a36Sopenharmony_cipush:
168862306a36Sopenharmony_ci	if (list_empty(&recv_ioctx->wait_list)) {
168962306a36Sopenharmony_ci		WARN_ON_ONCE(ch->processing_wait_list);
169062306a36Sopenharmony_ci		list_add_tail(&recv_ioctx->wait_list, &ch->cmd_wait_list);
169162306a36Sopenharmony_ci	}
169262306a36Sopenharmony_ci	goto out;
169362306a36Sopenharmony_ci}
169462306a36Sopenharmony_ci
169562306a36Sopenharmony_cistatic void srpt_recv_done(struct ib_cq *cq, struct ib_wc *wc)
169662306a36Sopenharmony_ci{
169762306a36Sopenharmony_ci	struct srpt_rdma_ch *ch = wc->qp->qp_context;
169862306a36Sopenharmony_ci	struct srpt_recv_ioctx *ioctx =
169962306a36Sopenharmony_ci		container_of(wc->wr_cqe, struct srpt_recv_ioctx, ioctx.cqe);
170062306a36Sopenharmony_ci
170162306a36Sopenharmony_ci	if (wc->status == IB_WC_SUCCESS) {
170262306a36Sopenharmony_ci		int req_lim;
170362306a36Sopenharmony_ci
170462306a36Sopenharmony_ci		req_lim = atomic_dec_return(&ch->req_lim);
170562306a36Sopenharmony_ci		if (unlikely(req_lim < 0))
170662306a36Sopenharmony_ci			pr_err("req_lim = %d < 0\n", req_lim);
170762306a36Sopenharmony_ci		ioctx->byte_len = wc->byte_len;
170862306a36Sopenharmony_ci		srpt_handle_new_iu(ch, ioctx);
170962306a36Sopenharmony_ci	} else {
171062306a36Sopenharmony_ci		pr_info_ratelimited("receiving failed for ioctx %p with status %d\n",
171162306a36Sopenharmony_ci				    ioctx, wc->status);
171262306a36Sopenharmony_ci	}
171362306a36Sopenharmony_ci}
171462306a36Sopenharmony_ci
171562306a36Sopenharmony_ci/*
171662306a36Sopenharmony_ci * This function must be called from the context in which RDMA completions are
171762306a36Sopenharmony_ci * processed because it accesses the wait list without protection against
171862306a36Sopenharmony_ci * access from other threads.
171962306a36Sopenharmony_ci */
172062306a36Sopenharmony_cistatic void srpt_process_wait_list(struct srpt_rdma_ch *ch)
172162306a36Sopenharmony_ci{
172262306a36Sopenharmony_ci	struct srpt_recv_ioctx *recv_ioctx, *tmp;
172362306a36Sopenharmony_ci
172462306a36Sopenharmony_ci	WARN_ON_ONCE(ch->state == CH_CONNECTING);
172562306a36Sopenharmony_ci
172662306a36Sopenharmony_ci	if (list_empty(&ch->cmd_wait_list))
172762306a36Sopenharmony_ci		return;
172862306a36Sopenharmony_ci
172962306a36Sopenharmony_ci	WARN_ON_ONCE(ch->processing_wait_list);
173062306a36Sopenharmony_ci	ch->processing_wait_list = true;
173162306a36Sopenharmony_ci	list_for_each_entry_safe(recv_ioctx, tmp, &ch->cmd_wait_list,
173262306a36Sopenharmony_ci				 wait_list) {
173362306a36Sopenharmony_ci		if (!srpt_handle_new_iu(ch, recv_ioctx))
173462306a36Sopenharmony_ci			break;
173562306a36Sopenharmony_ci	}
173662306a36Sopenharmony_ci	ch->processing_wait_list = false;
173762306a36Sopenharmony_ci}
173862306a36Sopenharmony_ci
173962306a36Sopenharmony_ci/**
174062306a36Sopenharmony_ci * srpt_send_done - send completion callback
174162306a36Sopenharmony_ci * @cq: Completion queue.
174262306a36Sopenharmony_ci * @wc: Work completion.
174362306a36Sopenharmony_ci *
174462306a36Sopenharmony_ci * Note: Although this has not yet been observed during tests, at least in
174562306a36Sopenharmony_ci * theory it is possible that the srpt_get_send_ioctx() call invoked by
174662306a36Sopenharmony_ci * srpt_handle_new_iu() fails. This is possible because the req_lim_delta
174762306a36Sopenharmony_ci * value in each response is set to one, and it is possible that this response
174862306a36Sopenharmony_ci * makes the initiator send a new request before the send completion for that
174962306a36Sopenharmony_ci * response has been processed. This could e.g. happen if the call to
175062306a36Sopenharmony_ci * srpt_put_send_iotcx() is delayed because of a higher priority interrupt or
175162306a36Sopenharmony_ci * if IB retransmission causes generation of the send completion to be
175262306a36Sopenharmony_ci * delayed. Incoming information units for which srpt_get_send_ioctx() fails
175362306a36Sopenharmony_ci * are queued on cmd_wait_list. The code below processes these delayed
175462306a36Sopenharmony_ci * requests one at a time.
175562306a36Sopenharmony_ci */
175662306a36Sopenharmony_cistatic void srpt_send_done(struct ib_cq *cq, struct ib_wc *wc)
175762306a36Sopenharmony_ci{
175862306a36Sopenharmony_ci	struct srpt_rdma_ch *ch = wc->qp->qp_context;
175962306a36Sopenharmony_ci	struct srpt_send_ioctx *ioctx =
176062306a36Sopenharmony_ci		container_of(wc->wr_cqe, struct srpt_send_ioctx, ioctx.cqe);
176162306a36Sopenharmony_ci	enum srpt_command_state state;
176262306a36Sopenharmony_ci
176362306a36Sopenharmony_ci	state = srpt_set_cmd_state(ioctx, SRPT_STATE_DONE);
176462306a36Sopenharmony_ci
176562306a36Sopenharmony_ci	WARN_ON(state != SRPT_STATE_CMD_RSP_SENT &&
176662306a36Sopenharmony_ci		state != SRPT_STATE_MGMT_RSP_SENT);
176762306a36Sopenharmony_ci
176862306a36Sopenharmony_ci	atomic_add(1 + ioctx->n_rdma, &ch->sq_wr_avail);
176962306a36Sopenharmony_ci
177062306a36Sopenharmony_ci	if (wc->status != IB_WC_SUCCESS)
177162306a36Sopenharmony_ci		pr_info("sending response for ioctx 0x%p failed with status %d\n",
177262306a36Sopenharmony_ci			ioctx, wc->status);
177362306a36Sopenharmony_ci
177462306a36Sopenharmony_ci	if (state != SRPT_STATE_DONE) {
177562306a36Sopenharmony_ci		transport_generic_free_cmd(&ioctx->cmd, 0);
177662306a36Sopenharmony_ci	} else {
177762306a36Sopenharmony_ci		pr_err("IB completion has been received too late for wr_id = %u.\n",
177862306a36Sopenharmony_ci		       ioctx->ioctx.index);
177962306a36Sopenharmony_ci	}
178062306a36Sopenharmony_ci
178162306a36Sopenharmony_ci	srpt_process_wait_list(ch);
178262306a36Sopenharmony_ci}
178362306a36Sopenharmony_ci
178462306a36Sopenharmony_ci/**
178562306a36Sopenharmony_ci * srpt_create_ch_ib - create receive and send completion queues
178662306a36Sopenharmony_ci * @ch: SRPT RDMA channel.
178762306a36Sopenharmony_ci */
178862306a36Sopenharmony_cistatic int srpt_create_ch_ib(struct srpt_rdma_ch *ch)
178962306a36Sopenharmony_ci{
179062306a36Sopenharmony_ci	struct ib_qp_init_attr *qp_init;
179162306a36Sopenharmony_ci	struct srpt_port *sport = ch->sport;
179262306a36Sopenharmony_ci	struct srpt_device *sdev = sport->sdev;
179362306a36Sopenharmony_ci	const struct ib_device_attr *attrs = &sdev->device->attrs;
179462306a36Sopenharmony_ci	int sq_size = sport->port_attrib.srp_sq_size;
179562306a36Sopenharmony_ci	int i, ret;
179662306a36Sopenharmony_ci
179762306a36Sopenharmony_ci	WARN_ON(ch->rq_size < 1);
179862306a36Sopenharmony_ci
179962306a36Sopenharmony_ci	ret = -ENOMEM;
180062306a36Sopenharmony_ci	qp_init = kzalloc(sizeof(*qp_init), GFP_KERNEL);
180162306a36Sopenharmony_ci	if (!qp_init)
180262306a36Sopenharmony_ci		goto out;
180362306a36Sopenharmony_ci
180462306a36Sopenharmony_ciretry:
180562306a36Sopenharmony_ci	ch->cq = ib_cq_pool_get(sdev->device, ch->rq_size + sq_size, -1,
180662306a36Sopenharmony_ci				 IB_POLL_WORKQUEUE);
180762306a36Sopenharmony_ci	if (IS_ERR(ch->cq)) {
180862306a36Sopenharmony_ci		ret = PTR_ERR(ch->cq);
180962306a36Sopenharmony_ci		pr_err("failed to create CQ cqe= %d ret= %d\n",
181062306a36Sopenharmony_ci		       ch->rq_size + sq_size, ret);
181162306a36Sopenharmony_ci		goto out;
181262306a36Sopenharmony_ci	}
181362306a36Sopenharmony_ci	ch->cq_size = ch->rq_size + sq_size;
181462306a36Sopenharmony_ci
181562306a36Sopenharmony_ci	qp_init->qp_context = (void *)ch;
181662306a36Sopenharmony_ci	qp_init->event_handler = srpt_qp_event;
181762306a36Sopenharmony_ci	qp_init->send_cq = ch->cq;
181862306a36Sopenharmony_ci	qp_init->recv_cq = ch->cq;
181962306a36Sopenharmony_ci	qp_init->sq_sig_type = IB_SIGNAL_REQ_WR;
182062306a36Sopenharmony_ci	qp_init->qp_type = IB_QPT_RC;
182162306a36Sopenharmony_ci	/*
182262306a36Sopenharmony_ci	 * We divide up our send queue size into half SEND WRs to send the
182362306a36Sopenharmony_ci	 * completions, and half R/W contexts to actually do the RDMA
182462306a36Sopenharmony_ci	 * READ/WRITE transfers.  Note that we need to allocate CQ slots for
182562306a36Sopenharmony_ci	 * both both, as RDMA contexts will also post completions for the
182662306a36Sopenharmony_ci	 * RDMA READ case.
182762306a36Sopenharmony_ci	 */
182862306a36Sopenharmony_ci	qp_init->cap.max_send_wr = min(sq_size / 2, attrs->max_qp_wr);
182962306a36Sopenharmony_ci	qp_init->cap.max_rdma_ctxs = sq_size / 2;
183062306a36Sopenharmony_ci	qp_init->cap.max_send_sge = attrs->max_send_sge;
183162306a36Sopenharmony_ci	qp_init->cap.max_recv_sge = 1;
183262306a36Sopenharmony_ci	qp_init->port_num = ch->sport->port;
183362306a36Sopenharmony_ci	if (sdev->use_srq)
183462306a36Sopenharmony_ci		qp_init->srq = sdev->srq;
183562306a36Sopenharmony_ci	else
183662306a36Sopenharmony_ci		qp_init->cap.max_recv_wr = ch->rq_size;
183762306a36Sopenharmony_ci
183862306a36Sopenharmony_ci	if (ch->using_rdma_cm) {
183962306a36Sopenharmony_ci		ret = rdma_create_qp(ch->rdma_cm.cm_id, sdev->pd, qp_init);
184062306a36Sopenharmony_ci		ch->qp = ch->rdma_cm.cm_id->qp;
184162306a36Sopenharmony_ci	} else {
184262306a36Sopenharmony_ci		ch->qp = ib_create_qp(sdev->pd, qp_init);
184362306a36Sopenharmony_ci		if (!IS_ERR(ch->qp)) {
184462306a36Sopenharmony_ci			ret = srpt_init_ch_qp(ch, ch->qp);
184562306a36Sopenharmony_ci			if (ret)
184662306a36Sopenharmony_ci				ib_destroy_qp(ch->qp);
184762306a36Sopenharmony_ci		} else {
184862306a36Sopenharmony_ci			ret = PTR_ERR(ch->qp);
184962306a36Sopenharmony_ci		}
185062306a36Sopenharmony_ci	}
185162306a36Sopenharmony_ci	if (ret) {
185262306a36Sopenharmony_ci		bool retry = sq_size > MIN_SRPT_SQ_SIZE;
185362306a36Sopenharmony_ci
185462306a36Sopenharmony_ci		if (retry) {
185562306a36Sopenharmony_ci			pr_debug("failed to create queue pair with sq_size = %d (%d) - retrying\n",
185662306a36Sopenharmony_ci				 sq_size, ret);
185762306a36Sopenharmony_ci			ib_cq_pool_put(ch->cq, ch->cq_size);
185862306a36Sopenharmony_ci			sq_size = max(sq_size / 2, MIN_SRPT_SQ_SIZE);
185962306a36Sopenharmony_ci			goto retry;
186062306a36Sopenharmony_ci		} else {
186162306a36Sopenharmony_ci			pr_err("failed to create queue pair with sq_size = %d (%d)\n",
186262306a36Sopenharmony_ci			       sq_size, ret);
186362306a36Sopenharmony_ci			goto err_destroy_cq;
186462306a36Sopenharmony_ci		}
186562306a36Sopenharmony_ci	}
186662306a36Sopenharmony_ci
186762306a36Sopenharmony_ci	atomic_set(&ch->sq_wr_avail, qp_init->cap.max_send_wr);
186862306a36Sopenharmony_ci
186962306a36Sopenharmony_ci	pr_debug("%s: max_cqe= %d max_sge= %d sq_size = %d ch= %p\n",
187062306a36Sopenharmony_ci		 __func__, ch->cq->cqe, qp_init->cap.max_send_sge,
187162306a36Sopenharmony_ci		 qp_init->cap.max_send_wr, ch);
187262306a36Sopenharmony_ci
187362306a36Sopenharmony_ci	if (!sdev->use_srq)
187462306a36Sopenharmony_ci		for (i = 0; i < ch->rq_size; i++)
187562306a36Sopenharmony_ci			srpt_post_recv(sdev, ch, ch->ioctx_recv_ring[i]);
187662306a36Sopenharmony_ci
187762306a36Sopenharmony_ciout:
187862306a36Sopenharmony_ci	kfree(qp_init);
187962306a36Sopenharmony_ci	return ret;
188062306a36Sopenharmony_ci
188162306a36Sopenharmony_cierr_destroy_cq:
188262306a36Sopenharmony_ci	ch->qp = NULL;
188362306a36Sopenharmony_ci	ib_cq_pool_put(ch->cq, ch->cq_size);
188462306a36Sopenharmony_ci	goto out;
188562306a36Sopenharmony_ci}
188662306a36Sopenharmony_ci
188762306a36Sopenharmony_cistatic void srpt_destroy_ch_ib(struct srpt_rdma_ch *ch)
188862306a36Sopenharmony_ci{
188962306a36Sopenharmony_ci	ib_destroy_qp(ch->qp);
189062306a36Sopenharmony_ci	ib_cq_pool_put(ch->cq, ch->cq_size);
189162306a36Sopenharmony_ci}
189262306a36Sopenharmony_ci
189362306a36Sopenharmony_ci/**
189462306a36Sopenharmony_ci * srpt_close_ch - close a RDMA channel
189562306a36Sopenharmony_ci * @ch: SRPT RDMA channel.
189662306a36Sopenharmony_ci *
189762306a36Sopenharmony_ci * Make sure all resources associated with the channel will be deallocated at
189862306a36Sopenharmony_ci * an appropriate time.
189962306a36Sopenharmony_ci *
190062306a36Sopenharmony_ci * Returns true if and only if the channel state has been modified into
190162306a36Sopenharmony_ci * CH_DRAINING.
190262306a36Sopenharmony_ci */
190362306a36Sopenharmony_cistatic bool srpt_close_ch(struct srpt_rdma_ch *ch)
190462306a36Sopenharmony_ci{
190562306a36Sopenharmony_ci	int ret;
190662306a36Sopenharmony_ci
190762306a36Sopenharmony_ci	if (!srpt_set_ch_state(ch, CH_DRAINING)) {
190862306a36Sopenharmony_ci		pr_debug("%s: already closed\n", ch->sess_name);
190962306a36Sopenharmony_ci		return false;
191062306a36Sopenharmony_ci	}
191162306a36Sopenharmony_ci
191262306a36Sopenharmony_ci	kref_get(&ch->kref);
191362306a36Sopenharmony_ci
191462306a36Sopenharmony_ci	ret = srpt_ch_qp_err(ch);
191562306a36Sopenharmony_ci	if (ret < 0)
191662306a36Sopenharmony_ci		pr_err("%s-%d: changing queue pair into error state failed: %d\n",
191762306a36Sopenharmony_ci		       ch->sess_name, ch->qp->qp_num, ret);
191862306a36Sopenharmony_ci
191962306a36Sopenharmony_ci	ret = srpt_zerolength_write(ch);
192062306a36Sopenharmony_ci	if (ret < 0) {
192162306a36Sopenharmony_ci		pr_err("%s-%d: queuing zero-length write failed: %d\n",
192262306a36Sopenharmony_ci		       ch->sess_name, ch->qp->qp_num, ret);
192362306a36Sopenharmony_ci		if (srpt_set_ch_state(ch, CH_DISCONNECTED))
192462306a36Sopenharmony_ci			schedule_work(&ch->release_work);
192562306a36Sopenharmony_ci		else
192662306a36Sopenharmony_ci			WARN_ON_ONCE(true);
192762306a36Sopenharmony_ci	}
192862306a36Sopenharmony_ci
192962306a36Sopenharmony_ci	kref_put(&ch->kref, srpt_free_ch);
193062306a36Sopenharmony_ci
193162306a36Sopenharmony_ci	return true;
193262306a36Sopenharmony_ci}
193362306a36Sopenharmony_ci
193462306a36Sopenharmony_ci/*
193562306a36Sopenharmony_ci * Change the channel state into CH_DISCONNECTING. If a channel has not yet
193662306a36Sopenharmony_ci * reached the connected state, close it. If a channel is in the connected
193762306a36Sopenharmony_ci * state, send a DREQ. If a DREQ has been received, send a DREP. Note: it is
193862306a36Sopenharmony_ci * the responsibility of the caller to ensure that this function is not
193962306a36Sopenharmony_ci * invoked concurrently with the code that accepts a connection. This means
194062306a36Sopenharmony_ci * that this function must either be invoked from inside a CM callback
194162306a36Sopenharmony_ci * function or that it must be invoked with the srpt_port.mutex held.
194262306a36Sopenharmony_ci */
194362306a36Sopenharmony_cistatic int srpt_disconnect_ch(struct srpt_rdma_ch *ch)
194462306a36Sopenharmony_ci{
194562306a36Sopenharmony_ci	int ret;
194662306a36Sopenharmony_ci
194762306a36Sopenharmony_ci	if (!srpt_set_ch_state(ch, CH_DISCONNECTING))
194862306a36Sopenharmony_ci		return -ENOTCONN;
194962306a36Sopenharmony_ci
195062306a36Sopenharmony_ci	if (ch->using_rdma_cm) {
195162306a36Sopenharmony_ci		ret = rdma_disconnect(ch->rdma_cm.cm_id);
195262306a36Sopenharmony_ci	} else {
195362306a36Sopenharmony_ci		ret = ib_send_cm_dreq(ch->ib_cm.cm_id, NULL, 0);
195462306a36Sopenharmony_ci		if (ret < 0)
195562306a36Sopenharmony_ci			ret = ib_send_cm_drep(ch->ib_cm.cm_id, NULL, 0);
195662306a36Sopenharmony_ci	}
195762306a36Sopenharmony_ci
195862306a36Sopenharmony_ci	if (ret < 0 && srpt_close_ch(ch))
195962306a36Sopenharmony_ci		ret = 0;
196062306a36Sopenharmony_ci
196162306a36Sopenharmony_ci	return ret;
196262306a36Sopenharmony_ci}
196362306a36Sopenharmony_ci
196462306a36Sopenharmony_ci/* Send DREQ and wait for DREP. */
196562306a36Sopenharmony_cistatic void srpt_disconnect_ch_sync(struct srpt_rdma_ch *ch)
196662306a36Sopenharmony_ci{
196762306a36Sopenharmony_ci	DECLARE_COMPLETION_ONSTACK(closed);
196862306a36Sopenharmony_ci	struct srpt_port *sport = ch->sport;
196962306a36Sopenharmony_ci
197062306a36Sopenharmony_ci	pr_debug("ch %s-%d state %d\n", ch->sess_name, ch->qp->qp_num,
197162306a36Sopenharmony_ci		 ch->state);
197262306a36Sopenharmony_ci
197362306a36Sopenharmony_ci	ch->closed = &closed;
197462306a36Sopenharmony_ci
197562306a36Sopenharmony_ci	mutex_lock(&sport->mutex);
197662306a36Sopenharmony_ci	srpt_disconnect_ch(ch);
197762306a36Sopenharmony_ci	mutex_unlock(&sport->mutex);
197862306a36Sopenharmony_ci
197962306a36Sopenharmony_ci	while (wait_for_completion_timeout(&closed, 5 * HZ) == 0)
198062306a36Sopenharmony_ci		pr_info("%s(%s-%d state %d): still waiting ...\n", __func__,
198162306a36Sopenharmony_ci			ch->sess_name, ch->qp->qp_num, ch->state);
198262306a36Sopenharmony_ci
198362306a36Sopenharmony_ci}
198462306a36Sopenharmony_ci
198562306a36Sopenharmony_cistatic void __srpt_close_all_ch(struct srpt_port *sport)
198662306a36Sopenharmony_ci{
198762306a36Sopenharmony_ci	struct srpt_nexus *nexus;
198862306a36Sopenharmony_ci	struct srpt_rdma_ch *ch;
198962306a36Sopenharmony_ci
199062306a36Sopenharmony_ci	lockdep_assert_held(&sport->mutex);
199162306a36Sopenharmony_ci
199262306a36Sopenharmony_ci	list_for_each_entry(nexus, &sport->nexus_list, entry) {
199362306a36Sopenharmony_ci		list_for_each_entry(ch, &nexus->ch_list, list) {
199462306a36Sopenharmony_ci			if (srpt_disconnect_ch(ch) >= 0)
199562306a36Sopenharmony_ci				pr_info("Closing channel %s-%d because target %s_%d has been disabled\n",
199662306a36Sopenharmony_ci					ch->sess_name, ch->qp->qp_num,
199762306a36Sopenharmony_ci					dev_name(&sport->sdev->device->dev),
199862306a36Sopenharmony_ci					sport->port);
199962306a36Sopenharmony_ci			srpt_close_ch(ch);
200062306a36Sopenharmony_ci		}
200162306a36Sopenharmony_ci	}
200262306a36Sopenharmony_ci}
200362306a36Sopenharmony_ci
200462306a36Sopenharmony_ci/*
200562306a36Sopenharmony_ci * Look up (i_port_id, t_port_id) in sport->nexus_list. Create an entry if
200662306a36Sopenharmony_ci * it does not yet exist.
200762306a36Sopenharmony_ci */
200862306a36Sopenharmony_cistatic struct srpt_nexus *srpt_get_nexus(struct srpt_port *sport,
200962306a36Sopenharmony_ci					 const u8 i_port_id[16],
201062306a36Sopenharmony_ci					 const u8 t_port_id[16])
201162306a36Sopenharmony_ci{
201262306a36Sopenharmony_ci	struct srpt_nexus *nexus = NULL, *tmp_nexus = NULL, *n;
201362306a36Sopenharmony_ci
201462306a36Sopenharmony_ci	for (;;) {
201562306a36Sopenharmony_ci		mutex_lock(&sport->mutex);
201662306a36Sopenharmony_ci		list_for_each_entry(n, &sport->nexus_list, entry) {
201762306a36Sopenharmony_ci			if (memcmp(n->i_port_id, i_port_id, 16) == 0 &&
201862306a36Sopenharmony_ci			    memcmp(n->t_port_id, t_port_id, 16) == 0) {
201962306a36Sopenharmony_ci				nexus = n;
202062306a36Sopenharmony_ci				break;
202162306a36Sopenharmony_ci			}
202262306a36Sopenharmony_ci		}
202362306a36Sopenharmony_ci		if (!nexus && tmp_nexus) {
202462306a36Sopenharmony_ci			list_add_tail_rcu(&tmp_nexus->entry,
202562306a36Sopenharmony_ci					  &sport->nexus_list);
202662306a36Sopenharmony_ci			swap(nexus, tmp_nexus);
202762306a36Sopenharmony_ci		}
202862306a36Sopenharmony_ci		mutex_unlock(&sport->mutex);
202962306a36Sopenharmony_ci
203062306a36Sopenharmony_ci		if (nexus)
203162306a36Sopenharmony_ci			break;
203262306a36Sopenharmony_ci		tmp_nexus = kzalloc(sizeof(*nexus), GFP_KERNEL);
203362306a36Sopenharmony_ci		if (!tmp_nexus) {
203462306a36Sopenharmony_ci			nexus = ERR_PTR(-ENOMEM);
203562306a36Sopenharmony_ci			break;
203662306a36Sopenharmony_ci		}
203762306a36Sopenharmony_ci		INIT_LIST_HEAD(&tmp_nexus->ch_list);
203862306a36Sopenharmony_ci		memcpy(tmp_nexus->i_port_id, i_port_id, 16);
203962306a36Sopenharmony_ci		memcpy(tmp_nexus->t_port_id, t_port_id, 16);
204062306a36Sopenharmony_ci	}
204162306a36Sopenharmony_ci
204262306a36Sopenharmony_ci	kfree(tmp_nexus);
204362306a36Sopenharmony_ci
204462306a36Sopenharmony_ci	return nexus;
204562306a36Sopenharmony_ci}
204662306a36Sopenharmony_ci
204762306a36Sopenharmony_cistatic void srpt_set_enabled(struct srpt_port *sport, bool enabled)
204862306a36Sopenharmony_ci	__must_hold(&sport->mutex)
204962306a36Sopenharmony_ci{
205062306a36Sopenharmony_ci	lockdep_assert_held(&sport->mutex);
205162306a36Sopenharmony_ci
205262306a36Sopenharmony_ci	if (sport->enabled == enabled)
205362306a36Sopenharmony_ci		return;
205462306a36Sopenharmony_ci	sport->enabled = enabled;
205562306a36Sopenharmony_ci	if (!enabled)
205662306a36Sopenharmony_ci		__srpt_close_all_ch(sport);
205762306a36Sopenharmony_ci}
205862306a36Sopenharmony_ci
205962306a36Sopenharmony_cistatic void srpt_drop_sport_ref(struct srpt_port *sport)
206062306a36Sopenharmony_ci{
206162306a36Sopenharmony_ci	if (atomic_dec_return(&sport->refcount) == 0 && sport->freed_channels)
206262306a36Sopenharmony_ci		complete(sport->freed_channels);
206362306a36Sopenharmony_ci}
206462306a36Sopenharmony_ci
206562306a36Sopenharmony_cistatic void srpt_free_ch(struct kref *kref)
206662306a36Sopenharmony_ci{
206762306a36Sopenharmony_ci	struct srpt_rdma_ch *ch = container_of(kref, struct srpt_rdma_ch, kref);
206862306a36Sopenharmony_ci
206962306a36Sopenharmony_ci	srpt_drop_sport_ref(ch->sport);
207062306a36Sopenharmony_ci	kfree_rcu(ch, rcu);
207162306a36Sopenharmony_ci}
207262306a36Sopenharmony_ci
207362306a36Sopenharmony_ci/*
207462306a36Sopenharmony_ci * Shut down the SCSI target session, tell the connection manager to
207562306a36Sopenharmony_ci * disconnect the associated RDMA channel, transition the QP to the error
207662306a36Sopenharmony_ci * state and remove the channel from the channel list. This function is
207762306a36Sopenharmony_ci * typically called from inside srpt_zerolength_write_done(). Concurrent
207862306a36Sopenharmony_ci * srpt_zerolength_write() calls from inside srpt_close_ch() are possible
207962306a36Sopenharmony_ci * as long as the channel is on sport->nexus_list.
208062306a36Sopenharmony_ci */
208162306a36Sopenharmony_cistatic void srpt_release_channel_work(struct work_struct *w)
208262306a36Sopenharmony_ci{
208362306a36Sopenharmony_ci	struct srpt_rdma_ch *ch;
208462306a36Sopenharmony_ci	struct srpt_device *sdev;
208562306a36Sopenharmony_ci	struct srpt_port *sport;
208662306a36Sopenharmony_ci	struct se_session *se_sess;
208762306a36Sopenharmony_ci
208862306a36Sopenharmony_ci	ch = container_of(w, struct srpt_rdma_ch, release_work);
208962306a36Sopenharmony_ci	pr_debug("%s-%d\n", ch->sess_name, ch->qp->qp_num);
209062306a36Sopenharmony_ci
209162306a36Sopenharmony_ci	sdev = ch->sport->sdev;
209262306a36Sopenharmony_ci	BUG_ON(!sdev);
209362306a36Sopenharmony_ci
209462306a36Sopenharmony_ci	se_sess = ch->sess;
209562306a36Sopenharmony_ci	BUG_ON(!se_sess);
209662306a36Sopenharmony_ci
209762306a36Sopenharmony_ci	target_stop_session(se_sess);
209862306a36Sopenharmony_ci	target_wait_for_sess_cmds(se_sess);
209962306a36Sopenharmony_ci
210062306a36Sopenharmony_ci	target_remove_session(se_sess);
210162306a36Sopenharmony_ci	ch->sess = NULL;
210262306a36Sopenharmony_ci
210362306a36Sopenharmony_ci	if (ch->using_rdma_cm)
210462306a36Sopenharmony_ci		rdma_destroy_id(ch->rdma_cm.cm_id);
210562306a36Sopenharmony_ci	else
210662306a36Sopenharmony_ci		ib_destroy_cm_id(ch->ib_cm.cm_id);
210762306a36Sopenharmony_ci
210862306a36Sopenharmony_ci	sport = ch->sport;
210962306a36Sopenharmony_ci	mutex_lock(&sport->mutex);
211062306a36Sopenharmony_ci	list_del_rcu(&ch->list);
211162306a36Sopenharmony_ci	mutex_unlock(&sport->mutex);
211262306a36Sopenharmony_ci
211362306a36Sopenharmony_ci	if (ch->closed)
211462306a36Sopenharmony_ci		complete(ch->closed);
211562306a36Sopenharmony_ci
211662306a36Sopenharmony_ci	srpt_destroy_ch_ib(ch);
211762306a36Sopenharmony_ci
211862306a36Sopenharmony_ci	srpt_free_ioctx_ring((struct srpt_ioctx **)ch->ioctx_ring,
211962306a36Sopenharmony_ci			     ch->sport->sdev, ch->rq_size,
212062306a36Sopenharmony_ci			     ch->rsp_buf_cache, DMA_TO_DEVICE);
212162306a36Sopenharmony_ci
212262306a36Sopenharmony_ci	kmem_cache_destroy(ch->rsp_buf_cache);
212362306a36Sopenharmony_ci
212462306a36Sopenharmony_ci	srpt_free_ioctx_ring((struct srpt_ioctx **)ch->ioctx_recv_ring,
212562306a36Sopenharmony_ci			     sdev, ch->rq_size,
212662306a36Sopenharmony_ci			     ch->req_buf_cache, DMA_FROM_DEVICE);
212762306a36Sopenharmony_ci
212862306a36Sopenharmony_ci	kmem_cache_destroy(ch->req_buf_cache);
212962306a36Sopenharmony_ci
213062306a36Sopenharmony_ci	kref_put(&ch->kref, srpt_free_ch);
213162306a36Sopenharmony_ci}
213262306a36Sopenharmony_ci
213362306a36Sopenharmony_ci/**
213462306a36Sopenharmony_ci * srpt_cm_req_recv - process the event IB_CM_REQ_RECEIVED
213562306a36Sopenharmony_ci * @sdev: HCA through which the login request was received.
213662306a36Sopenharmony_ci * @ib_cm_id: IB/CM connection identifier in case of IB/CM.
213762306a36Sopenharmony_ci * @rdma_cm_id: RDMA/CM connection identifier in case of RDMA/CM.
213862306a36Sopenharmony_ci * @port_num: Port through which the REQ message was received.
213962306a36Sopenharmony_ci * @pkey: P_Key of the incoming connection.
214062306a36Sopenharmony_ci * @req: SRP login request.
214162306a36Sopenharmony_ci * @src_addr: GID (IB/CM) or IP address (RDMA/CM) of the port that submitted
214262306a36Sopenharmony_ci * the login request.
214362306a36Sopenharmony_ci *
214462306a36Sopenharmony_ci * Ownership of the cm_id is transferred to the target session if this
214562306a36Sopenharmony_ci * function returns zero. Otherwise the caller remains the owner of cm_id.
214662306a36Sopenharmony_ci */
214762306a36Sopenharmony_cistatic int srpt_cm_req_recv(struct srpt_device *const sdev,
214862306a36Sopenharmony_ci			    struct ib_cm_id *ib_cm_id,
214962306a36Sopenharmony_ci			    struct rdma_cm_id *rdma_cm_id,
215062306a36Sopenharmony_ci			    u8 port_num, __be16 pkey,
215162306a36Sopenharmony_ci			    const struct srp_login_req *req,
215262306a36Sopenharmony_ci			    const char *src_addr)
215362306a36Sopenharmony_ci{
215462306a36Sopenharmony_ci	struct srpt_port *sport = &sdev->port[port_num - 1];
215562306a36Sopenharmony_ci	struct srpt_nexus *nexus;
215662306a36Sopenharmony_ci	struct srp_login_rsp *rsp = NULL;
215762306a36Sopenharmony_ci	struct srp_login_rej *rej = NULL;
215862306a36Sopenharmony_ci	union {
215962306a36Sopenharmony_ci		struct rdma_conn_param rdma_cm;
216062306a36Sopenharmony_ci		struct ib_cm_rep_param ib_cm;
216162306a36Sopenharmony_ci	} *rep_param = NULL;
216262306a36Sopenharmony_ci	struct srpt_rdma_ch *ch = NULL;
216362306a36Sopenharmony_ci	char i_port_id[36];
216462306a36Sopenharmony_ci	u32 it_iu_len;
216562306a36Sopenharmony_ci	int i, tag_num, tag_size, ret;
216662306a36Sopenharmony_ci	struct srpt_tpg *stpg;
216762306a36Sopenharmony_ci
216862306a36Sopenharmony_ci	WARN_ON_ONCE(irqs_disabled());
216962306a36Sopenharmony_ci
217062306a36Sopenharmony_ci	it_iu_len = be32_to_cpu(req->req_it_iu_len);
217162306a36Sopenharmony_ci
217262306a36Sopenharmony_ci	pr_info("Received SRP_LOGIN_REQ with i_port_id %pI6, t_port_id %pI6 and it_iu_len %d on port %d (guid=%pI6); pkey %#04x\n",
217362306a36Sopenharmony_ci		req->initiator_port_id, req->target_port_id, it_iu_len,
217462306a36Sopenharmony_ci		port_num, &sport->gid, be16_to_cpu(pkey));
217562306a36Sopenharmony_ci
217662306a36Sopenharmony_ci	nexus = srpt_get_nexus(sport, req->initiator_port_id,
217762306a36Sopenharmony_ci			       req->target_port_id);
217862306a36Sopenharmony_ci	if (IS_ERR(nexus)) {
217962306a36Sopenharmony_ci		ret = PTR_ERR(nexus);
218062306a36Sopenharmony_ci		goto out;
218162306a36Sopenharmony_ci	}
218262306a36Sopenharmony_ci
218362306a36Sopenharmony_ci	ret = -ENOMEM;
218462306a36Sopenharmony_ci	rsp = kzalloc(sizeof(*rsp), GFP_KERNEL);
218562306a36Sopenharmony_ci	rej = kzalloc(sizeof(*rej), GFP_KERNEL);
218662306a36Sopenharmony_ci	rep_param = kzalloc(sizeof(*rep_param), GFP_KERNEL);
218762306a36Sopenharmony_ci	if (!rsp || !rej || !rep_param)
218862306a36Sopenharmony_ci		goto out;
218962306a36Sopenharmony_ci
219062306a36Sopenharmony_ci	ret = -EINVAL;
219162306a36Sopenharmony_ci	if (it_iu_len > srp_max_req_size || it_iu_len < 64) {
219262306a36Sopenharmony_ci		rej->reason = cpu_to_be32(
219362306a36Sopenharmony_ci				SRP_LOGIN_REJ_REQ_IT_IU_LENGTH_TOO_LARGE);
219462306a36Sopenharmony_ci		pr_err("rejected SRP_LOGIN_REQ because its length (%d bytes) is out of range (%d .. %d)\n",
219562306a36Sopenharmony_ci		       it_iu_len, 64, srp_max_req_size);
219662306a36Sopenharmony_ci		goto reject;
219762306a36Sopenharmony_ci	}
219862306a36Sopenharmony_ci
219962306a36Sopenharmony_ci	if (!sport->enabled) {
220062306a36Sopenharmony_ci		rej->reason = cpu_to_be32(SRP_LOGIN_REJ_INSUFFICIENT_RESOURCES);
220162306a36Sopenharmony_ci		pr_info("rejected SRP_LOGIN_REQ because target port %s_%d has not yet been enabled\n",
220262306a36Sopenharmony_ci			dev_name(&sport->sdev->device->dev), port_num);
220362306a36Sopenharmony_ci		goto reject;
220462306a36Sopenharmony_ci	}
220562306a36Sopenharmony_ci
220662306a36Sopenharmony_ci	if (*(__be64 *)req->target_port_id != cpu_to_be64(srpt_service_guid)
220762306a36Sopenharmony_ci	    || *(__be64 *)(req->target_port_id + 8) !=
220862306a36Sopenharmony_ci	       cpu_to_be64(srpt_service_guid)) {
220962306a36Sopenharmony_ci		rej->reason = cpu_to_be32(
221062306a36Sopenharmony_ci				SRP_LOGIN_REJ_UNABLE_ASSOCIATE_CHANNEL);
221162306a36Sopenharmony_ci		pr_err("rejected SRP_LOGIN_REQ because it has an invalid target port identifier.\n");
221262306a36Sopenharmony_ci		goto reject;
221362306a36Sopenharmony_ci	}
221462306a36Sopenharmony_ci
221562306a36Sopenharmony_ci	ret = -ENOMEM;
221662306a36Sopenharmony_ci	ch = kzalloc(sizeof(*ch), GFP_KERNEL);
221762306a36Sopenharmony_ci	if (!ch) {
221862306a36Sopenharmony_ci		rej->reason = cpu_to_be32(SRP_LOGIN_REJ_INSUFFICIENT_RESOURCES);
221962306a36Sopenharmony_ci		pr_err("rejected SRP_LOGIN_REQ because out of memory.\n");
222062306a36Sopenharmony_ci		goto reject;
222162306a36Sopenharmony_ci	}
222262306a36Sopenharmony_ci
222362306a36Sopenharmony_ci	kref_init(&ch->kref);
222462306a36Sopenharmony_ci	ch->pkey = be16_to_cpu(pkey);
222562306a36Sopenharmony_ci	ch->nexus = nexus;
222662306a36Sopenharmony_ci	ch->zw_cqe.done = srpt_zerolength_write_done;
222762306a36Sopenharmony_ci	INIT_WORK(&ch->release_work, srpt_release_channel_work);
222862306a36Sopenharmony_ci	ch->sport = sport;
222962306a36Sopenharmony_ci	if (rdma_cm_id) {
223062306a36Sopenharmony_ci		ch->using_rdma_cm = true;
223162306a36Sopenharmony_ci		ch->rdma_cm.cm_id = rdma_cm_id;
223262306a36Sopenharmony_ci		rdma_cm_id->context = ch;
223362306a36Sopenharmony_ci	} else {
223462306a36Sopenharmony_ci		ch->ib_cm.cm_id = ib_cm_id;
223562306a36Sopenharmony_ci		ib_cm_id->context = ch;
223662306a36Sopenharmony_ci	}
223762306a36Sopenharmony_ci	/*
223862306a36Sopenharmony_ci	 * ch->rq_size should be at least as large as the initiator queue
223962306a36Sopenharmony_ci	 * depth to avoid that the initiator driver has to report QUEUE_FULL
224062306a36Sopenharmony_ci	 * to the SCSI mid-layer.
224162306a36Sopenharmony_ci	 */
224262306a36Sopenharmony_ci	ch->rq_size = min(MAX_SRPT_RQ_SIZE, sdev->device->attrs.max_qp_wr);
224362306a36Sopenharmony_ci	spin_lock_init(&ch->spinlock);
224462306a36Sopenharmony_ci	ch->state = CH_CONNECTING;
224562306a36Sopenharmony_ci	INIT_LIST_HEAD(&ch->cmd_wait_list);
224662306a36Sopenharmony_ci	ch->max_rsp_size = ch->sport->port_attrib.srp_max_rsp_size;
224762306a36Sopenharmony_ci
224862306a36Sopenharmony_ci	ch->rsp_buf_cache = kmem_cache_create("srpt-rsp-buf", ch->max_rsp_size,
224962306a36Sopenharmony_ci					      512, 0, NULL);
225062306a36Sopenharmony_ci	if (!ch->rsp_buf_cache)
225162306a36Sopenharmony_ci		goto free_ch;
225262306a36Sopenharmony_ci
225362306a36Sopenharmony_ci	ch->ioctx_ring = (struct srpt_send_ioctx **)
225462306a36Sopenharmony_ci		srpt_alloc_ioctx_ring(ch->sport->sdev, ch->rq_size,
225562306a36Sopenharmony_ci				      sizeof(*ch->ioctx_ring[0]),
225662306a36Sopenharmony_ci				      ch->rsp_buf_cache, 0, DMA_TO_DEVICE);
225762306a36Sopenharmony_ci	if (!ch->ioctx_ring) {
225862306a36Sopenharmony_ci		pr_err("rejected SRP_LOGIN_REQ because creating a new QP SQ ring failed.\n");
225962306a36Sopenharmony_ci		rej->reason = cpu_to_be32(SRP_LOGIN_REJ_INSUFFICIENT_RESOURCES);
226062306a36Sopenharmony_ci		goto free_rsp_cache;
226162306a36Sopenharmony_ci	}
226262306a36Sopenharmony_ci
226362306a36Sopenharmony_ci	for (i = 0; i < ch->rq_size; i++)
226462306a36Sopenharmony_ci		ch->ioctx_ring[i]->ch = ch;
226562306a36Sopenharmony_ci	if (!sdev->use_srq) {
226662306a36Sopenharmony_ci		u16 imm_data_offset = req->req_flags & SRP_IMMED_REQUESTED ?
226762306a36Sopenharmony_ci			be16_to_cpu(req->imm_data_offset) : 0;
226862306a36Sopenharmony_ci		u16 alignment_offset;
226962306a36Sopenharmony_ci		u32 req_sz;
227062306a36Sopenharmony_ci
227162306a36Sopenharmony_ci		if (req->req_flags & SRP_IMMED_REQUESTED)
227262306a36Sopenharmony_ci			pr_debug("imm_data_offset = %d\n",
227362306a36Sopenharmony_ci				 be16_to_cpu(req->imm_data_offset));
227462306a36Sopenharmony_ci		if (imm_data_offset >= sizeof(struct srp_cmd)) {
227562306a36Sopenharmony_ci			ch->imm_data_offset = imm_data_offset;
227662306a36Sopenharmony_ci			rsp->rsp_flags |= SRP_LOGIN_RSP_IMMED_SUPP;
227762306a36Sopenharmony_ci		} else {
227862306a36Sopenharmony_ci			ch->imm_data_offset = 0;
227962306a36Sopenharmony_ci		}
228062306a36Sopenharmony_ci		alignment_offset = round_up(imm_data_offset, 512) -
228162306a36Sopenharmony_ci			imm_data_offset;
228262306a36Sopenharmony_ci		req_sz = alignment_offset + imm_data_offset + srp_max_req_size;
228362306a36Sopenharmony_ci		ch->req_buf_cache = kmem_cache_create("srpt-req-buf", req_sz,
228462306a36Sopenharmony_ci						      512, 0, NULL);
228562306a36Sopenharmony_ci		if (!ch->req_buf_cache)
228662306a36Sopenharmony_ci			goto free_rsp_ring;
228762306a36Sopenharmony_ci
228862306a36Sopenharmony_ci		ch->ioctx_recv_ring = (struct srpt_recv_ioctx **)
228962306a36Sopenharmony_ci			srpt_alloc_ioctx_ring(ch->sport->sdev, ch->rq_size,
229062306a36Sopenharmony_ci					      sizeof(*ch->ioctx_recv_ring[0]),
229162306a36Sopenharmony_ci					      ch->req_buf_cache,
229262306a36Sopenharmony_ci					      alignment_offset,
229362306a36Sopenharmony_ci					      DMA_FROM_DEVICE);
229462306a36Sopenharmony_ci		if (!ch->ioctx_recv_ring) {
229562306a36Sopenharmony_ci			pr_err("rejected SRP_LOGIN_REQ because creating a new QP RQ ring failed.\n");
229662306a36Sopenharmony_ci			rej->reason =
229762306a36Sopenharmony_ci			    cpu_to_be32(SRP_LOGIN_REJ_INSUFFICIENT_RESOURCES);
229862306a36Sopenharmony_ci			goto free_recv_cache;
229962306a36Sopenharmony_ci		}
230062306a36Sopenharmony_ci		for (i = 0; i < ch->rq_size; i++)
230162306a36Sopenharmony_ci			INIT_LIST_HEAD(&ch->ioctx_recv_ring[i]->wait_list);
230262306a36Sopenharmony_ci	}
230362306a36Sopenharmony_ci
230462306a36Sopenharmony_ci	ret = srpt_create_ch_ib(ch);
230562306a36Sopenharmony_ci	if (ret) {
230662306a36Sopenharmony_ci		rej->reason = cpu_to_be32(SRP_LOGIN_REJ_INSUFFICIENT_RESOURCES);
230762306a36Sopenharmony_ci		pr_err("rejected SRP_LOGIN_REQ because creating a new RDMA channel failed.\n");
230862306a36Sopenharmony_ci		goto free_recv_ring;
230962306a36Sopenharmony_ci	}
231062306a36Sopenharmony_ci
231162306a36Sopenharmony_ci	strscpy(ch->sess_name, src_addr, sizeof(ch->sess_name));
231262306a36Sopenharmony_ci	snprintf(i_port_id, sizeof(i_port_id), "0x%016llx%016llx",
231362306a36Sopenharmony_ci			be64_to_cpu(*(__be64 *)nexus->i_port_id),
231462306a36Sopenharmony_ci			be64_to_cpu(*(__be64 *)(nexus->i_port_id + 8)));
231562306a36Sopenharmony_ci
231662306a36Sopenharmony_ci	pr_debug("registering src addr %s or i_port_id %s\n", ch->sess_name,
231762306a36Sopenharmony_ci		 i_port_id);
231862306a36Sopenharmony_ci
231962306a36Sopenharmony_ci	tag_num = ch->rq_size;
232062306a36Sopenharmony_ci	tag_size = 1; /* ib_srpt does not use se_sess->sess_cmd_map */
232162306a36Sopenharmony_ci
232262306a36Sopenharmony_ci	if (sport->guid_id) {
232362306a36Sopenharmony_ci		mutex_lock(&sport->guid_id->mutex);
232462306a36Sopenharmony_ci		list_for_each_entry(stpg, &sport->guid_id->tpg_list, entry) {
232562306a36Sopenharmony_ci			if (!IS_ERR_OR_NULL(ch->sess))
232662306a36Sopenharmony_ci				break;
232762306a36Sopenharmony_ci			ch->sess = target_setup_session(&stpg->tpg, tag_num,
232862306a36Sopenharmony_ci						tag_size, TARGET_PROT_NORMAL,
232962306a36Sopenharmony_ci						ch->sess_name, ch, NULL);
233062306a36Sopenharmony_ci		}
233162306a36Sopenharmony_ci		mutex_unlock(&sport->guid_id->mutex);
233262306a36Sopenharmony_ci	}
233362306a36Sopenharmony_ci
233462306a36Sopenharmony_ci	if (sport->gid_id) {
233562306a36Sopenharmony_ci		mutex_lock(&sport->gid_id->mutex);
233662306a36Sopenharmony_ci		list_for_each_entry(stpg, &sport->gid_id->tpg_list, entry) {
233762306a36Sopenharmony_ci			if (!IS_ERR_OR_NULL(ch->sess))
233862306a36Sopenharmony_ci				break;
233962306a36Sopenharmony_ci			ch->sess = target_setup_session(&stpg->tpg, tag_num,
234062306a36Sopenharmony_ci					tag_size, TARGET_PROT_NORMAL, i_port_id,
234162306a36Sopenharmony_ci					ch, NULL);
234262306a36Sopenharmony_ci			if (!IS_ERR_OR_NULL(ch->sess))
234362306a36Sopenharmony_ci				break;
234462306a36Sopenharmony_ci			/* Retry without leading "0x" */
234562306a36Sopenharmony_ci			ch->sess = target_setup_session(&stpg->tpg, tag_num,
234662306a36Sopenharmony_ci						tag_size, TARGET_PROT_NORMAL,
234762306a36Sopenharmony_ci						i_port_id + 2, ch, NULL);
234862306a36Sopenharmony_ci		}
234962306a36Sopenharmony_ci		mutex_unlock(&sport->gid_id->mutex);
235062306a36Sopenharmony_ci	}
235162306a36Sopenharmony_ci
235262306a36Sopenharmony_ci	if (IS_ERR_OR_NULL(ch->sess)) {
235362306a36Sopenharmony_ci		WARN_ON_ONCE(ch->sess == NULL);
235462306a36Sopenharmony_ci		ret = PTR_ERR(ch->sess);
235562306a36Sopenharmony_ci		ch->sess = NULL;
235662306a36Sopenharmony_ci		pr_info("Rejected login for initiator %s: ret = %d.\n",
235762306a36Sopenharmony_ci			ch->sess_name, ret);
235862306a36Sopenharmony_ci		rej->reason = cpu_to_be32(ret == -ENOMEM ?
235962306a36Sopenharmony_ci				SRP_LOGIN_REJ_INSUFFICIENT_RESOURCES :
236062306a36Sopenharmony_ci				SRP_LOGIN_REJ_CHANNEL_LIMIT_REACHED);
236162306a36Sopenharmony_ci		goto destroy_ib;
236262306a36Sopenharmony_ci	}
236362306a36Sopenharmony_ci
236462306a36Sopenharmony_ci	/*
236562306a36Sopenharmony_ci	 * Once a session has been created destruction of srpt_rdma_ch objects
236662306a36Sopenharmony_ci	 * will decrement sport->refcount. Hence increment sport->refcount now.
236762306a36Sopenharmony_ci	 */
236862306a36Sopenharmony_ci	atomic_inc(&sport->refcount);
236962306a36Sopenharmony_ci
237062306a36Sopenharmony_ci	mutex_lock(&sport->mutex);
237162306a36Sopenharmony_ci
237262306a36Sopenharmony_ci	if ((req->req_flags & SRP_MTCH_ACTION) == SRP_MULTICHAN_SINGLE) {
237362306a36Sopenharmony_ci		struct srpt_rdma_ch *ch2;
237462306a36Sopenharmony_ci
237562306a36Sopenharmony_ci		list_for_each_entry(ch2, &nexus->ch_list, list) {
237662306a36Sopenharmony_ci			if (srpt_disconnect_ch(ch2) < 0)
237762306a36Sopenharmony_ci				continue;
237862306a36Sopenharmony_ci			pr_info("Relogin - closed existing channel %s\n",
237962306a36Sopenharmony_ci				ch2->sess_name);
238062306a36Sopenharmony_ci			rsp->rsp_flags |= SRP_LOGIN_RSP_MULTICHAN_TERMINATED;
238162306a36Sopenharmony_ci		}
238262306a36Sopenharmony_ci	} else {
238362306a36Sopenharmony_ci		rsp->rsp_flags |= SRP_LOGIN_RSP_MULTICHAN_MAINTAINED;
238462306a36Sopenharmony_ci	}
238562306a36Sopenharmony_ci
238662306a36Sopenharmony_ci	list_add_tail_rcu(&ch->list, &nexus->ch_list);
238762306a36Sopenharmony_ci
238862306a36Sopenharmony_ci	if (!sport->enabled) {
238962306a36Sopenharmony_ci		rej->reason = cpu_to_be32(
239062306a36Sopenharmony_ci				SRP_LOGIN_REJ_INSUFFICIENT_RESOURCES);
239162306a36Sopenharmony_ci		pr_info("rejected SRP_LOGIN_REQ because target %s_%d is not enabled\n",
239262306a36Sopenharmony_ci			dev_name(&sdev->device->dev), port_num);
239362306a36Sopenharmony_ci		mutex_unlock(&sport->mutex);
239462306a36Sopenharmony_ci		ret = -EINVAL;
239562306a36Sopenharmony_ci		goto reject;
239662306a36Sopenharmony_ci	}
239762306a36Sopenharmony_ci
239862306a36Sopenharmony_ci	mutex_unlock(&sport->mutex);
239962306a36Sopenharmony_ci
240062306a36Sopenharmony_ci	ret = ch->using_rdma_cm ? 0 : srpt_ch_qp_rtr(ch, ch->qp);
240162306a36Sopenharmony_ci	if (ret) {
240262306a36Sopenharmony_ci		rej->reason = cpu_to_be32(SRP_LOGIN_REJ_INSUFFICIENT_RESOURCES);
240362306a36Sopenharmony_ci		pr_err("rejected SRP_LOGIN_REQ because enabling RTR failed (error code = %d)\n",
240462306a36Sopenharmony_ci		       ret);
240562306a36Sopenharmony_ci		goto reject;
240662306a36Sopenharmony_ci	}
240762306a36Sopenharmony_ci
240862306a36Sopenharmony_ci	pr_debug("Establish connection sess=%p name=%s ch=%p\n", ch->sess,
240962306a36Sopenharmony_ci		 ch->sess_name, ch);
241062306a36Sopenharmony_ci
241162306a36Sopenharmony_ci	/* create srp_login_response */
241262306a36Sopenharmony_ci	rsp->opcode = SRP_LOGIN_RSP;
241362306a36Sopenharmony_ci	rsp->tag = req->tag;
241462306a36Sopenharmony_ci	rsp->max_it_iu_len = cpu_to_be32(srp_max_req_size);
241562306a36Sopenharmony_ci	rsp->max_ti_iu_len = req->req_it_iu_len;
241662306a36Sopenharmony_ci	ch->max_ti_iu_len = it_iu_len;
241762306a36Sopenharmony_ci	rsp->buf_fmt = cpu_to_be16(SRP_BUF_FORMAT_DIRECT |
241862306a36Sopenharmony_ci				   SRP_BUF_FORMAT_INDIRECT);
241962306a36Sopenharmony_ci	rsp->req_lim_delta = cpu_to_be32(ch->rq_size);
242062306a36Sopenharmony_ci	atomic_set(&ch->req_lim, ch->rq_size);
242162306a36Sopenharmony_ci	atomic_set(&ch->req_lim_delta, 0);
242262306a36Sopenharmony_ci
242362306a36Sopenharmony_ci	/* create cm reply */
242462306a36Sopenharmony_ci	if (ch->using_rdma_cm) {
242562306a36Sopenharmony_ci		rep_param->rdma_cm.private_data = (void *)rsp;
242662306a36Sopenharmony_ci		rep_param->rdma_cm.private_data_len = sizeof(*rsp);
242762306a36Sopenharmony_ci		rep_param->rdma_cm.rnr_retry_count = 7;
242862306a36Sopenharmony_ci		rep_param->rdma_cm.flow_control = 1;
242962306a36Sopenharmony_ci		rep_param->rdma_cm.responder_resources = 4;
243062306a36Sopenharmony_ci		rep_param->rdma_cm.initiator_depth = 4;
243162306a36Sopenharmony_ci	} else {
243262306a36Sopenharmony_ci		rep_param->ib_cm.qp_num = ch->qp->qp_num;
243362306a36Sopenharmony_ci		rep_param->ib_cm.private_data = (void *)rsp;
243462306a36Sopenharmony_ci		rep_param->ib_cm.private_data_len = sizeof(*rsp);
243562306a36Sopenharmony_ci		rep_param->ib_cm.rnr_retry_count = 7;
243662306a36Sopenharmony_ci		rep_param->ib_cm.flow_control = 1;
243762306a36Sopenharmony_ci		rep_param->ib_cm.failover_accepted = 0;
243862306a36Sopenharmony_ci		rep_param->ib_cm.srq = 1;
243962306a36Sopenharmony_ci		rep_param->ib_cm.responder_resources = 4;
244062306a36Sopenharmony_ci		rep_param->ib_cm.initiator_depth = 4;
244162306a36Sopenharmony_ci	}
244262306a36Sopenharmony_ci
244362306a36Sopenharmony_ci	/*
244462306a36Sopenharmony_ci	 * Hold the sport mutex while accepting a connection to avoid that
244562306a36Sopenharmony_ci	 * srpt_disconnect_ch() is invoked concurrently with this code.
244662306a36Sopenharmony_ci	 */
244762306a36Sopenharmony_ci	mutex_lock(&sport->mutex);
244862306a36Sopenharmony_ci	if (sport->enabled && ch->state == CH_CONNECTING) {
244962306a36Sopenharmony_ci		if (ch->using_rdma_cm)
245062306a36Sopenharmony_ci			ret = rdma_accept(rdma_cm_id, &rep_param->rdma_cm);
245162306a36Sopenharmony_ci		else
245262306a36Sopenharmony_ci			ret = ib_send_cm_rep(ib_cm_id, &rep_param->ib_cm);
245362306a36Sopenharmony_ci	} else {
245462306a36Sopenharmony_ci		ret = -EINVAL;
245562306a36Sopenharmony_ci	}
245662306a36Sopenharmony_ci	mutex_unlock(&sport->mutex);
245762306a36Sopenharmony_ci
245862306a36Sopenharmony_ci	switch (ret) {
245962306a36Sopenharmony_ci	case 0:
246062306a36Sopenharmony_ci		break;
246162306a36Sopenharmony_ci	case -EINVAL:
246262306a36Sopenharmony_ci		goto reject;
246362306a36Sopenharmony_ci	default:
246462306a36Sopenharmony_ci		rej->reason = cpu_to_be32(SRP_LOGIN_REJ_INSUFFICIENT_RESOURCES);
246562306a36Sopenharmony_ci		pr_err("sending SRP_LOGIN_REQ response failed (error code = %d)\n",
246662306a36Sopenharmony_ci		       ret);
246762306a36Sopenharmony_ci		goto reject;
246862306a36Sopenharmony_ci	}
246962306a36Sopenharmony_ci
247062306a36Sopenharmony_ci	goto out;
247162306a36Sopenharmony_ci
247262306a36Sopenharmony_cidestroy_ib:
247362306a36Sopenharmony_ci	srpt_destroy_ch_ib(ch);
247462306a36Sopenharmony_ci
247562306a36Sopenharmony_cifree_recv_ring:
247662306a36Sopenharmony_ci	srpt_free_ioctx_ring((struct srpt_ioctx **)ch->ioctx_recv_ring,
247762306a36Sopenharmony_ci			     ch->sport->sdev, ch->rq_size,
247862306a36Sopenharmony_ci			     ch->req_buf_cache, DMA_FROM_DEVICE);
247962306a36Sopenharmony_ci
248062306a36Sopenharmony_cifree_recv_cache:
248162306a36Sopenharmony_ci	kmem_cache_destroy(ch->req_buf_cache);
248262306a36Sopenharmony_ci
248362306a36Sopenharmony_cifree_rsp_ring:
248462306a36Sopenharmony_ci	srpt_free_ioctx_ring((struct srpt_ioctx **)ch->ioctx_ring,
248562306a36Sopenharmony_ci			     ch->sport->sdev, ch->rq_size,
248662306a36Sopenharmony_ci			     ch->rsp_buf_cache, DMA_TO_DEVICE);
248762306a36Sopenharmony_ci
248862306a36Sopenharmony_cifree_rsp_cache:
248962306a36Sopenharmony_ci	kmem_cache_destroy(ch->rsp_buf_cache);
249062306a36Sopenharmony_ci
249162306a36Sopenharmony_cifree_ch:
249262306a36Sopenharmony_ci	if (rdma_cm_id)
249362306a36Sopenharmony_ci		rdma_cm_id->context = NULL;
249462306a36Sopenharmony_ci	else
249562306a36Sopenharmony_ci		ib_cm_id->context = NULL;
249662306a36Sopenharmony_ci	kfree(ch);
249762306a36Sopenharmony_ci	ch = NULL;
249862306a36Sopenharmony_ci
249962306a36Sopenharmony_ci	WARN_ON_ONCE(ret == 0);
250062306a36Sopenharmony_ci
250162306a36Sopenharmony_cireject:
250262306a36Sopenharmony_ci	pr_info("Rejecting login with reason %#x\n", be32_to_cpu(rej->reason));
250362306a36Sopenharmony_ci	rej->opcode = SRP_LOGIN_REJ;
250462306a36Sopenharmony_ci	rej->tag = req->tag;
250562306a36Sopenharmony_ci	rej->buf_fmt = cpu_to_be16(SRP_BUF_FORMAT_DIRECT |
250662306a36Sopenharmony_ci				   SRP_BUF_FORMAT_INDIRECT);
250762306a36Sopenharmony_ci
250862306a36Sopenharmony_ci	if (rdma_cm_id)
250962306a36Sopenharmony_ci		rdma_reject(rdma_cm_id, rej, sizeof(*rej),
251062306a36Sopenharmony_ci			    IB_CM_REJ_CONSUMER_DEFINED);
251162306a36Sopenharmony_ci	else
251262306a36Sopenharmony_ci		ib_send_cm_rej(ib_cm_id, IB_CM_REJ_CONSUMER_DEFINED, NULL, 0,
251362306a36Sopenharmony_ci			       rej, sizeof(*rej));
251462306a36Sopenharmony_ci
251562306a36Sopenharmony_ci	if (ch && ch->sess) {
251662306a36Sopenharmony_ci		srpt_close_ch(ch);
251762306a36Sopenharmony_ci		/*
251862306a36Sopenharmony_ci		 * Tell the caller not to free cm_id since
251962306a36Sopenharmony_ci		 * srpt_release_channel_work() will do that.
252062306a36Sopenharmony_ci		 */
252162306a36Sopenharmony_ci		ret = 0;
252262306a36Sopenharmony_ci	}
252362306a36Sopenharmony_ci
252462306a36Sopenharmony_ciout:
252562306a36Sopenharmony_ci	kfree(rep_param);
252662306a36Sopenharmony_ci	kfree(rsp);
252762306a36Sopenharmony_ci	kfree(rej);
252862306a36Sopenharmony_ci
252962306a36Sopenharmony_ci	return ret;
253062306a36Sopenharmony_ci}
253162306a36Sopenharmony_ci
253262306a36Sopenharmony_cistatic int srpt_ib_cm_req_recv(struct ib_cm_id *cm_id,
253362306a36Sopenharmony_ci			       const struct ib_cm_req_event_param *param,
253462306a36Sopenharmony_ci			       void *private_data)
253562306a36Sopenharmony_ci{
253662306a36Sopenharmony_ci	char sguid[40];
253762306a36Sopenharmony_ci
253862306a36Sopenharmony_ci	srpt_format_guid(sguid, sizeof(sguid),
253962306a36Sopenharmony_ci			 &param->primary_path->dgid.global.interface_id);
254062306a36Sopenharmony_ci
254162306a36Sopenharmony_ci	return srpt_cm_req_recv(cm_id->context, cm_id, NULL, param->port,
254262306a36Sopenharmony_ci				param->primary_path->pkey,
254362306a36Sopenharmony_ci				private_data, sguid);
254462306a36Sopenharmony_ci}
254562306a36Sopenharmony_ci
254662306a36Sopenharmony_cistatic int srpt_rdma_cm_req_recv(struct rdma_cm_id *cm_id,
254762306a36Sopenharmony_ci				 struct rdma_cm_event *event)
254862306a36Sopenharmony_ci{
254962306a36Sopenharmony_ci	struct srpt_device *sdev;
255062306a36Sopenharmony_ci	struct srp_login_req req;
255162306a36Sopenharmony_ci	const struct srp_login_req_rdma *req_rdma;
255262306a36Sopenharmony_ci	struct sa_path_rec *path_rec = cm_id->route.path_rec;
255362306a36Sopenharmony_ci	char src_addr[40];
255462306a36Sopenharmony_ci
255562306a36Sopenharmony_ci	sdev = ib_get_client_data(cm_id->device, &srpt_client);
255662306a36Sopenharmony_ci	if (!sdev)
255762306a36Sopenharmony_ci		return -ECONNREFUSED;
255862306a36Sopenharmony_ci
255962306a36Sopenharmony_ci	if (event->param.conn.private_data_len < sizeof(*req_rdma))
256062306a36Sopenharmony_ci		return -EINVAL;
256162306a36Sopenharmony_ci
256262306a36Sopenharmony_ci	/* Transform srp_login_req_rdma into srp_login_req. */
256362306a36Sopenharmony_ci	req_rdma = event->param.conn.private_data;
256462306a36Sopenharmony_ci	memset(&req, 0, sizeof(req));
256562306a36Sopenharmony_ci	req.opcode		= req_rdma->opcode;
256662306a36Sopenharmony_ci	req.tag			= req_rdma->tag;
256762306a36Sopenharmony_ci	req.req_it_iu_len	= req_rdma->req_it_iu_len;
256862306a36Sopenharmony_ci	req.req_buf_fmt		= req_rdma->req_buf_fmt;
256962306a36Sopenharmony_ci	req.req_flags		= req_rdma->req_flags;
257062306a36Sopenharmony_ci	memcpy(req.initiator_port_id, req_rdma->initiator_port_id, 16);
257162306a36Sopenharmony_ci	memcpy(req.target_port_id, req_rdma->target_port_id, 16);
257262306a36Sopenharmony_ci	req.imm_data_offset	= req_rdma->imm_data_offset;
257362306a36Sopenharmony_ci
257462306a36Sopenharmony_ci	snprintf(src_addr, sizeof(src_addr), "%pIS",
257562306a36Sopenharmony_ci		 &cm_id->route.addr.src_addr);
257662306a36Sopenharmony_ci
257762306a36Sopenharmony_ci	return srpt_cm_req_recv(sdev, NULL, cm_id, cm_id->port_num,
257862306a36Sopenharmony_ci				path_rec ? path_rec->pkey : 0, &req, src_addr);
257962306a36Sopenharmony_ci}
258062306a36Sopenharmony_ci
258162306a36Sopenharmony_cistatic void srpt_cm_rej_recv(struct srpt_rdma_ch *ch,
258262306a36Sopenharmony_ci			     enum ib_cm_rej_reason reason,
258362306a36Sopenharmony_ci			     const u8 *private_data,
258462306a36Sopenharmony_ci			     u8 private_data_len)
258562306a36Sopenharmony_ci{
258662306a36Sopenharmony_ci	char *priv = NULL;
258762306a36Sopenharmony_ci	int i;
258862306a36Sopenharmony_ci
258962306a36Sopenharmony_ci	if (private_data_len && (priv = kmalloc(private_data_len * 3 + 1,
259062306a36Sopenharmony_ci						GFP_KERNEL))) {
259162306a36Sopenharmony_ci		for (i = 0; i < private_data_len; i++)
259262306a36Sopenharmony_ci			sprintf(priv + 3 * i, " %02x", private_data[i]);
259362306a36Sopenharmony_ci	}
259462306a36Sopenharmony_ci	pr_info("Received CM REJ for ch %s-%d; reason %d%s%s.\n",
259562306a36Sopenharmony_ci		ch->sess_name, ch->qp->qp_num, reason, private_data_len ?
259662306a36Sopenharmony_ci		"; private data" : "", priv ? priv : " (?)");
259762306a36Sopenharmony_ci	kfree(priv);
259862306a36Sopenharmony_ci}
259962306a36Sopenharmony_ci
260062306a36Sopenharmony_ci/**
260162306a36Sopenharmony_ci * srpt_cm_rtu_recv - process an IB_CM_RTU_RECEIVED or USER_ESTABLISHED event
260262306a36Sopenharmony_ci * @ch: SRPT RDMA channel.
260362306a36Sopenharmony_ci *
260462306a36Sopenharmony_ci * An RTU (ready to use) message indicates that the connection has been
260562306a36Sopenharmony_ci * established and that the recipient may begin transmitting.
260662306a36Sopenharmony_ci */
260762306a36Sopenharmony_cistatic void srpt_cm_rtu_recv(struct srpt_rdma_ch *ch)
260862306a36Sopenharmony_ci{
260962306a36Sopenharmony_ci	int ret;
261062306a36Sopenharmony_ci
261162306a36Sopenharmony_ci	ret = ch->using_rdma_cm ? 0 : srpt_ch_qp_rts(ch, ch->qp);
261262306a36Sopenharmony_ci	if (ret < 0) {
261362306a36Sopenharmony_ci		pr_err("%s-%d: QP transition to RTS failed\n", ch->sess_name,
261462306a36Sopenharmony_ci		       ch->qp->qp_num);
261562306a36Sopenharmony_ci		srpt_close_ch(ch);
261662306a36Sopenharmony_ci		return;
261762306a36Sopenharmony_ci	}
261862306a36Sopenharmony_ci
261962306a36Sopenharmony_ci	/*
262062306a36Sopenharmony_ci	 * Note: calling srpt_close_ch() if the transition to the LIVE state
262162306a36Sopenharmony_ci	 * fails is not necessary since that means that that function has
262262306a36Sopenharmony_ci	 * already been invoked from another thread.
262362306a36Sopenharmony_ci	 */
262462306a36Sopenharmony_ci	if (!srpt_set_ch_state(ch, CH_LIVE)) {
262562306a36Sopenharmony_ci		pr_err("%s-%d: channel transition to LIVE state failed\n",
262662306a36Sopenharmony_ci		       ch->sess_name, ch->qp->qp_num);
262762306a36Sopenharmony_ci		return;
262862306a36Sopenharmony_ci	}
262962306a36Sopenharmony_ci
263062306a36Sopenharmony_ci	/* Trigger wait list processing. */
263162306a36Sopenharmony_ci	ret = srpt_zerolength_write(ch);
263262306a36Sopenharmony_ci	WARN_ONCE(ret < 0, "%d\n", ret);
263362306a36Sopenharmony_ci}
263462306a36Sopenharmony_ci
263562306a36Sopenharmony_ci/**
263662306a36Sopenharmony_ci * srpt_cm_handler - IB connection manager callback function
263762306a36Sopenharmony_ci * @cm_id: IB/CM connection identifier.
263862306a36Sopenharmony_ci * @event: IB/CM event.
263962306a36Sopenharmony_ci *
264062306a36Sopenharmony_ci * A non-zero return value will cause the caller destroy the CM ID.
264162306a36Sopenharmony_ci *
264262306a36Sopenharmony_ci * Note: srpt_cm_handler() must only return a non-zero value when transferring
264362306a36Sopenharmony_ci * ownership of the cm_id to a channel by srpt_cm_req_recv() failed. Returning
264462306a36Sopenharmony_ci * a non-zero value in any other case will trigger a race with the
264562306a36Sopenharmony_ci * ib_destroy_cm_id() call in srpt_release_channel().
264662306a36Sopenharmony_ci */
264762306a36Sopenharmony_cistatic int srpt_cm_handler(struct ib_cm_id *cm_id,
264862306a36Sopenharmony_ci			   const struct ib_cm_event *event)
264962306a36Sopenharmony_ci{
265062306a36Sopenharmony_ci	struct srpt_rdma_ch *ch = cm_id->context;
265162306a36Sopenharmony_ci	int ret;
265262306a36Sopenharmony_ci
265362306a36Sopenharmony_ci	ret = 0;
265462306a36Sopenharmony_ci	switch (event->event) {
265562306a36Sopenharmony_ci	case IB_CM_REQ_RECEIVED:
265662306a36Sopenharmony_ci		ret = srpt_ib_cm_req_recv(cm_id, &event->param.req_rcvd,
265762306a36Sopenharmony_ci					  event->private_data);
265862306a36Sopenharmony_ci		break;
265962306a36Sopenharmony_ci	case IB_CM_REJ_RECEIVED:
266062306a36Sopenharmony_ci		srpt_cm_rej_recv(ch, event->param.rej_rcvd.reason,
266162306a36Sopenharmony_ci				 event->private_data,
266262306a36Sopenharmony_ci				 IB_CM_REJ_PRIVATE_DATA_SIZE);
266362306a36Sopenharmony_ci		break;
266462306a36Sopenharmony_ci	case IB_CM_RTU_RECEIVED:
266562306a36Sopenharmony_ci	case IB_CM_USER_ESTABLISHED:
266662306a36Sopenharmony_ci		srpt_cm_rtu_recv(ch);
266762306a36Sopenharmony_ci		break;
266862306a36Sopenharmony_ci	case IB_CM_DREQ_RECEIVED:
266962306a36Sopenharmony_ci		srpt_disconnect_ch(ch);
267062306a36Sopenharmony_ci		break;
267162306a36Sopenharmony_ci	case IB_CM_DREP_RECEIVED:
267262306a36Sopenharmony_ci		pr_info("Received CM DREP message for ch %s-%d.\n",
267362306a36Sopenharmony_ci			ch->sess_name, ch->qp->qp_num);
267462306a36Sopenharmony_ci		srpt_close_ch(ch);
267562306a36Sopenharmony_ci		break;
267662306a36Sopenharmony_ci	case IB_CM_TIMEWAIT_EXIT:
267762306a36Sopenharmony_ci		pr_info("Received CM TimeWait exit for ch %s-%d.\n",
267862306a36Sopenharmony_ci			ch->sess_name, ch->qp->qp_num);
267962306a36Sopenharmony_ci		srpt_close_ch(ch);
268062306a36Sopenharmony_ci		break;
268162306a36Sopenharmony_ci	case IB_CM_REP_ERROR:
268262306a36Sopenharmony_ci		pr_info("Received CM REP error for ch %s-%d.\n", ch->sess_name,
268362306a36Sopenharmony_ci			ch->qp->qp_num);
268462306a36Sopenharmony_ci		break;
268562306a36Sopenharmony_ci	case IB_CM_DREQ_ERROR:
268662306a36Sopenharmony_ci		pr_info("Received CM DREQ ERROR event.\n");
268762306a36Sopenharmony_ci		break;
268862306a36Sopenharmony_ci	case IB_CM_MRA_RECEIVED:
268962306a36Sopenharmony_ci		pr_info("Received CM MRA event\n");
269062306a36Sopenharmony_ci		break;
269162306a36Sopenharmony_ci	default:
269262306a36Sopenharmony_ci		pr_err("received unrecognized CM event %d\n", event->event);
269362306a36Sopenharmony_ci		break;
269462306a36Sopenharmony_ci	}
269562306a36Sopenharmony_ci
269662306a36Sopenharmony_ci	return ret;
269762306a36Sopenharmony_ci}
269862306a36Sopenharmony_ci
269962306a36Sopenharmony_cistatic int srpt_rdma_cm_handler(struct rdma_cm_id *cm_id,
270062306a36Sopenharmony_ci				struct rdma_cm_event *event)
270162306a36Sopenharmony_ci{
270262306a36Sopenharmony_ci	struct srpt_rdma_ch *ch = cm_id->context;
270362306a36Sopenharmony_ci	int ret = 0;
270462306a36Sopenharmony_ci
270562306a36Sopenharmony_ci	switch (event->event) {
270662306a36Sopenharmony_ci	case RDMA_CM_EVENT_CONNECT_REQUEST:
270762306a36Sopenharmony_ci		ret = srpt_rdma_cm_req_recv(cm_id, event);
270862306a36Sopenharmony_ci		break;
270962306a36Sopenharmony_ci	case RDMA_CM_EVENT_REJECTED:
271062306a36Sopenharmony_ci		srpt_cm_rej_recv(ch, event->status,
271162306a36Sopenharmony_ci				 event->param.conn.private_data,
271262306a36Sopenharmony_ci				 event->param.conn.private_data_len);
271362306a36Sopenharmony_ci		break;
271462306a36Sopenharmony_ci	case RDMA_CM_EVENT_ESTABLISHED:
271562306a36Sopenharmony_ci		srpt_cm_rtu_recv(ch);
271662306a36Sopenharmony_ci		break;
271762306a36Sopenharmony_ci	case RDMA_CM_EVENT_DISCONNECTED:
271862306a36Sopenharmony_ci		if (ch->state < CH_DISCONNECTING)
271962306a36Sopenharmony_ci			srpt_disconnect_ch(ch);
272062306a36Sopenharmony_ci		else
272162306a36Sopenharmony_ci			srpt_close_ch(ch);
272262306a36Sopenharmony_ci		break;
272362306a36Sopenharmony_ci	case RDMA_CM_EVENT_TIMEWAIT_EXIT:
272462306a36Sopenharmony_ci		srpt_close_ch(ch);
272562306a36Sopenharmony_ci		break;
272662306a36Sopenharmony_ci	case RDMA_CM_EVENT_UNREACHABLE:
272762306a36Sopenharmony_ci		pr_info("Received CM REP error for ch %s-%d.\n", ch->sess_name,
272862306a36Sopenharmony_ci			ch->qp->qp_num);
272962306a36Sopenharmony_ci		break;
273062306a36Sopenharmony_ci	case RDMA_CM_EVENT_DEVICE_REMOVAL:
273162306a36Sopenharmony_ci	case RDMA_CM_EVENT_ADDR_CHANGE:
273262306a36Sopenharmony_ci		break;
273362306a36Sopenharmony_ci	default:
273462306a36Sopenharmony_ci		pr_err("received unrecognized RDMA CM event %d\n",
273562306a36Sopenharmony_ci		       event->event);
273662306a36Sopenharmony_ci		break;
273762306a36Sopenharmony_ci	}
273862306a36Sopenharmony_ci
273962306a36Sopenharmony_ci	return ret;
274062306a36Sopenharmony_ci}
274162306a36Sopenharmony_ci
274262306a36Sopenharmony_ci/*
274362306a36Sopenharmony_ci * srpt_write_pending - Start data transfer from initiator to target (write).
274462306a36Sopenharmony_ci */
274562306a36Sopenharmony_cistatic int srpt_write_pending(struct se_cmd *se_cmd)
274662306a36Sopenharmony_ci{
274762306a36Sopenharmony_ci	struct srpt_send_ioctx *ioctx =
274862306a36Sopenharmony_ci		container_of(se_cmd, struct srpt_send_ioctx, cmd);
274962306a36Sopenharmony_ci	struct srpt_rdma_ch *ch = ioctx->ch;
275062306a36Sopenharmony_ci	struct ib_send_wr *first_wr = NULL;
275162306a36Sopenharmony_ci	struct ib_cqe *cqe = &ioctx->rdma_cqe;
275262306a36Sopenharmony_ci	enum srpt_command_state new_state;
275362306a36Sopenharmony_ci	int ret, i;
275462306a36Sopenharmony_ci
275562306a36Sopenharmony_ci	if (ioctx->recv_ioctx) {
275662306a36Sopenharmony_ci		srpt_set_cmd_state(ioctx, SRPT_STATE_DATA_IN);
275762306a36Sopenharmony_ci		target_execute_cmd(&ioctx->cmd);
275862306a36Sopenharmony_ci		return 0;
275962306a36Sopenharmony_ci	}
276062306a36Sopenharmony_ci
276162306a36Sopenharmony_ci	new_state = srpt_set_cmd_state(ioctx, SRPT_STATE_NEED_DATA);
276262306a36Sopenharmony_ci	WARN_ON(new_state == SRPT_STATE_DONE);
276362306a36Sopenharmony_ci
276462306a36Sopenharmony_ci	if (atomic_sub_return(ioctx->n_rdma, &ch->sq_wr_avail) < 0) {
276562306a36Sopenharmony_ci		pr_warn("%s: IB send queue full (needed %d)\n",
276662306a36Sopenharmony_ci				__func__, ioctx->n_rdma);
276762306a36Sopenharmony_ci		ret = -ENOMEM;
276862306a36Sopenharmony_ci		goto out_undo;
276962306a36Sopenharmony_ci	}
277062306a36Sopenharmony_ci
277162306a36Sopenharmony_ci	cqe->done = srpt_rdma_read_done;
277262306a36Sopenharmony_ci	for (i = ioctx->n_rw_ctx - 1; i >= 0; i--) {
277362306a36Sopenharmony_ci		struct srpt_rw_ctx *ctx = &ioctx->rw_ctxs[i];
277462306a36Sopenharmony_ci
277562306a36Sopenharmony_ci		first_wr = rdma_rw_ctx_wrs(&ctx->rw, ch->qp, ch->sport->port,
277662306a36Sopenharmony_ci				cqe, first_wr);
277762306a36Sopenharmony_ci		cqe = NULL;
277862306a36Sopenharmony_ci	}
277962306a36Sopenharmony_ci
278062306a36Sopenharmony_ci	ret = ib_post_send(ch->qp, first_wr, NULL);
278162306a36Sopenharmony_ci	if (ret) {
278262306a36Sopenharmony_ci		pr_err("%s: ib_post_send() returned %d for %d (avail: %d)\n",
278362306a36Sopenharmony_ci			 __func__, ret, ioctx->n_rdma,
278462306a36Sopenharmony_ci			 atomic_read(&ch->sq_wr_avail));
278562306a36Sopenharmony_ci		goto out_undo;
278662306a36Sopenharmony_ci	}
278762306a36Sopenharmony_ci
278862306a36Sopenharmony_ci	return 0;
278962306a36Sopenharmony_ciout_undo:
279062306a36Sopenharmony_ci	atomic_add(ioctx->n_rdma, &ch->sq_wr_avail);
279162306a36Sopenharmony_ci	return ret;
279262306a36Sopenharmony_ci}
279362306a36Sopenharmony_ci
279462306a36Sopenharmony_cistatic u8 tcm_to_srp_tsk_mgmt_status(const int tcm_mgmt_status)
279562306a36Sopenharmony_ci{
279662306a36Sopenharmony_ci	switch (tcm_mgmt_status) {
279762306a36Sopenharmony_ci	case TMR_FUNCTION_COMPLETE:
279862306a36Sopenharmony_ci		return SRP_TSK_MGMT_SUCCESS;
279962306a36Sopenharmony_ci	case TMR_FUNCTION_REJECTED:
280062306a36Sopenharmony_ci		return SRP_TSK_MGMT_FUNC_NOT_SUPP;
280162306a36Sopenharmony_ci	}
280262306a36Sopenharmony_ci	return SRP_TSK_MGMT_FAILED;
280362306a36Sopenharmony_ci}
280462306a36Sopenharmony_ci
280562306a36Sopenharmony_ci/**
280662306a36Sopenharmony_ci * srpt_queue_response - transmit the response to a SCSI command
280762306a36Sopenharmony_ci * @cmd: SCSI target command.
280862306a36Sopenharmony_ci *
280962306a36Sopenharmony_ci * Callback function called by the TCM core. Must not block since it can be
281062306a36Sopenharmony_ci * invoked on the context of the IB completion handler.
281162306a36Sopenharmony_ci */
281262306a36Sopenharmony_cistatic void srpt_queue_response(struct se_cmd *cmd)
281362306a36Sopenharmony_ci{
281462306a36Sopenharmony_ci	struct srpt_send_ioctx *ioctx =
281562306a36Sopenharmony_ci		container_of(cmd, struct srpt_send_ioctx, cmd);
281662306a36Sopenharmony_ci	struct srpt_rdma_ch *ch = ioctx->ch;
281762306a36Sopenharmony_ci	struct srpt_device *sdev = ch->sport->sdev;
281862306a36Sopenharmony_ci	struct ib_send_wr send_wr, *first_wr = &send_wr;
281962306a36Sopenharmony_ci	struct ib_sge sge;
282062306a36Sopenharmony_ci	enum srpt_command_state state;
282162306a36Sopenharmony_ci	int resp_len, ret, i;
282262306a36Sopenharmony_ci	u8 srp_tm_status;
282362306a36Sopenharmony_ci
282462306a36Sopenharmony_ci	state = ioctx->state;
282562306a36Sopenharmony_ci	switch (state) {
282662306a36Sopenharmony_ci	case SRPT_STATE_NEW:
282762306a36Sopenharmony_ci	case SRPT_STATE_DATA_IN:
282862306a36Sopenharmony_ci		ioctx->state = SRPT_STATE_CMD_RSP_SENT;
282962306a36Sopenharmony_ci		break;
283062306a36Sopenharmony_ci	case SRPT_STATE_MGMT:
283162306a36Sopenharmony_ci		ioctx->state = SRPT_STATE_MGMT_RSP_SENT;
283262306a36Sopenharmony_ci		break;
283362306a36Sopenharmony_ci	default:
283462306a36Sopenharmony_ci		WARN(true, "ch %p; cmd %d: unexpected command state %d\n",
283562306a36Sopenharmony_ci			ch, ioctx->ioctx.index, ioctx->state);
283662306a36Sopenharmony_ci		break;
283762306a36Sopenharmony_ci	}
283862306a36Sopenharmony_ci
283962306a36Sopenharmony_ci	if (WARN_ON_ONCE(state == SRPT_STATE_CMD_RSP_SENT))
284062306a36Sopenharmony_ci		return;
284162306a36Sopenharmony_ci
284262306a36Sopenharmony_ci	/* For read commands, transfer the data to the initiator. */
284362306a36Sopenharmony_ci	if (ioctx->cmd.data_direction == DMA_FROM_DEVICE &&
284462306a36Sopenharmony_ci	    ioctx->cmd.data_length &&
284562306a36Sopenharmony_ci	    !ioctx->queue_status_only) {
284662306a36Sopenharmony_ci		for (i = ioctx->n_rw_ctx - 1; i >= 0; i--) {
284762306a36Sopenharmony_ci			struct srpt_rw_ctx *ctx = &ioctx->rw_ctxs[i];
284862306a36Sopenharmony_ci
284962306a36Sopenharmony_ci			first_wr = rdma_rw_ctx_wrs(&ctx->rw, ch->qp,
285062306a36Sopenharmony_ci					ch->sport->port, NULL, first_wr);
285162306a36Sopenharmony_ci		}
285262306a36Sopenharmony_ci	}
285362306a36Sopenharmony_ci
285462306a36Sopenharmony_ci	if (state != SRPT_STATE_MGMT)
285562306a36Sopenharmony_ci		resp_len = srpt_build_cmd_rsp(ch, ioctx, ioctx->cmd.tag,
285662306a36Sopenharmony_ci					      cmd->scsi_status);
285762306a36Sopenharmony_ci	else {
285862306a36Sopenharmony_ci		srp_tm_status
285962306a36Sopenharmony_ci			= tcm_to_srp_tsk_mgmt_status(cmd->se_tmr_req->response);
286062306a36Sopenharmony_ci		resp_len = srpt_build_tskmgmt_rsp(ch, ioctx, srp_tm_status,
286162306a36Sopenharmony_ci						 ioctx->cmd.tag);
286262306a36Sopenharmony_ci	}
286362306a36Sopenharmony_ci
286462306a36Sopenharmony_ci	atomic_inc(&ch->req_lim);
286562306a36Sopenharmony_ci
286662306a36Sopenharmony_ci	if (unlikely(atomic_sub_return(1 + ioctx->n_rdma,
286762306a36Sopenharmony_ci			&ch->sq_wr_avail) < 0)) {
286862306a36Sopenharmony_ci		pr_warn("%s: IB send queue full (needed %d)\n",
286962306a36Sopenharmony_ci				__func__, ioctx->n_rdma);
287062306a36Sopenharmony_ci		goto out;
287162306a36Sopenharmony_ci	}
287262306a36Sopenharmony_ci
287362306a36Sopenharmony_ci	ib_dma_sync_single_for_device(sdev->device, ioctx->ioctx.dma, resp_len,
287462306a36Sopenharmony_ci				      DMA_TO_DEVICE);
287562306a36Sopenharmony_ci
287662306a36Sopenharmony_ci	sge.addr = ioctx->ioctx.dma;
287762306a36Sopenharmony_ci	sge.length = resp_len;
287862306a36Sopenharmony_ci	sge.lkey = sdev->lkey;
287962306a36Sopenharmony_ci
288062306a36Sopenharmony_ci	ioctx->ioctx.cqe.done = srpt_send_done;
288162306a36Sopenharmony_ci	send_wr.next = NULL;
288262306a36Sopenharmony_ci	send_wr.wr_cqe = &ioctx->ioctx.cqe;
288362306a36Sopenharmony_ci	send_wr.sg_list = &sge;
288462306a36Sopenharmony_ci	send_wr.num_sge = 1;
288562306a36Sopenharmony_ci	send_wr.opcode = IB_WR_SEND;
288662306a36Sopenharmony_ci	send_wr.send_flags = IB_SEND_SIGNALED;
288762306a36Sopenharmony_ci
288862306a36Sopenharmony_ci	ret = ib_post_send(ch->qp, first_wr, NULL);
288962306a36Sopenharmony_ci	if (ret < 0) {
289062306a36Sopenharmony_ci		pr_err("%s: sending cmd response failed for tag %llu (%d)\n",
289162306a36Sopenharmony_ci			__func__, ioctx->cmd.tag, ret);
289262306a36Sopenharmony_ci		goto out;
289362306a36Sopenharmony_ci	}
289462306a36Sopenharmony_ci
289562306a36Sopenharmony_ci	return;
289662306a36Sopenharmony_ci
289762306a36Sopenharmony_ciout:
289862306a36Sopenharmony_ci	atomic_add(1 + ioctx->n_rdma, &ch->sq_wr_avail);
289962306a36Sopenharmony_ci	atomic_dec(&ch->req_lim);
290062306a36Sopenharmony_ci	srpt_set_cmd_state(ioctx, SRPT_STATE_DONE);
290162306a36Sopenharmony_ci	target_put_sess_cmd(&ioctx->cmd);
290262306a36Sopenharmony_ci}
290362306a36Sopenharmony_ci
290462306a36Sopenharmony_cistatic int srpt_queue_data_in(struct se_cmd *cmd)
290562306a36Sopenharmony_ci{
290662306a36Sopenharmony_ci	srpt_queue_response(cmd);
290762306a36Sopenharmony_ci	return 0;
290862306a36Sopenharmony_ci}
290962306a36Sopenharmony_ci
291062306a36Sopenharmony_cistatic void srpt_queue_tm_rsp(struct se_cmd *cmd)
291162306a36Sopenharmony_ci{
291262306a36Sopenharmony_ci	srpt_queue_response(cmd);
291362306a36Sopenharmony_ci}
291462306a36Sopenharmony_ci
291562306a36Sopenharmony_ci/*
291662306a36Sopenharmony_ci * This function is called for aborted commands if no response is sent to the
291762306a36Sopenharmony_ci * initiator. Make sure that the credits freed by aborting a command are
291862306a36Sopenharmony_ci * returned to the initiator the next time a response is sent by incrementing
291962306a36Sopenharmony_ci * ch->req_lim_delta.
292062306a36Sopenharmony_ci */
292162306a36Sopenharmony_cistatic void srpt_aborted_task(struct se_cmd *cmd)
292262306a36Sopenharmony_ci{
292362306a36Sopenharmony_ci	struct srpt_send_ioctx *ioctx = container_of(cmd,
292462306a36Sopenharmony_ci				struct srpt_send_ioctx, cmd);
292562306a36Sopenharmony_ci	struct srpt_rdma_ch *ch = ioctx->ch;
292662306a36Sopenharmony_ci
292762306a36Sopenharmony_ci	atomic_inc(&ch->req_lim_delta);
292862306a36Sopenharmony_ci}
292962306a36Sopenharmony_ci
293062306a36Sopenharmony_cistatic int srpt_queue_status(struct se_cmd *cmd)
293162306a36Sopenharmony_ci{
293262306a36Sopenharmony_ci	struct srpt_send_ioctx *ioctx;
293362306a36Sopenharmony_ci
293462306a36Sopenharmony_ci	ioctx = container_of(cmd, struct srpt_send_ioctx, cmd);
293562306a36Sopenharmony_ci	BUG_ON(ioctx->sense_data != cmd->sense_buffer);
293662306a36Sopenharmony_ci	if (cmd->se_cmd_flags &
293762306a36Sopenharmony_ci	    (SCF_TRANSPORT_TASK_SENSE | SCF_EMULATED_TASK_SENSE))
293862306a36Sopenharmony_ci		WARN_ON(cmd->scsi_status != SAM_STAT_CHECK_CONDITION);
293962306a36Sopenharmony_ci	ioctx->queue_status_only = true;
294062306a36Sopenharmony_ci	srpt_queue_response(cmd);
294162306a36Sopenharmony_ci	return 0;
294262306a36Sopenharmony_ci}
294362306a36Sopenharmony_ci
294462306a36Sopenharmony_cistatic void srpt_refresh_port_work(struct work_struct *work)
294562306a36Sopenharmony_ci{
294662306a36Sopenharmony_ci	struct srpt_port *sport = container_of(work, struct srpt_port, work);
294762306a36Sopenharmony_ci
294862306a36Sopenharmony_ci	srpt_refresh_port(sport);
294962306a36Sopenharmony_ci}
295062306a36Sopenharmony_ci
295162306a36Sopenharmony_ci/**
295262306a36Sopenharmony_ci * srpt_release_sport - disable login and wait for associated channels
295362306a36Sopenharmony_ci * @sport: SRPT HCA port.
295462306a36Sopenharmony_ci */
295562306a36Sopenharmony_cistatic int srpt_release_sport(struct srpt_port *sport)
295662306a36Sopenharmony_ci{
295762306a36Sopenharmony_ci	DECLARE_COMPLETION_ONSTACK(c);
295862306a36Sopenharmony_ci	struct srpt_nexus *nexus, *next_n;
295962306a36Sopenharmony_ci	struct srpt_rdma_ch *ch;
296062306a36Sopenharmony_ci
296162306a36Sopenharmony_ci	WARN_ON_ONCE(irqs_disabled());
296262306a36Sopenharmony_ci
296362306a36Sopenharmony_ci	sport->freed_channels = &c;
296462306a36Sopenharmony_ci
296562306a36Sopenharmony_ci	mutex_lock(&sport->mutex);
296662306a36Sopenharmony_ci	srpt_set_enabled(sport, false);
296762306a36Sopenharmony_ci	mutex_unlock(&sport->mutex);
296862306a36Sopenharmony_ci
296962306a36Sopenharmony_ci	while (atomic_read(&sport->refcount) > 0 &&
297062306a36Sopenharmony_ci	       wait_for_completion_timeout(&c, 5 * HZ) <= 0) {
297162306a36Sopenharmony_ci		pr_info("%s_%d: waiting for unregistration of %d sessions ...\n",
297262306a36Sopenharmony_ci			dev_name(&sport->sdev->device->dev), sport->port,
297362306a36Sopenharmony_ci			atomic_read(&sport->refcount));
297462306a36Sopenharmony_ci		rcu_read_lock();
297562306a36Sopenharmony_ci		list_for_each_entry(nexus, &sport->nexus_list, entry) {
297662306a36Sopenharmony_ci			list_for_each_entry(ch, &nexus->ch_list, list) {
297762306a36Sopenharmony_ci				pr_info("%s-%d: state %s\n",
297862306a36Sopenharmony_ci					ch->sess_name, ch->qp->qp_num,
297962306a36Sopenharmony_ci					get_ch_state_name(ch->state));
298062306a36Sopenharmony_ci			}
298162306a36Sopenharmony_ci		}
298262306a36Sopenharmony_ci		rcu_read_unlock();
298362306a36Sopenharmony_ci	}
298462306a36Sopenharmony_ci
298562306a36Sopenharmony_ci	mutex_lock(&sport->mutex);
298662306a36Sopenharmony_ci	list_for_each_entry_safe(nexus, next_n, &sport->nexus_list, entry) {
298762306a36Sopenharmony_ci		list_del(&nexus->entry);
298862306a36Sopenharmony_ci		kfree_rcu(nexus, rcu);
298962306a36Sopenharmony_ci	}
299062306a36Sopenharmony_ci	mutex_unlock(&sport->mutex);
299162306a36Sopenharmony_ci
299262306a36Sopenharmony_ci	return 0;
299362306a36Sopenharmony_ci}
299462306a36Sopenharmony_ci
299562306a36Sopenharmony_cistruct port_and_port_id {
299662306a36Sopenharmony_ci	struct srpt_port *sport;
299762306a36Sopenharmony_ci	struct srpt_port_id **port_id;
299862306a36Sopenharmony_ci};
299962306a36Sopenharmony_ci
300062306a36Sopenharmony_cistatic struct port_and_port_id __srpt_lookup_port(const char *name)
300162306a36Sopenharmony_ci{
300262306a36Sopenharmony_ci	struct ib_device *dev;
300362306a36Sopenharmony_ci	struct srpt_device *sdev;
300462306a36Sopenharmony_ci	struct srpt_port *sport;
300562306a36Sopenharmony_ci	int i;
300662306a36Sopenharmony_ci
300762306a36Sopenharmony_ci	list_for_each_entry(sdev, &srpt_dev_list, list) {
300862306a36Sopenharmony_ci		dev = sdev->device;
300962306a36Sopenharmony_ci		if (!dev)
301062306a36Sopenharmony_ci			continue;
301162306a36Sopenharmony_ci
301262306a36Sopenharmony_ci		for (i = 0; i < dev->phys_port_cnt; i++) {
301362306a36Sopenharmony_ci			sport = &sdev->port[i];
301462306a36Sopenharmony_ci
301562306a36Sopenharmony_ci			if (strcmp(sport->guid_name, name) == 0) {
301662306a36Sopenharmony_ci				kref_get(&sdev->refcnt);
301762306a36Sopenharmony_ci				return (struct port_and_port_id){
301862306a36Sopenharmony_ci					sport, &sport->guid_id};
301962306a36Sopenharmony_ci			}
302062306a36Sopenharmony_ci			if (strcmp(sport->gid_name, name) == 0) {
302162306a36Sopenharmony_ci				kref_get(&sdev->refcnt);
302262306a36Sopenharmony_ci				return (struct port_and_port_id){
302362306a36Sopenharmony_ci					sport, &sport->gid_id};
302462306a36Sopenharmony_ci			}
302562306a36Sopenharmony_ci		}
302662306a36Sopenharmony_ci	}
302762306a36Sopenharmony_ci
302862306a36Sopenharmony_ci	return (struct port_and_port_id){};
302962306a36Sopenharmony_ci}
303062306a36Sopenharmony_ci
303162306a36Sopenharmony_ci/**
303262306a36Sopenharmony_ci * srpt_lookup_port() - Look up an RDMA port by name
303362306a36Sopenharmony_ci * @name: ASCII port name
303462306a36Sopenharmony_ci *
303562306a36Sopenharmony_ci * Increments the RDMA port reference count if an RDMA port pointer is returned.
303662306a36Sopenharmony_ci * The caller must drop that reference count by calling srpt_port_put_ref().
303762306a36Sopenharmony_ci */
303862306a36Sopenharmony_cistatic struct port_and_port_id srpt_lookup_port(const char *name)
303962306a36Sopenharmony_ci{
304062306a36Sopenharmony_ci	struct port_and_port_id papi;
304162306a36Sopenharmony_ci
304262306a36Sopenharmony_ci	spin_lock(&srpt_dev_lock);
304362306a36Sopenharmony_ci	papi = __srpt_lookup_port(name);
304462306a36Sopenharmony_ci	spin_unlock(&srpt_dev_lock);
304562306a36Sopenharmony_ci
304662306a36Sopenharmony_ci	return papi;
304762306a36Sopenharmony_ci}
304862306a36Sopenharmony_ci
304962306a36Sopenharmony_cistatic void srpt_free_srq(struct srpt_device *sdev)
305062306a36Sopenharmony_ci{
305162306a36Sopenharmony_ci	if (!sdev->srq)
305262306a36Sopenharmony_ci		return;
305362306a36Sopenharmony_ci
305462306a36Sopenharmony_ci	ib_destroy_srq(sdev->srq);
305562306a36Sopenharmony_ci	srpt_free_ioctx_ring((struct srpt_ioctx **)sdev->ioctx_ring, sdev,
305662306a36Sopenharmony_ci			     sdev->srq_size, sdev->req_buf_cache,
305762306a36Sopenharmony_ci			     DMA_FROM_DEVICE);
305862306a36Sopenharmony_ci	kmem_cache_destroy(sdev->req_buf_cache);
305962306a36Sopenharmony_ci	sdev->srq = NULL;
306062306a36Sopenharmony_ci}
306162306a36Sopenharmony_ci
306262306a36Sopenharmony_cistatic int srpt_alloc_srq(struct srpt_device *sdev)
306362306a36Sopenharmony_ci{
306462306a36Sopenharmony_ci	struct ib_srq_init_attr srq_attr = {
306562306a36Sopenharmony_ci		.event_handler = srpt_srq_event,
306662306a36Sopenharmony_ci		.srq_context = (void *)sdev,
306762306a36Sopenharmony_ci		.attr.max_wr = sdev->srq_size,
306862306a36Sopenharmony_ci		.attr.max_sge = 1,
306962306a36Sopenharmony_ci		.srq_type = IB_SRQT_BASIC,
307062306a36Sopenharmony_ci	};
307162306a36Sopenharmony_ci	struct ib_device *device = sdev->device;
307262306a36Sopenharmony_ci	struct ib_srq *srq;
307362306a36Sopenharmony_ci	int i;
307462306a36Sopenharmony_ci
307562306a36Sopenharmony_ci	WARN_ON_ONCE(sdev->srq);
307662306a36Sopenharmony_ci	srq = ib_create_srq(sdev->pd, &srq_attr);
307762306a36Sopenharmony_ci	if (IS_ERR(srq)) {
307862306a36Sopenharmony_ci		pr_debug("ib_create_srq() failed: %ld\n", PTR_ERR(srq));
307962306a36Sopenharmony_ci		return PTR_ERR(srq);
308062306a36Sopenharmony_ci	}
308162306a36Sopenharmony_ci
308262306a36Sopenharmony_ci	pr_debug("create SRQ #wr= %d max_allow=%d dev= %s\n", sdev->srq_size,
308362306a36Sopenharmony_ci		 sdev->device->attrs.max_srq_wr, dev_name(&device->dev));
308462306a36Sopenharmony_ci
308562306a36Sopenharmony_ci	sdev->req_buf_cache = kmem_cache_create("srpt-srq-req-buf",
308662306a36Sopenharmony_ci						srp_max_req_size, 0, 0, NULL);
308762306a36Sopenharmony_ci	if (!sdev->req_buf_cache)
308862306a36Sopenharmony_ci		goto free_srq;
308962306a36Sopenharmony_ci
309062306a36Sopenharmony_ci	sdev->ioctx_ring = (struct srpt_recv_ioctx **)
309162306a36Sopenharmony_ci		srpt_alloc_ioctx_ring(sdev, sdev->srq_size,
309262306a36Sopenharmony_ci				      sizeof(*sdev->ioctx_ring[0]),
309362306a36Sopenharmony_ci				      sdev->req_buf_cache, 0, DMA_FROM_DEVICE);
309462306a36Sopenharmony_ci	if (!sdev->ioctx_ring)
309562306a36Sopenharmony_ci		goto free_cache;
309662306a36Sopenharmony_ci
309762306a36Sopenharmony_ci	sdev->use_srq = true;
309862306a36Sopenharmony_ci	sdev->srq = srq;
309962306a36Sopenharmony_ci
310062306a36Sopenharmony_ci	for (i = 0; i < sdev->srq_size; ++i) {
310162306a36Sopenharmony_ci		INIT_LIST_HEAD(&sdev->ioctx_ring[i]->wait_list);
310262306a36Sopenharmony_ci		srpt_post_recv(sdev, NULL, sdev->ioctx_ring[i]);
310362306a36Sopenharmony_ci	}
310462306a36Sopenharmony_ci
310562306a36Sopenharmony_ci	return 0;
310662306a36Sopenharmony_ci
310762306a36Sopenharmony_cifree_cache:
310862306a36Sopenharmony_ci	kmem_cache_destroy(sdev->req_buf_cache);
310962306a36Sopenharmony_ci
311062306a36Sopenharmony_cifree_srq:
311162306a36Sopenharmony_ci	ib_destroy_srq(srq);
311262306a36Sopenharmony_ci	return -ENOMEM;
311362306a36Sopenharmony_ci}
311462306a36Sopenharmony_ci
311562306a36Sopenharmony_cistatic int srpt_use_srq(struct srpt_device *sdev, bool use_srq)
311662306a36Sopenharmony_ci{
311762306a36Sopenharmony_ci	struct ib_device *device = sdev->device;
311862306a36Sopenharmony_ci	int ret = 0;
311962306a36Sopenharmony_ci
312062306a36Sopenharmony_ci	if (!use_srq) {
312162306a36Sopenharmony_ci		srpt_free_srq(sdev);
312262306a36Sopenharmony_ci		sdev->use_srq = false;
312362306a36Sopenharmony_ci	} else if (use_srq && !sdev->srq) {
312462306a36Sopenharmony_ci		ret = srpt_alloc_srq(sdev);
312562306a36Sopenharmony_ci	}
312662306a36Sopenharmony_ci	pr_debug("%s(%s): use_srq = %d; ret = %d\n", __func__,
312762306a36Sopenharmony_ci		 dev_name(&device->dev), sdev->use_srq, ret);
312862306a36Sopenharmony_ci	return ret;
312962306a36Sopenharmony_ci}
313062306a36Sopenharmony_ci
313162306a36Sopenharmony_cistatic void srpt_free_sdev(struct kref *refcnt)
313262306a36Sopenharmony_ci{
313362306a36Sopenharmony_ci	struct srpt_device *sdev = container_of(refcnt, typeof(*sdev), refcnt);
313462306a36Sopenharmony_ci
313562306a36Sopenharmony_ci	kfree(sdev);
313662306a36Sopenharmony_ci}
313762306a36Sopenharmony_ci
313862306a36Sopenharmony_cistatic void srpt_sdev_put(struct srpt_device *sdev)
313962306a36Sopenharmony_ci{
314062306a36Sopenharmony_ci	kref_put(&sdev->refcnt, srpt_free_sdev);
314162306a36Sopenharmony_ci}
314262306a36Sopenharmony_ci
314362306a36Sopenharmony_ci/**
314462306a36Sopenharmony_ci * srpt_add_one - InfiniBand device addition callback function
314562306a36Sopenharmony_ci * @device: Describes a HCA.
314662306a36Sopenharmony_ci */
314762306a36Sopenharmony_cistatic int srpt_add_one(struct ib_device *device)
314862306a36Sopenharmony_ci{
314962306a36Sopenharmony_ci	struct srpt_device *sdev;
315062306a36Sopenharmony_ci	struct srpt_port *sport;
315162306a36Sopenharmony_ci	int ret;
315262306a36Sopenharmony_ci	u32 i;
315362306a36Sopenharmony_ci
315462306a36Sopenharmony_ci	pr_debug("device = %p\n", device);
315562306a36Sopenharmony_ci
315662306a36Sopenharmony_ci	sdev = kzalloc(struct_size(sdev, port, device->phys_port_cnt),
315762306a36Sopenharmony_ci		       GFP_KERNEL);
315862306a36Sopenharmony_ci	if (!sdev)
315962306a36Sopenharmony_ci		return -ENOMEM;
316062306a36Sopenharmony_ci
316162306a36Sopenharmony_ci	kref_init(&sdev->refcnt);
316262306a36Sopenharmony_ci	sdev->device = device;
316362306a36Sopenharmony_ci	mutex_init(&sdev->sdev_mutex);
316462306a36Sopenharmony_ci
316562306a36Sopenharmony_ci	sdev->pd = ib_alloc_pd(device, 0);
316662306a36Sopenharmony_ci	if (IS_ERR(sdev->pd)) {
316762306a36Sopenharmony_ci		ret = PTR_ERR(sdev->pd);
316862306a36Sopenharmony_ci		goto free_dev;
316962306a36Sopenharmony_ci	}
317062306a36Sopenharmony_ci
317162306a36Sopenharmony_ci	sdev->lkey = sdev->pd->local_dma_lkey;
317262306a36Sopenharmony_ci
317362306a36Sopenharmony_ci	sdev->srq_size = min(srpt_srq_size, sdev->device->attrs.max_srq_wr);
317462306a36Sopenharmony_ci
317562306a36Sopenharmony_ci	srpt_use_srq(sdev, sdev->port[0].port_attrib.use_srq);
317662306a36Sopenharmony_ci
317762306a36Sopenharmony_ci	if (!srpt_service_guid)
317862306a36Sopenharmony_ci		srpt_service_guid = be64_to_cpu(device->node_guid);
317962306a36Sopenharmony_ci
318062306a36Sopenharmony_ci	if (rdma_port_get_link_layer(device, 1) == IB_LINK_LAYER_INFINIBAND)
318162306a36Sopenharmony_ci		sdev->cm_id = ib_create_cm_id(device, srpt_cm_handler, sdev);
318262306a36Sopenharmony_ci	if (IS_ERR(sdev->cm_id)) {
318362306a36Sopenharmony_ci		pr_info("ib_create_cm_id() failed: %ld\n",
318462306a36Sopenharmony_ci			PTR_ERR(sdev->cm_id));
318562306a36Sopenharmony_ci		ret = PTR_ERR(sdev->cm_id);
318662306a36Sopenharmony_ci		sdev->cm_id = NULL;
318762306a36Sopenharmony_ci		if (!rdma_cm_id)
318862306a36Sopenharmony_ci			goto err_ring;
318962306a36Sopenharmony_ci	}
319062306a36Sopenharmony_ci
319162306a36Sopenharmony_ci	/* print out target login information */
319262306a36Sopenharmony_ci	pr_debug("Target login info: id_ext=%016llx,ioc_guid=%016llx,pkey=ffff,service_id=%016llx\n",
319362306a36Sopenharmony_ci		 srpt_service_guid, srpt_service_guid, srpt_service_guid);
319462306a36Sopenharmony_ci
319562306a36Sopenharmony_ci	/*
319662306a36Sopenharmony_ci	 * We do not have a consistent service_id (ie. also id_ext of target_id)
319762306a36Sopenharmony_ci	 * to identify this target. We currently use the guid of the first HCA
319862306a36Sopenharmony_ci	 * in the system as service_id; therefore, the target_id will change
319962306a36Sopenharmony_ci	 * if this HCA is gone bad and replaced by different HCA
320062306a36Sopenharmony_ci	 */
320162306a36Sopenharmony_ci	ret = sdev->cm_id ?
320262306a36Sopenharmony_ci		ib_cm_listen(sdev->cm_id, cpu_to_be64(srpt_service_guid)) :
320362306a36Sopenharmony_ci		0;
320462306a36Sopenharmony_ci	if (ret < 0) {
320562306a36Sopenharmony_ci		pr_err("ib_cm_listen() failed: %d (cm_id state = %d)\n", ret,
320662306a36Sopenharmony_ci		       sdev->cm_id->state);
320762306a36Sopenharmony_ci		goto err_cm;
320862306a36Sopenharmony_ci	}
320962306a36Sopenharmony_ci
321062306a36Sopenharmony_ci	INIT_IB_EVENT_HANDLER(&sdev->event_handler, sdev->device,
321162306a36Sopenharmony_ci			      srpt_event_handler);
321262306a36Sopenharmony_ci
321362306a36Sopenharmony_ci	for (i = 1; i <= sdev->device->phys_port_cnt; i++) {
321462306a36Sopenharmony_ci		sport = &sdev->port[i - 1];
321562306a36Sopenharmony_ci		INIT_LIST_HEAD(&sport->nexus_list);
321662306a36Sopenharmony_ci		mutex_init(&sport->mutex);
321762306a36Sopenharmony_ci		sport->sdev = sdev;
321862306a36Sopenharmony_ci		sport->port = i;
321962306a36Sopenharmony_ci		sport->port_attrib.srp_max_rdma_size = DEFAULT_MAX_RDMA_SIZE;
322062306a36Sopenharmony_ci		sport->port_attrib.srp_max_rsp_size = DEFAULT_MAX_RSP_SIZE;
322162306a36Sopenharmony_ci		sport->port_attrib.srp_sq_size = DEF_SRPT_SQ_SIZE;
322262306a36Sopenharmony_ci		sport->port_attrib.use_srq = false;
322362306a36Sopenharmony_ci		INIT_WORK(&sport->work, srpt_refresh_port_work);
322462306a36Sopenharmony_ci
322562306a36Sopenharmony_ci		ret = srpt_refresh_port(sport);
322662306a36Sopenharmony_ci		if (ret) {
322762306a36Sopenharmony_ci			pr_err("MAD registration failed for %s-%d.\n",
322862306a36Sopenharmony_ci			       dev_name(&sdev->device->dev), i);
322962306a36Sopenharmony_ci			i--;
323062306a36Sopenharmony_ci			goto err_port;
323162306a36Sopenharmony_ci		}
323262306a36Sopenharmony_ci	}
323362306a36Sopenharmony_ci
323462306a36Sopenharmony_ci	ib_register_event_handler(&sdev->event_handler);
323562306a36Sopenharmony_ci	spin_lock(&srpt_dev_lock);
323662306a36Sopenharmony_ci	list_add_tail(&sdev->list, &srpt_dev_list);
323762306a36Sopenharmony_ci	spin_unlock(&srpt_dev_lock);
323862306a36Sopenharmony_ci
323962306a36Sopenharmony_ci	ib_set_client_data(device, &srpt_client, sdev);
324062306a36Sopenharmony_ci	pr_debug("added %s.\n", dev_name(&device->dev));
324162306a36Sopenharmony_ci	return 0;
324262306a36Sopenharmony_ci
324362306a36Sopenharmony_cierr_port:
324462306a36Sopenharmony_ci	srpt_unregister_mad_agent(sdev, i);
324562306a36Sopenharmony_cierr_cm:
324662306a36Sopenharmony_ci	if (sdev->cm_id)
324762306a36Sopenharmony_ci		ib_destroy_cm_id(sdev->cm_id);
324862306a36Sopenharmony_cierr_ring:
324962306a36Sopenharmony_ci	srpt_free_srq(sdev);
325062306a36Sopenharmony_ci	ib_dealloc_pd(sdev->pd);
325162306a36Sopenharmony_cifree_dev:
325262306a36Sopenharmony_ci	srpt_sdev_put(sdev);
325362306a36Sopenharmony_ci	pr_info("%s(%s) failed.\n", __func__, dev_name(&device->dev));
325462306a36Sopenharmony_ci	return ret;
325562306a36Sopenharmony_ci}
325662306a36Sopenharmony_ci
325762306a36Sopenharmony_ci/**
325862306a36Sopenharmony_ci * srpt_remove_one - InfiniBand device removal callback function
325962306a36Sopenharmony_ci * @device: Describes a HCA.
326062306a36Sopenharmony_ci * @client_data: The value passed as the third argument to ib_set_client_data().
326162306a36Sopenharmony_ci */
326262306a36Sopenharmony_cistatic void srpt_remove_one(struct ib_device *device, void *client_data)
326362306a36Sopenharmony_ci{
326462306a36Sopenharmony_ci	struct srpt_device *sdev = client_data;
326562306a36Sopenharmony_ci	int i;
326662306a36Sopenharmony_ci
326762306a36Sopenharmony_ci	srpt_unregister_mad_agent(sdev, sdev->device->phys_port_cnt);
326862306a36Sopenharmony_ci
326962306a36Sopenharmony_ci	ib_unregister_event_handler(&sdev->event_handler);
327062306a36Sopenharmony_ci
327162306a36Sopenharmony_ci	/* Cancel any work queued by the just unregistered IB event handler. */
327262306a36Sopenharmony_ci	for (i = 0; i < sdev->device->phys_port_cnt; i++)
327362306a36Sopenharmony_ci		cancel_work_sync(&sdev->port[i].work);
327462306a36Sopenharmony_ci
327562306a36Sopenharmony_ci	if (sdev->cm_id)
327662306a36Sopenharmony_ci		ib_destroy_cm_id(sdev->cm_id);
327762306a36Sopenharmony_ci
327862306a36Sopenharmony_ci	ib_set_client_data(device, &srpt_client, NULL);
327962306a36Sopenharmony_ci
328062306a36Sopenharmony_ci	/*
328162306a36Sopenharmony_ci	 * Unregistering a target must happen after destroying sdev->cm_id
328262306a36Sopenharmony_ci	 * such that no new SRP_LOGIN_REQ information units can arrive while
328362306a36Sopenharmony_ci	 * destroying the target.
328462306a36Sopenharmony_ci	 */
328562306a36Sopenharmony_ci	spin_lock(&srpt_dev_lock);
328662306a36Sopenharmony_ci	list_del(&sdev->list);
328762306a36Sopenharmony_ci	spin_unlock(&srpt_dev_lock);
328862306a36Sopenharmony_ci
328962306a36Sopenharmony_ci	for (i = 0; i < sdev->device->phys_port_cnt; i++)
329062306a36Sopenharmony_ci		srpt_release_sport(&sdev->port[i]);
329162306a36Sopenharmony_ci
329262306a36Sopenharmony_ci	srpt_free_srq(sdev);
329362306a36Sopenharmony_ci
329462306a36Sopenharmony_ci	ib_dealloc_pd(sdev->pd);
329562306a36Sopenharmony_ci
329662306a36Sopenharmony_ci	srpt_sdev_put(sdev);
329762306a36Sopenharmony_ci}
329862306a36Sopenharmony_ci
329962306a36Sopenharmony_cistatic struct ib_client srpt_client = {
330062306a36Sopenharmony_ci	.name = DRV_NAME,
330162306a36Sopenharmony_ci	.add = srpt_add_one,
330262306a36Sopenharmony_ci	.remove = srpt_remove_one
330362306a36Sopenharmony_ci};
330462306a36Sopenharmony_ci
330562306a36Sopenharmony_cistatic int srpt_check_true(struct se_portal_group *se_tpg)
330662306a36Sopenharmony_ci{
330762306a36Sopenharmony_ci	return 1;
330862306a36Sopenharmony_ci}
330962306a36Sopenharmony_ci
331062306a36Sopenharmony_cistatic struct srpt_port *srpt_tpg_to_sport(struct se_portal_group *tpg)
331162306a36Sopenharmony_ci{
331262306a36Sopenharmony_ci	return tpg->se_tpg_wwn->priv;
331362306a36Sopenharmony_ci}
331462306a36Sopenharmony_ci
331562306a36Sopenharmony_cistatic struct srpt_port_id *srpt_wwn_to_sport_id(struct se_wwn *wwn)
331662306a36Sopenharmony_ci{
331762306a36Sopenharmony_ci	struct srpt_port *sport = wwn->priv;
331862306a36Sopenharmony_ci
331962306a36Sopenharmony_ci	if (sport->guid_id && &sport->guid_id->wwn == wwn)
332062306a36Sopenharmony_ci		return sport->guid_id;
332162306a36Sopenharmony_ci	if (sport->gid_id && &sport->gid_id->wwn == wwn)
332262306a36Sopenharmony_ci		return sport->gid_id;
332362306a36Sopenharmony_ci	WARN_ON_ONCE(true);
332462306a36Sopenharmony_ci	return NULL;
332562306a36Sopenharmony_ci}
332662306a36Sopenharmony_ci
332762306a36Sopenharmony_cistatic char *srpt_get_fabric_wwn(struct se_portal_group *tpg)
332862306a36Sopenharmony_ci{
332962306a36Sopenharmony_ci	struct srpt_tpg *stpg = container_of(tpg, typeof(*stpg), tpg);
333062306a36Sopenharmony_ci
333162306a36Sopenharmony_ci	return stpg->sport_id->name;
333262306a36Sopenharmony_ci}
333362306a36Sopenharmony_ci
333462306a36Sopenharmony_cistatic u16 srpt_get_tag(struct se_portal_group *tpg)
333562306a36Sopenharmony_ci{
333662306a36Sopenharmony_ci	return 1;
333762306a36Sopenharmony_ci}
333862306a36Sopenharmony_ci
333962306a36Sopenharmony_cistatic void srpt_release_cmd(struct se_cmd *se_cmd)
334062306a36Sopenharmony_ci{
334162306a36Sopenharmony_ci	struct srpt_send_ioctx *ioctx = container_of(se_cmd,
334262306a36Sopenharmony_ci				struct srpt_send_ioctx, cmd);
334362306a36Sopenharmony_ci	struct srpt_rdma_ch *ch = ioctx->ch;
334462306a36Sopenharmony_ci	struct srpt_recv_ioctx *recv_ioctx = ioctx->recv_ioctx;
334562306a36Sopenharmony_ci
334662306a36Sopenharmony_ci	WARN_ON_ONCE(ioctx->state != SRPT_STATE_DONE &&
334762306a36Sopenharmony_ci		     !(ioctx->cmd.transport_state & CMD_T_ABORTED));
334862306a36Sopenharmony_ci
334962306a36Sopenharmony_ci	if (recv_ioctx) {
335062306a36Sopenharmony_ci		WARN_ON_ONCE(!list_empty(&recv_ioctx->wait_list));
335162306a36Sopenharmony_ci		ioctx->recv_ioctx = NULL;
335262306a36Sopenharmony_ci		srpt_post_recv(ch->sport->sdev, ch, recv_ioctx);
335362306a36Sopenharmony_ci	}
335462306a36Sopenharmony_ci
335562306a36Sopenharmony_ci	if (ioctx->n_rw_ctx) {
335662306a36Sopenharmony_ci		srpt_free_rw_ctxs(ch, ioctx);
335762306a36Sopenharmony_ci		ioctx->n_rw_ctx = 0;
335862306a36Sopenharmony_ci	}
335962306a36Sopenharmony_ci
336062306a36Sopenharmony_ci	target_free_tag(se_cmd->se_sess, se_cmd);
336162306a36Sopenharmony_ci}
336262306a36Sopenharmony_ci
336362306a36Sopenharmony_ci/**
336462306a36Sopenharmony_ci * srpt_close_session - forcibly close a session
336562306a36Sopenharmony_ci * @se_sess: SCSI target session.
336662306a36Sopenharmony_ci *
336762306a36Sopenharmony_ci * Callback function invoked by the TCM core to clean up sessions associated
336862306a36Sopenharmony_ci * with a node ACL when the user invokes
336962306a36Sopenharmony_ci * rmdir /sys/kernel/config/target/$driver/$port/$tpg/acls/$i_port_id
337062306a36Sopenharmony_ci */
337162306a36Sopenharmony_cistatic void srpt_close_session(struct se_session *se_sess)
337262306a36Sopenharmony_ci{
337362306a36Sopenharmony_ci	struct srpt_rdma_ch *ch = se_sess->fabric_sess_ptr;
337462306a36Sopenharmony_ci
337562306a36Sopenharmony_ci	srpt_disconnect_ch_sync(ch);
337662306a36Sopenharmony_ci}
337762306a36Sopenharmony_ci
337862306a36Sopenharmony_ci/* Note: only used from inside debug printk's by the TCM core. */
337962306a36Sopenharmony_cistatic int srpt_get_tcm_cmd_state(struct se_cmd *se_cmd)
338062306a36Sopenharmony_ci{
338162306a36Sopenharmony_ci	struct srpt_send_ioctx *ioctx;
338262306a36Sopenharmony_ci
338362306a36Sopenharmony_ci	ioctx = container_of(se_cmd, struct srpt_send_ioctx, cmd);
338462306a36Sopenharmony_ci	return ioctx->state;
338562306a36Sopenharmony_ci}
338662306a36Sopenharmony_ci
338762306a36Sopenharmony_cistatic int srpt_parse_guid(u64 *guid, const char *name)
338862306a36Sopenharmony_ci{
338962306a36Sopenharmony_ci	u16 w[4];
339062306a36Sopenharmony_ci	int ret = -EINVAL;
339162306a36Sopenharmony_ci
339262306a36Sopenharmony_ci	if (sscanf(name, "%hx:%hx:%hx:%hx", &w[0], &w[1], &w[2], &w[3]) != 4)
339362306a36Sopenharmony_ci		goto out;
339462306a36Sopenharmony_ci	*guid = get_unaligned_be64(w);
339562306a36Sopenharmony_ci	ret = 0;
339662306a36Sopenharmony_ciout:
339762306a36Sopenharmony_ci	return ret;
339862306a36Sopenharmony_ci}
339962306a36Sopenharmony_ci
340062306a36Sopenharmony_ci/**
340162306a36Sopenharmony_ci * srpt_parse_i_port_id - parse an initiator port ID
340262306a36Sopenharmony_ci * @name: ASCII representation of a 128-bit initiator port ID.
340362306a36Sopenharmony_ci * @i_port_id: Binary 128-bit port ID.
340462306a36Sopenharmony_ci */
340562306a36Sopenharmony_cistatic int srpt_parse_i_port_id(u8 i_port_id[16], const char *name)
340662306a36Sopenharmony_ci{
340762306a36Sopenharmony_ci	const char *p;
340862306a36Sopenharmony_ci	unsigned len, count, leading_zero_bytes;
340962306a36Sopenharmony_ci	int ret;
341062306a36Sopenharmony_ci
341162306a36Sopenharmony_ci	p = name;
341262306a36Sopenharmony_ci	if (strncasecmp(p, "0x", 2) == 0)
341362306a36Sopenharmony_ci		p += 2;
341462306a36Sopenharmony_ci	ret = -EINVAL;
341562306a36Sopenharmony_ci	len = strlen(p);
341662306a36Sopenharmony_ci	if (len % 2)
341762306a36Sopenharmony_ci		goto out;
341862306a36Sopenharmony_ci	count = min(len / 2, 16U);
341962306a36Sopenharmony_ci	leading_zero_bytes = 16 - count;
342062306a36Sopenharmony_ci	memset(i_port_id, 0, leading_zero_bytes);
342162306a36Sopenharmony_ci	ret = hex2bin(i_port_id + leading_zero_bytes, p, count);
342262306a36Sopenharmony_ci
342362306a36Sopenharmony_ciout:
342462306a36Sopenharmony_ci	return ret;
342562306a36Sopenharmony_ci}
342662306a36Sopenharmony_ci
342762306a36Sopenharmony_ci/*
342862306a36Sopenharmony_ci * configfs callback function invoked for mkdir
342962306a36Sopenharmony_ci * /sys/kernel/config/target/$driver/$port/$tpg/acls/$i_port_id
343062306a36Sopenharmony_ci *
343162306a36Sopenharmony_ci * i_port_id must be an initiator port GUID, GID or IP address. See also the
343262306a36Sopenharmony_ci * target_alloc_session() calls in this driver. Examples of valid initiator
343362306a36Sopenharmony_ci * port IDs:
343462306a36Sopenharmony_ci * 0x0000000000000000505400fffe4a0b7b
343562306a36Sopenharmony_ci * 0000000000000000505400fffe4a0b7b
343662306a36Sopenharmony_ci * 5054:00ff:fe4a:0b7b
343762306a36Sopenharmony_ci * 192.168.122.76
343862306a36Sopenharmony_ci */
343962306a36Sopenharmony_cistatic int srpt_init_nodeacl(struct se_node_acl *se_nacl, const char *name)
344062306a36Sopenharmony_ci{
344162306a36Sopenharmony_ci	struct sockaddr_storage sa;
344262306a36Sopenharmony_ci	u64 guid;
344362306a36Sopenharmony_ci	u8 i_port_id[16];
344462306a36Sopenharmony_ci	int ret;
344562306a36Sopenharmony_ci
344662306a36Sopenharmony_ci	ret = srpt_parse_guid(&guid, name);
344762306a36Sopenharmony_ci	if (ret < 0)
344862306a36Sopenharmony_ci		ret = srpt_parse_i_port_id(i_port_id, name);
344962306a36Sopenharmony_ci	if (ret < 0)
345062306a36Sopenharmony_ci		ret = inet_pton_with_scope(&init_net, AF_UNSPEC, name, NULL,
345162306a36Sopenharmony_ci					   &sa);
345262306a36Sopenharmony_ci	if (ret < 0)
345362306a36Sopenharmony_ci		pr_err("invalid initiator port ID %s\n", name);
345462306a36Sopenharmony_ci	return ret;
345562306a36Sopenharmony_ci}
345662306a36Sopenharmony_ci
345762306a36Sopenharmony_cistatic ssize_t srpt_tpg_attrib_srp_max_rdma_size_show(struct config_item *item,
345862306a36Sopenharmony_ci		char *page)
345962306a36Sopenharmony_ci{
346062306a36Sopenharmony_ci	struct se_portal_group *se_tpg = attrib_to_tpg(item);
346162306a36Sopenharmony_ci	struct srpt_port *sport = srpt_tpg_to_sport(se_tpg);
346262306a36Sopenharmony_ci
346362306a36Sopenharmony_ci	return sysfs_emit(page, "%u\n", sport->port_attrib.srp_max_rdma_size);
346462306a36Sopenharmony_ci}
346562306a36Sopenharmony_ci
346662306a36Sopenharmony_cistatic ssize_t srpt_tpg_attrib_srp_max_rdma_size_store(struct config_item *item,
346762306a36Sopenharmony_ci		const char *page, size_t count)
346862306a36Sopenharmony_ci{
346962306a36Sopenharmony_ci	struct se_portal_group *se_tpg = attrib_to_tpg(item);
347062306a36Sopenharmony_ci	struct srpt_port *sport = srpt_tpg_to_sport(se_tpg);
347162306a36Sopenharmony_ci	unsigned long val;
347262306a36Sopenharmony_ci	int ret;
347362306a36Sopenharmony_ci
347462306a36Sopenharmony_ci	ret = kstrtoul(page, 0, &val);
347562306a36Sopenharmony_ci	if (ret < 0) {
347662306a36Sopenharmony_ci		pr_err("kstrtoul() failed with ret: %d\n", ret);
347762306a36Sopenharmony_ci		return -EINVAL;
347862306a36Sopenharmony_ci	}
347962306a36Sopenharmony_ci	if (val > MAX_SRPT_RDMA_SIZE) {
348062306a36Sopenharmony_ci		pr_err("val: %lu exceeds MAX_SRPT_RDMA_SIZE: %d\n", val,
348162306a36Sopenharmony_ci			MAX_SRPT_RDMA_SIZE);
348262306a36Sopenharmony_ci		return -EINVAL;
348362306a36Sopenharmony_ci	}
348462306a36Sopenharmony_ci	if (val < DEFAULT_MAX_RDMA_SIZE) {
348562306a36Sopenharmony_ci		pr_err("val: %lu smaller than DEFAULT_MAX_RDMA_SIZE: %d\n",
348662306a36Sopenharmony_ci			val, DEFAULT_MAX_RDMA_SIZE);
348762306a36Sopenharmony_ci		return -EINVAL;
348862306a36Sopenharmony_ci	}
348962306a36Sopenharmony_ci	sport->port_attrib.srp_max_rdma_size = val;
349062306a36Sopenharmony_ci
349162306a36Sopenharmony_ci	return count;
349262306a36Sopenharmony_ci}
349362306a36Sopenharmony_ci
349462306a36Sopenharmony_cistatic ssize_t srpt_tpg_attrib_srp_max_rsp_size_show(struct config_item *item,
349562306a36Sopenharmony_ci		char *page)
349662306a36Sopenharmony_ci{
349762306a36Sopenharmony_ci	struct se_portal_group *se_tpg = attrib_to_tpg(item);
349862306a36Sopenharmony_ci	struct srpt_port *sport = srpt_tpg_to_sport(se_tpg);
349962306a36Sopenharmony_ci
350062306a36Sopenharmony_ci	return sysfs_emit(page, "%u\n", sport->port_attrib.srp_max_rsp_size);
350162306a36Sopenharmony_ci}
350262306a36Sopenharmony_ci
350362306a36Sopenharmony_cistatic ssize_t srpt_tpg_attrib_srp_max_rsp_size_store(struct config_item *item,
350462306a36Sopenharmony_ci		const char *page, size_t count)
350562306a36Sopenharmony_ci{
350662306a36Sopenharmony_ci	struct se_portal_group *se_tpg = attrib_to_tpg(item);
350762306a36Sopenharmony_ci	struct srpt_port *sport = srpt_tpg_to_sport(se_tpg);
350862306a36Sopenharmony_ci	unsigned long val;
350962306a36Sopenharmony_ci	int ret;
351062306a36Sopenharmony_ci
351162306a36Sopenharmony_ci	ret = kstrtoul(page, 0, &val);
351262306a36Sopenharmony_ci	if (ret < 0) {
351362306a36Sopenharmony_ci		pr_err("kstrtoul() failed with ret: %d\n", ret);
351462306a36Sopenharmony_ci		return -EINVAL;
351562306a36Sopenharmony_ci	}
351662306a36Sopenharmony_ci	if (val > MAX_SRPT_RSP_SIZE) {
351762306a36Sopenharmony_ci		pr_err("val: %lu exceeds MAX_SRPT_RSP_SIZE: %d\n", val,
351862306a36Sopenharmony_ci			MAX_SRPT_RSP_SIZE);
351962306a36Sopenharmony_ci		return -EINVAL;
352062306a36Sopenharmony_ci	}
352162306a36Sopenharmony_ci	if (val < MIN_MAX_RSP_SIZE) {
352262306a36Sopenharmony_ci		pr_err("val: %lu smaller than MIN_MAX_RSP_SIZE: %d\n", val,
352362306a36Sopenharmony_ci			MIN_MAX_RSP_SIZE);
352462306a36Sopenharmony_ci		return -EINVAL;
352562306a36Sopenharmony_ci	}
352662306a36Sopenharmony_ci	sport->port_attrib.srp_max_rsp_size = val;
352762306a36Sopenharmony_ci
352862306a36Sopenharmony_ci	return count;
352962306a36Sopenharmony_ci}
353062306a36Sopenharmony_ci
353162306a36Sopenharmony_cistatic ssize_t srpt_tpg_attrib_srp_sq_size_show(struct config_item *item,
353262306a36Sopenharmony_ci		char *page)
353362306a36Sopenharmony_ci{
353462306a36Sopenharmony_ci	struct se_portal_group *se_tpg = attrib_to_tpg(item);
353562306a36Sopenharmony_ci	struct srpt_port *sport = srpt_tpg_to_sport(se_tpg);
353662306a36Sopenharmony_ci
353762306a36Sopenharmony_ci	return sysfs_emit(page, "%u\n", sport->port_attrib.srp_sq_size);
353862306a36Sopenharmony_ci}
353962306a36Sopenharmony_ci
354062306a36Sopenharmony_cistatic ssize_t srpt_tpg_attrib_srp_sq_size_store(struct config_item *item,
354162306a36Sopenharmony_ci		const char *page, size_t count)
354262306a36Sopenharmony_ci{
354362306a36Sopenharmony_ci	struct se_portal_group *se_tpg = attrib_to_tpg(item);
354462306a36Sopenharmony_ci	struct srpt_port *sport = srpt_tpg_to_sport(se_tpg);
354562306a36Sopenharmony_ci	unsigned long val;
354662306a36Sopenharmony_ci	int ret;
354762306a36Sopenharmony_ci
354862306a36Sopenharmony_ci	ret = kstrtoul(page, 0, &val);
354962306a36Sopenharmony_ci	if (ret < 0) {
355062306a36Sopenharmony_ci		pr_err("kstrtoul() failed with ret: %d\n", ret);
355162306a36Sopenharmony_ci		return -EINVAL;
355262306a36Sopenharmony_ci	}
355362306a36Sopenharmony_ci	if (val > MAX_SRPT_SRQ_SIZE) {
355462306a36Sopenharmony_ci		pr_err("val: %lu exceeds MAX_SRPT_SRQ_SIZE: %d\n", val,
355562306a36Sopenharmony_ci			MAX_SRPT_SRQ_SIZE);
355662306a36Sopenharmony_ci		return -EINVAL;
355762306a36Sopenharmony_ci	}
355862306a36Sopenharmony_ci	if (val < MIN_SRPT_SRQ_SIZE) {
355962306a36Sopenharmony_ci		pr_err("val: %lu smaller than MIN_SRPT_SRQ_SIZE: %d\n", val,
356062306a36Sopenharmony_ci			MIN_SRPT_SRQ_SIZE);
356162306a36Sopenharmony_ci		return -EINVAL;
356262306a36Sopenharmony_ci	}
356362306a36Sopenharmony_ci	sport->port_attrib.srp_sq_size = val;
356462306a36Sopenharmony_ci
356562306a36Sopenharmony_ci	return count;
356662306a36Sopenharmony_ci}
356762306a36Sopenharmony_ci
356862306a36Sopenharmony_cistatic ssize_t srpt_tpg_attrib_use_srq_show(struct config_item *item,
356962306a36Sopenharmony_ci					    char *page)
357062306a36Sopenharmony_ci{
357162306a36Sopenharmony_ci	struct se_portal_group *se_tpg = attrib_to_tpg(item);
357262306a36Sopenharmony_ci	struct srpt_port *sport = srpt_tpg_to_sport(se_tpg);
357362306a36Sopenharmony_ci
357462306a36Sopenharmony_ci	return sysfs_emit(page, "%d\n", sport->port_attrib.use_srq);
357562306a36Sopenharmony_ci}
357662306a36Sopenharmony_ci
357762306a36Sopenharmony_cistatic ssize_t srpt_tpg_attrib_use_srq_store(struct config_item *item,
357862306a36Sopenharmony_ci					     const char *page, size_t count)
357962306a36Sopenharmony_ci{
358062306a36Sopenharmony_ci	struct se_portal_group *se_tpg = attrib_to_tpg(item);
358162306a36Sopenharmony_ci	struct srpt_port *sport = srpt_tpg_to_sport(se_tpg);
358262306a36Sopenharmony_ci	struct srpt_device *sdev = sport->sdev;
358362306a36Sopenharmony_ci	unsigned long val;
358462306a36Sopenharmony_ci	bool enabled;
358562306a36Sopenharmony_ci	int ret;
358662306a36Sopenharmony_ci
358762306a36Sopenharmony_ci	ret = kstrtoul(page, 0, &val);
358862306a36Sopenharmony_ci	if (ret < 0)
358962306a36Sopenharmony_ci		return ret;
359062306a36Sopenharmony_ci	if (val != !!val)
359162306a36Sopenharmony_ci		return -EINVAL;
359262306a36Sopenharmony_ci
359362306a36Sopenharmony_ci	ret = mutex_lock_interruptible(&sdev->sdev_mutex);
359462306a36Sopenharmony_ci	if (ret < 0)
359562306a36Sopenharmony_ci		return ret;
359662306a36Sopenharmony_ci	ret = mutex_lock_interruptible(&sport->mutex);
359762306a36Sopenharmony_ci	if (ret < 0)
359862306a36Sopenharmony_ci		goto unlock_sdev;
359962306a36Sopenharmony_ci	enabled = sport->enabled;
360062306a36Sopenharmony_ci	/* Log out all initiator systems before changing 'use_srq'. */
360162306a36Sopenharmony_ci	srpt_set_enabled(sport, false);
360262306a36Sopenharmony_ci	sport->port_attrib.use_srq = val;
360362306a36Sopenharmony_ci	srpt_use_srq(sdev, sport->port_attrib.use_srq);
360462306a36Sopenharmony_ci	srpt_set_enabled(sport, enabled);
360562306a36Sopenharmony_ci	ret = count;
360662306a36Sopenharmony_ci	mutex_unlock(&sport->mutex);
360762306a36Sopenharmony_ciunlock_sdev:
360862306a36Sopenharmony_ci	mutex_unlock(&sdev->sdev_mutex);
360962306a36Sopenharmony_ci
361062306a36Sopenharmony_ci	return ret;
361162306a36Sopenharmony_ci}
361262306a36Sopenharmony_ci
361362306a36Sopenharmony_ciCONFIGFS_ATTR(srpt_tpg_attrib_,  srp_max_rdma_size);
361462306a36Sopenharmony_ciCONFIGFS_ATTR(srpt_tpg_attrib_,  srp_max_rsp_size);
361562306a36Sopenharmony_ciCONFIGFS_ATTR(srpt_tpg_attrib_,  srp_sq_size);
361662306a36Sopenharmony_ciCONFIGFS_ATTR(srpt_tpg_attrib_,  use_srq);
361762306a36Sopenharmony_ci
361862306a36Sopenharmony_cistatic struct configfs_attribute *srpt_tpg_attrib_attrs[] = {
361962306a36Sopenharmony_ci	&srpt_tpg_attrib_attr_srp_max_rdma_size,
362062306a36Sopenharmony_ci	&srpt_tpg_attrib_attr_srp_max_rsp_size,
362162306a36Sopenharmony_ci	&srpt_tpg_attrib_attr_srp_sq_size,
362262306a36Sopenharmony_ci	&srpt_tpg_attrib_attr_use_srq,
362362306a36Sopenharmony_ci	NULL,
362462306a36Sopenharmony_ci};
362562306a36Sopenharmony_ci
362662306a36Sopenharmony_cistatic struct rdma_cm_id *srpt_create_rdma_id(struct sockaddr *listen_addr)
362762306a36Sopenharmony_ci{
362862306a36Sopenharmony_ci	struct rdma_cm_id *rdma_cm_id;
362962306a36Sopenharmony_ci	int ret;
363062306a36Sopenharmony_ci
363162306a36Sopenharmony_ci	rdma_cm_id = rdma_create_id(&init_net, srpt_rdma_cm_handler,
363262306a36Sopenharmony_ci				    NULL, RDMA_PS_TCP, IB_QPT_RC);
363362306a36Sopenharmony_ci	if (IS_ERR(rdma_cm_id)) {
363462306a36Sopenharmony_ci		pr_err("RDMA/CM ID creation failed: %ld\n",
363562306a36Sopenharmony_ci		       PTR_ERR(rdma_cm_id));
363662306a36Sopenharmony_ci		goto out;
363762306a36Sopenharmony_ci	}
363862306a36Sopenharmony_ci
363962306a36Sopenharmony_ci	ret = rdma_bind_addr(rdma_cm_id, listen_addr);
364062306a36Sopenharmony_ci	if (ret) {
364162306a36Sopenharmony_ci		char addr_str[64];
364262306a36Sopenharmony_ci
364362306a36Sopenharmony_ci		snprintf(addr_str, sizeof(addr_str), "%pISp", listen_addr);
364462306a36Sopenharmony_ci		pr_err("Binding RDMA/CM ID to address %s failed: %d\n",
364562306a36Sopenharmony_ci		       addr_str, ret);
364662306a36Sopenharmony_ci		rdma_destroy_id(rdma_cm_id);
364762306a36Sopenharmony_ci		rdma_cm_id = ERR_PTR(ret);
364862306a36Sopenharmony_ci		goto out;
364962306a36Sopenharmony_ci	}
365062306a36Sopenharmony_ci
365162306a36Sopenharmony_ci	ret = rdma_listen(rdma_cm_id, 128);
365262306a36Sopenharmony_ci	if (ret) {
365362306a36Sopenharmony_ci		pr_err("rdma_listen() failed: %d\n", ret);
365462306a36Sopenharmony_ci		rdma_destroy_id(rdma_cm_id);
365562306a36Sopenharmony_ci		rdma_cm_id = ERR_PTR(ret);
365662306a36Sopenharmony_ci	}
365762306a36Sopenharmony_ci
365862306a36Sopenharmony_ciout:
365962306a36Sopenharmony_ci	return rdma_cm_id;
366062306a36Sopenharmony_ci}
366162306a36Sopenharmony_ci
366262306a36Sopenharmony_cistatic ssize_t srpt_rdma_cm_port_show(struct config_item *item, char *page)
366362306a36Sopenharmony_ci{
366462306a36Sopenharmony_ci	return sysfs_emit(page, "%d\n", rdma_cm_port);
366562306a36Sopenharmony_ci}
366662306a36Sopenharmony_ci
366762306a36Sopenharmony_cistatic ssize_t srpt_rdma_cm_port_store(struct config_item *item,
366862306a36Sopenharmony_ci				       const char *page, size_t count)
366962306a36Sopenharmony_ci{
367062306a36Sopenharmony_ci	struct sockaddr_in  addr4 = { .sin_family  = AF_INET  };
367162306a36Sopenharmony_ci	struct sockaddr_in6 addr6 = { .sin6_family = AF_INET6 };
367262306a36Sopenharmony_ci	struct rdma_cm_id *new_id = NULL;
367362306a36Sopenharmony_ci	u16 val;
367462306a36Sopenharmony_ci	int ret;
367562306a36Sopenharmony_ci
367662306a36Sopenharmony_ci	ret = kstrtou16(page, 0, &val);
367762306a36Sopenharmony_ci	if (ret < 0)
367862306a36Sopenharmony_ci		return ret;
367962306a36Sopenharmony_ci	ret = count;
368062306a36Sopenharmony_ci	if (rdma_cm_port == val)
368162306a36Sopenharmony_ci		goto out;
368262306a36Sopenharmony_ci
368362306a36Sopenharmony_ci	if (val) {
368462306a36Sopenharmony_ci		addr6.sin6_port = cpu_to_be16(val);
368562306a36Sopenharmony_ci		new_id = srpt_create_rdma_id((struct sockaddr *)&addr6);
368662306a36Sopenharmony_ci		if (IS_ERR(new_id)) {
368762306a36Sopenharmony_ci			addr4.sin_port = cpu_to_be16(val);
368862306a36Sopenharmony_ci			new_id = srpt_create_rdma_id((struct sockaddr *)&addr4);
368962306a36Sopenharmony_ci			if (IS_ERR(new_id)) {
369062306a36Sopenharmony_ci				ret = PTR_ERR(new_id);
369162306a36Sopenharmony_ci				goto out;
369262306a36Sopenharmony_ci			}
369362306a36Sopenharmony_ci		}
369462306a36Sopenharmony_ci	}
369562306a36Sopenharmony_ci
369662306a36Sopenharmony_ci	mutex_lock(&rdma_cm_mutex);
369762306a36Sopenharmony_ci	rdma_cm_port = val;
369862306a36Sopenharmony_ci	swap(rdma_cm_id, new_id);
369962306a36Sopenharmony_ci	mutex_unlock(&rdma_cm_mutex);
370062306a36Sopenharmony_ci
370162306a36Sopenharmony_ci	if (new_id)
370262306a36Sopenharmony_ci		rdma_destroy_id(new_id);
370362306a36Sopenharmony_ci	ret = count;
370462306a36Sopenharmony_ciout:
370562306a36Sopenharmony_ci	return ret;
370662306a36Sopenharmony_ci}
370762306a36Sopenharmony_ci
370862306a36Sopenharmony_ciCONFIGFS_ATTR(srpt_, rdma_cm_port);
370962306a36Sopenharmony_ci
371062306a36Sopenharmony_cistatic struct configfs_attribute *srpt_da_attrs[] = {
371162306a36Sopenharmony_ci	&srpt_attr_rdma_cm_port,
371262306a36Sopenharmony_ci	NULL,
371362306a36Sopenharmony_ci};
371462306a36Sopenharmony_ci
371562306a36Sopenharmony_cistatic int srpt_enable_tpg(struct se_portal_group *se_tpg, bool enable)
371662306a36Sopenharmony_ci{
371762306a36Sopenharmony_ci	struct srpt_port *sport = srpt_tpg_to_sport(se_tpg);
371862306a36Sopenharmony_ci
371962306a36Sopenharmony_ci	mutex_lock(&sport->mutex);
372062306a36Sopenharmony_ci	srpt_set_enabled(sport, enable);
372162306a36Sopenharmony_ci	mutex_unlock(&sport->mutex);
372262306a36Sopenharmony_ci
372362306a36Sopenharmony_ci	return 0;
372462306a36Sopenharmony_ci}
372562306a36Sopenharmony_ci
372662306a36Sopenharmony_ci/**
372762306a36Sopenharmony_ci * srpt_make_tpg - configfs callback invoked for mkdir /sys/kernel/config/target/$driver/$port/$tpg
372862306a36Sopenharmony_ci * @wwn: Corresponds to $driver/$port.
372962306a36Sopenharmony_ci * @name: $tpg.
373062306a36Sopenharmony_ci */
373162306a36Sopenharmony_cistatic struct se_portal_group *srpt_make_tpg(struct se_wwn *wwn,
373262306a36Sopenharmony_ci					     const char *name)
373362306a36Sopenharmony_ci{
373462306a36Sopenharmony_ci	struct srpt_port_id *sport_id = srpt_wwn_to_sport_id(wwn);
373562306a36Sopenharmony_ci	struct srpt_tpg *stpg;
373662306a36Sopenharmony_ci	int res = -ENOMEM;
373762306a36Sopenharmony_ci
373862306a36Sopenharmony_ci	stpg = kzalloc(sizeof(*stpg), GFP_KERNEL);
373962306a36Sopenharmony_ci	if (!stpg)
374062306a36Sopenharmony_ci		return ERR_PTR(res);
374162306a36Sopenharmony_ci	stpg->sport_id = sport_id;
374262306a36Sopenharmony_ci	res = core_tpg_register(wwn, &stpg->tpg, SCSI_PROTOCOL_SRP);
374362306a36Sopenharmony_ci	if (res) {
374462306a36Sopenharmony_ci		kfree(stpg);
374562306a36Sopenharmony_ci		return ERR_PTR(res);
374662306a36Sopenharmony_ci	}
374762306a36Sopenharmony_ci
374862306a36Sopenharmony_ci	mutex_lock(&sport_id->mutex);
374962306a36Sopenharmony_ci	list_add_tail(&stpg->entry, &sport_id->tpg_list);
375062306a36Sopenharmony_ci	mutex_unlock(&sport_id->mutex);
375162306a36Sopenharmony_ci
375262306a36Sopenharmony_ci	return &stpg->tpg;
375362306a36Sopenharmony_ci}
375462306a36Sopenharmony_ci
375562306a36Sopenharmony_ci/**
375662306a36Sopenharmony_ci * srpt_drop_tpg - configfs callback invoked for rmdir /sys/kernel/config/target/$driver/$port/$tpg
375762306a36Sopenharmony_ci * @tpg: Target portal group to deregister.
375862306a36Sopenharmony_ci */
375962306a36Sopenharmony_cistatic void srpt_drop_tpg(struct se_portal_group *tpg)
376062306a36Sopenharmony_ci{
376162306a36Sopenharmony_ci	struct srpt_tpg *stpg = container_of(tpg, typeof(*stpg), tpg);
376262306a36Sopenharmony_ci	struct srpt_port_id *sport_id = stpg->sport_id;
376362306a36Sopenharmony_ci	struct srpt_port *sport = srpt_tpg_to_sport(tpg);
376462306a36Sopenharmony_ci
376562306a36Sopenharmony_ci	mutex_lock(&sport_id->mutex);
376662306a36Sopenharmony_ci	list_del(&stpg->entry);
376762306a36Sopenharmony_ci	mutex_unlock(&sport_id->mutex);
376862306a36Sopenharmony_ci
376962306a36Sopenharmony_ci	sport->enabled = false;
377062306a36Sopenharmony_ci	core_tpg_deregister(tpg);
377162306a36Sopenharmony_ci	kfree(stpg);
377262306a36Sopenharmony_ci}
377362306a36Sopenharmony_ci
377462306a36Sopenharmony_ci/**
377562306a36Sopenharmony_ci * srpt_make_tport - configfs callback invoked for mkdir /sys/kernel/config/target/$driver/$port
377662306a36Sopenharmony_ci * @tf: Not used.
377762306a36Sopenharmony_ci * @group: Not used.
377862306a36Sopenharmony_ci * @name: $port.
377962306a36Sopenharmony_ci */
378062306a36Sopenharmony_cistatic struct se_wwn *srpt_make_tport(struct target_fabric_configfs *tf,
378162306a36Sopenharmony_ci				      struct config_group *group,
378262306a36Sopenharmony_ci				      const char *name)
378362306a36Sopenharmony_ci{
378462306a36Sopenharmony_ci	struct port_and_port_id papi = srpt_lookup_port(name);
378562306a36Sopenharmony_ci	struct srpt_port *sport = papi.sport;
378662306a36Sopenharmony_ci	struct srpt_port_id *port_id;
378762306a36Sopenharmony_ci
378862306a36Sopenharmony_ci	if (!papi.port_id)
378962306a36Sopenharmony_ci		return ERR_PTR(-EINVAL);
379062306a36Sopenharmony_ci	if (*papi.port_id) {
379162306a36Sopenharmony_ci		/* Attempt to create a directory that already exists. */
379262306a36Sopenharmony_ci		WARN_ON_ONCE(true);
379362306a36Sopenharmony_ci		return &(*papi.port_id)->wwn;
379462306a36Sopenharmony_ci	}
379562306a36Sopenharmony_ci	port_id = kzalloc(sizeof(*port_id), GFP_KERNEL);
379662306a36Sopenharmony_ci	if (!port_id) {
379762306a36Sopenharmony_ci		srpt_sdev_put(sport->sdev);
379862306a36Sopenharmony_ci		return ERR_PTR(-ENOMEM);
379962306a36Sopenharmony_ci	}
380062306a36Sopenharmony_ci	mutex_init(&port_id->mutex);
380162306a36Sopenharmony_ci	INIT_LIST_HEAD(&port_id->tpg_list);
380262306a36Sopenharmony_ci	port_id->wwn.priv = sport;
380362306a36Sopenharmony_ci	memcpy(port_id->name, port_id == sport->guid_id ? sport->guid_name :
380462306a36Sopenharmony_ci	       sport->gid_name, ARRAY_SIZE(port_id->name));
380562306a36Sopenharmony_ci
380662306a36Sopenharmony_ci	*papi.port_id = port_id;
380762306a36Sopenharmony_ci
380862306a36Sopenharmony_ci	return &port_id->wwn;
380962306a36Sopenharmony_ci}
381062306a36Sopenharmony_ci
381162306a36Sopenharmony_ci/**
381262306a36Sopenharmony_ci * srpt_drop_tport - configfs callback invoked for rmdir /sys/kernel/config/target/$driver/$port
381362306a36Sopenharmony_ci * @wwn: $port.
381462306a36Sopenharmony_ci */
381562306a36Sopenharmony_cistatic void srpt_drop_tport(struct se_wwn *wwn)
381662306a36Sopenharmony_ci{
381762306a36Sopenharmony_ci	struct srpt_port_id *port_id = container_of(wwn, typeof(*port_id), wwn);
381862306a36Sopenharmony_ci	struct srpt_port *sport = wwn->priv;
381962306a36Sopenharmony_ci
382062306a36Sopenharmony_ci	if (sport->guid_id == port_id)
382162306a36Sopenharmony_ci		sport->guid_id = NULL;
382262306a36Sopenharmony_ci	else if (sport->gid_id == port_id)
382362306a36Sopenharmony_ci		sport->gid_id = NULL;
382462306a36Sopenharmony_ci	else
382562306a36Sopenharmony_ci		WARN_ON_ONCE(true);
382662306a36Sopenharmony_ci
382762306a36Sopenharmony_ci	srpt_sdev_put(sport->sdev);
382862306a36Sopenharmony_ci	kfree(port_id);
382962306a36Sopenharmony_ci}
383062306a36Sopenharmony_ci
383162306a36Sopenharmony_cistatic ssize_t srpt_wwn_version_show(struct config_item *item, char *buf)
383262306a36Sopenharmony_ci{
383362306a36Sopenharmony_ci	return sysfs_emit(buf, "\n");
383462306a36Sopenharmony_ci}
383562306a36Sopenharmony_ci
383662306a36Sopenharmony_ciCONFIGFS_ATTR_RO(srpt_wwn_, version);
383762306a36Sopenharmony_ci
383862306a36Sopenharmony_cistatic struct configfs_attribute *srpt_wwn_attrs[] = {
383962306a36Sopenharmony_ci	&srpt_wwn_attr_version,
384062306a36Sopenharmony_ci	NULL,
384162306a36Sopenharmony_ci};
384262306a36Sopenharmony_ci
384362306a36Sopenharmony_cistatic const struct target_core_fabric_ops srpt_template = {
384462306a36Sopenharmony_ci	.module				= THIS_MODULE,
384562306a36Sopenharmony_ci	.fabric_name			= "srpt",
384662306a36Sopenharmony_ci	.tpg_get_wwn			= srpt_get_fabric_wwn,
384762306a36Sopenharmony_ci	.tpg_get_tag			= srpt_get_tag,
384862306a36Sopenharmony_ci	.tpg_check_demo_mode_cache	= srpt_check_true,
384962306a36Sopenharmony_ci	.tpg_check_demo_mode_write_protect = srpt_check_true,
385062306a36Sopenharmony_ci	.release_cmd			= srpt_release_cmd,
385162306a36Sopenharmony_ci	.check_stop_free		= srpt_check_stop_free,
385262306a36Sopenharmony_ci	.close_session			= srpt_close_session,
385362306a36Sopenharmony_ci	.sess_get_initiator_sid		= NULL,
385462306a36Sopenharmony_ci	.write_pending			= srpt_write_pending,
385562306a36Sopenharmony_ci	.get_cmd_state			= srpt_get_tcm_cmd_state,
385662306a36Sopenharmony_ci	.queue_data_in			= srpt_queue_data_in,
385762306a36Sopenharmony_ci	.queue_status			= srpt_queue_status,
385862306a36Sopenharmony_ci	.queue_tm_rsp			= srpt_queue_tm_rsp,
385962306a36Sopenharmony_ci	.aborted_task			= srpt_aborted_task,
386062306a36Sopenharmony_ci	/*
386162306a36Sopenharmony_ci	 * Setup function pointers for generic logic in
386262306a36Sopenharmony_ci	 * target_core_fabric_configfs.c
386362306a36Sopenharmony_ci	 */
386462306a36Sopenharmony_ci	.fabric_make_wwn		= srpt_make_tport,
386562306a36Sopenharmony_ci	.fabric_drop_wwn		= srpt_drop_tport,
386662306a36Sopenharmony_ci	.fabric_make_tpg		= srpt_make_tpg,
386762306a36Sopenharmony_ci	.fabric_enable_tpg		= srpt_enable_tpg,
386862306a36Sopenharmony_ci	.fabric_drop_tpg		= srpt_drop_tpg,
386962306a36Sopenharmony_ci	.fabric_init_nodeacl		= srpt_init_nodeacl,
387062306a36Sopenharmony_ci
387162306a36Sopenharmony_ci	.tfc_discovery_attrs		= srpt_da_attrs,
387262306a36Sopenharmony_ci	.tfc_wwn_attrs			= srpt_wwn_attrs,
387362306a36Sopenharmony_ci	.tfc_tpg_attrib_attrs		= srpt_tpg_attrib_attrs,
387462306a36Sopenharmony_ci};
387562306a36Sopenharmony_ci
387662306a36Sopenharmony_ci/**
387762306a36Sopenharmony_ci * srpt_init_module - kernel module initialization
387862306a36Sopenharmony_ci *
387962306a36Sopenharmony_ci * Note: Since ib_register_client() registers callback functions, and since at
388062306a36Sopenharmony_ci * least one of these callback functions (srpt_add_one()) calls target core
388162306a36Sopenharmony_ci * functions, this driver must be registered with the target core before
388262306a36Sopenharmony_ci * ib_register_client() is called.
388362306a36Sopenharmony_ci */
388462306a36Sopenharmony_cistatic int __init srpt_init_module(void)
388562306a36Sopenharmony_ci{
388662306a36Sopenharmony_ci	int ret;
388762306a36Sopenharmony_ci
388862306a36Sopenharmony_ci	ret = -EINVAL;
388962306a36Sopenharmony_ci	if (srp_max_req_size < MIN_MAX_REQ_SIZE) {
389062306a36Sopenharmony_ci		pr_err("invalid value %d for kernel module parameter srp_max_req_size -- must be at least %d.\n",
389162306a36Sopenharmony_ci		       srp_max_req_size, MIN_MAX_REQ_SIZE);
389262306a36Sopenharmony_ci		goto out;
389362306a36Sopenharmony_ci	}
389462306a36Sopenharmony_ci
389562306a36Sopenharmony_ci	if (srpt_srq_size < MIN_SRPT_SRQ_SIZE
389662306a36Sopenharmony_ci	    || srpt_srq_size > MAX_SRPT_SRQ_SIZE) {
389762306a36Sopenharmony_ci		pr_err("invalid value %d for kernel module parameter srpt_srq_size -- must be in the range [%d..%d].\n",
389862306a36Sopenharmony_ci		       srpt_srq_size, MIN_SRPT_SRQ_SIZE, MAX_SRPT_SRQ_SIZE);
389962306a36Sopenharmony_ci		goto out;
390062306a36Sopenharmony_ci	}
390162306a36Sopenharmony_ci
390262306a36Sopenharmony_ci	ret = target_register_template(&srpt_template);
390362306a36Sopenharmony_ci	if (ret)
390462306a36Sopenharmony_ci		goto out;
390562306a36Sopenharmony_ci
390662306a36Sopenharmony_ci	ret = ib_register_client(&srpt_client);
390762306a36Sopenharmony_ci	if (ret) {
390862306a36Sopenharmony_ci		pr_err("couldn't register IB client\n");
390962306a36Sopenharmony_ci		goto out_unregister_target;
391062306a36Sopenharmony_ci	}
391162306a36Sopenharmony_ci
391262306a36Sopenharmony_ci	return 0;
391362306a36Sopenharmony_ci
391462306a36Sopenharmony_ciout_unregister_target:
391562306a36Sopenharmony_ci	target_unregister_template(&srpt_template);
391662306a36Sopenharmony_ciout:
391762306a36Sopenharmony_ci	return ret;
391862306a36Sopenharmony_ci}
391962306a36Sopenharmony_ci
392062306a36Sopenharmony_cistatic void __exit srpt_cleanup_module(void)
392162306a36Sopenharmony_ci{
392262306a36Sopenharmony_ci	if (rdma_cm_id)
392362306a36Sopenharmony_ci		rdma_destroy_id(rdma_cm_id);
392462306a36Sopenharmony_ci	ib_unregister_client(&srpt_client);
392562306a36Sopenharmony_ci	target_unregister_template(&srpt_template);
392662306a36Sopenharmony_ci}
392762306a36Sopenharmony_ci
392862306a36Sopenharmony_cimodule_init(srpt_init_module);
392962306a36Sopenharmony_cimodule_exit(srpt_cleanup_module);
3930