Blame view

drivers/vhost/scsi.c 61.6 KB
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1
2
3
  /*******************************************************************************
   * Vhost kernel TCM fabric driver for virtio SCSI initiators
   *
4c76251e8   Nicholas Bellinger   target: Update co...
4
   * (C) Copyright 2010-2013 Datera, Inc.
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
5
6
7
8
   * (C) Copyright 2010-2012 IBM Corp.
   *
   * Licensed to the Linux Foundation under the General Public License (GPL) version 2.
   *
4c76251e8   Nicholas Bellinger   target: Update co...
9
   * Authors: Nicholas A. Bellinger <nab@daterainc.com>
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
   *          Stefan Hajnoczi <stefanha@linux.vnet.ibm.com>
   *
   * This program is free software; you can redistribute it and/or modify
   * it under the terms of the GNU General Public License as published by
   * the Free Software Foundation; either version 2 of the License, or
   * (at your option) any later version.
   *
   * This program is distributed in the hope that it will be useful,
   * but WITHOUT ANY WARRANTY; without even the implied warranty of
   * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   * GNU General Public License for more details.
   *
   ****************************************************************************/
  
  #include <linux/module.h>
  #include <linux/moduleparam.h>
  #include <generated/utsrelease.h>
  #include <linux/utsname.h>
  #include <linux/init.h>
  #include <linux/slab.h>
  #include <linux/kthread.h>
  #include <linux/types.h>
  #include <linux/string.h>
  #include <linux/configfs.h>
  #include <linux/ctype.h>
  #include <linux/compat.h>
  #include <linux/eventfd.h>
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
37
38
39
40
41
42
43
44
45
46
47
  #include <linux/fs.h>
  #include <linux/miscdevice.h>
  #include <asm/unaligned.h>
  #include <scsi/scsi.h>
  #include <scsi/scsi_tcq.h>
  #include <target/target_core_base.h>
  #include <target/target_core_fabric.h>
  #include <target/target_core_fabric_configfs.h>
  #include <target/target_core_configfs.h>
  #include <target/configfs_macros.h>
  #include <linux/vhost.h>
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
48
  #include <linux/virtio_scsi.h>
9d6064a34   Asias He   tcm_vhost: Use ll...
49
  #include <linux/llist.h>
1b7f390eb   Asias He   tcm_vhost: Multi-...
50
  #include <linux/bitmap.h>
4824d3bfb   Nicholas Bellinger   vhost/scsi: Conve...
51
  #include <linux/percpu_ida.h>
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
52

057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
53
  #include "vhost.h"
5012a3a38   Michael S. Tsirkin   tcm_vhost: header...
54
55
56
57
  
  #define TCM_VHOST_VERSION  "v0.1"
  #define TCM_VHOST_NAMELEN 256
  #define TCM_VHOST_MAX_CDB_SIZE 32
4824d3bfb   Nicholas Bellinger   vhost/scsi: Conve...
58
  #define TCM_VHOST_DEFAULT_TAGS 256
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
59
  #define TCM_VHOST_PREALLOC_SGLS 2048
5a01d0821   Nicholas Bellinger   vhost/scsi: Move ...
60
  #define TCM_VHOST_PREALLOC_UPAGES 2048
b1935f687   Nicholas Bellinger   vhost/scsi: Add p...
61
  #define TCM_VHOST_PREALLOC_PROT_SGLS 512
5012a3a38   Michael S. Tsirkin   tcm_vhost: header...
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
  
  struct vhost_scsi_inflight {
  	/* Wait for the flush operation to finish */
  	struct completion comp;
  	/* Refcount for the inflight reqs */
  	struct kref kref;
  };
  
  struct tcm_vhost_cmd {
  	/* Descriptor from vhost_get_vq_desc() for virt_queue segment */
  	int tvc_vq_desc;
  	/* virtio-scsi initiator task attribute */
  	int tvc_task_attr;
  	/* virtio-scsi initiator data direction */
  	enum dma_data_direction tvc_data_direction;
  	/* Expected data transfer length from virtio-scsi header */
  	u32 tvc_exp_data_len;
  	/* The Tag from include/linux/virtio_scsi.h:struct virtio_scsi_cmd_req */
  	u64 tvc_tag;
  	/* The number of scatterlists associated with this cmd */
  	u32 tvc_sgl_count;
e31885dd9   Nicholas Bellinger   vhost/scsi: Add T...
83
  	u32 tvc_prot_sgl_count;
5012a3a38   Michael S. Tsirkin   tcm_vhost: header...
84
85
86
87
  	/* Saved unpacked SCSI LUN for tcm_vhost_submission_work() */
  	u32 tvc_lun;
  	/* Pointer to the SGL formatted memory from virtio-scsi */
  	struct scatterlist *tvc_sgl;
b1935f687   Nicholas Bellinger   vhost/scsi: Add p...
88
  	struct scatterlist *tvc_prot_sgl;
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
89
  	struct page **tvc_upages;
5012a3a38   Michael S. Tsirkin   tcm_vhost: header...
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
  	/* Pointer to response */
  	struct virtio_scsi_cmd_resp __user *tvc_resp;
  	/* Pointer to vhost_scsi for our device */
  	struct vhost_scsi *tvc_vhost;
  	/* Pointer to vhost_virtqueue for the cmd */
  	struct vhost_virtqueue *tvc_vq;
  	/* Pointer to vhost nexus memory */
  	struct tcm_vhost_nexus *tvc_nexus;
  	/* The TCM I/O descriptor that is accessed via container_of() */
  	struct se_cmd tvc_se_cmd;
  	/* work item used for cmwq dispatch to tcm_vhost_submission_work() */
  	struct work_struct work;
  	/* Copy of the incoming SCSI command descriptor block (CDB) */
  	unsigned char tvc_cdb[TCM_VHOST_MAX_CDB_SIZE];
  	/* Sense buffer that will be mapped into outgoing status */
  	unsigned char tvc_sense_buf[TRANSPORT_SENSE_BUFFER];
  	/* Completed commands list, serviced from vhost worker thread */
  	struct llist_node tvc_completion_list;
  	/* Used to track inflight cmd */
  	struct vhost_scsi_inflight *inflight;
  };
  
  struct tcm_vhost_nexus {
  	/* Pointer to TCM session for I_T Nexus */
  	struct se_session *tvn_se_sess;
  };
  
  struct tcm_vhost_nacl {
  	/* Binary World Wide unique Port Name for Vhost Initiator port */
  	u64 iport_wwpn;
  	/* ASCII formatted WWPN for Sas Initiator port */
  	char iport_name[TCM_VHOST_NAMELEN];
  	/* Returned by tcm_vhost_make_nodeacl() */
  	struct se_node_acl se_node_acl;
  };
5012a3a38   Michael S. Tsirkin   tcm_vhost: header...
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
  struct tcm_vhost_tpg {
  	/* Vhost port target portal group tag for TCM */
  	u16 tport_tpgt;
  	/* Used to track number of TPG Port/Lun Links wrt to explict I_T Nexus shutdown */
  	int tv_tpg_port_count;
  	/* Used for vhost_scsi device reference to tpg_nexus, protected by tv_tpg_mutex */
  	int tv_tpg_vhost_count;
  	/* list for tcm_vhost_list */
  	struct list_head tv_tpg_list;
  	/* Used to protect access for tpg_nexus */
  	struct mutex tv_tpg_mutex;
  	/* Pointer to the TCM VHost I_T Nexus for this TPG endpoint */
  	struct tcm_vhost_nexus *tpg_nexus;
  	/* Pointer back to tcm_vhost_tport */
  	struct tcm_vhost_tport *tport;
  	/* Returned by tcm_vhost_make_tpg() */
  	struct se_portal_group se_tpg;
  	/* Pointer back to vhost_scsi, protected by tv_tpg_mutex */
  	struct vhost_scsi *vhost_scsi;
  };
  
  struct tcm_vhost_tport {
  	/* SCSI protocol the tport is providing */
  	u8 tport_proto_id;
  	/* Binary World Wide unique Port Name for Vhost Target port */
  	u64 tport_wwpn;
  	/* ASCII formatted WWPN for Vhost Target port */
  	char tport_name[TCM_VHOST_NAMELEN];
  	/* Returned by tcm_vhost_make_tport() */
  	struct se_wwn tport_wwn;
  };
  
  struct tcm_vhost_evt {
  	/* event to be sent to guest */
  	struct virtio_scsi_event event;
  	/* event list, serviced from vhost worker thread */
  	struct llist_node list;
  };
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
163

101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
164
165
166
167
168
  enum {
  	VHOST_SCSI_VQ_CTL = 0,
  	VHOST_SCSI_VQ_EVT = 1,
  	VHOST_SCSI_VQ_IO = 2,
  };
5dade7105   Nicholas Bellinger   tcm_vhost: Avoid ...
169
  enum {
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
170
171
  	VHOST_SCSI_FEATURES = VHOST_FEATURES | (1ULL << VIRTIO_SCSI_F_HOTPLUG) |
  					       (1ULL << VIRTIO_SCSI_F_T10_PI)
5dade7105   Nicholas Bellinger   tcm_vhost: Avoid ...
172
  };
1b7f390eb   Asias He   tcm_vhost: Multi-...
173
174
  #define VHOST_SCSI_MAX_TARGET	256
  #define VHOST_SCSI_MAX_VQ	128
a6c9af873   Asias He   tcm_vhost: Add ho...
175
  #define VHOST_SCSI_MAX_EVENT	128
67e18cf9a   Asias He   tcm_vhost: Multi-...
176

3ab2e420e   Asias He   vhost: Allow devi...
177
178
  struct vhost_scsi_virtqueue {
  	struct vhost_virtqueue vq;
3dfbff328   Michael S. Tsirkin   tcm_vhost: docume...
179
180
181
182
183
  	/*
  	 * Reference counting for inflight reqs, used for flush operation. At
  	 * each time, one reference tracks new commands submitted, while we
  	 * wait for another one to reach 0.
  	 */
f2f0173d6   Asias He   tcm_vhost: Wait f...
184
  	struct vhost_scsi_inflight inflights[2];
3dfbff328   Michael S. Tsirkin   tcm_vhost: docume...
185
186
187
188
  	/*
  	 * Indicate current inflight in use, protected by vq->mutex.
  	 * Writers must also take dev mutex and flush under it.
  	 */
f2f0173d6   Asias He   tcm_vhost: Wait f...
189
  	int inflight_idx;
3ab2e420e   Asias He   vhost: Allow devi...
190
  };
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
191
  struct vhost_scsi {
67e18cf9a   Asias He   tcm_vhost: Multi-...
192
  	/* Protected by vhost_scsi->dev.mutex */
4f7f46d32   Asias He   tcm_vhost: Use vq...
193
  	struct tcm_vhost_tpg **vs_tpg;
67e18cf9a   Asias He   tcm_vhost: Multi-...
194
  	char vs_vhost_wwpn[TRANSPORT_IQN_LEN];
67e18cf9a   Asias He   tcm_vhost: Multi-...
195

057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
196
  	struct vhost_dev dev;
3ab2e420e   Asias He   vhost: Allow devi...
197
  	struct vhost_scsi_virtqueue vqs[VHOST_SCSI_MAX_VQ];
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
198
199
  
  	struct vhost_work vs_completion_work; /* cmd completion work item */
9d6064a34   Asias He   tcm_vhost: Use ll...
200
  	struct llist_head vs_completion_list; /* cmd completion queue */
a6c9af873   Asias He   tcm_vhost: Add ho...
201
202
203
204
205
206
  
  	struct vhost_work vs_event_work; /* evt injection work item */
  	struct llist_head vs_event_list; /* evt injection queue */
  
  	bool vs_events_missed; /* any missed events, protected by vq->mutex */
  	int vs_events_nr; /* num of pending events, protected by vq->mutex */
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
207
208
209
210
211
212
213
214
215
216
  };
  
  /* Local pointer to allocated TCM configfs fabric module */
  static struct target_fabric_configfs *tcm_vhost_fabric_configfs;
  
  static struct workqueue_struct *tcm_vhost_workqueue;
  
  /* Global spinlock to protect tcm_vhost TPG list for vhost IOCTL access */
  static DEFINE_MUTEX(tcm_vhost_mutex);
  static LIST_HEAD(tcm_vhost_list);
765b34fdc   Asias He   tcm_vhost: Introd...
217
218
219
220
221
  static int iov_num_pages(struct iovec *iov)
  {
  	return (PAGE_ALIGN((unsigned long)iov->iov_base + iov->iov_len) -
  	       ((unsigned long)iov->iov_base & PAGE_MASK)) >> PAGE_SHIFT;
  }
0a1febf7b   Asias He   vhost: Make local...
222
  static void tcm_vhost_done_inflight(struct kref *kref)
