xref: /freebsd/lib/libnvmf/nvmf_transport.c (revision dd21556857e8d40f66bf5ad54754d9d52669ebf7)
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause
3  *
4  * Copyright (c) 2022-2024 Chelsio Communications, Inc.
5  * Written by: John Baldwin <jhb@FreeBSD.org>
6  */
7 
8 #include <sys/refcount.h>
9 #include <assert.h>
10 #include <errno.h>
11 #include <stdarg.h>
12 #include <stdio.h>
13 #include <stdlib.h>
14 #include <string.h>
15 
16 #include "libnvmf.h"
17 #include "internal.h"
18 
19 struct nvmf_association *
20 nvmf_allocate_association(enum nvmf_trtype trtype, bool controller,
21     const struct nvmf_association_params *params)
22 {
23 	struct nvmf_transport_ops *ops;
24 	struct nvmf_association *na;
25 
26 	switch (trtype) {
27 	case NVMF_TRTYPE_TCP:
28 		ops = &tcp_ops;
29 		break;
30 	default:
31 		errno = EINVAL;
32 		return (NULL);
33 	}
34 
35 	na = ops->allocate_association(controller, params);
36 	if (na == NULL)
37 		return (NULL);
38 
39 	na->na_ops = ops;
40 	na->na_trtype = trtype;
41 	na->na_controller = controller;
42 	na->na_params = *params;
43 	na->na_last_error = NULL;
44 	refcount_init(&na->na_refs, 1);
45 	return (na);
46 }
47 
48 void
49 nvmf_update_assocation(struct nvmf_association *na,
50     const struct nvme_controller_data *cdata)
51 {
52 	na->na_ops->update_association(na, cdata);
53 }
54 
55 void
56 nvmf_free_association(struct nvmf_association *na)
57 {
58 	if (refcount_release(&na->na_refs)) {
59 		free(na->na_last_error);
60 		na->na_ops->free_association(na);
61 	}
62 }
63 
64 const char *
65 nvmf_association_error(const struct nvmf_association *na)
66 {
67 	return (na->na_last_error);
68 }
69 
70 void
71 na_clear_error(struct nvmf_association *na)
72 {
73 	free(na->na_last_error);
74 	na->na_last_error = NULL;
75 }
76 
77 void
78 na_error(struct nvmf_association *na, const char *fmt, ...)
79 {
80 	va_list ap;
81 	char *str;
82 
83 	if (na->na_last_error != NULL)
84 		return;
85 	va_start(ap, fmt);
86 	vasprintf(&str, fmt, ap);
87 	va_end(ap);
88 	na->na_last_error = str;
89 }
90 
91 struct nvmf_qpair *
92 nvmf_allocate_qpair(struct nvmf_association *na,
93     const struct nvmf_qpair_params *params)
94 {
95 	struct nvmf_qpair *qp;
96 
97 	na_clear_error(na);
98 	qp = na->na_ops->allocate_qpair(na, params);
99 	if (qp == NULL)
100 		return (NULL);
101 
102 	refcount_acquire(&na->na_refs);
103 	qp->nq_association = na;
104 	qp->nq_admin = params->admin;
105 	TAILQ_INIT(&qp->nq_rx_capsules);
106 	return (qp);
107 }
108 
109 void
110 nvmf_free_qpair(struct nvmf_qpair *qp)
111 {
112 	struct nvmf_association *na;
113 	struct nvmf_capsule *nc, *tc;
114 
115 	TAILQ_FOREACH_SAFE(nc, &qp->nq_rx_capsules, nc_link, tc) {
116 		TAILQ_REMOVE(&qp->nq_rx_capsules, nc, nc_link);
117 		nvmf_free_capsule(nc);
118 	}
119 	na = qp->nq_association;
120 	na->na_ops->free_qpair(qp);
121 	nvmf_free_association(na);
122 }
123 
124 struct nvmf_capsule *
125 nvmf_allocate_command(struct nvmf_qpair *qp, const void *sqe)
126 {
127 	struct nvmf_capsule *nc;
128 
129 	nc = qp->nq_association->na_ops->allocate_capsule(qp);
130 	if (nc == NULL)
131 		return (NULL);
132 
133 	nc->nc_qpair = qp;
134 	nc->nc_qe_len = sizeof(struct nvme_command);
135 	memcpy(&nc->nc_sqe, sqe, nc->nc_qe_len);
136 
137 	/* 4.2 of NVMe base spec: Fabrics always uses SGL. */
138 	nc->nc_sqe.fuse &= ~NVMEM(NVME_CMD_PSDT);
139 	nc->nc_sqe.fuse |= NVMEF(NVME_CMD_PSDT, NVME_PSDT_SGL);
140 	return (nc);
141 }
142 
143 struct nvmf_capsule *
144 nvmf_allocate_response(struct nvmf_qpair *qp, const void *cqe)
145 {
146 	struct nvmf_capsule *nc;
147 
148 	nc = qp->nq_association->na_ops->allocate_capsule(qp);
149 	if (nc == NULL)
150 		return (NULL);
151 
152 	nc->nc_qpair = qp;
153 	nc->nc_qe_len = sizeof(struct nvme_completion);
154 	memcpy(&nc->nc_cqe, cqe, nc->nc_qe_len);
155 	return (nc);
156 }
157 
158 int
159 nvmf_capsule_append_data(struct nvmf_capsule *nc, void *buf, size_t len,
160     bool send)
161 {
162 	if (nc->nc_qe_len == sizeof(struct nvme_completion))
163 		return (EINVAL);
164 	if (nc->nc_data_len != 0)
165 		return (EBUSY);
166 
167 	nc->nc_data = buf;
168 	nc->nc_data_len = len;
169 	nc->nc_send_data = send;
170 	return (0);
171 }
172 
173 void
174 nvmf_free_capsule(struct nvmf_capsule *nc)
175 {
176 	nc->nc_qpair->nq_association->na_ops->free_capsule(nc);
177 }
178 
179 int
180 nvmf_transmit_capsule(struct nvmf_capsule *nc)
181 {
182 	return (nc->nc_qpair->nq_association->na_ops->transmit_capsule(nc));
183 }
184 
185 int
186 nvmf_receive_capsule(struct nvmf_qpair *qp, struct nvmf_capsule **ncp)
187 {
188 	return (qp->nq_association->na_ops->receive_capsule(qp, ncp));
189 }
190 
191 const void *
192 nvmf_capsule_sqe(const struct nvmf_capsule *nc)
193 {
194 	assert(nc->nc_qe_len == sizeof(struct nvme_command));
195 	return (&nc->nc_sqe);
196 }
197 
198 const void *
199 nvmf_capsule_cqe(const struct nvmf_capsule *nc)
200 {
201 	assert(nc->nc_qe_len == sizeof(struct nvme_completion));
202 	return (&nc->nc_cqe);
203 }
204 
205 uint8_t
206 nvmf_validate_command_capsule(const struct nvmf_capsule *nc)
207 {
208 	assert(nc->nc_qe_len == sizeof(struct nvme_command));
209 
210 	if (NVMEV(NVME_CMD_PSDT, nc->nc_sqe.fuse) != NVME_PSDT_SGL)
211 		return (NVME_SC_INVALID_FIELD);
212 
213 	return (nc->nc_qpair->nq_association->na_ops->validate_command_capsule(nc));
214 }
215 
216 size_t
217 nvmf_capsule_data_len(const struct nvmf_capsule *nc)
218 {
219 	return (nc->nc_qpair->nq_association->na_ops->capsule_data_len(nc));
220 }
221 
222 int
223 nvmf_receive_controller_data(const struct nvmf_capsule *nc,
224     uint32_t data_offset, void *buf, size_t len)
225 {
226 	return (nc->nc_qpair->nq_association->na_ops->receive_controller_data(nc,
227 	    data_offset, buf, len));
228 }
229 
230 int
231 nvmf_send_controller_data(const struct nvmf_capsule *nc, const void *buf,
232     size_t len)
233 {
234 	return (nc->nc_qpair->nq_association->na_ops->send_controller_data(nc,
235 	    buf, len));
236 }
237 
238 int
239 nvmf_kernel_handoff_params(struct nvmf_qpair *qp, nvlist_t **nvlp)
240 {
241 	nvlist_t *nvl;
242 	int error;
243 
244 	nvl = nvlist_create(0);
245 	nvlist_add_bool(nvl, "admin", qp->nq_admin);
246 	nvlist_add_bool(nvl, "sq_flow_control", qp->nq_flow_control);
247 	nvlist_add_number(nvl, "qsize", qp->nq_qsize);
248 	nvlist_add_number(nvl, "sqhd", qp->nq_sqhd);
249 	if (!qp->nq_association->na_controller)
250 		nvlist_add_number(nvl, "sqtail", qp->nq_sqtail);
251 	qp->nq_association->na_ops->kernel_handoff_params(qp, nvl);
252 	error = nvlist_error(nvl);
253 	if (error != 0) {
254 		nvlist_destroy(nvl);
255 		return (error);
256 	}
257 
258 	*nvlp = nvl;
259 	return (0);
260 }
261 
262 const char *
263 nvmf_transport_type(uint8_t trtype)
264 {
265 	static _Thread_local char buf[8];
266 
267 	switch (trtype) {
268 	case NVMF_TRTYPE_RDMA:
269 		return ("RDMA");
270 	case NVMF_TRTYPE_FC:
271 		return ("Fibre Channel");
272 	case NVMF_TRTYPE_TCP:
273 		return ("TCP");
274 	case NVMF_TRTYPE_INTRA_HOST:
275 		return ("Intra-host");
276 	default:
277 		snprintf(buf, sizeof(buf), "0x%02x\n", trtype);
278 		return (buf);
279 	}
280 }
281 
282 int
283 nvmf_pack_ioc_nvlist(struct nvmf_ioc_nv *nv, nvlist_t *nvl)
284 {
285 	int error;
286 
287 	memset(nv, 0, sizeof(*nv));
288 
289 	error = nvlist_error(nvl);
290 	if (error)
291 		return (error);
292 
293 	nv->data = nvlist_pack(nvl, &nv->size);
294 	if (nv->data == NULL)
295 		return (ENOMEM);
296 
297 	return (0);
298 }
299