f2f0173d6   Asias He   tcm_vhost: Wait f...
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
  {
  	struct vhost_scsi_inflight *inflight;
  
  	inflight = container_of(kref, struct vhost_scsi_inflight, kref);
  	complete(&inflight->comp);
  }
  
  static void tcm_vhost_init_inflight(struct vhost_scsi *vs,
  				    struct vhost_scsi_inflight *old_inflight[])
  {
  	struct vhost_scsi_inflight *new_inflight;
  	struct vhost_virtqueue *vq;
  	int idx, i;
  
  	for (i = 0; i < VHOST_SCSI_MAX_VQ; i++) {
  		vq = &vs->vqs[i].vq;
  
  		mutex_lock(&vq->mutex);
  
  		/* store old infight */
  		idx = vs->vqs[i].inflight_idx;
  		if (old_inflight)
  			old_inflight[i] = &vs->vqs[i].inflights[idx];
  
  		/* setup new infight */
  		vs->vqs[i].inflight_idx = idx ^ 1;
  		new_inflight = &vs->vqs[i].inflights[idx ^ 1];
  		kref_init(&new_inflight->kref);
  		init_completion(&new_inflight->comp);
  
  		mutex_unlock(&vq->mutex);
  	}
  }
  
  static struct vhost_scsi_inflight *
  tcm_vhost_get_inflight(struct vhost_virtqueue *vq)
  {
  	struct vhost_scsi_inflight *inflight;
  	struct vhost_scsi_virtqueue *svq;
  
  	svq = container_of(vq, struct vhost_scsi_virtqueue, vq);
  	inflight = &svq->inflights[svq->inflight_idx];
  	kref_get(&inflight->kref);
  
  	return inflight;
  }
  
  static void tcm_vhost_put_inflight(struct vhost_scsi_inflight *inflight)
  {
  	kref_put(&inflight->kref, tcm_vhost_done_inflight);
  }
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
  static int tcm_vhost_check_true(struct se_portal_group *se_tpg)
  {
  	return 1;
  }
  
  static int tcm_vhost_check_false(struct se_portal_group *se_tpg)
  {
  	return 0;
  }
  
  static char *tcm_vhost_get_fabric_name(void)
  {
  	return "vhost";
  }
  
  static u8 tcm_vhost_get_fabric_proto_ident(struct se_portal_group *se_tpg)
  {
  	struct tcm_vhost_tpg *tpg = container_of(se_tpg,
  				struct tcm_vhost_tpg, se_tpg);
  	struct tcm_vhost_tport *tport = tpg->tport;
  
  	switch (tport->tport_proto_id) {
  	case SCSI_PROTOCOL_SAS:
  		return sas_get_fabric_proto_ident(se_tpg);
  	case SCSI_PROTOCOL_FCP:
  		return fc_get_fabric_proto_ident(se_tpg);
  	case SCSI_PROTOCOL_ISCSI:
  		return iscsi_get_fabric_proto_ident(se_tpg);
  	default:
  		pr_err("Unknown tport_proto_id: 0x%02x, using"
  			" SAS emulation
  ", tport->tport_proto_id);
  		break;
  	}
  
  	return sas_get_fabric_proto_ident(se_tpg);
  }
  
  static char *tcm_vhost_get_fabric_wwn(struct se_portal_group *se_tpg)
  {
  	struct tcm_vhost_tpg *tpg = container_of(se_tpg,
  				struct tcm_vhost_tpg, se_tpg);
  	struct tcm_vhost_tport *tport = tpg->tport;
  
  	return &tport->tport_name[0];
  }
  
  static u16 tcm_vhost_get_tag(struct se_portal_group *se_tpg)
  {
  	struct tcm_vhost_tpg *tpg = container_of(se_tpg,
  				struct tcm_vhost_tpg, se_tpg);
  	return tpg->tport_tpgt;
  }
  
  static u32 tcm_vhost_get_default_depth(struct se_portal_group *se_tpg)
  {
  	return 1;
  }
683bd967d   Asias He   vhost-scsi: Make ...
332
333
334
335
336
337
  static u32
  tcm_vhost_get_pr_transport_id(struct se_portal_group *se_tpg,
  			      struct se_node_acl *se_nacl,
  			      struct t10_pr_registration *pr_reg,
  			      int *format_code,
  			      unsigned char *buf)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
  {
  	struct tcm_vhost_tpg *tpg = container_of(se_tpg,
  				struct tcm_vhost_tpg, se_tpg);
  	struct tcm_vhost_tport *tport = tpg->tport;
  
  	switch (tport->tport_proto_id) {
  	case SCSI_PROTOCOL_SAS:
  		return sas_get_pr_transport_id(se_tpg, se_nacl, pr_reg,
  					format_code, buf);
  	case SCSI_PROTOCOL_FCP:
  		return fc_get_pr_transport_id(se_tpg, se_nacl, pr_reg,
  					format_code, buf);
  	case SCSI_PROTOCOL_ISCSI:
  		return iscsi_get_pr_transport_id(se_tpg, se_nacl, pr_reg,
  					format_code, buf);
  	default:
  		pr_err("Unknown tport_proto_id: 0x%02x, using"
  			" SAS emulation
  ", tport->tport_proto_id);
  		break;
  	}
  
  	return sas_get_pr_transport_id(se_tpg, se_nacl, pr_reg,
  			format_code, buf);
  }
683bd967d   Asias He   vhost-scsi: Make ...
363
364
365
366
367
  static u32
  tcm_vhost_get_pr_transport_id_len(struct se_portal_group *se_tpg,
  				  struct se_node_acl *se_nacl,
  				  struct t10_pr_registration *pr_reg,
  				  int *format_code)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
  {
  	struct tcm_vhost_tpg *tpg = container_of(se_tpg,
  				struct tcm_vhost_tpg, se_tpg);
  	struct tcm_vhost_tport *tport = tpg->tport;
  
  	switch (tport->tport_proto_id) {
  	case SCSI_PROTOCOL_SAS:
  		return sas_get_pr_transport_id_len(se_tpg, se_nacl, pr_reg,
  					format_code);
  	case SCSI_PROTOCOL_FCP:
  		return fc_get_pr_transport_id_len(se_tpg, se_nacl, pr_reg,
  					format_code);
  	case SCSI_PROTOCOL_ISCSI:
  		return iscsi_get_pr_transport_id_len(se_tpg, se_nacl, pr_reg,
  					format_code);
  	default:
  		pr_err("Unknown tport_proto_id: 0x%02x, using"
  			" SAS emulation
  ", tport->tport_proto_id);
  		break;
  	}
  
  	return sas_get_pr_transport_id_len(se_tpg, se_nacl, pr_reg,
  			format_code);
  }
683bd967d   Asias He   vhost-scsi: Make ...
393
394
395
396
397
  static char *
  tcm_vhost_parse_pr_out_transport_id(struct se_portal_group *se_tpg,
  				    const char *buf,
  				    u32 *out_tid_len,
  				    char **port_nexus_ptr)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
  {
  	struct tcm_vhost_tpg *tpg = container_of(se_tpg,
  				struct tcm_vhost_tpg, se_tpg);
  	struct tcm_vhost_tport *tport = tpg->tport;
  
  	switch (tport->tport_proto_id) {
  	case SCSI_PROTOCOL_SAS:
  		return sas_parse_pr_out_transport_id(se_tpg, buf, out_tid_len,
  					port_nexus_ptr);
  	case SCSI_PROTOCOL_FCP:
  		return fc_parse_pr_out_transport_id(se_tpg, buf, out_tid_len,
  					port_nexus_ptr);
  	case SCSI_PROTOCOL_ISCSI:
  		return iscsi_parse_pr_out_transport_id(se_tpg, buf, out_tid_len,
  					port_nexus_ptr);
  	default:
  		pr_err("Unknown tport_proto_id: 0x%02x, using"
  			" SAS emulation
  ", tport->tport_proto_id);
  		break;
  	}
  
  	return sas_parse_pr_out_transport_id(se_tpg, buf, out_tid_len,
  			port_nexus_ptr);
  }
683bd967d   Asias He   vhost-scsi: Make ...
423
424
  static struct se_node_acl *
  tcm_vhost_alloc_fabric_acl(struct se_portal_group *se_tpg)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
425
426
427
428
429
  {
  	struct tcm_vhost_nacl *nacl;
  
  	nacl = kzalloc(sizeof(struct tcm_vhost_nacl), GFP_KERNEL);
  	if (!nacl) {
744627e91   Masanari Iida   treewide: fix pri...
430
431
  		pr_err("Unable to allocate struct tcm_vhost_nacl
  ");
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
432
433
434
435
436
  		return NULL;
  	}
  
  	return &nacl->se_node_acl;
  }
683bd967d   Asias He   vhost-scsi: Make ...
437
438
439
  static void
  tcm_vhost_release_fabric_acl(struct se_portal_group *se_tpg,
  			     struct se_node_acl *se_nacl)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
440
441
442
443
444
445
446
447
448
449
450
451
452
  {
  	struct tcm_vhost_nacl *nacl = container_of(se_nacl,
  			struct tcm_vhost_nacl, se_node_acl);
  	kfree(nacl);
  }
  
  static u32 tcm_vhost_tpg_get_inst_index(struct se_portal_group *se_tpg)
  {
  	return 1;
  }
  
  static void tcm_vhost_release_cmd(struct se_cmd *se_cmd)
  {
084ed45b3   Nicholas Bellinger   vhost/scsi: Conve...
453
454
  	struct tcm_vhost_cmd *tv_cmd = container_of(se_cmd,
  				struct tcm_vhost_cmd, tvc_se_cmd);
4824d3bfb   Nicholas Bellinger   vhost/scsi: Conve...
455
  	struct se_session *se_sess = se_cmd->se_sess;
e31885dd9   Nicholas Bellinger   vhost/scsi: Add T...
456
  	int i;
084ed45b3   Nicholas Bellinger   vhost/scsi: Conve...
457
458
  
  	if (tv_cmd->tvc_sgl_count) {
084ed45b3   Nicholas Bellinger   vhost/scsi: Conve...
459
460
  		for (i = 0; i < tv_cmd->tvc_sgl_count; i++)
  			put_page(sg_page(&tv_cmd->tvc_sgl[i]));
d3d665a65   Michael S. Tsirkin   vhost-scsi: white...
461
  	}
e31885dd9   Nicholas Bellinger   vhost/scsi: Add T...
462
463
464
465
  	if (tv_cmd->tvc_prot_sgl_count) {
  		for (i = 0; i < tv_cmd->tvc_prot_sgl_count; i++)
  			put_page(sg_page(&tv_cmd->tvc_prot_sgl[i]));
  	}
084ed45b3   Nicholas Bellinger   vhost/scsi: Conve...
466
467
  
  	tcm_vhost_put_inflight(tv_cmd->inflight);
4824d3bfb   Nicholas Bellinger   vhost/scsi: Conve...
468
  	percpu_ida_free(&se_sess->sess_tag_pool, se_cmd->map_tag);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
  }
  
  static int tcm_vhost_shutdown_session(struct se_session *se_sess)
  {
  	return 0;
  }
  
  static void tcm_vhost_close_session(struct se_session *se_sess)
  {
  	return;
  }
  
  static u32 tcm_vhost_sess_get_index(struct se_session *se_sess)
  {
  	return 0;
  }
  
  static int tcm_vhost_write_pending(struct se_cmd *se_cmd)
  {
  	/* Go ahead and process the write immediately */
  	target_execute_cmd(se_cmd);
  	return 0;
  }
  
  static int tcm_vhost_write_pending_status(struct se_cmd *se_cmd)
  {
  	return 0;
  }
  
  static void tcm_vhost_set_default_node_attrs(struct se_node_acl *nacl)
  {
  	return;
  }
  
  static u32 tcm_vhost_get_task_tag(struct se_cmd *se_cmd)
  {
  	return 0;
  }
  
  static int tcm_vhost_get_cmd_state(struct se_cmd *se_cmd)
  {
  	return 0;
  }
3c63f66a0   Asias He   vhost-scsi: Renam...
512
  static void vhost_scsi_complete_cmd(struct tcm_vhost_cmd *cmd)
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
513
  {
3c63f66a0   Asias He   vhost-scsi: Renam...
514
  	struct vhost_scsi *vs = cmd->tvc_vhost;
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
515

3c63f66a0   Asias He   vhost-scsi: Renam...
516
  	llist_add(&cmd->tvc_completion_list, &vs->vs_completion_list);
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
517
518
519
  
  	vhost_work_queue(&vs->dev, &vs->vs_completion_work);
  }
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
520
521
522
  
  static int tcm_vhost_queue_data_in(struct se_cmd *se_cmd)
  {
3c63f66a0   Asias He   vhost-scsi: Renam...
523
  	struct tcm_vhost_cmd *cmd = container_of(se_cmd,
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
524
  				struct tcm_vhost_cmd, tvc_se_cmd);
3c63f66a0   Asias He   vhost-scsi: Renam...
525
  	vhost_scsi_complete_cmd(cmd);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
526
527
528
529
530
  	return 0;
  }
  
  static int tcm_vhost_queue_status(struct se_cmd *se_cmd)
  {
3c63f66a0   Asias He   vhost-scsi: Renam...
531
  	struct tcm_vhost_cmd *cmd = container_of(se_cmd,
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
532
  				struct tcm_vhost_cmd, tvc_se_cmd);
3c63f66a0   Asias He   vhost-scsi: Renam...
533
  	vhost_scsi_complete_cmd(cmd);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
534
535
  	return 0;
  }
b79fafac7   Joern Engel   target: make queu...
536
  static void tcm_vhost_queue_tm_rsp(struct se_cmd *se_cmd)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
537
  {
b79fafac7   Joern Engel   target: make queu...
538
  	return;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
539
  }
131e6abc6   Nicholas Bellinger   target: Add TFO->...
540
541
542
543
  static void tcm_vhost_aborted_task(struct se_cmd *se_cmd)
  {
  	return;
  }
a6c9af873   Asias He   tcm_vhost: Add ho...
544
545
546
547
548
  static void tcm_vhost_free_evt(struct vhost_scsi *vs, struct tcm_vhost_evt *evt)
  {
  	vs->vs_events_nr--;
  	kfree(evt);
  }
683bd967d   Asias He   vhost-scsi: Make ...
549
550
551
  static struct tcm_vhost_evt *
  tcm_vhost_allocate_evt(struct vhost_scsi *vs,
  		       u32 event, u32 reason)
a6c9af873   Asias He   tcm_vhost: Add ho...
552
  {
3ab2e420e   Asias He   vhost: Allow devi...
553
  	struct vhost_virtqueue *vq = &vs->vqs[VHOST_SCSI_VQ_EVT].vq;
a6c9af873   Asias He   tcm_vhost: Add ho...
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
  	struct tcm_vhost_evt *evt;
  
  	if (vs->vs_events_nr > VHOST_SCSI_MAX_EVENT) {
  		vs->vs_events_missed = true;
  		return NULL;
  	}
  
  	evt = kzalloc(sizeof(*evt), GFP_KERNEL);
  	if (!evt) {
  		vq_err(vq, "Failed to allocate tcm_vhost_evt
  ");
  		vs->vs_events_missed = true;
  		return NULL;
  	}
  
  	evt->event.event = event;
  	evt->event.reason = reason;
  	vs->vs_events_nr++;
  
  	return evt;
  }
3c63f66a0   Asias He   vhost-scsi: Renam...
575
  static void vhost_scsi_free_cmd(struct tcm_vhost_cmd *cmd)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
576
  {
3c63f66a0   Asias He   vhost-scsi: Renam...
577
  	struct se_cmd *se_cmd = &cmd->tvc_se_cmd;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
578
579
  
  	/* TODO locking against target/backend threads? */
6c131d0c5   Nicholas Bellinger   vhost/scsi: Drop ...
580
  	transport_generic_free_cmd(se_cmd, 0);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
581

084ed45b3   Nicholas Bellinger   vhost/scsi: Conve...
582
  }
f2f0173d6   Asias He   tcm_vhost: Wait f...
583

084ed45b3   Nicholas Bellinger   vhost/scsi: Conve...
584
585
586
  static int vhost_scsi_check_stop_free(struct se_cmd *se_cmd)
  {
  	return target_put_sess_cmd(se_cmd->se_sess, se_cmd);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
587
  }
683bd967d   Asias He   vhost-scsi: Make ...
588
589
  static void
  tcm_vhost_do_evt_work(struct vhost_scsi *vs, struct tcm_vhost_evt *evt)
a6c9af873   Asias He   tcm_vhost: Add ho...
590
  {
3ab2e420e   Asias He   vhost: Allow devi...
591
  	struct vhost_virtqueue *vq = &vs->vqs[VHOST_SCSI_VQ_EVT].vq;
a6c9af873   Asias He   tcm_vhost: Add ho...
592
593
594
595
596
597
598
599
600
601
602
603
  	struct virtio_scsi_event *event = &evt->event;
  	struct virtio_scsi_event __user *eventp;
  	unsigned out, in;
  	int head, ret;
  
  	if (!vq->private_data) {
  		vs->vs_events_missed = true;
  		return;
  	}
  
  again:
  	vhost_disable_notify(&vs->dev, vq);
47283bef7   Michael S. Tsirkin   vhost: move memor...
604
  	head = vhost_get_vq_desc(vq, vq->iov,
a6c9af873   Asias He   tcm_vhost: Add ho...
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
  			ARRAY_SIZE(vq->iov), &out, &in,
  			NULL, NULL);
  	if (head < 0) {
  		vs->vs_events_missed = true;
  		return;
  	}
  	if (head == vq->num) {
  		if (vhost_enable_notify(&vs->dev, vq))
  			goto again;
  		vs->vs_events_missed = true;
  		return;
  	}
  
  	if ((vq->iov[out].iov_len != sizeof(struct virtio_scsi_event))) {
  		vq_err(vq, "Expecting virtio_scsi_event, got %zu bytes
  ",
  				vq->iov[out].iov_len);
  		vs->vs_events_missed = true;
  		return;
  	}
  
  	if (vs->vs_events_missed) {
  		event->event |= VIRTIO_SCSI_T_EVENTS_MISSED;
  		vs->vs_events_missed = false;
  	}
  
  	eventp = vq->iov[out].iov_base;
  	ret = __copy_to_user(eventp, event, sizeof(*event));
  	if (!ret)
  		vhost_add_used_and_signal(&vs->dev, vq, head, 0);
  	else
  		vq_err(vq, "Faulted on tcm_vhost_send_event
  ");
  }
  
  static void tcm_vhost_evt_work(struct vhost_work *work)
  {
  	struct vhost_scsi *vs = container_of(work, struct vhost_scsi,
  					vs_event_work);
3ab2e420e   Asias He   vhost: Allow devi...
644
  	struct vhost_virtqueue *vq = &vs->vqs[VHOST_SCSI_VQ_EVT].vq;
a6c9af873   Asias He   tcm_vhost: Add ho...
645
646
647
648
649
650
651
652
653
654
655
656
657
  	struct tcm_vhost_evt *evt;
  	struct llist_node *llnode;
  
  	mutex_lock(&vq->mutex);
  	llnode = llist_del_all(&vs->vs_event_list);
  	while (llnode) {
  		evt = llist_entry(llnode, struct tcm_vhost_evt, list);
  		llnode = llist_next(llnode);
  		tcm_vhost_do_evt_work(vs, evt);
  		tcm_vhost_free_evt(vs, evt);
  	}
  	mutex_unlock(&vq->mutex);
  }
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
658
659
660
661
662
663
664
665
666
  /* Fill in status and signal that we are done processing this command
   *
   * This is scheduled in the vhost work queue so we are called with the owner
   * process mm and can access the vring.
   */
  static void vhost_scsi_complete_cmd_work(struct vhost_work *work)
  {
  	struct vhost_scsi *vs = container_of(work, struct vhost_scsi,
  					vs_completion_work);
1b7f390eb   Asias He   tcm_vhost: Multi-...
667
  	DECLARE_BITMAP(signal, VHOST_SCSI_MAX_VQ);
9d6064a34   Asias He   tcm_vhost: Use ll...
668
  	struct virtio_scsi_cmd_resp v_rsp;
3c63f66a0   Asias He   vhost-scsi: Renam...
669
  	struct tcm_vhost_cmd *cmd;
9d6064a34   Asias He   tcm_vhost: Use ll...
670
671
  	struct llist_node *llnode;
  	struct se_cmd *se_cmd;
1b7f390eb   Asias He   tcm_vhost: Multi-...
672
  	int ret, vq;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
673

1b7f390eb   Asias He   tcm_vhost: Multi-...
674
  	bitmap_zero(signal, VHOST_SCSI_MAX_VQ);
9d6064a34   Asias He   tcm_vhost: Use ll...
675
676
  	llnode = llist_del_all(&vs->vs_completion_list);
  	while (llnode) {
3c63f66a0   Asias He   vhost-scsi: Renam...
677
  		cmd = llist_entry(llnode, struct tcm_vhost_cmd,
9d6064a34   Asias He   tcm_vhost: Use ll...
678
679
  				     tvc_completion_list);
  		llnode = llist_next(llnode);
3c63f66a0   Asias He   vhost-scsi: Renam...
680
  		se_cmd = &cmd->tvc_se_cmd;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
681
682
683
  
  		pr_debug("%s tv_cmd %p resid %u status %#02x
  ", __func__,
3c63f66a0   Asias He   vhost-scsi: Renam...
684
  			cmd, se_cmd->residual_count, se_cmd->scsi_status);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
685
686
687
688
689
690
  
  		memset(&v_rsp, 0, sizeof(v_rsp));
  		v_rsp.resid = se_cmd->residual_count;
  		/* TODO is status_qualifier field needed? */
  		v_rsp.status = se_cmd->scsi_status;
  		v_rsp.sense_len = se_cmd->scsi_sense_length;
3c63f66a0   Asias He   vhost-scsi: Renam...
691
  		memcpy(v_rsp.sense, cmd->tvc_sense_buf,
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
692
  		       v_rsp.sense_len);
3c63f66a0   Asias He   vhost-scsi: Renam...
693
  		ret = copy_to_user(cmd->tvc_resp, &v_rsp, sizeof(v_rsp));
1b7f390eb   Asias He   tcm_vhost: Multi-...
694
  		if (likely(ret == 0)) {
3ab2e420e   Asias He   vhost: Allow devi...
695
  			struct vhost_scsi_virtqueue *q;
3c63f66a0   Asias He   vhost-scsi: Renam...
696
697
  			vhost_add_used(cmd->tvc_vq, cmd->tvc_vq_desc, 0);
  			q = container_of(cmd->tvc_vq, struct vhost_scsi_virtqueue, vq);
3ab2e420e   Asias He   vhost: Allow devi...
698
  			vq = q - vs->vqs;
1b7f390eb   Asias He   tcm_vhost: Multi-...
699
700
  			__set_bit(vq, signal);
  		} else
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
701
702
  			pr_err("Faulted on virtio_scsi_cmd_resp
  ");
3c63f66a0   Asias He   vhost-scsi: Renam...
703
  		vhost_scsi_free_cmd(cmd);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
704
  	}
1b7f390eb   Asias He   tcm_vhost: Multi-...
705
706
707
  	vq = -1;
  	while ((vq = find_next_bit(signal, VHOST_SCSI_MAX_VQ, vq + 1))
  		< VHOST_SCSI_MAX_VQ)
3ab2e420e   Asias He   vhost: Allow devi...
708
  		vhost_signal(&vs->dev, &vs->vqs[vq].vq);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
709
  }
683bd967d   Asias He   vhost-scsi: Make ...
710
  static struct tcm_vhost_cmd *
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
711
712
713
  vhost_scsi_get_tag(struct vhost_virtqueue *vq, struct tcm_vhost_tpg *tpg,
  		   unsigned char *cdb, u64 scsi_tag, u16 lun, u8 task_attr,
  		   u32 exp_data_len, int data_direction)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
714
  {
3c63f66a0   Asias He   vhost-scsi: Renam...
715
  	struct tcm_vhost_cmd *cmd;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
716
  	struct tcm_vhost_nexus *tv_nexus;
4824d3bfb   Nicholas Bellinger   vhost/scsi: Conve...
717
  	struct se_session *se_sess;
b1935f687   Nicholas Bellinger   vhost/scsi: Add p...
718
  	struct scatterlist *sg, *prot_sg;
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
719
  	struct page **pages;
4824d3bfb   Nicholas Bellinger   vhost/scsi: Conve...
720
  	int tag;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
721

987183128   Asias He   vhost-scsi: Renam...
722
  	tv_nexus = tpg->tpg_nexus;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
723
724
725
726
727
  	if (!tv_nexus) {
  		pr_err("Unable to locate active struct tcm_vhost_nexus
  ");
  		return ERR_PTR(-EIO);
  	}
4824d3bfb   Nicholas Bellinger   vhost/scsi: Conve...
728
  	se_sess = tv_nexus->tvn_se_sess;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
729

6f6b5d1ec   Kent Overstreet   percpu_ida: Make ...
730
  	tag = percpu_ida_alloc(&se_sess->sess_tag_pool, TASK_RUNNING);
4a47d3a1f   Nicholas Bellinger   vhost/scsi: Use G...
731
732
733
734
735
  	if (tag < 0) {
  		pr_err("Unable to obtain tag for tcm_vhost_cmd
  ");
  		return ERR_PTR(-ENOMEM);
  	}
4824d3bfb   Nicholas Bellinger   vhost/scsi: Conve...
736
  	cmd = &((struct tcm_vhost_cmd *)se_sess->sess_cmd_map)[tag];
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
737
  	sg = cmd->tvc_sgl;
b1935f687   Nicholas Bellinger   vhost/scsi: Add p...
738
  	prot_sg = cmd->tvc_prot_sgl;
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
739
  	pages = cmd->tvc_upages;
4824d3bfb   Nicholas Bellinger   vhost/scsi: Conve...
740
  	memset(cmd, 0, sizeof(struct tcm_vhost_cmd));
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
741
  	cmd->tvc_sgl = sg;
b1935f687   Nicholas Bellinger   vhost/scsi: Add p...
742
  	cmd->tvc_prot_sgl = prot_sg;
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
743
  	cmd->tvc_upages = pages;
4824d3bfb   Nicholas Bellinger   vhost/scsi: Conve...
744
  	cmd->tvc_se_cmd.map_tag = tag;
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
745
746
747
  	cmd->tvc_tag = scsi_tag;
  	cmd->tvc_lun = lun;
  	cmd->tvc_task_attr = task_attr;
3c63f66a0   Asias He   vhost-scsi: Renam...
748
749
750
751
  	cmd->tvc_exp_data_len = exp_data_len;
  	cmd->tvc_data_direction = data_direction;
  	cmd->tvc_nexus = tv_nexus;
  	cmd->inflight = tcm_vhost_get_inflight(vq);
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
752
  	memcpy(cmd->tvc_cdb, cdb, TCM_VHOST_MAX_CDB_SIZE);
3c63f66a0   Asias He   vhost-scsi: Renam...
753
  	return cmd;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
754
755
756
757
758
759
760
  }
  
  /*
   * Map a user memory range into a scatterlist
   *
   * Returns the number of scatterlist entries used or -errno on error.
   */
683bd967d   Asias He   vhost-scsi: Make ...
761
  static int
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
762
763
  vhost_scsi_map_to_sgl(struct tcm_vhost_cmd *tv_cmd,
  		      struct scatterlist *sgl,
683bd967d   Asias He   vhost-scsi: Make ...
764
765
  		      unsigned int sgl_count,
  		      struct iovec *iov,
5a01d0821   Nicholas Bellinger   vhost/scsi: Move ...
766
767
  		      struct page **pages,
  		      bool write)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
768
  {
1810053e8   Asias He   tcm_vhost: Optimi...
769
  	unsigned int npages = 0, pages_nr, offset, nbytes;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
770
  	struct scatterlist *sg = sgl;
1810053e8   Asias He   tcm_vhost: Optimi...
771
772
  	void __user *ptr = iov->iov_base;
  	size_t len = iov->iov_len;
1810053e8   Asias He   tcm_vhost: Optimi...
773
  	int ret, i;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
774

1810053e8   Asias He   tcm_vhost: Optimi...
775
  	pages_nr = iov_num_pages(iov);
5a01d0821   Nicholas Bellinger   vhost/scsi: Move ...
776
777
778
779
  	if (pages_nr > sgl_count) {
  		pr_err("vhost_scsi_map_to_sgl() pages_nr: %u greater than"
  		       " sgl_count: %u
  ", pages_nr, sgl_count);
1810053e8   Asias He   tcm_vhost: Optimi...
780
  		return -ENOBUFS;
5a01d0821   Nicholas Bellinger   vhost/scsi: Move ...
781
782
  	}
  	if (pages_nr > TCM_VHOST_PREALLOC_UPAGES) {
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
783
  		pr_err("vhost_scsi_map_to_sgl() pages_nr: %u greater than"
5a01d0821   Nicholas Bellinger   vhost/scsi: Move ...
784
785
786
  		       " preallocated TCM_VHOST_PREALLOC_UPAGES: %u
  ",
  			pages_nr, TCM_VHOST_PREALLOC_UPAGES);
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
787
788
  		return -ENOBUFS;
  	}
1810053e8   Asias He   tcm_vhost: Optimi...
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
  	ret = get_user_pages_fast((unsigned long)ptr, pages_nr, write, pages);
  	/* No pages were pinned */
  	if (ret < 0)
  		goto out;
  	/* Less pages pinned than wanted */
  	if (ret != pages_nr) {
  		for (i = 0; i < ret; i++)
  			put_page(pages[i]);
  		ret = -EFAULT;
  		goto out;
  	}
  
  	while (len > 0) {
  		offset = (uintptr_t)ptr & ~PAGE_MASK;
  		nbytes = min_t(unsigned int, PAGE_SIZE - offset, len);
  		sg_set_page(sg, pages[npages], nbytes, offset);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
805
806
807
808
809
  		ptr += nbytes;
  		len -= nbytes;
  		sg++;
  		npages++;
  	}
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
810

1810053e8   Asias He   tcm_vhost: Optimi...
811
  out:
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
812
813
  	return ret;
  }
683bd967d   Asias He   vhost-scsi: Make ...
814
  static int
3c63f66a0   Asias He   vhost-scsi: Renam...
815
  vhost_scsi_map_iov_to_sgl(struct tcm_vhost_cmd *cmd,
683bd967d   Asias He   vhost-scsi: Make ...
816
  			  struct iovec *iov,
5a01d0821   Nicholas Bellinger   vhost/scsi: Move ...
817
818
  			  int niov,
  			  bool write)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
819
  {
5a01d0821   Nicholas Bellinger   vhost/scsi: Move ...
820
821
822
  	struct scatterlist *sg = cmd->tvc_sgl;
  	unsigned int sgl_count = 0;
  	int ret, i;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
823

f3158f362   Asias He   tcm_vhost: Use io...
824
825
  	for (i = 0; i < niov; i++)
  		sgl_count += iov_num_pages(&iov[i]);
5a01d0821   Nicholas Bellinger   vhost/scsi: Move ...
826
827
828
829
830
831
832
  	if (sgl_count > TCM_VHOST_PREALLOC_SGLS) {
  		pr_err("vhost_scsi_map_iov_to_sgl() sgl_count: %u greater than"
  			" preallocated TCM_VHOST_PREALLOC_SGLS: %u
  ",
  			sgl_count, TCM_VHOST_PREALLOC_SGLS);
  		return -ENOBUFS;
  	}
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
833

3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
834
835
  	pr_debug("%s sg %p sgl_count %u
  ", __func__, sg, sgl_count);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
836
  	sg_init_table(sg, sgl_count);
3c63f66a0   Asias He   vhost-scsi: Renam...
837
  	cmd->tvc_sgl_count = sgl_count;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
838

5a01d0821   Nicholas Bellinger   vhost/scsi: Move ...
839
840
  	pr_debug("Mapping iovec %p for %u pages
  ", &iov[0], sgl_count);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
841
  	for (i = 0; i < niov; i++) {
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
842
  		ret = vhost_scsi_map_to_sgl(cmd, sg, sgl_count, &iov[i],
5a01d0821   Nicholas Bellinger   vhost/scsi: Move ...
843
  					    cmd->tvc_upages, write);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
844
  		if (ret < 0) {
3c63f66a0   Asias He   vhost-scsi: Renam...
845
846
  			for (i = 0; i < cmd->tvc_sgl_count; i++)
  				put_page(sg_page(&cmd->tvc_sgl[i]));
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
847

3c63f66a0   Asias He   vhost-scsi: Renam...
848
  			cmd->tvc_sgl_count = 0;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
849
850
  			return ret;
  		}
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
851
852
853
854
855
  		sg += ret;
  		sgl_count -= ret;
  	}
  	return 0;
  }
e31885dd9   Nicholas Bellinger   vhost/scsi: Add T...
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
  static int
  vhost_scsi_map_iov_to_prot(struct tcm_vhost_cmd *cmd,
  			   struct iovec *iov,
  			   int niov,
  			   bool write)
  {
  	struct scatterlist *prot_sg = cmd->tvc_prot_sgl;
  	unsigned int prot_sgl_count = 0;
  	int ret, i;
  
  	for (i = 0; i < niov; i++)
  		prot_sgl_count += iov_num_pages(&iov[i]);
  
  	if (prot_sgl_count > TCM_VHOST_PREALLOC_PROT_SGLS) {
  		pr_err("vhost_scsi_map_iov_to_prot() sgl_count: %u greater than"
  			" preallocated TCM_VHOST_PREALLOC_PROT_SGLS: %u
  ",
  			prot_sgl_count, TCM_VHOST_PREALLOC_PROT_SGLS);
  		return -ENOBUFS;
  	}
  
  	pr_debug("%s prot_sg %p prot_sgl_count %u
  ", __func__,
  		 prot_sg, prot_sgl_count);
  	sg_init_table(prot_sg, prot_sgl_count);
  	cmd->tvc_prot_sgl_count = prot_sgl_count;
  
  	for (i = 0; i < niov; i++) {
  		ret = vhost_scsi_map_to_sgl(cmd, prot_sg, prot_sgl_count, &iov[i],
  					    cmd->tvc_upages, write);
  		if (ret < 0) {
  			for (i = 0; i < cmd->tvc_prot_sgl_count; i++)
  				put_page(sg_page(&cmd->tvc_prot_sgl[i]));
  
  			cmd->tvc_prot_sgl_count = 0;
  			return ret;
  		}
  		prot_sg += ret;
  		prot_sgl_count -= ret;
  	}
  	return 0;
  }
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
898
899
  static void tcm_vhost_submission_work(struct work_struct *work)
  {
3c63f66a0   Asias He   vhost-scsi: Renam...
900
  	struct tcm_vhost_cmd *cmd =
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
901
  		container_of(work, struct tcm_vhost_cmd, work);
9f0abc155   Nicholas Bellinger   tcm_vhost: Conver...
902
  	struct tcm_vhost_nexus *tv_nexus;
3c63f66a0   Asias He   vhost-scsi: Renam...
903
  	struct se_cmd *se_cmd = &cmd->tvc_se_cmd;
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
904
905
  	struct scatterlist *sg_ptr, *sg_prot_ptr = NULL;
  	int rc;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
906

95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
907
  	/* FIXME: BIDI operation */
3c63f66a0   Asias He   vhost-scsi: Renam...
908
909
  	if (cmd->tvc_sgl_count) {
  		sg_ptr = cmd->tvc_sgl;
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
910
911
912
913
914
  
  		if (cmd->tvc_prot_sgl_count)
  			sg_prot_ptr = cmd->tvc_prot_sgl;
  		else
  			se_cmd->prot_pto = true;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
915
916
917
  	} else {
  		sg_ptr = NULL;
  	}
3c63f66a0   Asias He   vhost-scsi: Renam...
918
  	tv_nexus = cmd->tvc_nexus;
9f0abc155   Nicholas Bellinger   tcm_vhost: Conver...
919
920
  
  	rc = target_submit_cmd_map_sgls(se_cmd, tv_nexus->tvn_se_sess,
3c63f66a0   Asias He   vhost-scsi: Renam...
921
922
923
  			cmd->tvc_cdb, &cmd->tvc_sense_buf[0],
  			cmd->tvc_lun, cmd->tvc_exp_data_len,
  			cmd->tvc_task_attr, cmd->tvc_data_direction,
6d2fa9e14   Linus Torvalds   Merge branch 'for...
924
  			TARGET_SCF_ACK_KREF, sg_ptr, cmd->tvc_sgl_count,
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
925
  			NULL, 0, sg_prot_ptr, cmd->tvc_prot_sgl_count);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
926
927
  	if (rc < 0) {
  		transport_send_check_condition_and_sense(se_cmd,
9f0abc155   Nicholas Bellinger   tcm_vhost: Conver...
928
  				TCM_LOGICAL_UNIT_COMMUNICATION_FAILURE, 0);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
929
  		transport_generic_free_cmd(se_cmd, 0);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
930
  	}
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
931
  }
683bd967d   Asias He   vhost-scsi: Make ...
932
933
934
935
  static void
  vhost_scsi_send_bad_target(struct vhost_scsi *vs,
  			   struct vhost_virtqueue *vq,
  			   int head, unsigned out)
637ab21e2   Asias He   tcm_vhost: Add vh...
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
  {
  	struct virtio_scsi_cmd_resp __user *resp;
  	struct virtio_scsi_cmd_resp rsp;
  	int ret;
  
  	memset(&rsp, 0, sizeof(rsp));
  	rsp.response = VIRTIO_SCSI_S_BAD_TARGET;
  	resp = vq->iov[out].iov_base;
  	ret = __copy_to_user(resp, &rsp, sizeof(rsp));
  	if (!ret)
  		vhost_add_used_and_signal(&vs->dev, vq, head, 0);
  	else
  		pr_err("Faulted on virtio_scsi_cmd_resp
  ");
  }
683bd967d   Asias He   vhost-scsi: Make ...
951
952
  static void
  vhost_scsi_handle_vq(struct vhost_scsi *vs, struct vhost_virtqueue *vq)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
953
  {
4f7f46d32   Asias He   tcm_vhost: Use vq...
954
  	struct tcm_vhost_tpg **vs_tpg;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
955
  	struct virtio_scsi_cmd_req v_req;
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
956
  	struct virtio_scsi_cmd_req_pi v_req_pi;
987183128   Asias He   vhost-scsi: Renam...
957
  	struct tcm_vhost_tpg *tpg;
3c63f66a0   Asias He   vhost-scsi: Renam...
958
  	struct tcm_vhost_cmd *cmd;
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
959
960
  	u64 tag;
  	u32 exp_data_len, data_first, data_num, data_direction, prot_first;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
961
  	unsigned out, in, i;
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
962
963
964
965
966
967
  	int head, ret, data_niov, prot_niov, prot_bytes;
  	size_t req_size;
  	u16 lun;
  	u8 *target, *lunp, task_attr;
  	bool hdr_pi;
  	void *req, *cdb;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
968

e7802212e   Asias He   vhost-scsi: Alway...
969
  	mutex_lock(&vq->mutex);
4f7f46d32   Asias He   tcm_vhost: Use vq...
970
971
972
  	/*
  	 * We can handle the vq only after the endpoint is setup by calling the
  	 * VHOST_SCSI_SET_ENDPOINT ioctl.
4f7f46d32   Asias He   tcm_vhost: Use vq...
973
  	 */
e7802212e   Asias He   vhost-scsi: Alway...
974
  	vs_tpg = vq->private_data;
4f7f46d32   Asias He   tcm_vhost: Use vq...
975
  	if (!vs_tpg)
e7802212e   Asias He   vhost-scsi: Alway...
976
  		goto out;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
977

057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
978
979
980
  	vhost_disable_notify(&vs->dev, vq);
  
  	for (;;) {
47283bef7   Michael S. Tsirkin   vhost: move memor...
981
  		head = vhost_get_vq_desc(vq, vq->iov,
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
  					ARRAY_SIZE(vq->iov), &out, &in,
  					NULL, NULL);
  		pr_debug("vhost_get_vq_desc: head: %d, out: %u in: %u
  ",
  					head, out, in);
  		/* On error, stop handling until the next kick. */
  		if (unlikely(head < 0))
  			break;
  		/* Nothing new?  Wait for eventfd to tell us they refilled. */
  		if (head == vq->num) {
  			if (unlikely(vhost_enable_notify(&vs->dev, vq))) {
  				vhost_disable_notify(&vs->dev, vq);
  				continue;
  			}
  			break;
  		}
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
998
  		/* FIXME: BIDI operation */
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
999
1000
1001
1002
1003
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
1025
1026
1027
1028
  		if (out == 1 && in == 1) {
  			data_direction = DMA_NONE;
  			data_first = 0;
  			data_num = 0;
  		} else if (out == 1 && in > 1) {
  			data_direction = DMA_FROM_DEVICE;
  			data_first = out + 1;
  			data_num = in - 1;
  		} else if (out > 1 && in == 1) {
  			data_direction = DMA_TO_DEVICE;
  			data_first = 1;
  			data_num = out - 1;
  		} else {
  			vq_err(vq, "Invalid buffer layout out: %u in: %u
  ",
  					out, in);
  			break;
  		}
  
  		/*
  		 * Check for a sane resp buffer so we can report errors to
  		 * the guest.
  		 */
  		if (unlikely(vq->iov[out].iov_len !=
  					sizeof(struct virtio_scsi_cmd_resp))) {
  			vq_err(vq, "Expecting virtio_scsi_cmd_resp, got %zu"
  				" bytes
  ", vq->iov[out].iov_len);
  			break;
  		}
ed9ea4ed3   Linus Torvalds   Merge branch 'for...
1029
  		if (vhost_has_feature(vq, VIRTIO_SCSI_F_T10_PI)) {
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
1030
1031
1032
1033
1034
1035
1036
1037
1038
1039
1040
1041
1042
1043
1044
1045
1046
  			req = &v_req_pi;
  			lunp = &v_req_pi.lun[0];
  			target = &v_req_pi.lun[1];
  			req_size = sizeof(v_req_pi);
  			hdr_pi = true;
  		} else {
  			req = &v_req;
  			lunp = &v_req.lun[0];
  			target = &v_req.lun[1];
  			req_size = sizeof(v_req);
  			hdr_pi = false;
  		}
  
  		if (unlikely(vq->iov[0].iov_len < req_size)) {
  			pr_err("Expecting virtio-scsi header: %zu, got %zu
  ",
  			       req_size, vq->iov[0].iov_len);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1047
1048
  			break;
  		}
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
1049
  		ret = memcpy_fromiovecend(req, &vq->iov[0], 0, req_size);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1050
1051
1052
1053
1054
  		if (unlikely(ret)) {
  			vq_err(vq, "Faulted on virtio_scsi_cmd_req
  ");
  			break;
  		}
7fe412d07   Venkatesh Srinivas   vhost/scsi: Check...
1055
  		/* virtio-scsi spec requires byte 0 of the lun to be 1 */
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
1056
  		if (unlikely(*lunp != 1)) {
7fe412d07   Venkatesh Srinivas   vhost/scsi: Check...
1057
1058
1059
  			vhost_scsi_send_bad_target(vs, vq, head, out);
  			continue;
  		}
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
1060
  		tpg = ACCESS_ONCE(vs_tpg[*target]);
67e18cf9a   Asias He   tcm_vhost: Multi-...
1061
1062
  
  		/* Target does not exist, fail the request */
987183128   Asias He   vhost-scsi: Renam...
1063
  		if (unlikely(!tpg)) {
637ab21e2   Asias He   tcm_vhost: Add vh...
1064
  			vhost_scsi_send_bad_target(vs, vq, head, out);
67e18cf9a   Asias He   tcm_vhost: Multi-...
1065
1066
  			continue;
  		}
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
1067
1068
1069
1070
1071
1072
1073
1074
1075
1076
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089
1090
1091
1092
1093
1094
1095
1096
1097
1098
1099
1100
1101
1102
1103
1104
1105
1106
1107
1108
1109
1110
1111
1112
1113
1114
1115
1116
  		data_niov = data_num;
  		prot_niov = prot_first = prot_bytes = 0;
  		/*
  		 * Determine if any protection information iovecs are preceeding
  		 * the actual data payload, and adjust data_first + data_niov
  		 * values accordingly for vhost_scsi_map_iov_to_sgl() below.
  		 *
  		 * Also extract virtio_scsi header bits for vhost_scsi_get_tag()
  		 */
  		if (hdr_pi) {
  			if (v_req_pi.pi_bytesout) {
  				if (data_direction != DMA_TO_DEVICE) {
  					vq_err(vq, "Received non zero do_pi_niov"
  						", but wrong data_direction
  ");
  					goto err_cmd;
  				}
  				prot_bytes = v_req_pi.pi_bytesout;
  			} else if (v_req_pi.pi_bytesin) {
  				if (data_direction != DMA_FROM_DEVICE) {
  					vq_err(vq, "Received non zero di_pi_niov"
  						", but wrong data_direction
  ");
  					goto err_cmd;
  				}
  				prot_bytes = v_req_pi.pi_bytesin;
  			}
  			if (prot_bytes) {
  				int tmp = 0;
  
  				for (i = 0; i < data_num; i++) {
  					tmp += vq->iov[data_first + i].iov_len;
  					prot_niov++;
  					if (tmp >= prot_bytes)
  						break;
  				}
  				prot_first = data_first;
  				data_first += prot_niov;
  				data_niov = data_num - prot_niov;
  			}
  			tag = v_req_pi.tag;
  			task_attr = v_req_pi.task_attr;
  			cdb = &v_req_pi.cdb[0];
  			lun = ((v_req_pi.lun[2] << 8) | v_req_pi.lun[3]) & 0x3FFF;
  		} else {
  			tag = v_req.tag;
  			task_attr = v_req.task_attr;
  			cdb = &v_req.cdb[0];
  			lun = ((v_req.lun[2] << 8) | v_req.lun[3]) & 0x3FFF;
  		}
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1117
  		exp_data_len = 0;
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
1118
  		for (i = 0; i < data_niov; i++)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1119
  			exp_data_len += vq->iov[data_first + i].iov_len;
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
1120
1121
1122
1123
1124
1125
1126
1127
1128
1129
1130
1131
1132
  		/*
  		 * Check that the recieved CDB size does not exceeded our
  		 * hardcoded max for vhost-scsi
  		 *
  		 * TODO what if cdb was too small for varlen cdb header?
  		 */
  		if (unlikely(scsi_command_size(cdb) > TCM_VHOST_MAX_CDB_SIZE)) {
  			vq_err(vq, "Received SCSI CDB with command_size: %d that"
  				" exceeds SCSI_MAX_VARLEN_CDB_SIZE: %d
  ",
  				scsi_command_size(cdb), TCM_VHOST_MAX_CDB_SIZE);
  			goto err_cmd;
  		}
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1133

95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
1134
  		cmd = vhost_scsi_get_tag(vq, tpg, cdb, tag, lun, task_attr,
9f977ef7b   Nicholas Bellinger   vhost-scsi: Inclu...
1135
1136
  					 exp_data_len + prot_bytes,
  					 data_direction);
3c63f66a0   Asias He   vhost-scsi: Renam...
1137
  		if (IS_ERR(cmd)) {
4824d3bfb   Nicholas Bellinger   vhost/scsi: Conve...
1138
1139
  			vq_err(vq, "vhost_scsi_get_tag failed %ld
  ",
3c63f66a0   Asias He   vhost-scsi: Renam...
1140
  					PTR_ERR(cmd));
055f648c4   Asias He   tcm_vhost: Send b...
1141
  			goto err_cmd;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1142
  		}
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
1143

057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1144
  		pr_debug("Allocated tv_cmd: %p exp_data_len: %d, data_direction"
3c63f66a0   Asias He   vhost-scsi: Renam...
1145
1146
  			": %d
  ", cmd, exp_data_len, data_direction);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1147

3c63f66a0   Asias He   vhost-scsi: Renam...
1148
1149
1150
  		cmd->tvc_vhost = vs;
  		cmd->tvc_vq = vq;
  		cmd->tvc_resp = vq->iov[out].iov_base;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1151

057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1152
1153
  		pr_debug("vhost_scsi got command opcode: %#02x, lun: %d
  ",
3c63f66a0   Asias He   vhost-scsi: Renam...
1154
  			cmd->tvc_cdb[0], cmd->tvc_lun);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1155

95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
1156
1157
1158
1159
1160
1161
1162
1163
1164
1165
1166
  		if (prot_niov) {
  			ret = vhost_scsi_map_iov_to_prot(cmd,
  					&vq->iov[prot_first], prot_niov,
  					data_direction == DMA_FROM_DEVICE);
  			if (unlikely(ret)) {
  				vq_err(vq, "Failed to map iov to"
  					" prot_sgl
  ");
  				goto err_free;
  			}
  		}
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1167
  		if (data_direction != DMA_NONE) {
3c63f66a0   Asias He   vhost-scsi: Renam...
1168
  			ret = vhost_scsi_map_iov_to_sgl(cmd,
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
1169
  					&vq->iov[data_first], data_niov,
60a01f558   Nicholas Bellinger   vhost/scsi: Fix i...
1170
  					data_direction == DMA_FROM_DEVICE);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1171
1172
1173
  			if (unlikely(ret)) {
  				vq_err(vq, "Failed to map iov to sgl
  ");
055f648c4   Asias He   tcm_vhost: Send b...
1174
  				goto err_free;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1175
1176
  			}
  		}
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1177
1178
1179
1180
1181
  		/*
  		 * Save the descriptor from vhost_get_vq_desc() to be used to
  		 * complete the virtio-scsi request in TCM callback context via
  		 * tcm_vhost_queue_data_in() and tcm_vhost_queue_status()
  		 */
3c63f66a0   Asias He   vhost-scsi: Renam...
1182
  		cmd->tvc_vq_desc = head;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1183
1184
1185
1186
1187
1188
  		/*
  		 * Dispatch tv_cmd descriptor for cmwq execution in process
  		 * context provided by tcm_vhost_workqueue.  This also ensures
  		 * tv_cmd is executed on the same kworker CPU as this vhost
  		 * thread to gain positive L2 cache locality effects..
  		 */
3c63f66a0   Asias He   vhost-scsi: Renam...
1189
1190
  		INIT_WORK(&cmd->work, tcm_vhost_submission_work);
  		queue_work(tcm_vhost_workqueue, &cmd->work);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1191
1192
1193
  	}
  
  	mutex_unlock(&vq->mutex);
7ea206cf3   Asias He   tcm_vhost: Fix tv...
1194
  	return;
055f648c4   Asias He   tcm_vhost: Send b...
1195
  err_free:
3c63f66a0   Asias He   vhost-scsi: Renam...
1196
  	vhost_scsi_free_cmd(cmd);
055f648c4   Asias He   tcm_vhost: Send b...
1197
1198
  err_cmd:
  	vhost_scsi_send_bad_target(vs, vq, head, out);
e7802212e   Asias He   vhost-scsi: Alway...
1199
  out:
7ea206cf3   Asias He   tcm_vhost: Fix tv...
1200
  	mutex_unlock(&vq->mutex);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1201
1202
1203
1204
  }
  
  static void vhost_scsi_ctl_handle_kick(struct vhost_work *work)
  {
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1205
1206
  	pr_debug("%s: The handling func for control queue.
  ", __func__);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1207
  }
683bd967d   Asias He   vhost-scsi: Make ...
1208
1209
1210
1211
1212
1213
  static void
  tcm_vhost_send_evt(struct vhost_scsi *vs,
  		   struct tcm_vhost_tpg *tpg,
  		   struct se_lun *lun,
  		   u32 event,
  		   u32 reason)
a6c9af873   Asias He   tcm_vhost: Add ho...
1214
1215
1216
1217
1218
1219
1220
1221
1222
1223
1224
1225
1226
1227
1228
1229
1230
1231
1232
1233
1234
1235
1236
  {
  	struct tcm_vhost_evt *evt;
  
  	evt = tcm_vhost_allocate_evt(vs, event, reason);
  	if (!evt)
  		return;
  
  	if (tpg && lun) {
  		/* TODO: share lun setup code with virtio-scsi.ko */
  		/*
  		 * Note: evt->event is zeroed when we allocate it and
  		 * lun[4-7] need to be zero according to virtio-scsi spec.
  		 */
  		evt->event.lun[0] = 0x01;
  		evt->event.lun[1] = tpg->tport_tpgt & 0xFF;
  		if (lun->unpacked_lun >= 256)
  			evt->event.lun[2] = lun->unpacked_lun >> 8 | 0x40 ;
  		evt->event.lun[3] = lun->unpacked_lun & 0xFF;
  	}
  
  	llist_add(&evt->list, &vs->vs_event_list);
  	vhost_work_queue(&vs->dev, &vs->vs_event_work);
  }
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1237
1238
  static void vhost_scsi_evt_handle_kick(struct vhost_work *work)
  {
a6c9af873   Asias He   tcm_vhost: Add ho...
1239
1240
1241
1242
1243
1244
1245
1246
1247
1248
1249
1250
  	struct vhost_virtqueue *vq = container_of(work, struct vhost_virtqueue,
  						poll.work);
  	struct vhost_scsi *vs = container_of(vq->dev, struct vhost_scsi, dev);
  
  	mutex_lock(&vq->mutex);
  	if (!vq->private_data)
  		goto out;
  
  	if (vs->vs_events_missed)
  		tcm_vhost_send_evt(vs, NULL, NULL, VIRTIO_SCSI_T_NO_EVENT, 0);
  out:
  	mutex_unlock(&vq->mutex);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1251
1252
1253
1254
1255
1256
1257
  }
  
  static void vhost_scsi_handle_kick(struct vhost_work *work)
  {
  	struct vhost_virtqueue *vq = container_of(work, struct vhost_virtqueue,
  						poll.work);
  	struct vhost_scsi *vs = container_of(vq->dev, struct vhost_scsi, dev);
1b7f390eb   Asias He   tcm_vhost: Multi-...
1258
  	vhost_scsi_handle_vq(vs, vq);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1259
  }
4f7f46d32   Asias He   tcm_vhost: Use vq...
1260
1261
  static void vhost_scsi_flush_vq(struct vhost_scsi *vs, int index)
  {
3ab2e420e   Asias He   vhost: Allow devi...
1262
  	vhost_poll_flush(&vs->vqs[index].vq.poll);
4f7f46d32   Asias He   tcm_vhost: Use vq...
1263
  }
3dfbff328   Michael S. Tsirkin   tcm_vhost: docume...
1264
  /* Callers must hold dev mutex */
4f7f46d32   Asias He   tcm_vhost: Use vq...
1265
1266
  static void vhost_scsi_flush(struct vhost_scsi *vs)
  {
f2f0173d6   Asias He   tcm_vhost: Wait f...
1267
  	struct vhost_scsi_inflight *old_inflight[VHOST_SCSI_MAX_VQ];
4f7f46d32   Asias He   tcm_vhost: Use vq...
1268
  	int i;
f2f0173d6   Asias He   tcm_vhost: Wait f...
1269
1270
1271
1272
1273
1274
1275
1276
1277
1278
1279
1280
  	/* Init new inflight and remember the old inflight */
  	tcm_vhost_init_inflight(vs, old_inflight);
  
  	/*
  	 * The inflight->kref was initialized to 1. We decrement it here to
  	 * indicate the start of the flush operation so that it will reach 0
  	 * when all the reqs are finished.
  	 */
  	for (i = 0; i < VHOST_SCSI_MAX_VQ; i++)
  		kref_put(&old_inflight[i]->kref, tcm_vhost_done_inflight);
  
  	/* Flush both the vhost poll and vhost work */
4f7f46d32   Asias He   tcm_vhost: Use vq...
1281
1282
1283
  	for (i = 0; i < VHOST_SCSI_MAX_VQ; i++)
  		vhost_scsi_flush_vq(vs, i);
  	vhost_work_flush(&vs->dev, &vs->vs_completion_work);
a6c9af873   Asias He   tcm_vhost: Add ho...
1284
  	vhost_work_flush(&vs->dev, &vs->vs_event_work);
f2f0173d6   Asias He   tcm_vhost: Wait f...
1285
1286
1287
1288
  
  	/* Wait for all reqs issued before the flush to be finished */
  	for (i = 0; i < VHOST_SCSI_MAX_VQ; i++)
  		wait_for_completion(&old_inflight[i]->comp);
4f7f46d32   Asias He   tcm_vhost: Use vq...
1289
  }
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1290
1291
1292
  /*
   * Called from vhost_scsi_ioctl() context to walk the list of available
   * tcm_vhost_tpg with an active struct tcm_vhost_nexus
f2b7daf5b   Asias He   tcm_vhost: Refact...
1293
1294
1295
   *
   *  The lock nesting rule is:
   *    tcm_vhost_mutex -> vs->dev.mutex -> tpg->tv_tpg_mutex -> vq->mutex
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1296
   */
683bd967d   Asias He   vhost-scsi: Make ...
1297
1298
1299
  static int
  vhost_scsi_set_endpoint(struct vhost_scsi *vs,
  			struct vhost_scsi_target *t)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1300
1301
  {
  	struct tcm_vhost_tport *tv_tport;
987183128   Asias He   vhost-scsi: Renam...
1302
  	struct tcm_vhost_tpg *tpg;
4f7f46d32   Asias He   tcm_vhost: Use vq...
1303
1304
1305
  	struct tcm_vhost_tpg **vs_tpg;
  	struct vhost_virtqueue *vq;
  	int index, ret, i, len;
67e18cf9a   Asias He   tcm_vhost: Multi-...
1306
  	bool match = false;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1307

f2b7daf5b   Asias He   tcm_vhost: Refact...
1308
  	mutex_lock(&tcm_vhost_mutex);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1309
  	mutex_lock(&vs->dev.mutex);
f2b7daf5b   Asias He   tcm_vhost: Refact...
1310

057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1311
1312
1313
  	/* Verify that ring has been setup correctly. */
  	for (index = 0; index < vs->dev.nvqs; ++index) {
  		/* Verify that ring has been setup correctly. */
3ab2e420e   Asias He   vhost: Allow devi...
1314
  		if (!vhost_vq_access_ok(&vs->vqs[index].vq)) {
f2b7daf5b   Asias He   tcm_vhost: Refact...
1315
1316
  			ret = -EFAULT;
  			goto out;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1317
1318
  		}
  	}
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1319

4f7f46d32   Asias He   tcm_vhost: Use vq...
1320
1321
1322
  	len = sizeof(vs_tpg[0]) * VHOST_SCSI_MAX_TARGET;
  	vs_tpg = kzalloc(len, GFP_KERNEL);
  	if (!vs_tpg) {
f2b7daf5b   Asias He   tcm_vhost: Refact...
1323
1324
  		ret = -ENOMEM;
  		goto out;
4f7f46d32   Asias He   tcm_vhost: Use vq...
1325
1326
1327
  	}
  	if (vs->vs_tpg)
  		memcpy(vs_tpg, vs->vs_tpg, len);
987183128   Asias He   vhost-scsi: Renam...
1328
1329
1330
1331
  	list_for_each_entry(tpg, &tcm_vhost_list, tv_tpg_list) {
  		mutex_lock(&tpg->tv_tpg_mutex);
  		if (!tpg->tpg_nexus) {
  			mutex_unlock(&tpg->tv_tpg_mutex);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1332
1333
  			continue;
  		}
987183128   Asias He   vhost-scsi: Renam...
1334
1335
  		if (tpg->tv_tpg_vhost_count != 0) {
  			mutex_unlock(&tpg->tv_tpg_mutex);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1336
1337
  			continue;
  		}
987183128   Asias He   vhost-scsi: Renam...
1338
  		tv_tport = tpg->tport;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1339

67e18cf9a   Asias He   tcm_vhost: Multi-...
1340
  		if (!strcmp(tv_tport->tport_name, t->vhost_wwpn)) {
987183128   Asias He   vhost-scsi: Renam...
1341
  			if (vs->vs_tpg && vs->vs_tpg[tpg->tport_tpgt]) {
4f7f46d32   Asias He   tcm_vhost: Use vq...
1342
  				kfree(vs_tpg);
987183128   Asias He   vhost-scsi: Renam...
1343
  				mutex_unlock(&tpg->tv_tpg_mutex);
f2b7daf5b   Asias He   tcm_vhost: Refact...
1344
1345
  				ret = -EEXIST;
  				goto out;
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1346
  			}
987183128   Asias He   vhost-scsi: Renam...
1347
1348
1349
  			tpg->tv_tpg_vhost_count++;
  			tpg->vhost_scsi = vs;
  			vs_tpg[tpg->tport_tpgt] = tpg;
4e857c58e   Peter Zijlstra   arch: Mass conver...
1350
  			smp_mb__after_atomic();
67e18cf9a   Asias He   tcm_vhost: Multi-...
1351
  			match = true;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1352
  		}
987183128   Asias He   vhost-scsi: Renam...
1353
  		mutex_unlock(&tpg->tv_tpg_mutex);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1354
  	}
67e18cf9a   Asias He   tcm_vhost: Multi-...
1355
1356
1357
1358
  
  	if (match) {
  		memcpy(vs->vs_vhost_wwpn, t->vhost_wwpn,
  		       sizeof(vs->vs_vhost_wwpn));
4f7f46d32   Asias He   tcm_vhost: Use vq...
1359
  		for (i = 0; i < VHOST_SCSI_MAX_VQ; i++) {
3ab2e420e   Asias He   vhost: Allow devi...
1360
  			vq = &vs->vqs[i].vq;
4f7f46d32   Asias He   tcm_vhost: Use vq...
1361
  			mutex_lock(&vq->mutex);
22fa90c7f   Asias He   vhost: Remove cus...
1362
  			vq->private_data = vs_tpg;
dfd5d5692   Asias He   tcm_vhost: Initia...
1363
  			vhost_init_used(vq);
4f7f46d32   Asias He   tcm_vhost: Use vq...
1364
1365
  			mutex_unlock(&vq->mutex);
  		}
67e18cf9a   Asias He   tcm_vhost: Multi-...
1366
1367
1368
1369
  		ret = 0;
  	} else {
  		ret = -EEXIST;
  	}
4f7f46d32   Asias He   tcm_vhost: Use vq...
1370
1371
1372
1373
1374
1375
1376
  	/*
  	 * Act as synchronize_rcu to make sure access to
  	 * old vs->vs_tpg is finished.
  	 */
  	vhost_scsi_flush(vs);
  	kfree(vs->vs_tpg);
  	vs->vs_tpg = vs_tpg;
f2b7daf5b   Asias He   tcm_vhost: Refact...
1377
  out:
67e18cf9a   Asias He   tcm_vhost: Multi-...
1378
  	mutex_unlock(&vs->dev.mutex);
f2b7daf5b   Asias He   tcm_vhost: Refact...
1379
  	mutex_unlock(&tcm_vhost_mutex);
67e18cf9a   Asias He   tcm_vhost: Multi-...
1380
  	return ret;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1381
  }
683bd967d   Asias He   vhost-scsi: Make ...
1382
1383
1384
  static int
  vhost_scsi_clear_endpoint(struct vhost_scsi *vs,
  			  struct vhost_scsi_target *t)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1385
1386
  {
  	struct tcm_vhost_tport *tv_tport;
987183128   Asias He   vhost-scsi: Renam...
1387
  	struct tcm_vhost_tpg *tpg;
4f7f46d32   Asias He   tcm_vhost: Use vq...
1388
1389
  	struct vhost_virtqueue *vq;
  	bool match = false;
67e18cf9a   Asias He   tcm_vhost: Multi-...
1390
1391
  	int index, ret, i;
  	u8 target;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1392

f2b7daf5b   Asias He   tcm_vhost: Refact...
1393
  	mutex_lock(&tcm_vhost_mutex);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1394
1395
1396
  	mutex_lock(&vs->dev.mutex);
  	/* Verify that ring has been setup correctly. */
  	for (index = 0; index < vs->dev.nvqs; ++index) {
3ab2e420e   Asias He   vhost: Allow devi...
1397
  		if (!vhost_vq_access_ok(&vs->vqs[index].vq)) {
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1398
  			ret = -EFAULT;
038e0af4a   Asias He   tcm_vhost: Add mi...
1399
  			goto err_dev;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1400
1401
  		}
  	}
4f7f46d32   Asias He   tcm_vhost: Use vq...
1402
1403
  
  	if (!vs->vs_tpg) {
f2b7daf5b   Asias He   tcm_vhost: Refact...
1404
1405
  		ret = 0;
  		goto err_dev;
4f7f46d32   Asias He   tcm_vhost: Use vq...
1406
  	}
67e18cf9a   Asias He   tcm_vhost: Multi-...
1407
1408
  	for (i = 0; i < VHOST_SCSI_MAX_TARGET; i++) {
  		target = i;
987183128   Asias He   vhost-scsi: Renam...
1409
1410
  		tpg = vs->vs_tpg[target];
  		if (!tpg)
67e18cf9a   Asias He   tcm_vhost: Multi-...
1411
  			continue;
987183128   Asias He   vhost-scsi: Renam...
1412
1413
  		mutex_lock(&tpg->tv_tpg_mutex);
  		tv_tport = tpg->tport;
67e18cf9a   Asias He   tcm_vhost: Multi-...
1414
1415
  		if (!tv_tport) {
  			ret = -ENODEV;
038e0af4a   Asias He   tcm_vhost: Add mi...
1416
  			goto err_tpg;
67e18cf9a   Asias He   tcm_vhost: Multi-...
1417
1418
1419
  		}
  
  		if (strcmp(tv_tport->tport_name, t->vhost_wwpn)) {
987183128   Asias He   vhost-scsi: Renam...
1420
  			pr_warn("tv_tport->tport_name: %s, tpg->tport_tpgt: %hu"
67e18cf9a   Asias He   tcm_vhost: Multi-...
1421
1422
  				" does not match t->vhost_wwpn: %s, t->vhost_tpgt: %hu
  ",
987183128   Asias He   vhost-scsi: Renam...
1423
  				tv_tport->tport_name, tpg->tport_tpgt,
67e18cf9a   Asias He   tcm_vhost: Multi-...
1424
1425
  				t->vhost_wwpn, t->vhost_tpgt);
  			ret = -EINVAL;
038e0af4a   Asias He   tcm_vhost: Add mi...
1426
  			goto err_tpg;
67e18cf9a   Asias He   tcm_vhost: Multi-...
1427
  		}
987183128   Asias He   vhost-scsi: Renam...
1428
1429
  		tpg->tv_tpg_vhost_count--;
  		tpg->vhost_scsi = NULL;
67e18cf9a   Asias He   tcm_vhost: Multi-...
1430
  		vs->vs_tpg[target] = NULL;
4f7f46d32   Asias He   tcm_vhost: Use vq...
1431
  		match = true;
987183128   Asias He   vhost-scsi: Renam...
1432
  		mutex_unlock(&tpg->tv_tpg_mutex);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1433
  	}
4f7f46d32   Asias He   tcm_vhost: Use vq...
1434
1435
  	if (match) {
  		for (i = 0; i < VHOST_SCSI_MAX_VQ; i++) {
3ab2e420e   Asias He   vhost: Allow devi...
1436
  			vq = &vs->vqs[i].vq;
4f7f46d32   Asias He   tcm_vhost: Use vq...
1437
  			mutex_lock(&vq->mutex);
22fa90c7f   Asias He   vhost: Remove cus...
1438
  			vq->private_data = NULL;
4f7f46d32   Asias He   tcm_vhost: Use vq...
1439
1440
1441
1442
1443
1444
1445
1446
1447
1448
  			mutex_unlock(&vq->mutex);
  		}
  	}
  	/*
  	 * Act as synchronize_rcu to make sure access to
  	 * old vs->vs_tpg is finished.
  	 */
  	vhost_scsi_flush(vs);
  	kfree(vs->vs_tpg);
  	vs->vs_tpg = NULL;
a6c9af873   Asias He   tcm_vhost: Add ho...
1449
  	WARN_ON(vs->vs_events_nr);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1450
  	mutex_unlock(&vs->dev.mutex);
f2b7daf5b   Asias He   tcm_vhost: Refact...
1451
  	mutex_unlock(&tcm_vhost_mutex);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1452
  	return 0;
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1453

038e0af4a   Asias He   tcm_vhost: Add mi...
1454
  err_tpg:
987183128   Asias He   vhost-scsi: Renam...
1455
  	mutex_unlock(&tpg->tv_tpg_mutex);
038e0af4a   Asias He   tcm_vhost: Add mi...
1456
  err_dev:
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1457
  	mutex_unlock(&vs->dev.mutex);
f2b7daf5b   Asias He   tcm_vhost: Refact...
1458
  	mutex_unlock(&tcm_vhost_mutex);
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1459
  	return ret;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1460
  }
4f7f46d32   Asias He   tcm_vhost: Use vq...
1461
1462
  static int vhost_scsi_set_features(struct vhost_scsi *vs, u64 features)
  {
ea16c5143   Michael S. Tsirkin   vhost: move acked...
1463
1464
  	struct vhost_virtqueue *vq;
  	int i;
4f7f46d32   Asias He   tcm_vhost: Use vq...
1465
1466
1467
1468
1469
1470
1471
1472
1473
  	if (features & ~VHOST_SCSI_FEATURES)
  		return -EOPNOTSUPP;
  
  	mutex_lock(&vs->dev.mutex);
  	if ((features & (1 << VHOST_F_LOG_ALL)) &&
  	    !vhost_log_access_ok(&vs->dev)) {
  		mutex_unlock(&vs->dev.mutex);
  		return -EFAULT;
  	}
ea16c5143   Michael S. Tsirkin   vhost: move acked...
1474
1475
1476
1477
1478
1479
1480
  
  	for (i = 0; i < VHOST_SCSI_MAX_VQ; i++) {
  		vq = &vs->vqs[i].vq;
  		mutex_lock(&vq->mutex);
  		vq->acked_features = features;
  		mutex_unlock(&vq->mutex);
  	}
4f7f46d32   Asias He   tcm_vhost: Use vq...
1481
1482
1483
  	mutex_unlock(&vs->dev.mutex);
  	return 0;
  }
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1484
1485
  static int vhost_scsi_open(struct inode *inode, struct file *f)
  {
c7289312f   Asias He   vhost-scsi: Renam...
1486
  	struct vhost_scsi *vs;
3ab2e420e   Asias He   vhost: Allow devi...
1487
  	struct vhost_virtqueue **vqs;
595cb7549   Michael S. Tsirkin   vhost/scsi: use v...
1488
  	int r = -ENOMEM, i;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1489

595cb7549   Michael S. Tsirkin   vhost/scsi: use v...
1490
1491
1492
1493
1494
1495
  	vs = kzalloc(sizeof(*vs), GFP_KERNEL | __GFP_NOWARN | __GFP_REPEAT);
  	if (!vs) {
  		vs = vzalloc(sizeof(*vs));
  		if (!vs)
  			goto err_vs;
  	}
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1496

3ab2e420e   Asias He   vhost: Allow devi...
1497
  	vqs = kmalloc(VHOST_SCSI_MAX_VQ * sizeof(*vqs), GFP_KERNEL);
595cb7549   Michael S. Tsirkin   vhost/scsi: use v...
1498
1499
  	if (!vqs)
  		goto err_vqs;
3ab2e420e   Asias He   vhost: Allow devi...
1500

c7289312f   Asias He   vhost-scsi: Renam...
1501
1502
  	vhost_work_init(&vs->vs_completion_work, vhost_scsi_complete_cmd_work);
  	vhost_work_init(&vs->vs_event_work, tcm_vhost_evt_work);
a6c9af873   Asias He   tcm_vhost: Add ho...
1503

c7289312f   Asias He   vhost-scsi: Renam...
1504
1505
  	vs->vs_events_nr = 0;
  	vs->vs_events_missed = false;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1506

c7289312f   Asias He   vhost-scsi: Renam...
1507
1508
1509
1510
  	vqs[VHOST_SCSI_VQ_CTL] = &vs->vqs[VHOST_SCSI_VQ_CTL].vq;
  	vqs[VHOST_SCSI_VQ_EVT] = &vs->vqs[VHOST_SCSI_VQ_EVT].vq;
  	vs->vqs[VHOST_SCSI_VQ_CTL].vq.handle_kick = vhost_scsi_ctl_handle_kick;
  	vs->vqs[VHOST_SCSI_VQ_EVT].vq.handle_kick = vhost_scsi_evt_handle_kick;
3ab2e420e   Asias He   vhost: Allow devi...
1511
  	for (i = VHOST_SCSI_VQ_IO; i < VHOST_SCSI_MAX_VQ; i++) {
c7289312f   Asias He   vhost-scsi: Renam...
1512
1513
  		vqs[i] = &vs->vqs[i].vq;
  		vs->vqs[i].vq.handle_kick = vhost_scsi_handle_kick;
3ab2e420e   Asias He   vhost: Allow devi...
1514
  	}
59566b6e8   Zhi Yong Wu   vhost: remove the...
1515
  	vhost_dev_init(&vs->dev, vqs, VHOST_SCSI_MAX_VQ);
f2f0173d6   Asias He   tcm_vhost: Wait f...
1516

c7289312f   Asias He   vhost-scsi: Renam...
1517
  	tcm_vhost_init_inflight(vs, NULL);
f2f0173d6   Asias He   tcm_vhost: Wait f...
1518

c7289312f   Asias He   vhost-scsi: Renam...
1519
  	f->private_data = vs;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1520
  	return 0;
595cb7549   Michael S. Tsirkin   vhost/scsi: use v...
1521

595cb7549   Michael S. Tsirkin   vhost/scsi: use v...
1522
  err_vqs:
684044415   Michael S. Tsirkin   vhost-scsi: don't...
1523
  	kvfree(vs);
595cb7549   Michael S. Tsirkin   vhost/scsi: use v...
1524
1525
  err_vs:
  	return r;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1526
1527
1528
1529
  }
  
  static int vhost_scsi_release(struct inode *inode, struct file *f)
  {
c7289312f   Asias He   vhost-scsi: Renam...
1530
  	struct vhost_scsi *vs = f->private_data;
67e18cf9a   Asias He   tcm_vhost: Multi-...
1531
  	struct vhost_scsi_target t;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1532

c7289312f   Asias He   vhost-scsi: Renam...
1533
1534
1535
1536
1537
1538
  	mutex_lock(&vs->dev.mutex);
  	memcpy(t.vhost_wwpn, vs->vs_vhost_wwpn, sizeof(t.vhost_wwpn));
  	mutex_unlock(&vs->dev.mutex);
  	vhost_scsi_clear_endpoint(vs, &t);
  	vhost_dev_stop(&vs->dev);
  	vhost_dev_cleanup(&vs->dev, false);
a6c9af873   Asias He   tcm_vhost: Add ho...
1539
  	/* Jobs can re-queue themselves in evt kick handler. Do extra flush. */
c7289312f   Asias He   vhost-scsi: Renam...
1540
1541
  	vhost_scsi_flush(vs);
  	kfree(vs->dev.vqs);
684044415   Michael S. Tsirkin   vhost-scsi: don't...
1542
  	kvfree(vs);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1543
1544
  	return 0;
  }
683bd967d   Asias He   vhost-scsi: Make ...
1545
1546
1547
1548
  static long
  vhost_scsi_ioctl(struct file *f,
  		 unsigned int ioctl,
  		 unsigned long arg)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1549
1550
1551
1552
1553
  {
  	struct vhost_scsi *vs = f->private_data;
  	struct vhost_scsi_target backend;
  	void __user *argp = (void __user *)arg;
  	u64 __user *featurep = argp;
11c634183   Asias He   tcm_vhost: Add io...
1554
1555
  	u32 __user *eventsp = argp;
  	u32 events_missed;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1556
  	u64 features;
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1557
  	int r, abi_version = VHOST_SCSI_ABI_VERSION;
3ab2e420e   Asias He   vhost: Allow devi...
1558
  	struct vhost_virtqueue *vq = &vs->vqs[VHOST_SCSI_VQ_EVT].vq;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1559
1560
1561
1562
1563
  
  	switch (ioctl) {
  	case VHOST_SCSI_SET_ENDPOINT:
  		if (copy_from_user(&backend, argp, sizeof backend))
  			return -EFAULT;
6de7145ca   Michael S. Tsirkin   tcm_vhost: Fix vh...
1564
1565
  		if (backend.reserved != 0)
  			return -EOPNOTSUPP;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1566
1567
1568
1569
1570
  
  		return vhost_scsi_set_endpoint(vs, &backend);
  	case VHOST_SCSI_CLEAR_ENDPOINT:
  		if (copy_from_user(&backend, argp, sizeof backend))
  			return -EFAULT;
6de7145ca   Michael S. Tsirkin   tcm_vhost: Fix vh...
1571
1572
  		if (backend.reserved != 0)
  			return -EOPNOTSUPP;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1573
1574
1575
  
  		return vhost_scsi_clear_endpoint(vs, &backend);
  	case VHOST_SCSI_GET_ABI_VERSION:
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1576
  		if (copy_to_user(argp, &abi_version, sizeof abi_version))
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1577
1578
  			return -EFAULT;
  		return 0;
11c634183   Asias He   tcm_vhost: Add io...
1579
1580
1581
1582
1583
1584
1585
1586
1587
1588
1589
1590
1591
1592
  	case VHOST_SCSI_SET_EVENTS_MISSED:
  		if (get_user(events_missed, eventsp))
  			return -EFAULT;
  		mutex_lock(&vq->mutex);
  		vs->vs_events_missed = events_missed;
  		mutex_unlock(&vq->mutex);
  		return 0;
  	case VHOST_SCSI_GET_EVENTS_MISSED:
  		mutex_lock(&vq->mutex);
  		events_missed = vs->vs_events_missed;
  		mutex_unlock(&vq->mutex);
  		if (put_user(events_missed, eventsp))
  			return -EFAULT;
  		return 0;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1593
  	case VHOST_GET_FEATURES:
5dade7105   Nicholas Bellinger   tcm_vhost: Avoid ...
1594
  		features = VHOST_SCSI_FEATURES;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1595
1596
1597
1598
1599
1600
1601
1602
1603
  		if (copy_to_user(featurep, &features, sizeof features))
  			return -EFAULT;
  		return 0;
  	case VHOST_SET_FEATURES:
  		if (copy_from_user(&features, featurep, sizeof features))
  			return -EFAULT;
  		return vhost_scsi_set_features(vs, features);
  	default:
  		mutex_lock(&vs->dev.mutex);
935cdee7e   Michael S. Tsirkin   vhost: avoid back...
1604
1605
1606
1607
  		r = vhost_dev_ioctl(&vs->dev, ioctl, argp);
  		/* TODO: flush backend after dev ioctl. */
  		if (r == -ENOIOCTLCMD)
  			r = vhost_vring_ioctl(&vs->dev, ioctl, argp);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1608
1609
1610
1611
  		mutex_unlock(&vs->dev.mutex);
  		return r;
  	}
  }
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1612
1613
1614
1615
1616
1617
1618
  #ifdef CONFIG_COMPAT
  static long vhost_scsi_compat_ioctl(struct file *f, unsigned int ioctl,
  				unsigned long arg)
  {
  	return vhost_scsi_ioctl(f, ioctl, (unsigned long)compat_ptr(arg));
  }
  #endif
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1619
1620
1621
1622
  static const struct file_operations vhost_scsi_fops = {
  	.owner          = THIS_MODULE,
  	.release        = vhost_scsi_release,
  	.unlocked_ioctl = vhost_scsi_ioctl,
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1623
1624
1625
  #ifdef CONFIG_COMPAT
  	.compat_ioctl	= vhost_scsi_compat_ioctl,
  #endif
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1626
1627
1628
1629
1630
1631
1632
1633
1634
1635
1636
1637
1638
1639
1640
1641
1642
1643
1644
1645
1646
1647
1648
1649
1650
1651
1652
1653
1654
1655
1656
1657
1658
1659
1660
  	.open           = vhost_scsi_open,
  	.llseek		= noop_llseek,
  };
  
  static struct miscdevice vhost_scsi_misc = {
  	MISC_DYNAMIC_MINOR,
  	"vhost-scsi",
  	&vhost_scsi_fops,
  };
  
  static int __init vhost_scsi_register(void)
  {
  	return misc_register(&vhost_scsi_misc);
  }
  
  static int vhost_scsi_deregister(void)
  {
  	return misc_deregister(&vhost_scsi_misc);
  }
  
  static char *tcm_vhost_dump_proto_id(struct tcm_vhost_tport *tport)
  {
  	switch (tport->tport_proto_id) {
  	case SCSI_PROTOCOL_SAS:
  		return "SAS";
  	case SCSI_PROTOCOL_FCP:
  		return "FCP";
  	case SCSI_PROTOCOL_ISCSI:
  		return "iSCSI";
  	default:
  		break;
  	}
  
  	return "Unknown";
  }
683bd967d   Asias He   vhost-scsi: Make ...
1661
1662
1663
  static void
  tcm_vhost_do_plug(struct tcm_vhost_tpg *tpg,
  		  struct se_lun *lun, bool plug)
a6c9af873   Asias He   tcm_vhost: Add ho...
1664
1665
1666
1667
1668
1669
1670
1671
1672
1673
  {
  
  	struct vhost_scsi *vs = tpg->vhost_scsi;
  	struct vhost_virtqueue *vq;
  	u32 reason;
  
  	if (!vs)
  		return;
  
  	mutex_lock(&vs->dev.mutex);
a6c9af873   Asias He   tcm_vhost: Add ho...
1674
1675
1676
1677
1678
  
  	if (plug)
  		reason = VIRTIO_SCSI_EVT_RESET_RESCAN;
  	else
  		reason = VIRTIO_SCSI_EVT_RESET_REMOVED;
3ab2e420e   Asias He   vhost: Allow devi...
1679
  	vq = &vs->vqs[VHOST_SCSI_VQ_EVT].vq;
a6c9af873   Asias He   tcm_vhost: Add ho...
1680
  	mutex_lock(&vq->mutex);
ea16c5143   Michael S. Tsirkin   vhost: move acked...
1681
1682
1683
  	if (vhost_has_feature(vq, VIRTIO_SCSI_F_HOTPLUG))
  		tcm_vhost_send_evt(vs, tpg, lun,
  				   VIRTIO_SCSI_T_TRANSPORT_RESET, reason);
a6c9af873   Asias He   tcm_vhost: Add ho...
1684
1685
1686
1687
1688
1689
1690
1691
1692
1693
1694
1695
1696
  	mutex_unlock(&vq->mutex);
  	mutex_unlock(&vs->dev.mutex);
  }
  
  static void tcm_vhost_hotplug(struct tcm_vhost_tpg *tpg, struct se_lun *lun)
  {
  	tcm_vhost_do_plug(tpg, lun, true);
  }
  
  static void tcm_vhost_hotunplug(struct tcm_vhost_tpg *tpg, struct se_lun *lun)
  {
  	tcm_vhost_do_plug(tpg, lun, false);
  }
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1697
  static int tcm_vhost_port_link(struct se_portal_group *se_tpg,
683bd967d   Asias He   vhost-scsi: Make ...
1698
  			       struct se_lun *lun)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1699
  {
987183128   Asias He   vhost-scsi: Renam...
1700
  	struct tcm_vhost_tpg *tpg = container_of(se_tpg,
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1701
  				struct tcm_vhost_tpg, se_tpg);
a6c9af873   Asias He   tcm_vhost: Add ho...
1702
  	mutex_lock(&tcm_vhost_mutex);
987183128   Asias He   vhost-scsi: Renam...
1703
1704
1705
  	mutex_lock(&tpg->tv_tpg_mutex);
  	tpg->tv_tpg_port_count++;
  	mutex_unlock(&tpg->tv_tpg_mutex);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1706

987183128   Asias He   vhost-scsi: Renam...
1707
  	tcm_vhost_hotplug(tpg, lun);
a6c9af873   Asias He   tcm_vhost: Add ho...
1708
1709
  
  	mutex_unlock(&tcm_vhost_mutex);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1710
1711
  	return 0;
  }
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1712
  static void tcm_vhost_port_unlink(struct se_portal_group *se_tpg,
683bd967d   Asias He   vhost-scsi: Make ...
1713
  				  struct se_lun *lun)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1714
  {
987183128   Asias He   vhost-scsi: Renam...
1715
  	struct tcm_vhost_tpg *tpg = container_of(se_tpg,
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1716
  				struct tcm_vhost_tpg, se_tpg);
a6c9af873   Asias He   tcm_vhost: Add ho...
1717
  	mutex_lock(&tcm_vhost_mutex);
987183128   Asias He   vhost-scsi: Renam...
1718
1719
1720
  	mutex_lock(&tpg->tv_tpg_mutex);
  	tpg->tv_tpg_port_count--;
  	mutex_unlock(&tpg->tv_tpg_mutex);
a6c9af873   Asias He   tcm_vhost: Add ho...
1721

987183128   Asias He   vhost-scsi: Renam...
1722
  	tcm_vhost_hotunplug(tpg, lun);
a6c9af873   Asias He   tcm_vhost: Add ho...
1723
1724
  
  	mutex_unlock(&tcm_vhost_mutex);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1725
  }
683bd967d   Asias He   vhost-scsi: Make ...
1726
1727
1728
1729
  static struct se_node_acl *
  tcm_vhost_make_nodeacl(struct se_portal_group *se_tpg,
  		       struct config_group *group,
  		       const char *name)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1730
1731
1732
1733
1734
1735
1736
1737
1738
1739
1740
1741
1742
1743
1744
1745
1746
1747
1748
1749
1750
1751
1752
1753
1754
1755
1756
1757
1758
1759
1760
1761
1762
1763
1764
1765
1766
1767
1768
  {
  	struct se_node_acl *se_nacl, *se_nacl_new;
  	struct tcm_vhost_nacl *nacl;
  	u64 wwpn = 0;
  	u32 nexus_depth;
  
  	/* tcm_vhost_parse_wwn(name, &wwpn, 1) < 0)
  		return ERR_PTR(-EINVAL); */
  	se_nacl_new = tcm_vhost_alloc_fabric_acl(se_tpg);
  	if (!se_nacl_new)
  		return ERR_PTR(-ENOMEM);
  
  	nexus_depth = 1;
  	/*
  	 * se_nacl_new may be released by core_tpg_add_initiator_node_acl()
  	 * when converting a NodeACL from demo mode -> explict
  	 */
  	se_nacl = core_tpg_add_initiator_node_acl(se_tpg, se_nacl_new,
  				name, nexus_depth);
  	if (IS_ERR(se_nacl)) {
  		tcm_vhost_release_fabric_acl(se_tpg, se_nacl_new);
  		return se_nacl;
  	}
  	/*
  	 * Locate our struct tcm_vhost_nacl and set the FC Nport WWPN
  	 */
  	nacl = container_of(se_nacl, struct tcm_vhost_nacl, se_node_acl);
  	nacl->iport_wwpn = wwpn;
  
  	return se_nacl;
  }
  
  static void tcm_vhost_drop_nodeacl(struct se_node_acl *se_acl)
  {
  	struct tcm_vhost_nacl *nacl = container_of(se_acl,
  				struct tcm_vhost_nacl, se_node_acl);
  	core_tpg_del_initiator_node_acl(se_acl->se_tpg, se_acl, 1);
  	kfree(nacl);
  }
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
1769
1770
1771
1772
1773
1774
1775
1776
1777
1778
1779
1780
1781
  static void tcm_vhost_free_cmd_map_res(struct tcm_vhost_nexus *nexus,
  				       struct se_session *se_sess)
  {
  	struct tcm_vhost_cmd *tv_cmd;
  	unsigned int i;
  
  	if (!se_sess->sess_cmd_map)
  		return;
  
  	for (i = 0; i < TCM_VHOST_DEFAULT_TAGS; i++) {
  		tv_cmd = &((struct tcm_vhost_cmd *)se_sess->sess_cmd_map)[i];
  
  		kfree(tv_cmd->tvc_sgl);
b1935f687   Nicholas Bellinger   vhost/scsi: Add p...
1782
  		kfree(tv_cmd->tvc_prot_sgl);
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
1783
1784
1785
  		kfree(tv_cmd->tvc_upages);
  	}
  }
987183128   Asias He   vhost-scsi: Renam...
1786
  static int tcm_vhost_make_nexus(struct tcm_vhost_tpg *tpg,
683bd967d   Asias He   vhost-scsi: Make ...
1787
  				const char *name)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1788
1789
  {
  	struct se_portal_group *se_tpg;
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
1790
  	struct se_session *se_sess;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1791
  	struct tcm_vhost_nexus *tv_nexus;
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
1792
1793
  	struct tcm_vhost_cmd *tv_cmd;
  	unsigned int i;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1794

987183128   Asias He   vhost-scsi: Renam...
1795
1796
1797
1798
1799
  	mutex_lock(&tpg->tv_tpg_mutex);
  	if (tpg->tpg_nexus) {
  		mutex_unlock(&tpg->tv_tpg_mutex);
  		pr_debug("tpg->tpg_nexus already exists
  ");
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1800
1801
  		return -EEXIST;
  	}
987183128   Asias He   vhost-scsi: Renam...
1802
  	se_tpg = &tpg->se_tpg;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1803
1804
1805
  
  	tv_nexus = kzalloc(sizeof(struct tcm_vhost_nexus), GFP_KERNEL);
  	if (!tv_nexus) {
987183128   Asias He   vhost-scsi: Renam...
1806
  		mutex_unlock(&tpg->tv_tpg_mutex);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1807
1808
1809
1810
1811
  		pr_err("Unable to allocate struct tcm_vhost_nexus
  ");
  		return -ENOMEM;
  	}
  	/*
4824d3bfb   Nicholas Bellinger   vhost/scsi: Conve...
1812
1813
  	 *  Initialize the struct se_session pointer and setup tagpool
  	 *  for struct tcm_vhost_cmd descriptors
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1814
  	 */
4824d3bfb   Nicholas Bellinger   vhost/scsi: Conve...
1815
1816
  	tv_nexus->tvn_se_sess = transport_init_session_tags(
  					TCM_VHOST_DEFAULT_TAGS,
e70beee78   Nicholas Bellinger   target: Pass in t...
1817
  					sizeof(struct tcm_vhost_cmd),
95e7c4341   Nicholas Bellinger   vhost/scsi: Enabl...
1818
  					TARGET_PROT_DIN_PASS | TARGET_PROT_DOUT_PASS);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1819
  	if (IS_ERR(tv_nexus->tvn_se_sess)) {
987183128   Asias He   vhost-scsi: Renam...
1820
  		mutex_unlock(&tpg->tv_tpg_mutex);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1821
1822
1823
  		kfree(tv_nexus);
  		return -ENOMEM;
  	}
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
1824
1825
1826
1827
1828
1829
1830
1831
1832
1833
1834
1835
1836
1837
  	se_sess = tv_nexus->tvn_se_sess;
  	for (i = 0; i < TCM_VHOST_DEFAULT_TAGS; i++) {
  		tv_cmd = &((struct tcm_vhost_cmd *)se_sess->sess_cmd_map)[i];
  
  		tv_cmd->tvc_sgl = kzalloc(sizeof(struct scatterlist) *
  					TCM_VHOST_PREALLOC_SGLS, GFP_KERNEL);
  		if (!tv_cmd->tvc_sgl) {
  			mutex_unlock(&tpg->tv_tpg_mutex);
  			pr_err("Unable to allocate tv_cmd->tvc_sgl
  ");
  			goto out;
  		}
  
  		tv_cmd->tvc_upages = kzalloc(sizeof(struct page *) *
5a01d0821   Nicholas Bellinger   vhost/scsi: Move ...
1838
  					TCM_VHOST_PREALLOC_UPAGES, GFP_KERNEL);
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
1839
1840
1841
1842
1843
1844
  		if (!tv_cmd->tvc_upages) {
  			mutex_unlock(&tpg->tv_tpg_mutex);
  			pr_err("Unable to allocate tv_cmd->tvc_upages
  ");
  			goto out;
  		}
b1935f687   Nicholas Bellinger   vhost/scsi: Add p...
1845
1846
1847
1848
1849
1850
1851
1852
1853
  
  		tv_cmd->tvc_prot_sgl = kzalloc(sizeof(struct scatterlist) *
  					TCM_VHOST_PREALLOC_PROT_SGLS, GFP_KERNEL);
  		if (!tv_cmd->tvc_prot_sgl) {
  			mutex_unlock(&tpg->tv_tpg_mutex);
  			pr_err("Unable to allocate tv_cmd->tvc_prot_sgl
  ");
  			goto out;
  		}
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
1854
  	}
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1855
1856
1857
1858
1859
1860
1861
1862
  	/*
  	 * Since we are running in 'demo mode' this call with generate a
  	 * struct se_node_acl for the tcm_vhost struct se_portal_group with
  	 * the SCSI Initiator port name of the passed configfs group 'name'.
  	 */
  	tv_nexus->tvn_se_sess->se_node_acl = core_tpg_check_initiator_node_acl(
  				se_tpg, (unsigned char *)name);
  	if (!tv_nexus->tvn_se_sess->se_node_acl) {
987183128   Asias He   vhost-scsi: Renam...
1863
  		mutex_unlock(&tpg->tv_tpg_mutex);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1864
1865
1866
  		pr_debug("core_tpg_check_initiator_node_acl() failed"
  				" for %s
  ", name);
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
1867
  		goto out;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1868
1869
  	}
  	/*
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1870
  	 * Now register the TCM vhost virtual I_T Nexus as active with the
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1871
1872
1873
1874
  	 * call to __transport_register_session()
  	 */
  	__transport_register_session(se_tpg, tv_nexus->tvn_se_sess->se_node_acl,
  			tv_nexus->tvn_se_sess, tv_nexus);
987183128   Asias He   vhost-scsi: Renam...
1875
  	tpg->tpg_nexus = tv_nexus;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1876

987183128   Asias He   vhost-scsi: Renam...
1877
  	mutex_unlock(&tpg->tv_tpg_mutex);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1878
  	return 0;
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
1879
1880
1881
1882
1883
1884
  
  out:
  	tcm_vhost_free_cmd_map_res(tv_nexus, se_sess);
  	transport_free_session(se_sess);
  	kfree(tv_nexus);
  	return -ENOMEM;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1885
  }
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1886
  static int tcm_vhost_drop_nexus(struct tcm_vhost_tpg *tpg)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1887
1888
1889
1890
1891
1892
1893
1894
1895
1896
1897
1898
1899
1900
1901
1902
  {
  	struct se_session *se_sess;
  	struct tcm_vhost_nexus *tv_nexus;
  
  	mutex_lock(&tpg->tv_tpg_mutex);
  	tv_nexus = tpg->tpg_nexus;
  	if (!tv_nexus) {
  		mutex_unlock(&tpg->tv_tpg_mutex);
  		return -ENODEV;
  	}
  
  	se_sess = tv_nexus->tvn_se_sess;
  	if (!se_sess) {
  		mutex_unlock(&tpg->tv_tpg_mutex);
  		return -ENODEV;
  	}
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1903
  	if (tpg->tv_tpg_port_count != 0) {
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1904
  		mutex_unlock(&tpg->tv_tpg_mutex);
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1905
  		pr_err("Unable to remove TCM_vhost I_T Nexus with"
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1906
1907
  			" active TPG port count: %d
  ",
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1908
1909
  			tpg->tv_tpg_port_count);
  		return -EBUSY;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1910
  	}
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1911
  	if (tpg->tv_tpg_vhost_count != 0) {
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1912
  		mutex_unlock(&tpg->tv_tpg_mutex);
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1913
  		pr_err("Unable to remove TCM_vhost I_T Nexus with"
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1914
1915
  			" active TPG vhost count: %d
  ",
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1916
1917
  			tpg->tv_tpg_vhost_count);
  		return -EBUSY;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1918
  	}
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1919
  	pr_debug("TCM_vhost_ConfigFS: Removing I_T Nexus to emulated"
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1920
1921
1922
  		" %s Initiator Port: %s
  ", tcm_vhost_dump_proto_id(tpg->tport),
  		tv_nexus->tvn_se_sess->se_node_acl->initiatorname);
3aee26b4a   Nicholas Bellinger   vhost/scsi: Add p...
1923
1924
  
  	tcm_vhost_free_cmd_map_res(tv_nexus, se_sess);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1925
  	/*
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1926
  	 * Release the SCSI I_T Nexus to the emulated vhost Target Port
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1927
1928
1929
1930
1931
1932
1933
1934
  	 */
  	transport_deregister_session(tv_nexus->tvn_se_sess);
  	tpg->tpg_nexus = NULL;
  	mutex_unlock(&tpg->tv_tpg_mutex);
  
  	kfree(tv_nexus);
  	return 0;
  }
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1935
  static ssize_t tcm_vhost_tpg_show_nexus(struct se_portal_group *se_tpg,
683bd967d   Asias He   vhost-scsi: Make ...
1936
  					char *page)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1937
  {
987183128   Asias He   vhost-scsi: Renam...
1938
  	struct tcm_vhost_tpg *tpg = container_of(se_tpg,
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1939
1940
1941
  				struct tcm_vhost_tpg, se_tpg);
  	struct tcm_vhost_nexus *tv_nexus;
  	ssize_t ret;
987183128   Asias He   vhost-scsi: Renam...
1942
1943
  	mutex_lock(&tpg->tv_tpg_mutex);
  	tv_nexus = tpg->tpg_nexus;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1944
  	if (!tv_nexus) {
987183128   Asias He   vhost-scsi: Renam...
1945
  		mutex_unlock(&tpg->tv_tpg_mutex);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1946
1947
1948
1949
1950
  		return -ENODEV;
  	}
  	ret = snprintf(page, PAGE_SIZE, "%s
  ",
  			tv_nexus->tvn_se_sess->se_node_acl->initiatorname);
987183128   Asias He   vhost-scsi: Renam...
1951
  	mutex_unlock(&tpg->tv_tpg_mutex);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1952
1953
1954
  
  	return ret;
  }
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
1955
  static ssize_t tcm_vhost_tpg_store_nexus(struct se_portal_group *se_tpg,
683bd967d   Asias He   vhost-scsi: Make ...
1956
1957
  					 const char *page,
  					 size_t count)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1958
  {
987183128   Asias He   vhost-scsi: Renam...
1959
  	struct tcm_vhost_tpg *tpg = container_of(se_tpg,
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1960
  				struct tcm_vhost_tpg, se_tpg);
987183128   Asias He   vhost-scsi: Renam...
1961
  	struct tcm_vhost_tport *tport_wwn = tpg->tport;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1962
1963
1964
1965
1966
1967
  	unsigned char i_port[TCM_VHOST_NAMELEN], *ptr, *port_ptr;
  	int ret;
  	/*
  	 * Shutdown the active I_T nexus if 'NULL' is passed..
  	 */
  	if (!strncmp(page, "NULL", 4)) {
987183128   Asias He   vhost-scsi: Renam...
1968
  		ret = tcm_vhost_drop_nexus(tpg);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
1969
1970
1971
1972
1973
1974
1975
1976
1977
1978
1979
1980
1981
1982
1983
1984
1985
1986
1987
1988
1989
1990
1991
1992
1993
1994
1995
1996
1997
1998
1999
2000
2001
2002
2003
2004
2005
2006
2007
2008
2009
2010
2011
2012
2013
2014
2015
2016
2017
2018
2019
2020
2021
2022
2023
2024
2025
2026
2027
2028
2029
2030
  		return (!ret) ? count : ret;
  	}
  	/*
  	 * Otherwise make sure the passed virtual Initiator port WWN matches
  	 * the fabric protocol_id set in tcm_vhost_make_tport(), and call
  	 * tcm_vhost_make_nexus().
  	 */
  	if (strlen(page) >= TCM_VHOST_NAMELEN) {
  		pr_err("Emulated NAA Sas Address: %s, exceeds"
  				" max: %d
  ", page, TCM_VHOST_NAMELEN);
  		return -EINVAL;
  	}
  	snprintf(&i_port[0], TCM_VHOST_NAMELEN, "%s", page);
  
  	ptr = strstr(i_port, "naa.");
  	if (ptr) {
  		if (tport_wwn->tport_proto_id != SCSI_PROTOCOL_SAS) {
  			pr_err("Passed SAS Initiator Port %s does not"
  				" match target port protoid: %s
  ", i_port,
  				tcm_vhost_dump_proto_id(tport_wwn));
  			return -EINVAL;
  		}
  		port_ptr = &i_port[0];
  		goto check_newline;
  	}
  	ptr = strstr(i_port, "fc.");
  	if (ptr) {
  		if (tport_wwn->tport_proto_id != SCSI_PROTOCOL_FCP) {
  			pr_err("Passed FCP Initiator Port %s does not"
  				" match target port protoid: %s
  ", i_port,
  				tcm_vhost_dump_proto_id(tport_wwn));
  			return -EINVAL;
  		}
  		port_ptr = &i_port[3]; /* Skip over "fc." */
  		goto check_newline;
  	}
  	ptr = strstr(i_port, "iqn.");
  	if (ptr) {
  		if (tport_wwn->tport_proto_id != SCSI_PROTOCOL_ISCSI) {
  			pr_err("Passed iSCSI Initiator Port %s does not"
  				" match target port protoid: %s
  ", i_port,
  				tcm_vhost_dump_proto_id(tport_wwn));
  			return -EINVAL;
  		}
  		port_ptr = &i_port[0];
  		goto check_newline;
  	}
  	pr_err("Unable to locate prefix for emulated Initiator Port:"
  			" %s
  ", i_port);
  	return -EINVAL;
  	/*
  	 * Clear any trailing newline for the NAA WWN
  	 */
  check_newline:
  	if (i_port[strlen(i_port)-1] == '
  ')
  		i_port[strlen(i_port)-1] = '\0';
987183128   Asias He   vhost-scsi: Renam...
2031
  	ret = tcm_vhost_make_nexus(tpg, port_ptr);
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
2032
2033
2034
2035
2036
2037
2038
2039
2040
2041
2042
2043
  	if (ret < 0)
  		return ret;
  
  	return count;
  }
  
  TF_TPG_BASE_ATTR(tcm_vhost, nexus, S_IRUGO | S_IWUSR);
  
  static struct configfs_attribute *tcm_vhost_tpg_attrs[] = {
  	&tcm_vhost_tpg_nexus.attr,
  	NULL,
  };
683bd967d   Asias He   vhost-scsi: Make ...
2044
2045
2046
2047
  static struct se_portal_group *
  tcm_vhost_make_tpg(struct se_wwn *wwn,
  		   struct config_group *group,
  		   const char *name)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
2048
2049
2050
2051
2052
2053
2054
2055
2056
2057
2058
2059
2060
2061
2062
2063
2064
2065
2066
2067
2068
2069
2070
2071
2072
2073
2074
2075
2076
2077
2078
2079
2080
2081
2082
2083
2084
2085
2086
2087
2088
2089
2090
2091
2092
  {
  	struct tcm_vhost_tport *tport = container_of(wwn,
  			struct tcm_vhost_tport, tport_wwn);
  
  	struct tcm_vhost_tpg *tpg;
  	unsigned long tpgt;
  	int ret;
  
  	if (strstr(name, "tpgt_") != name)
  		return ERR_PTR(-EINVAL);
  	if (kstrtoul(name + 5, 10, &tpgt) || tpgt > UINT_MAX)
  		return ERR_PTR(-EINVAL);
  
  	tpg = kzalloc(sizeof(struct tcm_vhost_tpg), GFP_KERNEL);
  	if (!tpg) {
  		pr_err("Unable to allocate struct tcm_vhost_tpg");
  		return ERR_PTR(-ENOMEM);
  	}
  	mutex_init(&tpg->tv_tpg_mutex);
  	INIT_LIST_HEAD(&tpg->tv_tpg_list);
  	tpg->tport = tport;
  	tpg->tport_tpgt = tpgt;
  
  	ret = core_tpg_register(&tcm_vhost_fabric_configfs->tf_ops, wwn,
  				&tpg->se_tpg, tpg, TRANSPORT_TPG_TYPE_NORMAL);
  	if (ret < 0) {
  		kfree(tpg);
  		return NULL;
  	}
  	mutex_lock(&tcm_vhost_mutex);
  	list_add_tail(&tpg->tv_tpg_list, &tcm_vhost_list);
  	mutex_unlock(&tcm_vhost_mutex);
  
  	return &tpg->se_tpg;
  }
  
  static void tcm_vhost_drop_tpg(struct se_portal_group *se_tpg)
  {
  	struct tcm_vhost_tpg *tpg = container_of(se_tpg,
  				struct tcm_vhost_tpg, se_tpg);
  
  	mutex_lock(&tcm_vhost_mutex);
  	list_del(&tpg->tv_tpg_list);
  	mutex_unlock(&tcm_vhost_mutex);
  	/*
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
2093
  	 * Release the virtual I_T Nexus for this vhost TPG
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
2094
2095
2096
2097
2098
2099
2100
2101
  	 */
  	tcm_vhost_drop_nexus(tpg);
  	/*
  	 * Deregister the se_tpg from TCM..
  	 */
  	core_tpg_deregister(se_tpg);
  	kfree(tpg);
  }
683bd967d   Asias He   vhost-scsi: Make ...
2102
2103
2104
2105
  static struct se_wwn *
  tcm_vhost_make_tport(struct target_fabric_configfs *tf,
  		     struct config_group *group,
  		     const char *name)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
2106
2107
2108
2109
2110
2111
2112
2113
2114
2115
2116
2117
2118
2119
2120
2121
2122
2123
2124
2125
2126
2127
2128
2129
2130
2131
2132
2133
2134
2135
2136
2137
2138
2139
2140
2141
2142
2143
2144
2145
2146
2147
2148
2149
2150
2151
2152
2153
2154
2155
2156
2157
2158
2159
2160
2161
2162
2163
2164
2165
2166
2167
2168
2169
2170
2171
2172
2173
2174
2175
2176
2177
  {
  	struct tcm_vhost_tport *tport;
  	char *ptr;
  	u64 wwpn = 0;
  	int off = 0;
  
  	/* if (tcm_vhost_parse_wwn(name, &wwpn, 1) < 0)
  		return ERR_PTR(-EINVAL); */
  
  	tport = kzalloc(sizeof(struct tcm_vhost_tport), GFP_KERNEL);
  	if (!tport) {
  		pr_err("Unable to allocate struct tcm_vhost_tport");
  		return ERR_PTR(-ENOMEM);
  	}
  	tport->tport_wwpn = wwpn;
  	/*
  	 * Determine the emulated Protocol Identifier and Target Port Name
  	 * based on the incoming configfs directory name.
  	 */
  	ptr = strstr(name, "naa.");
  	if (ptr) {
  		tport->tport_proto_id = SCSI_PROTOCOL_SAS;
  		goto check_len;
  	}
  	ptr = strstr(name, "fc.");
  	if (ptr) {
  		tport->tport_proto_id = SCSI_PROTOCOL_FCP;
  		off = 3; /* Skip over "fc." */
  		goto check_len;
  	}
  	ptr = strstr(name, "iqn.");
  	if (ptr) {
  		tport->tport_proto_id = SCSI_PROTOCOL_ISCSI;
  		goto check_len;
  	}
  
  	pr_err("Unable to locate prefix for emulated Target Port:"
  			" %s
  ", name);
  	kfree(tport);
  	return ERR_PTR(-EINVAL);
  
  check_len:
  	if (strlen(name) >= TCM_VHOST_NAMELEN) {
  		pr_err("Emulated %s Address: %s, exceeds"
  			" max: %d
  ", name, tcm_vhost_dump_proto_id(tport),
  			TCM_VHOST_NAMELEN);
  		kfree(tport);
  		return ERR_PTR(-EINVAL);
  	}
  	snprintf(&tport->tport_name[0], TCM_VHOST_NAMELEN, "%s", &name[off]);
  
  	pr_debug("TCM_VHost_ConfigFS: Allocated emulated Target"
  		" %s Address: %s
  ", tcm_vhost_dump_proto_id(tport), name);
  
  	return &tport->tport_wwn;
  }
  
  static void tcm_vhost_drop_tport(struct se_wwn *wwn)
  {
  	struct tcm_vhost_tport *tport = container_of(wwn,
  				struct tcm_vhost_tport, tport_wwn);
  
  	pr_debug("TCM_VHost_ConfigFS: Deallocating emulated Target"
  		" %s Address: %s
  ", tcm_vhost_dump_proto_id(tport),
  		tport->tport_name);
  
  	kfree(tport);
  }
683bd967d   Asias He   vhost-scsi: Make ...
2178
2179
2180
  static ssize_t
  tcm_vhost_wwn_show_attr_version(struct target_fabric_configfs *tf,
  				char *page)
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
2181
2182
2183
2184
2185
2186
2187
2188
2189
2190
2191
2192
2193
2194
2195
2196
2197
2198
2199
2200
2201
2202
2203
2204
2205
2206
2207
2208
2209
2210
2211
  {
  	return sprintf(page, "TCM_VHOST fabric module %s on %s/%s"
  		"on "UTS_RELEASE"
  ", TCM_VHOST_VERSION, utsname()->sysname,
  		utsname()->machine);
  }
  
  TF_WWN_ATTR_RO(tcm_vhost, version);
  
  static struct configfs_attribute *tcm_vhost_wwn_attrs[] = {
  	&tcm_vhost_wwn_version.attr,
  	NULL,
  };
  
  static struct target_core_fabric_ops tcm_vhost_ops = {
  	.get_fabric_name		= tcm_vhost_get_fabric_name,
  	.get_fabric_proto_ident		= tcm_vhost_get_fabric_proto_ident,
  	.tpg_get_wwn			= tcm_vhost_get_fabric_wwn,
  	.tpg_get_tag			= tcm_vhost_get_tag,
  	.tpg_get_default_depth		= tcm_vhost_get_default_depth,
  	.tpg_get_pr_transport_id	= tcm_vhost_get_pr_transport_id,
  	.tpg_get_pr_transport_id_len	= tcm_vhost_get_pr_transport_id_len,
  	.tpg_parse_pr_out_transport_id	= tcm_vhost_parse_pr_out_transport_id,
  	.tpg_check_demo_mode		= tcm_vhost_check_true,
  	.tpg_check_demo_mode_cache	= tcm_vhost_check_true,
  	.tpg_check_demo_mode_write_protect = tcm_vhost_check_false,
  	.tpg_check_prod_mode_write_protect = tcm_vhost_check_false,
  	.tpg_alloc_fabric_acl		= tcm_vhost_alloc_fabric_acl,
  	.tpg_release_fabric_acl		= tcm_vhost_release_fabric_acl,
  	.tpg_get_inst_index		= tcm_vhost_tpg_get_inst_index,
  	.release_cmd			= tcm_vhost_release_cmd,
084ed45b3   Nicholas Bellinger   vhost/scsi: Conve...
2212
  	.check_stop_free		= vhost_scsi_check_stop_free,
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
2213
2214
2215
2216
2217
2218
2219
2220
2221
2222
2223
2224
  	.shutdown_session		= tcm_vhost_shutdown_session,
  	.close_session			= tcm_vhost_close_session,
  	.sess_get_index			= tcm_vhost_sess_get_index,
  	.sess_get_initiator_sid		= NULL,
  	.write_pending			= tcm_vhost_write_pending,
  	.write_pending_status		= tcm_vhost_write_pending_status,
  	.set_default_node_attributes	= tcm_vhost_set_default_node_attrs,
  	.get_task_tag			= tcm_vhost_get_task_tag,
  	.get_cmd_state			= tcm_vhost_get_cmd_state,
  	.queue_data_in			= tcm_vhost_queue_data_in,
  	.queue_status			= tcm_vhost_queue_status,
  	.queue_tm_rsp			= tcm_vhost_queue_tm_rsp,
131e6abc6   Nicholas Bellinger   target: Add TFO->...
2225
  	.aborted_task			= tcm_vhost_aborted_task,
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
2226
2227
2228
2229
2230
2231
2232
2233
2234
2235
2236
2237
2238
2239
2240
2241
2242
2243
2244
2245
2246
2247
2248
2249
2250
2251
2252
2253
2254
2255
2256
2257
2258
2259
2260
2261
2262
2263
2264
2265
  	/*
  	 * Setup callers for generic logic in target_core_fabric_configfs.c
  	 */
  	.fabric_make_wwn		= tcm_vhost_make_tport,
  	.fabric_drop_wwn		= tcm_vhost_drop_tport,
  	.fabric_make_tpg		= tcm_vhost_make_tpg,
  	.fabric_drop_tpg		= tcm_vhost_drop_tpg,
  	.fabric_post_link		= tcm_vhost_port_link,
  	.fabric_pre_unlink		= tcm_vhost_port_unlink,
  	.fabric_make_np			= NULL,
  	.fabric_drop_np			= NULL,
  	.fabric_make_nodeacl		= tcm_vhost_make_nodeacl,
  	.fabric_drop_nodeacl		= tcm_vhost_drop_nodeacl,
  };
  
  static int tcm_vhost_register_configfs(void)
  {
  	struct target_fabric_configfs *fabric;
  	int ret;
  
  	pr_debug("TCM_VHOST fabric module %s on %s/%s"
  		" on "UTS_RELEASE"
  ", TCM_VHOST_VERSION, utsname()->sysname,
  		utsname()->machine);
  	/*
  	 * Register the top level struct config_item_type with TCM core
  	 */
  	fabric = target_fabric_configfs_init(THIS_MODULE, "vhost");
  	if (IS_ERR(fabric)) {
  		pr_err("target_fabric_configfs_init() failed
  ");
  		return PTR_ERR(fabric);
  	}
  	/*
  	 * Setup fabric->tf_ops from our local tcm_vhost_ops
  	 */
  	fabric->tf_ops = tcm_vhost_ops;
  	/*
  	 * Setup default attribute lists for various fabric->tf_cit_tmpl
  	 */
d80e224dd   Andy Grover   target: Remove TF...
2266
2267
2268
2269
2270
2271
2272
2273
2274
  	fabric->tf_cit_tmpl.tfc_wwn_cit.ct_attrs = tcm_vhost_wwn_attrs;
  	fabric->tf_cit_tmpl.tfc_tpg_base_cit.ct_attrs = tcm_vhost_tpg_attrs;
  	fabric->tf_cit_tmpl.tfc_tpg_attrib_cit.ct_attrs = NULL;
  	fabric->tf_cit_tmpl.tfc_tpg_param_cit.ct_attrs = NULL;
  	fabric->tf_cit_tmpl.tfc_tpg_np_base_cit.ct_attrs = NULL;
  	fabric->tf_cit_tmpl.tfc_tpg_nacl_base_cit.ct_attrs = NULL;
  	fabric->tf_cit_tmpl.tfc_tpg_nacl_attrib_cit.ct_attrs = NULL;
  	fabric->tf_cit_tmpl.tfc_tpg_nacl_auth_cit.ct_attrs = NULL;
  	fabric->tf_cit_tmpl.tfc_tpg_nacl_param_cit.ct_attrs = NULL;
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
2275
2276
2277
2278
2279
2280
2281
2282
2283
2284
2285
2286
2287
2288
2289
2290
2291
2292
2293
2294
2295
2296
2297
2298
2299
2300
2301
2302
2303
2304
2305
2306
2307
  	/*
  	 * Register the fabric for use within TCM
  	 */
  	ret = target_fabric_configfs_register(fabric);
  	if (ret < 0) {
  		pr_err("target_fabric_configfs_register() failed"
  				" for TCM_VHOST
  ");
  		return ret;
  	}
  	/*
  	 * Setup our local pointer to *fabric
  	 */
  	tcm_vhost_fabric_configfs = fabric;
  	pr_debug("TCM_VHOST[0] - Set fabric -> tcm_vhost_fabric_configfs
  ");
  	return 0;
  };
  
  static void tcm_vhost_deregister_configfs(void)
  {
  	if (!tcm_vhost_fabric_configfs)
  		return;
  
  	target_fabric_configfs_deregister(tcm_vhost_fabric_configfs);
  	tcm_vhost_fabric_configfs = NULL;
  	pr_debug("TCM_VHOST[0] - Cleared tcm_vhost_fabric_configfs
  ");
  };
  
  static int __init tcm_vhost_init(void)
  {
  	int ret = -ENOMEM;
101998f6f   Nicholas Bellinger   tcm_vhost: Post-m...
2308
2309
2310
2311
  	/*
  	 * Use our own dedicated workqueue for submitting I/O into
  	 * target core to avoid contention within system_wq.
  	 */
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
2312
2313
2314
2315
2316
2317
2318
2319
2320
2321
2322
2323
2324
2325
2326
2327
2328
2329
2330
2331
2332
2333
2334
2335
2336
2337
2338
2339
  	tcm_vhost_workqueue = alloc_workqueue("tcm_vhost", 0, 0);
  	if (!tcm_vhost_workqueue)
  		goto out;
  
  	ret = vhost_scsi_register();
  	if (ret < 0)
  		goto out_destroy_workqueue;
  
  	ret = tcm_vhost_register_configfs();
  	if (ret < 0)
  		goto out_vhost_scsi_deregister;
  
  	return 0;
  
  out_vhost_scsi_deregister:
  	vhost_scsi_deregister();
  out_destroy_workqueue:
  	destroy_workqueue(tcm_vhost_workqueue);
  out:
  	return ret;
  };
  
  static void tcm_vhost_exit(void)
  {
  	tcm_vhost_deregister_configfs();
  	vhost_scsi_deregister();
  	destroy_workqueue(tcm_vhost_workqueue);
  };
181c04a35   Michael S. Tsirkin   vhost_scsi: modul...
2340
2341
  MODULE_DESCRIPTION("VHOST_SCSI series fabric driver");
  MODULE_ALIAS("tcm_vhost");
057cbf49a   Nicholas Bellinger   tcm_vhost: Initia...
2342
2343
2344
  MODULE_LICENSE("GPL");
  module_init(tcm_vhost_init);
  module_exit(tcm_vhost_exit);