xref: /openbmc/linux/drivers/nvme/target/configfs.c (revision ba76af67)
177141dc6SChristoph Hellwig // SPDX-License-Identifier: GPL-2.0
2a07b4970SChristoph Hellwig /*
3a07b4970SChristoph Hellwig  * Configfs interface for the NVMe target.
4a07b4970SChristoph Hellwig  * Copyright (c) 2015-2016 HGST, a Western Digital Company.
5a07b4970SChristoph Hellwig  */
6a07b4970SChristoph Hellwig #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
7a07b4970SChristoph Hellwig #include <linux/kernel.h>
8a07b4970SChristoph Hellwig #include <linux/module.h>
9a07b4970SChristoph Hellwig #include <linux/slab.h>
10a07b4970SChristoph Hellwig #include <linux/stat.h>
11a07b4970SChristoph Hellwig #include <linux/ctype.h>
12c6925093SLogan Gunthorpe #include <linux/pci.h>
13c6925093SLogan Gunthorpe #include <linux/pci-p2pdma.h>
14a07b4970SChristoph Hellwig 
15a07b4970SChristoph Hellwig #include "nvmet.h"
16a07b4970SChristoph Hellwig 
1766603a31SBhumika Goyal static const struct config_item_type nvmet_host_type;
1866603a31SBhumika Goyal static const struct config_item_type nvmet_subsys_type;
19a07b4970SChristoph Hellwig 
20b662a078SJay Sternberg static LIST_HEAD(nvmet_ports_list);
21b662a078SJay Sternberg struct list_head *nvmet_ports = &nvmet_ports_list;
22b662a078SJay Sternberg 
2345e2f3c2SChaitanya Kulkarni struct nvmet_type_name_map {
24a5d18612SChristoph Hellwig 	u8		type;
25a5d18612SChristoph Hellwig 	const char	*name;
2645e2f3c2SChaitanya Kulkarni };
2745e2f3c2SChaitanya Kulkarni 
2845e2f3c2SChaitanya Kulkarni static struct nvmet_type_name_map nvmet_transport[] = {
29a5d18612SChristoph Hellwig 	{ NVMF_TRTYPE_RDMA,	"rdma" },
30a5d18612SChristoph Hellwig 	{ NVMF_TRTYPE_FC,	"fc" },
31ad4f530eSSagi Grimberg 	{ NVMF_TRTYPE_TCP,	"tcp" },
32a5d18612SChristoph Hellwig 	{ NVMF_TRTYPE_LOOP,	"loop" },
33a5d18612SChristoph Hellwig };
34a5d18612SChristoph Hellwig 
357e764179SChaitanya Kulkarni static const struct nvmet_type_name_map nvmet_addr_family[] = {
367e764179SChaitanya Kulkarni 	{ NVMF_ADDR_FAMILY_PCI,		"pcie" },
377e764179SChaitanya Kulkarni 	{ NVMF_ADDR_FAMILY_IP4,		"ipv4" },
387e764179SChaitanya Kulkarni 	{ NVMF_ADDR_FAMILY_IP6,		"ipv6" },
397e764179SChaitanya Kulkarni 	{ NVMF_ADDR_FAMILY_IB,		"ib" },
407e764179SChaitanya Kulkarni 	{ NVMF_ADDR_FAMILY_FC,		"fc" },
41d02abd19SChaitanya Kulkarni 	{ NVMF_ADDR_FAMILY_LOOP,	"loop" },
427e764179SChaitanya Kulkarni };
437e764179SChaitanya Kulkarni 
443ecb5faaSChaitanya Kulkarni static bool nvmet_is_port_enabled(struct nvmet_port *p, const char *caller)
453ecb5faaSChaitanya Kulkarni {
463ecb5faaSChaitanya Kulkarni 	if (p->enabled)
473ecb5faaSChaitanya Kulkarni 		pr_err("Disable port '%u' before changing attribute in %s\n",
483ecb5faaSChaitanya Kulkarni 				le16_to_cpu(p->disc_addr.portid), caller);
493ecb5faaSChaitanya Kulkarni 	return p->enabled;
503ecb5faaSChaitanya Kulkarni }
513ecb5faaSChaitanya Kulkarni 
52a07b4970SChristoph Hellwig /*
53a07b4970SChristoph Hellwig  * nvmet_port Generic ConfigFS definitions.
54a07b4970SChristoph Hellwig  * Used in any place in the ConfigFS tree that refers to an address.
55a07b4970SChristoph Hellwig  */
567e764179SChaitanya Kulkarni static ssize_t nvmet_addr_adrfam_show(struct config_item *item, char *page)
57a07b4970SChristoph Hellwig {
587e764179SChaitanya Kulkarni 	u8 adrfam = to_nvmet_port(item)->disc_addr.adrfam;
597e764179SChaitanya Kulkarni 	int i;
607e764179SChaitanya Kulkarni 
617e764179SChaitanya Kulkarni 	for (i = 1; i < ARRAY_SIZE(nvmet_addr_family); i++) {
627e764179SChaitanya Kulkarni 		if (nvmet_addr_family[i].type == adrfam)
637e764179SChaitanya Kulkarni 			return sprintf(page, "%s\n", nvmet_addr_family[i].name);
64a07b4970SChristoph Hellwig 	}
657e764179SChaitanya Kulkarni 
667e764179SChaitanya Kulkarni 	return sprintf(page, "\n");
67a07b4970SChristoph Hellwig }
68a07b4970SChristoph Hellwig 
69a07b4970SChristoph Hellwig static ssize_t nvmet_addr_adrfam_store(struct config_item *item,
70a07b4970SChristoph Hellwig 		const char *page, size_t count)
71a07b4970SChristoph Hellwig {
72a07b4970SChristoph Hellwig 	struct nvmet_port *port = to_nvmet_port(item);
737e764179SChaitanya Kulkarni 	int i;
74a07b4970SChristoph Hellwig 
753ecb5faaSChaitanya Kulkarni 	if (nvmet_is_port_enabled(port, __func__))
76a07b4970SChristoph Hellwig 		return -EACCES;
77a07b4970SChristoph Hellwig 
787e764179SChaitanya Kulkarni 	for (i = 1; i < ARRAY_SIZE(nvmet_addr_family); i++) {
797e764179SChaitanya Kulkarni 		if (sysfs_streq(page, nvmet_addr_family[i].name))
807e764179SChaitanya Kulkarni 			goto found;
81a07b4970SChristoph Hellwig 	}
82a07b4970SChristoph Hellwig 
837e764179SChaitanya Kulkarni 	pr_err("Invalid value '%s' for adrfam\n", page);
847e764179SChaitanya Kulkarni 	return -EINVAL;
857e764179SChaitanya Kulkarni 
867e764179SChaitanya Kulkarni found:
87d02abd19SChaitanya Kulkarni 	port->disc_addr.adrfam = nvmet_addr_family[i].type;
88a07b4970SChristoph Hellwig 	return count;
89a07b4970SChristoph Hellwig }
90a07b4970SChristoph Hellwig 
91a07b4970SChristoph Hellwig CONFIGFS_ATTR(nvmet_, addr_adrfam);
92a07b4970SChristoph Hellwig 
93a07b4970SChristoph Hellwig static ssize_t nvmet_addr_portid_show(struct config_item *item,
94a07b4970SChristoph Hellwig 		char *page)
95a07b4970SChristoph Hellwig {
96a07b4970SChristoph Hellwig 	struct nvmet_port *port = to_nvmet_port(item);
97a07b4970SChristoph Hellwig 
98a07b4970SChristoph Hellwig 	return snprintf(page, PAGE_SIZE, "%d\n",
99a07b4970SChristoph Hellwig 			le16_to_cpu(port->disc_addr.portid));
100a07b4970SChristoph Hellwig }
101a07b4970SChristoph Hellwig 
102a07b4970SChristoph Hellwig static ssize_t nvmet_addr_portid_store(struct config_item *item,
103a07b4970SChristoph Hellwig 		const char *page, size_t count)
104a07b4970SChristoph Hellwig {
105a07b4970SChristoph Hellwig 	struct nvmet_port *port = to_nvmet_port(item);
106a07b4970SChristoph Hellwig 	u16 portid = 0;
107a07b4970SChristoph Hellwig 
108a07b4970SChristoph Hellwig 	if (kstrtou16(page, 0, &portid)) {
109a07b4970SChristoph Hellwig 		pr_err("Invalid value '%s' for portid\n", page);
110a07b4970SChristoph Hellwig 		return -EINVAL;
111a07b4970SChristoph Hellwig 	}
112a07b4970SChristoph Hellwig 
1133ecb5faaSChaitanya Kulkarni 	if (nvmet_is_port_enabled(port, __func__))
114a07b4970SChristoph Hellwig 		return -EACCES;
1153ecb5faaSChaitanya Kulkarni 
116a07b4970SChristoph Hellwig 	port->disc_addr.portid = cpu_to_le16(portid);
117a07b4970SChristoph Hellwig 	return count;
118a07b4970SChristoph Hellwig }
119a07b4970SChristoph Hellwig 
120a07b4970SChristoph Hellwig CONFIGFS_ATTR(nvmet_, addr_portid);
121a07b4970SChristoph Hellwig 
122a07b4970SChristoph Hellwig static ssize_t nvmet_addr_traddr_show(struct config_item *item,
123a07b4970SChristoph Hellwig 		char *page)
124a07b4970SChristoph Hellwig {
125a07b4970SChristoph Hellwig 	struct nvmet_port *port = to_nvmet_port(item);
126a07b4970SChristoph Hellwig 
127a07b4970SChristoph Hellwig 	return snprintf(page, PAGE_SIZE, "%s\n",
128a07b4970SChristoph Hellwig 			port->disc_addr.traddr);
129a07b4970SChristoph Hellwig }
130a07b4970SChristoph Hellwig 
131a07b4970SChristoph Hellwig static ssize_t nvmet_addr_traddr_store(struct config_item *item,
132a07b4970SChristoph Hellwig 		const char *page, size_t count)
133a07b4970SChristoph Hellwig {
134a07b4970SChristoph Hellwig 	struct nvmet_port *port = to_nvmet_port(item);
135a07b4970SChristoph Hellwig 
136a07b4970SChristoph Hellwig 	if (count > NVMF_TRADDR_SIZE) {
137a07b4970SChristoph Hellwig 		pr_err("Invalid value '%s' for traddr\n", page);
138a07b4970SChristoph Hellwig 		return -EINVAL;
139a07b4970SChristoph Hellwig 	}
140a07b4970SChristoph Hellwig 
1413ecb5faaSChaitanya Kulkarni 	if (nvmet_is_port_enabled(port, __func__))
142a07b4970SChristoph Hellwig 		return -EACCES;
1439ba2a5cbSSagi Grimberg 
1449ba2a5cbSSagi Grimberg 	if (sscanf(page, "%s\n", port->disc_addr.traddr) != 1)
1459ba2a5cbSSagi Grimberg 		return -EINVAL;
1469ba2a5cbSSagi Grimberg 	return count;
147a07b4970SChristoph Hellwig }
148a07b4970SChristoph Hellwig 
149a07b4970SChristoph Hellwig CONFIGFS_ATTR(nvmet_, addr_traddr);
150a07b4970SChristoph Hellwig 
15187628e28SChaitanya Kulkarni static const struct nvmet_type_name_map nvmet_addr_treq[] = {
15287628e28SChaitanya Kulkarni 	{ NVMF_TREQ_NOT_SPECIFIED,	"not specified" },
15387628e28SChaitanya Kulkarni 	{ NVMF_TREQ_REQUIRED,		"required" },
15487628e28SChaitanya Kulkarni 	{ NVMF_TREQ_NOT_REQUIRED,	"not required" },
15587628e28SChaitanya Kulkarni };
15687628e28SChaitanya Kulkarni 
15787628e28SChaitanya Kulkarni static ssize_t nvmet_addr_treq_show(struct config_item *item, char *page)
158a07b4970SChristoph Hellwig {
15987628e28SChaitanya Kulkarni 	u8 treq = to_nvmet_port(item)->disc_addr.treq &
16087628e28SChaitanya Kulkarni 		NVME_TREQ_SECURE_CHANNEL_MASK;
16187628e28SChaitanya Kulkarni 	int i;
16287628e28SChaitanya Kulkarni 
16387628e28SChaitanya Kulkarni 	for (i = 0; i < ARRAY_SIZE(nvmet_addr_treq); i++) {
16487628e28SChaitanya Kulkarni 		if (treq == nvmet_addr_treq[i].type)
16587628e28SChaitanya Kulkarni 			return sprintf(page, "%s\n", nvmet_addr_treq[i].name);
166a07b4970SChristoph Hellwig 	}
16787628e28SChaitanya Kulkarni 
16887628e28SChaitanya Kulkarni 	return sprintf(page, "\n");
169a07b4970SChristoph Hellwig }
170a07b4970SChristoph Hellwig 
171a07b4970SChristoph Hellwig static ssize_t nvmet_addr_treq_store(struct config_item *item,
172a07b4970SChristoph Hellwig 		const char *page, size_t count)
173a07b4970SChristoph Hellwig {
174a07b4970SChristoph Hellwig 	struct nvmet_port *port = to_nvmet_port(item);
1750445e1b5SSagi Grimberg 	u8 treq = port->disc_addr.treq & ~NVME_TREQ_SECURE_CHANNEL_MASK;
17687628e28SChaitanya Kulkarni 	int i;
177a07b4970SChristoph Hellwig 
1783ecb5faaSChaitanya Kulkarni 	if (nvmet_is_port_enabled(port, __func__))
179a07b4970SChristoph Hellwig 		return -EACCES;
180a07b4970SChristoph Hellwig 
18187628e28SChaitanya Kulkarni 	for (i = 0; i < ARRAY_SIZE(nvmet_addr_treq); i++) {
18287628e28SChaitanya Kulkarni 		if (sysfs_streq(page, nvmet_addr_treq[i].name))
18387628e28SChaitanya Kulkarni 			goto found;
18487628e28SChaitanya Kulkarni 	}
18587628e28SChaitanya Kulkarni 
186a07b4970SChristoph Hellwig 	pr_err("Invalid value '%s' for treq\n", page);
187a07b4970SChristoph Hellwig 	return -EINVAL;
188a07b4970SChristoph Hellwig 
18987628e28SChaitanya Kulkarni found:
19087628e28SChaitanya Kulkarni 	treq |= nvmet_addr_treq[i].type;
19187628e28SChaitanya Kulkarni 	port->disc_addr.treq = treq;
192a07b4970SChristoph Hellwig 	return count;
193a07b4970SChristoph Hellwig }
194a07b4970SChristoph Hellwig 
195a07b4970SChristoph Hellwig CONFIGFS_ATTR(nvmet_, addr_treq);
196a07b4970SChristoph Hellwig 
197a07b4970SChristoph Hellwig static ssize_t nvmet_addr_trsvcid_show(struct config_item *item,
198a07b4970SChristoph Hellwig 		char *page)
199a07b4970SChristoph Hellwig {
200a07b4970SChristoph Hellwig 	struct nvmet_port *port = to_nvmet_port(item);
201a07b4970SChristoph Hellwig 
202a07b4970SChristoph Hellwig 	return snprintf(page, PAGE_SIZE, "%s\n",
203a07b4970SChristoph Hellwig 			port->disc_addr.trsvcid);
204a07b4970SChristoph Hellwig }
205a07b4970SChristoph Hellwig 
206a07b4970SChristoph Hellwig static ssize_t nvmet_addr_trsvcid_store(struct config_item *item,
207a07b4970SChristoph Hellwig 		const char *page, size_t count)
208a07b4970SChristoph Hellwig {
209a07b4970SChristoph Hellwig 	struct nvmet_port *port = to_nvmet_port(item);
210a07b4970SChristoph Hellwig 
211a07b4970SChristoph Hellwig 	if (count > NVMF_TRSVCID_SIZE) {
212a07b4970SChristoph Hellwig 		pr_err("Invalid value '%s' for trsvcid\n", page);
213a07b4970SChristoph Hellwig 		return -EINVAL;
214a07b4970SChristoph Hellwig 	}
2153ecb5faaSChaitanya Kulkarni 	if (nvmet_is_port_enabled(port, __func__))
216a07b4970SChristoph Hellwig 		return -EACCES;
2179ba2a5cbSSagi Grimberg 
2189ba2a5cbSSagi Grimberg 	if (sscanf(page, "%s\n", port->disc_addr.trsvcid) != 1)
2199ba2a5cbSSagi Grimberg 		return -EINVAL;
2209ba2a5cbSSagi Grimberg 	return count;
221a07b4970SChristoph Hellwig }
222a07b4970SChristoph Hellwig 
223a07b4970SChristoph Hellwig CONFIGFS_ATTR(nvmet_, addr_trsvcid);
224a07b4970SChristoph Hellwig 
2250d5ee2b2SSteve Wise static ssize_t nvmet_param_inline_data_size_show(struct config_item *item,
2260d5ee2b2SSteve Wise 		char *page)
2270d5ee2b2SSteve Wise {
2280d5ee2b2SSteve Wise 	struct nvmet_port *port = to_nvmet_port(item);
2290d5ee2b2SSteve Wise 
2300d5ee2b2SSteve Wise 	return snprintf(page, PAGE_SIZE, "%d\n", port->inline_data_size);
2310d5ee2b2SSteve Wise }
2320d5ee2b2SSteve Wise 
2330d5ee2b2SSteve Wise static ssize_t nvmet_param_inline_data_size_store(struct config_item *item,
2340d5ee2b2SSteve Wise 		const char *page, size_t count)
2350d5ee2b2SSteve Wise {
2360d5ee2b2SSteve Wise 	struct nvmet_port *port = to_nvmet_port(item);
2370d5ee2b2SSteve Wise 	int ret;
2380d5ee2b2SSteve Wise 
2393ecb5faaSChaitanya Kulkarni 	if (nvmet_is_port_enabled(port, __func__))
2400d5ee2b2SSteve Wise 		return -EACCES;
2410d5ee2b2SSteve Wise 	ret = kstrtoint(page, 0, &port->inline_data_size);
2420d5ee2b2SSteve Wise 	if (ret) {
2430d5ee2b2SSteve Wise 		pr_err("Invalid value '%s' for inline_data_size\n", page);
2440d5ee2b2SSteve Wise 		return -EINVAL;
2450d5ee2b2SSteve Wise 	}
2460d5ee2b2SSteve Wise 	return count;
2470d5ee2b2SSteve Wise }
2480d5ee2b2SSteve Wise 
2490d5ee2b2SSteve Wise CONFIGFS_ATTR(nvmet_, param_inline_data_size);
2500d5ee2b2SSteve Wise 
251ea52ac1cSIsrael Rukshin #ifdef CONFIG_BLK_DEV_INTEGRITY
252ea52ac1cSIsrael Rukshin static ssize_t nvmet_param_pi_enable_show(struct config_item *item,
253ea52ac1cSIsrael Rukshin 		char *page)
254ea52ac1cSIsrael Rukshin {
255ea52ac1cSIsrael Rukshin 	struct nvmet_port *port = to_nvmet_port(item);
256ea52ac1cSIsrael Rukshin 
257ea52ac1cSIsrael Rukshin 	return snprintf(page, PAGE_SIZE, "%d\n", port->pi_enable);
258ea52ac1cSIsrael Rukshin }
259ea52ac1cSIsrael Rukshin 
260ea52ac1cSIsrael Rukshin static ssize_t nvmet_param_pi_enable_store(struct config_item *item,
261ea52ac1cSIsrael Rukshin 		const char *page, size_t count)
262ea52ac1cSIsrael Rukshin {
263ea52ac1cSIsrael Rukshin 	struct nvmet_port *port = to_nvmet_port(item);
264ea52ac1cSIsrael Rukshin 	bool val;
265ea52ac1cSIsrael Rukshin 
266ea52ac1cSIsrael Rukshin 	if (strtobool(page, &val))
267ea52ac1cSIsrael Rukshin 		return -EINVAL;
268ea52ac1cSIsrael Rukshin 
269ea52ac1cSIsrael Rukshin 	if (port->enabled) {
270ea52ac1cSIsrael Rukshin 		pr_err("Disable port before setting pi_enable value.\n");
271ea52ac1cSIsrael Rukshin 		return -EACCES;
272ea52ac1cSIsrael Rukshin 	}
273ea52ac1cSIsrael Rukshin 
274ea52ac1cSIsrael Rukshin 	port->pi_enable = val;
275ea52ac1cSIsrael Rukshin 	return count;
276ea52ac1cSIsrael Rukshin }
277ea52ac1cSIsrael Rukshin 
278ea52ac1cSIsrael Rukshin CONFIGFS_ATTR(nvmet_, param_pi_enable);
279ea52ac1cSIsrael Rukshin #endif
280ea52ac1cSIsrael Rukshin 
281a07b4970SChristoph Hellwig static ssize_t nvmet_addr_trtype_show(struct config_item *item,
282a07b4970SChristoph Hellwig 		char *page)
283a07b4970SChristoph Hellwig {
284a5d18612SChristoph Hellwig 	struct nvmet_port *port = to_nvmet_port(item);
285a5d18612SChristoph Hellwig 	int i;
286a5d18612SChristoph Hellwig 
28745e2f3c2SChaitanya Kulkarni 	for (i = 0; i < ARRAY_SIZE(nvmet_transport); i++) {
28845e2f3c2SChaitanya Kulkarni 		if (port->disc_addr.trtype == nvmet_transport[i].type)
28945e2f3c2SChaitanya Kulkarni 			return sprintf(page, "%s\n", nvmet_transport[i].name);
290a07b4970SChristoph Hellwig 	}
291a5d18612SChristoph Hellwig 
292a5d18612SChristoph Hellwig 	return sprintf(page, "\n");
293a07b4970SChristoph Hellwig }
294a07b4970SChristoph Hellwig 
295a07b4970SChristoph Hellwig static void nvmet_port_init_tsas_rdma(struct nvmet_port *port)
296a07b4970SChristoph Hellwig {
297a07b4970SChristoph Hellwig 	port->disc_addr.tsas.rdma.qptype = NVMF_RDMA_QPTYPE_CONNECTED;
298a07b4970SChristoph Hellwig 	port->disc_addr.tsas.rdma.prtype = NVMF_RDMA_PRTYPE_NOT_SPECIFIED;
299a07b4970SChristoph Hellwig 	port->disc_addr.tsas.rdma.cms = NVMF_RDMA_CMS_RDMA_CM;
300a07b4970SChristoph Hellwig }
301a07b4970SChristoph Hellwig 
302a07b4970SChristoph Hellwig static ssize_t nvmet_addr_trtype_store(struct config_item *item,
303a07b4970SChristoph Hellwig 		const char *page, size_t count)
304a07b4970SChristoph Hellwig {
305a07b4970SChristoph Hellwig 	struct nvmet_port *port = to_nvmet_port(item);
306a5d18612SChristoph Hellwig 	int i;
307a07b4970SChristoph Hellwig 
3083ecb5faaSChaitanya Kulkarni 	if (nvmet_is_port_enabled(port, __func__))
309a07b4970SChristoph Hellwig 		return -EACCES;
310a07b4970SChristoph Hellwig 
31145e2f3c2SChaitanya Kulkarni 	for (i = 0; i < ARRAY_SIZE(nvmet_transport); i++) {
31245e2f3c2SChaitanya Kulkarni 		if (sysfs_streq(page, nvmet_transport[i].name))
313a5d18612SChristoph Hellwig 			goto found;
314a07b4970SChristoph Hellwig 	}
315a07b4970SChristoph Hellwig 
316a5d18612SChristoph Hellwig 	pr_err("Invalid value '%s' for trtype\n", page);
317a5d18612SChristoph Hellwig 	return -EINVAL;
3187e764179SChaitanya Kulkarni 
319a5d18612SChristoph Hellwig found:
320a5d18612SChristoph Hellwig 	memset(&port->disc_addr.tsas, 0, NVMF_TSAS_SIZE);
32145e2f3c2SChaitanya Kulkarni 	port->disc_addr.trtype = nvmet_transport[i].type;
322a5d18612SChristoph Hellwig 	if (port->disc_addr.trtype == NVMF_TRTYPE_RDMA)
323a5d18612SChristoph Hellwig 		nvmet_port_init_tsas_rdma(port);
324a07b4970SChristoph Hellwig 	return count;
325a07b4970SChristoph Hellwig }
326a07b4970SChristoph Hellwig 
327a07b4970SChristoph Hellwig CONFIGFS_ATTR(nvmet_, addr_trtype);
328a07b4970SChristoph Hellwig 
329a07b4970SChristoph Hellwig /*
330a07b4970SChristoph Hellwig  * Namespace structures & file operation functions below
331a07b4970SChristoph Hellwig  */
332a07b4970SChristoph Hellwig static ssize_t nvmet_ns_device_path_show(struct config_item *item, char *page)
333a07b4970SChristoph Hellwig {
334a07b4970SChristoph Hellwig 	return sprintf(page, "%s\n", to_nvmet_ns(item)->device_path);
335a07b4970SChristoph Hellwig }
336a07b4970SChristoph Hellwig 
337a07b4970SChristoph Hellwig static ssize_t nvmet_ns_device_path_store(struct config_item *item,
338a07b4970SChristoph Hellwig 		const char *page, size_t count)
339a07b4970SChristoph Hellwig {
340a07b4970SChristoph Hellwig 	struct nvmet_ns *ns = to_nvmet_ns(item);
341a07b4970SChristoph Hellwig 	struct nvmet_subsys *subsys = ns->subsys;
3425613d312SHannes Reinecke 	size_t len;
343a07b4970SChristoph Hellwig 	int ret;
344a07b4970SChristoph Hellwig 
345a07b4970SChristoph Hellwig 	mutex_lock(&subsys->lock);
346a07b4970SChristoph Hellwig 	ret = -EBUSY;
347e4fcf07cSSolganik Alexander 	if (ns->enabled)
348a07b4970SChristoph Hellwig 		goto out_unlock;
349a07b4970SChristoph Hellwig 
3505613d312SHannes Reinecke 	ret = -EINVAL;
3515613d312SHannes Reinecke 	len = strcspn(page, "\n");
3525613d312SHannes Reinecke 	if (!len)
3535613d312SHannes Reinecke 		goto out_unlock;
354a07b4970SChristoph Hellwig 
3555613d312SHannes Reinecke 	kfree(ns->device_path);
356a07b4970SChristoph Hellwig 	ret = -ENOMEM;
35709bb8986SChen Zhou 	ns->device_path = kmemdup_nul(page, len, GFP_KERNEL);
358a07b4970SChristoph Hellwig 	if (!ns->device_path)
359a07b4970SChristoph Hellwig 		goto out_unlock;
360a07b4970SChristoph Hellwig 
361a07b4970SChristoph Hellwig 	mutex_unlock(&subsys->lock);
362a07b4970SChristoph Hellwig 	return count;
363a07b4970SChristoph Hellwig 
364a07b4970SChristoph Hellwig out_unlock:
365a07b4970SChristoph Hellwig 	mutex_unlock(&subsys->lock);
366a07b4970SChristoph Hellwig 	return ret;
367a07b4970SChristoph Hellwig }
368a07b4970SChristoph Hellwig 
369a07b4970SChristoph Hellwig CONFIGFS_ATTR(nvmet_ns_, device_path);
370a07b4970SChristoph Hellwig 
371c6925093SLogan Gunthorpe #ifdef CONFIG_PCI_P2PDMA
372c6925093SLogan Gunthorpe static ssize_t nvmet_ns_p2pmem_show(struct config_item *item, char *page)
373c6925093SLogan Gunthorpe {
374c6925093SLogan Gunthorpe 	struct nvmet_ns *ns = to_nvmet_ns(item);
375c6925093SLogan Gunthorpe 
376c6925093SLogan Gunthorpe 	return pci_p2pdma_enable_show(page, ns->p2p_dev, ns->use_p2pmem);
377c6925093SLogan Gunthorpe }
378c6925093SLogan Gunthorpe 
379c6925093SLogan Gunthorpe static ssize_t nvmet_ns_p2pmem_store(struct config_item *item,
380c6925093SLogan Gunthorpe 		const char *page, size_t count)
381c6925093SLogan Gunthorpe {
382c6925093SLogan Gunthorpe 	struct nvmet_ns *ns = to_nvmet_ns(item);
383c6925093SLogan Gunthorpe 	struct pci_dev *p2p_dev = NULL;
384c6925093SLogan Gunthorpe 	bool use_p2pmem;
385c6925093SLogan Gunthorpe 	int ret = count;
386c6925093SLogan Gunthorpe 	int error;
387c6925093SLogan Gunthorpe 
388c6925093SLogan Gunthorpe 	mutex_lock(&ns->subsys->lock);
389c6925093SLogan Gunthorpe 	if (ns->enabled) {
390c6925093SLogan Gunthorpe 		ret = -EBUSY;
391c6925093SLogan Gunthorpe 		goto out_unlock;
392c6925093SLogan Gunthorpe 	}
393c6925093SLogan Gunthorpe 
394c6925093SLogan Gunthorpe 	error = pci_p2pdma_enable_store(page, &p2p_dev, &use_p2pmem);
395c6925093SLogan Gunthorpe 	if (error) {
396c6925093SLogan Gunthorpe 		ret = error;
397c6925093SLogan Gunthorpe 		goto out_unlock;
398c6925093SLogan Gunthorpe 	}
399c6925093SLogan Gunthorpe 
400c6925093SLogan Gunthorpe 	ns->use_p2pmem = use_p2pmem;
401c6925093SLogan Gunthorpe 	pci_dev_put(ns->p2p_dev);
402c6925093SLogan Gunthorpe 	ns->p2p_dev = p2p_dev;
403c6925093SLogan Gunthorpe 
404c6925093SLogan Gunthorpe out_unlock:
405c6925093SLogan Gunthorpe 	mutex_unlock(&ns->subsys->lock);
406c6925093SLogan Gunthorpe 
407c6925093SLogan Gunthorpe 	return ret;
408c6925093SLogan Gunthorpe }
409c6925093SLogan Gunthorpe 
410c6925093SLogan Gunthorpe CONFIGFS_ATTR(nvmet_ns_, p2pmem);
411c6925093SLogan Gunthorpe #endif /* CONFIG_PCI_P2PDMA */
412c6925093SLogan Gunthorpe 
413430c7befSJohannes Thumshirn static ssize_t nvmet_ns_device_uuid_show(struct config_item *item, char *page)
414430c7befSJohannes Thumshirn {
415430c7befSJohannes Thumshirn 	return sprintf(page, "%pUb\n", &to_nvmet_ns(item)->uuid);
416430c7befSJohannes Thumshirn }
417430c7befSJohannes Thumshirn 
418430c7befSJohannes Thumshirn static ssize_t nvmet_ns_device_uuid_store(struct config_item *item,
419430c7befSJohannes Thumshirn 					  const char *page, size_t count)
420430c7befSJohannes Thumshirn {
421430c7befSJohannes Thumshirn 	struct nvmet_ns *ns = to_nvmet_ns(item);
422430c7befSJohannes Thumshirn 	struct nvmet_subsys *subsys = ns->subsys;
423430c7befSJohannes Thumshirn 	int ret = 0;
424430c7befSJohannes Thumshirn 
425430c7befSJohannes Thumshirn 	mutex_lock(&subsys->lock);
426430c7befSJohannes Thumshirn 	if (ns->enabled) {
427430c7befSJohannes Thumshirn 		ret = -EBUSY;
428430c7befSJohannes Thumshirn 		goto out_unlock;
429430c7befSJohannes Thumshirn 	}
430430c7befSJohannes Thumshirn 
431430c7befSJohannes Thumshirn 	if (uuid_parse(page, &ns->uuid))
432430c7befSJohannes Thumshirn 		ret = -EINVAL;
433430c7befSJohannes Thumshirn 
434430c7befSJohannes Thumshirn out_unlock:
435430c7befSJohannes Thumshirn 	mutex_unlock(&subsys->lock);
436430c7befSJohannes Thumshirn 	return ret ? ret : count;
437430c7befSJohannes Thumshirn }
438430c7befSJohannes Thumshirn 
439f871749aSMax Gurtovoy CONFIGFS_ATTR(nvmet_ns_, device_uuid);
440f871749aSMax Gurtovoy 
441a07b4970SChristoph Hellwig static ssize_t nvmet_ns_device_nguid_show(struct config_item *item, char *page)
442a07b4970SChristoph Hellwig {
443a07b4970SChristoph Hellwig 	return sprintf(page, "%pUb\n", &to_nvmet_ns(item)->nguid);
444a07b4970SChristoph Hellwig }
445a07b4970SChristoph Hellwig 
446a07b4970SChristoph Hellwig static ssize_t nvmet_ns_device_nguid_store(struct config_item *item,
447a07b4970SChristoph Hellwig 		const char *page, size_t count)
448a07b4970SChristoph Hellwig {
449a07b4970SChristoph Hellwig 	struct nvmet_ns *ns = to_nvmet_ns(item);
450a07b4970SChristoph Hellwig 	struct nvmet_subsys *subsys = ns->subsys;
451a07b4970SChristoph Hellwig 	u8 nguid[16];
452a07b4970SChristoph Hellwig 	const char *p = page;
453a07b4970SChristoph Hellwig 	int i;
454a07b4970SChristoph Hellwig 	int ret = 0;
455a07b4970SChristoph Hellwig 
456a07b4970SChristoph Hellwig 	mutex_lock(&subsys->lock);
457e4fcf07cSSolganik Alexander 	if (ns->enabled) {
458a07b4970SChristoph Hellwig 		ret = -EBUSY;
459a07b4970SChristoph Hellwig 		goto out_unlock;
460a07b4970SChristoph Hellwig 	}
461a07b4970SChristoph Hellwig 
462a07b4970SChristoph Hellwig 	for (i = 0; i < 16; i++) {
463a07b4970SChristoph Hellwig 		if (p + 2 > page + count) {
464a07b4970SChristoph Hellwig 			ret = -EINVAL;
465a07b4970SChristoph Hellwig 			goto out_unlock;
466a07b4970SChristoph Hellwig 		}
467a07b4970SChristoph Hellwig 		if (!isxdigit(p[0]) || !isxdigit(p[1])) {
468a07b4970SChristoph Hellwig 			ret = -EINVAL;
469a07b4970SChristoph Hellwig 			goto out_unlock;
470a07b4970SChristoph Hellwig 		}
471a07b4970SChristoph Hellwig 
472a07b4970SChristoph Hellwig 		nguid[i] = (hex_to_bin(p[0]) << 4) | hex_to_bin(p[1]);
473a07b4970SChristoph Hellwig 		p += 2;
474a07b4970SChristoph Hellwig 
475a07b4970SChristoph Hellwig 		if (*p == '-' || *p == ':')
476a07b4970SChristoph Hellwig 			p++;
477a07b4970SChristoph Hellwig 	}
478a07b4970SChristoph Hellwig 
479a07b4970SChristoph Hellwig 	memcpy(&ns->nguid, nguid, sizeof(nguid));
480a07b4970SChristoph Hellwig out_unlock:
481a07b4970SChristoph Hellwig 	mutex_unlock(&subsys->lock);
482a07b4970SChristoph Hellwig 	return ret ? ret : count;
483a07b4970SChristoph Hellwig }
484a07b4970SChristoph Hellwig 
485a07b4970SChristoph Hellwig CONFIGFS_ATTR(nvmet_ns_, device_nguid);
486a07b4970SChristoph Hellwig 
48762ac0d32SChristoph Hellwig static ssize_t nvmet_ns_ana_grpid_show(struct config_item *item, char *page)
48862ac0d32SChristoph Hellwig {
48962ac0d32SChristoph Hellwig 	return sprintf(page, "%u\n", to_nvmet_ns(item)->anagrpid);
49062ac0d32SChristoph Hellwig }
49162ac0d32SChristoph Hellwig 
49262ac0d32SChristoph Hellwig static ssize_t nvmet_ns_ana_grpid_store(struct config_item *item,
49362ac0d32SChristoph Hellwig 		const char *page, size_t count)
49462ac0d32SChristoph Hellwig {
49562ac0d32SChristoph Hellwig 	struct nvmet_ns *ns = to_nvmet_ns(item);
49662ac0d32SChristoph Hellwig 	u32 oldgrpid, newgrpid;
49762ac0d32SChristoph Hellwig 	int ret;
49862ac0d32SChristoph Hellwig 
49962ac0d32SChristoph Hellwig 	ret = kstrtou32(page, 0, &newgrpid);
50062ac0d32SChristoph Hellwig 	if (ret)
50162ac0d32SChristoph Hellwig 		return ret;
50262ac0d32SChristoph Hellwig 
50362ac0d32SChristoph Hellwig 	if (newgrpid < 1 || newgrpid > NVMET_MAX_ANAGRPS)
50462ac0d32SChristoph Hellwig 		return -EINVAL;
50562ac0d32SChristoph Hellwig 
50662ac0d32SChristoph Hellwig 	down_write(&nvmet_ana_sem);
50762ac0d32SChristoph Hellwig 	oldgrpid = ns->anagrpid;
50862ac0d32SChristoph Hellwig 	nvmet_ana_group_enabled[newgrpid]++;
50962ac0d32SChristoph Hellwig 	ns->anagrpid = newgrpid;
51062ac0d32SChristoph Hellwig 	nvmet_ana_group_enabled[oldgrpid]--;
51162ac0d32SChristoph Hellwig 	nvmet_ana_chgcnt++;
51262ac0d32SChristoph Hellwig 	up_write(&nvmet_ana_sem);
51362ac0d32SChristoph Hellwig 
51462ac0d32SChristoph Hellwig 	nvmet_send_ana_event(ns->subsys, NULL);
51562ac0d32SChristoph Hellwig 	return count;
51662ac0d32SChristoph Hellwig }
51762ac0d32SChristoph Hellwig 
51862ac0d32SChristoph Hellwig CONFIGFS_ATTR(nvmet_ns_, ana_grpid);
51962ac0d32SChristoph Hellwig 
520a07b4970SChristoph Hellwig static ssize_t nvmet_ns_enable_show(struct config_item *item, char *page)
521a07b4970SChristoph Hellwig {
522e4fcf07cSSolganik Alexander 	return sprintf(page, "%d\n", to_nvmet_ns(item)->enabled);
523a07b4970SChristoph Hellwig }
524a07b4970SChristoph Hellwig 
525a07b4970SChristoph Hellwig static ssize_t nvmet_ns_enable_store(struct config_item *item,
526a07b4970SChristoph Hellwig 		const char *page, size_t count)
527a07b4970SChristoph Hellwig {
528a07b4970SChristoph Hellwig 	struct nvmet_ns *ns = to_nvmet_ns(item);
529a07b4970SChristoph Hellwig 	bool enable;
530a07b4970SChristoph Hellwig 	int ret = 0;
531a07b4970SChristoph Hellwig 
532a07b4970SChristoph Hellwig 	if (strtobool(page, &enable))
533a07b4970SChristoph Hellwig 		return -EINVAL;
534a07b4970SChristoph Hellwig 
535a07b4970SChristoph Hellwig 	if (enable)
536a07b4970SChristoph Hellwig 		ret = nvmet_ns_enable(ns);
537a07b4970SChristoph Hellwig 	else
538a07b4970SChristoph Hellwig 		nvmet_ns_disable(ns);
539a07b4970SChristoph Hellwig 
540a07b4970SChristoph Hellwig 	return ret ? ret : count;
541a07b4970SChristoph Hellwig }
542a07b4970SChristoph Hellwig 
543a07b4970SChristoph Hellwig CONFIGFS_ATTR(nvmet_ns_, enable);
544a07b4970SChristoph Hellwig 
54555eb942eSChaitanya Kulkarni static ssize_t nvmet_ns_buffered_io_show(struct config_item *item, char *page)
54655eb942eSChaitanya Kulkarni {
54755eb942eSChaitanya Kulkarni 	return sprintf(page, "%d\n", to_nvmet_ns(item)->buffered_io);
54855eb942eSChaitanya Kulkarni }
54955eb942eSChaitanya Kulkarni 
55055eb942eSChaitanya Kulkarni static ssize_t nvmet_ns_buffered_io_store(struct config_item *item,
55155eb942eSChaitanya Kulkarni 		const char *page, size_t count)
55255eb942eSChaitanya Kulkarni {
55355eb942eSChaitanya Kulkarni 	struct nvmet_ns *ns = to_nvmet_ns(item);
55455eb942eSChaitanya Kulkarni 	bool val;
55555eb942eSChaitanya Kulkarni 
55655eb942eSChaitanya Kulkarni 	if (strtobool(page, &val))
55755eb942eSChaitanya Kulkarni 		return -EINVAL;
55855eb942eSChaitanya Kulkarni 
55955eb942eSChaitanya Kulkarni 	mutex_lock(&ns->subsys->lock);
56055eb942eSChaitanya Kulkarni 	if (ns->enabled) {
56155eb942eSChaitanya Kulkarni 		pr_err("disable ns before setting buffered_io value.\n");
56255eb942eSChaitanya Kulkarni 		mutex_unlock(&ns->subsys->lock);
56355eb942eSChaitanya Kulkarni 		return -EINVAL;
56455eb942eSChaitanya Kulkarni 	}
56555eb942eSChaitanya Kulkarni 
56655eb942eSChaitanya Kulkarni 	ns->buffered_io = val;
56755eb942eSChaitanya Kulkarni 	mutex_unlock(&ns->subsys->lock);
56855eb942eSChaitanya Kulkarni 	return count;
56955eb942eSChaitanya Kulkarni }
57055eb942eSChaitanya Kulkarni 
57155eb942eSChaitanya Kulkarni CONFIGFS_ATTR(nvmet_ns_, buffered_io);
57255eb942eSChaitanya Kulkarni 
5731f357548SChaitanya Kulkarni static ssize_t nvmet_ns_revalidate_size_store(struct config_item *item,
5741f357548SChaitanya Kulkarni 		const char *page, size_t count)
5751f357548SChaitanya Kulkarni {
5761f357548SChaitanya Kulkarni 	struct nvmet_ns *ns = to_nvmet_ns(item);
5771f357548SChaitanya Kulkarni 	bool val;
5781f357548SChaitanya Kulkarni 
5791f357548SChaitanya Kulkarni 	if (strtobool(page, &val))
5801f357548SChaitanya Kulkarni 		return -EINVAL;
5811f357548SChaitanya Kulkarni 
5821f357548SChaitanya Kulkarni 	if (!val)
5831f357548SChaitanya Kulkarni 		return -EINVAL;
5841f357548SChaitanya Kulkarni 
5851f357548SChaitanya Kulkarni 	mutex_lock(&ns->subsys->lock);
5861f357548SChaitanya Kulkarni 	if (!ns->enabled) {
5871f357548SChaitanya Kulkarni 		pr_err("enable ns before revalidate.\n");
5881f357548SChaitanya Kulkarni 		mutex_unlock(&ns->subsys->lock);
5891f357548SChaitanya Kulkarni 		return -EINVAL;
5901f357548SChaitanya Kulkarni 	}
5911f357548SChaitanya Kulkarni 	nvmet_ns_revalidate(ns);
5921f357548SChaitanya Kulkarni 	mutex_unlock(&ns->subsys->lock);
5931f357548SChaitanya Kulkarni 	return count;
5941f357548SChaitanya Kulkarni }
5951f357548SChaitanya Kulkarni 
5961f357548SChaitanya Kulkarni CONFIGFS_ATTR_WO(nvmet_ns_, revalidate_size);
5971f357548SChaitanya Kulkarni 
598a07b4970SChristoph Hellwig static struct configfs_attribute *nvmet_ns_attrs[] = {
599a07b4970SChristoph Hellwig 	&nvmet_ns_attr_device_path,
600a07b4970SChristoph Hellwig 	&nvmet_ns_attr_device_nguid,
601430c7befSJohannes Thumshirn 	&nvmet_ns_attr_device_uuid,
60262ac0d32SChristoph Hellwig 	&nvmet_ns_attr_ana_grpid,
603a07b4970SChristoph Hellwig 	&nvmet_ns_attr_enable,
60455eb942eSChaitanya Kulkarni 	&nvmet_ns_attr_buffered_io,
6051f357548SChaitanya Kulkarni 	&nvmet_ns_attr_revalidate_size,
606c6925093SLogan Gunthorpe #ifdef CONFIG_PCI_P2PDMA
607c6925093SLogan Gunthorpe 	&nvmet_ns_attr_p2pmem,
608c6925093SLogan Gunthorpe #endif
609a07b4970SChristoph Hellwig 	NULL,
610a07b4970SChristoph Hellwig };
611a07b4970SChristoph Hellwig 
612a07b4970SChristoph Hellwig static void nvmet_ns_release(struct config_item *item)
613a07b4970SChristoph Hellwig {
614a07b4970SChristoph Hellwig 	struct nvmet_ns *ns = to_nvmet_ns(item);
615a07b4970SChristoph Hellwig 
616a07b4970SChristoph Hellwig 	nvmet_ns_free(ns);
617a07b4970SChristoph Hellwig }
618a07b4970SChristoph Hellwig 
619a07b4970SChristoph Hellwig static struct configfs_item_operations nvmet_ns_item_ops = {
620a07b4970SChristoph Hellwig 	.release		= nvmet_ns_release,
621a07b4970SChristoph Hellwig };
622a07b4970SChristoph Hellwig 
62366603a31SBhumika Goyal static const struct config_item_type nvmet_ns_type = {
624a07b4970SChristoph Hellwig 	.ct_item_ops		= &nvmet_ns_item_ops,
625a07b4970SChristoph Hellwig 	.ct_attrs		= nvmet_ns_attrs,
626a07b4970SChristoph Hellwig 	.ct_owner		= THIS_MODULE,
627a07b4970SChristoph Hellwig };
628a07b4970SChristoph Hellwig 
629a07b4970SChristoph Hellwig static struct config_group *nvmet_ns_make(struct config_group *group,
630a07b4970SChristoph Hellwig 		const char *name)
631a07b4970SChristoph Hellwig {
632a07b4970SChristoph Hellwig 	struct nvmet_subsys *subsys = namespaces_to_subsys(&group->cg_item);
633a07b4970SChristoph Hellwig 	struct nvmet_ns *ns;
634a07b4970SChristoph Hellwig 	int ret;
635a07b4970SChristoph Hellwig 	u32 nsid;
636a07b4970SChristoph Hellwig 
637a07b4970SChristoph Hellwig 	ret = kstrtou32(name, 0, &nsid);
638a07b4970SChristoph Hellwig 	if (ret)
639a07b4970SChristoph Hellwig 		goto out;
640a07b4970SChristoph Hellwig 
641a07b4970SChristoph Hellwig 	ret = -EINVAL;
6425ba89503SMikhail Skorzhinskii 	if (nsid == 0 || nsid == NVME_NSID_ALL) {
6435ba89503SMikhail Skorzhinskii 		pr_err("invalid nsid %#x", nsid);
644a07b4970SChristoph Hellwig 		goto out;
6455ba89503SMikhail Skorzhinskii 	}
646a07b4970SChristoph Hellwig 
647a07b4970SChristoph Hellwig 	ret = -ENOMEM;
648a07b4970SChristoph Hellwig 	ns = nvmet_ns_alloc(subsys, nsid);
649a07b4970SChristoph Hellwig 	if (!ns)
650a07b4970SChristoph Hellwig 		goto out;
651a07b4970SChristoph Hellwig 	config_group_init_type_name(&ns->group, name, &nvmet_ns_type);
652a07b4970SChristoph Hellwig 
653a07b4970SChristoph Hellwig 	pr_info("adding nsid %d to subsystem %s\n", nsid, subsys->subsysnqn);
654a07b4970SChristoph Hellwig 
655a07b4970SChristoph Hellwig 	return &ns->group;
656a07b4970SChristoph Hellwig out:
657a07b4970SChristoph Hellwig 	return ERR_PTR(ret);
658a07b4970SChristoph Hellwig }
659a07b4970SChristoph Hellwig 
660a07b4970SChristoph Hellwig static struct configfs_group_operations nvmet_namespaces_group_ops = {
661a07b4970SChristoph Hellwig 	.make_group		= nvmet_ns_make,
662a07b4970SChristoph Hellwig };
663a07b4970SChristoph Hellwig 
66466603a31SBhumika Goyal static const struct config_item_type nvmet_namespaces_type = {
665a07b4970SChristoph Hellwig 	.ct_group_ops		= &nvmet_namespaces_group_ops,
666a07b4970SChristoph Hellwig 	.ct_owner		= THIS_MODULE,
667a07b4970SChristoph Hellwig };
668a07b4970SChristoph Hellwig 
669a07b4970SChristoph Hellwig static int nvmet_port_subsys_allow_link(struct config_item *parent,
670a07b4970SChristoph Hellwig 		struct config_item *target)
671a07b4970SChristoph Hellwig {
672a07b4970SChristoph Hellwig 	struct nvmet_port *port = to_nvmet_port(parent->ci_parent);
673a07b4970SChristoph Hellwig 	struct nvmet_subsys *subsys;
674a07b4970SChristoph Hellwig 	struct nvmet_subsys_link *link, *p;
675a07b4970SChristoph Hellwig 	int ret;
676a07b4970SChristoph Hellwig 
677a07b4970SChristoph Hellwig 	if (target->ci_type != &nvmet_subsys_type) {
678a07b4970SChristoph Hellwig 		pr_err("can only link subsystems into the subsystems dir.!\n");
679a07b4970SChristoph Hellwig 		return -EINVAL;
680a07b4970SChristoph Hellwig 	}
681a07b4970SChristoph Hellwig 	subsys = to_subsys(target);
682a07b4970SChristoph Hellwig 	link = kmalloc(sizeof(*link), GFP_KERNEL);
683a07b4970SChristoph Hellwig 	if (!link)
684a07b4970SChristoph Hellwig 		return -ENOMEM;
685a07b4970SChristoph Hellwig 	link->subsys = subsys;
686a07b4970SChristoph Hellwig 
687a07b4970SChristoph Hellwig 	down_write(&nvmet_config_sem);
688a07b4970SChristoph Hellwig 	ret = -EEXIST;
689a07b4970SChristoph Hellwig 	list_for_each_entry(p, &port->subsystems, entry) {
690a07b4970SChristoph Hellwig 		if (p->subsys == subsys)
691a07b4970SChristoph Hellwig 			goto out_free_link;
692a07b4970SChristoph Hellwig 	}
693a07b4970SChristoph Hellwig 
694a07b4970SChristoph Hellwig 	if (list_empty(&port->subsystems)) {
695a07b4970SChristoph Hellwig 		ret = nvmet_enable_port(port);
696a07b4970SChristoph Hellwig 		if (ret)
697a07b4970SChristoph Hellwig 			goto out_free_link;
698a07b4970SChristoph Hellwig 	}
699a07b4970SChristoph Hellwig 
700a07b4970SChristoph Hellwig 	list_add_tail(&link->entry, &port->subsystems);
701b662a078SJay Sternberg 	nvmet_port_disc_changed(port, subsys);
702b662a078SJay Sternberg 
703a07b4970SChristoph Hellwig 	up_write(&nvmet_config_sem);
704a07b4970SChristoph Hellwig 	return 0;
705a07b4970SChristoph Hellwig 
706a07b4970SChristoph Hellwig out_free_link:
707a07b4970SChristoph Hellwig 	up_write(&nvmet_config_sem);
708a07b4970SChristoph Hellwig 	kfree(link);
709a07b4970SChristoph Hellwig 	return ret;
710a07b4970SChristoph Hellwig }
711a07b4970SChristoph Hellwig 
712e16769d4SAndrzej Pietrasiewicz static void nvmet_port_subsys_drop_link(struct config_item *parent,
713a07b4970SChristoph Hellwig 		struct config_item *target)
714a07b4970SChristoph Hellwig {
715a07b4970SChristoph Hellwig 	struct nvmet_port *port = to_nvmet_port(parent->ci_parent);
716a07b4970SChristoph Hellwig 	struct nvmet_subsys *subsys = to_subsys(target);
717a07b4970SChristoph Hellwig 	struct nvmet_subsys_link *p;
718a07b4970SChristoph Hellwig 
719a07b4970SChristoph Hellwig 	down_write(&nvmet_config_sem);
720a07b4970SChristoph Hellwig 	list_for_each_entry(p, &port->subsystems, entry) {
721a07b4970SChristoph Hellwig 		if (p->subsys == subsys)
722a07b4970SChristoph Hellwig 			goto found;
723a07b4970SChristoph Hellwig 	}
724a07b4970SChristoph Hellwig 	up_write(&nvmet_config_sem);
725e16769d4SAndrzej Pietrasiewicz 	return;
726a07b4970SChristoph Hellwig 
727a07b4970SChristoph Hellwig found:
728a07b4970SChristoph Hellwig 	list_del(&p->entry);
7293aed8673SLogan Gunthorpe 	nvmet_port_del_ctrls(port, subsys);
730b662a078SJay Sternberg 	nvmet_port_disc_changed(port, subsys);
731b662a078SJay Sternberg 
732a07b4970SChristoph Hellwig 	if (list_empty(&port->subsystems))
733a07b4970SChristoph Hellwig 		nvmet_disable_port(port);
734a07b4970SChristoph Hellwig 	up_write(&nvmet_config_sem);
735a07b4970SChristoph Hellwig 	kfree(p);
736a07b4970SChristoph Hellwig }
737a07b4970SChristoph Hellwig 
738a07b4970SChristoph Hellwig static struct configfs_item_operations nvmet_port_subsys_item_ops = {
739a07b4970SChristoph Hellwig 	.allow_link		= nvmet_port_subsys_allow_link,
740a07b4970SChristoph Hellwig 	.drop_link		= nvmet_port_subsys_drop_link,
741a07b4970SChristoph Hellwig };
742a07b4970SChristoph Hellwig 
74366603a31SBhumika Goyal static const struct config_item_type nvmet_port_subsys_type = {
744a07b4970SChristoph Hellwig 	.ct_item_ops		= &nvmet_port_subsys_item_ops,
745a07b4970SChristoph Hellwig 	.ct_owner		= THIS_MODULE,
746a07b4970SChristoph Hellwig };
747a07b4970SChristoph Hellwig 
748a07b4970SChristoph Hellwig static int nvmet_allowed_hosts_allow_link(struct config_item *parent,
749a07b4970SChristoph Hellwig 		struct config_item *target)
750a07b4970SChristoph Hellwig {
751a07b4970SChristoph Hellwig 	struct nvmet_subsys *subsys = to_subsys(parent->ci_parent);
752a07b4970SChristoph Hellwig 	struct nvmet_host *host;
753a07b4970SChristoph Hellwig 	struct nvmet_host_link *link, *p;
754a07b4970SChristoph Hellwig 	int ret;
755a07b4970SChristoph Hellwig 
756a07b4970SChristoph Hellwig 	if (target->ci_type != &nvmet_host_type) {
757a07b4970SChristoph Hellwig 		pr_err("can only link hosts into the allowed_hosts directory!\n");
758a07b4970SChristoph Hellwig 		return -EINVAL;
759a07b4970SChristoph Hellwig 	}
760a07b4970SChristoph Hellwig 
761a07b4970SChristoph Hellwig 	host = to_host(target);
762a07b4970SChristoph Hellwig 	link = kmalloc(sizeof(*link), GFP_KERNEL);
763a07b4970SChristoph Hellwig 	if (!link)
764a07b4970SChristoph Hellwig 		return -ENOMEM;
765a07b4970SChristoph Hellwig 	link->host = host;
766a07b4970SChristoph Hellwig 
767a07b4970SChristoph Hellwig 	down_write(&nvmet_config_sem);
768a07b4970SChristoph Hellwig 	ret = -EINVAL;
769a07b4970SChristoph Hellwig 	if (subsys->allow_any_host) {
770a07b4970SChristoph Hellwig 		pr_err("can't add hosts when allow_any_host is set!\n");
771a07b4970SChristoph Hellwig 		goto out_free_link;
772a07b4970SChristoph Hellwig 	}
773a07b4970SChristoph Hellwig 
774a07b4970SChristoph Hellwig 	ret = -EEXIST;
775a07b4970SChristoph Hellwig 	list_for_each_entry(p, &subsys->hosts, entry) {
776a07b4970SChristoph Hellwig 		if (!strcmp(nvmet_host_name(p->host), nvmet_host_name(host)))
777a07b4970SChristoph Hellwig 			goto out_free_link;
778a07b4970SChristoph Hellwig 	}
779a07b4970SChristoph Hellwig 	list_add_tail(&link->entry, &subsys->hosts);
780b662a078SJay Sternberg 	nvmet_subsys_disc_changed(subsys, host);
781b662a078SJay Sternberg 
782a07b4970SChristoph Hellwig 	up_write(&nvmet_config_sem);
783a07b4970SChristoph Hellwig 	return 0;
784a07b4970SChristoph Hellwig out_free_link:
785a07b4970SChristoph Hellwig 	up_write(&nvmet_config_sem);
786a07b4970SChristoph Hellwig 	kfree(link);
787a07b4970SChristoph Hellwig 	return ret;
788a07b4970SChristoph Hellwig }
789a07b4970SChristoph Hellwig 
790e16769d4SAndrzej Pietrasiewicz static void nvmet_allowed_hosts_drop_link(struct config_item *parent,
791a07b4970SChristoph Hellwig 		struct config_item *target)
792a07b4970SChristoph Hellwig {
793a07b4970SChristoph Hellwig 	struct nvmet_subsys *subsys = to_subsys(parent->ci_parent);
794a07b4970SChristoph Hellwig 	struct nvmet_host *host = to_host(target);
795a07b4970SChristoph Hellwig 	struct nvmet_host_link *p;
796a07b4970SChristoph Hellwig 
797a07b4970SChristoph Hellwig 	down_write(&nvmet_config_sem);
798a07b4970SChristoph Hellwig 	list_for_each_entry(p, &subsys->hosts, entry) {
799a07b4970SChristoph Hellwig 		if (!strcmp(nvmet_host_name(p->host), nvmet_host_name(host)))
800a07b4970SChristoph Hellwig 			goto found;
801a07b4970SChristoph Hellwig 	}
802a07b4970SChristoph Hellwig 	up_write(&nvmet_config_sem);
803e16769d4SAndrzej Pietrasiewicz 	return;
804a07b4970SChristoph Hellwig 
805a07b4970SChristoph Hellwig found:
806a07b4970SChristoph Hellwig 	list_del(&p->entry);
807b662a078SJay Sternberg 	nvmet_subsys_disc_changed(subsys, host);
808b662a078SJay Sternberg 
809a07b4970SChristoph Hellwig 	up_write(&nvmet_config_sem);
810a07b4970SChristoph Hellwig 	kfree(p);
811a07b4970SChristoph Hellwig }
812a07b4970SChristoph Hellwig 
813a07b4970SChristoph Hellwig static struct configfs_item_operations nvmet_allowed_hosts_item_ops = {
814a07b4970SChristoph Hellwig 	.allow_link		= nvmet_allowed_hosts_allow_link,
815a07b4970SChristoph Hellwig 	.drop_link		= nvmet_allowed_hosts_drop_link,
816a07b4970SChristoph Hellwig };
817a07b4970SChristoph Hellwig 
81866603a31SBhumika Goyal static const struct config_item_type nvmet_allowed_hosts_type = {
819a07b4970SChristoph Hellwig 	.ct_item_ops		= &nvmet_allowed_hosts_item_ops,
820a07b4970SChristoph Hellwig 	.ct_owner		= THIS_MODULE,
821a07b4970SChristoph Hellwig };
822a07b4970SChristoph Hellwig 
823a07b4970SChristoph Hellwig static ssize_t nvmet_subsys_attr_allow_any_host_show(struct config_item *item,
824a07b4970SChristoph Hellwig 		char *page)
825a07b4970SChristoph Hellwig {
826a07b4970SChristoph Hellwig 	return snprintf(page, PAGE_SIZE, "%d\n",
827a07b4970SChristoph Hellwig 		to_subsys(item)->allow_any_host);
828a07b4970SChristoph Hellwig }
829a07b4970SChristoph Hellwig 
830a07b4970SChristoph Hellwig static ssize_t nvmet_subsys_attr_allow_any_host_store(struct config_item *item,
831a07b4970SChristoph Hellwig 		const char *page, size_t count)
832a07b4970SChristoph Hellwig {
833a07b4970SChristoph Hellwig 	struct nvmet_subsys *subsys = to_subsys(item);
834a07b4970SChristoph Hellwig 	bool allow_any_host;
835a07b4970SChristoph Hellwig 	int ret = 0;
836a07b4970SChristoph Hellwig 
837a07b4970SChristoph Hellwig 	if (strtobool(page, &allow_any_host))
838a07b4970SChristoph Hellwig 		return -EINVAL;
839a07b4970SChristoph Hellwig 
840a07b4970SChristoph Hellwig 	down_write(&nvmet_config_sem);
841a07b4970SChristoph Hellwig 	if (allow_any_host && !list_empty(&subsys->hosts)) {
842a07b4970SChristoph Hellwig 		pr_err("Can't set allow_any_host when explicit hosts are set!\n");
843a07b4970SChristoph Hellwig 		ret = -EINVAL;
844a07b4970SChristoph Hellwig 		goto out_unlock;
845a07b4970SChristoph Hellwig 	}
846a07b4970SChristoph Hellwig 
847b662a078SJay Sternberg 	if (subsys->allow_any_host != allow_any_host) {
848a07b4970SChristoph Hellwig 		subsys->allow_any_host = allow_any_host;
849b662a078SJay Sternberg 		nvmet_subsys_disc_changed(subsys, NULL);
850b662a078SJay Sternberg 	}
851b662a078SJay Sternberg 
852a07b4970SChristoph Hellwig out_unlock:
853a07b4970SChristoph Hellwig 	up_write(&nvmet_config_sem);
854a07b4970SChristoph Hellwig 	return ret ? ret : count;
855a07b4970SChristoph Hellwig }
856a07b4970SChristoph Hellwig 
857a07b4970SChristoph Hellwig CONFIGFS_ATTR(nvmet_subsys_, attr_allow_any_host);
858a07b4970SChristoph Hellwig 
85941528f80SJohannes Thumshirn static ssize_t nvmet_subsys_attr_version_show(struct config_item *item,
860c61d788bSJohannes Thumshirn 					      char *page)
861c61d788bSJohannes Thumshirn {
862c61d788bSJohannes Thumshirn 	struct nvmet_subsys *subsys = to_subsys(item);
863c61d788bSJohannes Thumshirn 
864c61d788bSJohannes Thumshirn 	if (NVME_TERTIARY(subsys->ver))
865a0f0dbaaSChaitanya Kulkarni 		return snprintf(page, PAGE_SIZE, "%llu.%llu.%llu\n",
866a0f0dbaaSChaitanya Kulkarni 				NVME_MAJOR(subsys->ver),
867a0f0dbaaSChaitanya Kulkarni 				NVME_MINOR(subsys->ver),
868a0f0dbaaSChaitanya Kulkarni 				NVME_TERTIARY(subsys->ver));
869527123c7SChaitanya Kulkarni 
870a0f0dbaaSChaitanya Kulkarni 	return snprintf(page, PAGE_SIZE, "%llu.%llu\n",
871a0f0dbaaSChaitanya Kulkarni 			NVME_MAJOR(subsys->ver),
872a0f0dbaaSChaitanya Kulkarni 			NVME_MINOR(subsys->ver));
873c61d788bSJohannes Thumshirn }
874c61d788bSJohannes Thumshirn 
87541528f80SJohannes Thumshirn static ssize_t nvmet_subsys_attr_version_store(struct config_item *item,
876c61d788bSJohannes Thumshirn 					       const char *page, size_t count)
877c61d788bSJohannes Thumshirn {
878c61d788bSJohannes Thumshirn 	struct nvmet_subsys *subsys = to_subsys(item);
879c61d788bSJohannes Thumshirn 	int major, minor, tertiary = 0;
880c61d788bSJohannes Thumshirn 	int ret;
881c61d788bSJohannes Thumshirn 
882ba76af67SLogan Gunthorpe 	/* passthru subsystems use the underlying controller's version */
883ba76af67SLogan Gunthorpe 	if (nvmet_passthru_ctrl(subsys))
884ba76af67SLogan Gunthorpe 		return -EINVAL;
885ba76af67SLogan Gunthorpe 
886c61d788bSJohannes Thumshirn 	ret = sscanf(page, "%d.%d.%d\n", &major, &minor, &tertiary);
887c61d788bSJohannes Thumshirn 	if (ret != 2 && ret != 3)
888c61d788bSJohannes Thumshirn 		return -EINVAL;
889c61d788bSJohannes Thumshirn 
890c61d788bSJohannes Thumshirn 	down_write(&nvmet_config_sem);
891c61d788bSJohannes Thumshirn 	subsys->ver = NVME_VS(major, minor, tertiary);
892c61d788bSJohannes Thumshirn 	up_write(&nvmet_config_sem);
893c61d788bSJohannes Thumshirn 
894c61d788bSJohannes Thumshirn 	return count;
895c61d788bSJohannes Thumshirn }
89641528f80SJohannes Thumshirn CONFIGFS_ATTR(nvmet_subsys_, attr_version);
897c61d788bSJohannes Thumshirn 
898fcbc5459SJohannes Thumshirn static ssize_t nvmet_subsys_attr_serial_show(struct config_item *item,
899fcbc5459SJohannes Thumshirn 					     char *page)
900fcbc5459SJohannes Thumshirn {
901fcbc5459SJohannes Thumshirn 	struct nvmet_subsys *subsys = to_subsys(item);
902fcbc5459SJohannes Thumshirn 
903fcbc5459SJohannes Thumshirn 	return snprintf(page, PAGE_SIZE, "%llx\n", subsys->serial);
904fcbc5459SJohannes Thumshirn }
905fcbc5459SJohannes Thumshirn 
906fcbc5459SJohannes Thumshirn static ssize_t nvmet_subsys_attr_serial_store(struct config_item *item,
907fcbc5459SJohannes Thumshirn 					      const char *page, size_t count)
908fcbc5459SJohannes Thumshirn {
909d3a9b0caSChaitanya Kulkarni 	u64 serial;
910d3a9b0caSChaitanya Kulkarni 
911d3a9b0caSChaitanya Kulkarni 	if (sscanf(page, "%llx\n", &serial) != 1)
912d3a9b0caSChaitanya Kulkarni 		return -EINVAL;
913fcbc5459SJohannes Thumshirn 
914fcbc5459SJohannes Thumshirn 	down_write(&nvmet_config_sem);
915d3a9b0caSChaitanya Kulkarni 	to_subsys(item)->serial = serial;
916fcbc5459SJohannes Thumshirn 	up_write(&nvmet_config_sem);
917fcbc5459SJohannes Thumshirn 
918fcbc5459SJohannes Thumshirn 	return count;
919fcbc5459SJohannes Thumshirn }
920fcbc5459SJohannes Thumshirn CONFIGFS_ATTR(nvmet_subsys_, attr_serial);
921fcbc5459SJohannes Thumshirn 
92294a39d61SChaitanya Kulkarni static ssize_t nvmet_subsys_attr_cntlid_min_show(struct config_item *item,
92394a39d61SChaitanya Kulkarni 						 char *page)
92494a39d61SChaitanya Kulkarni {
92594a39d61SChaitanya Kulkarni 	return snprintf(page, PAGE_SIZE, "%u\n", to_subsys(item)->cntlid_min);
92694a39d61SChaitanya Kulkarni }
92794a39d61SChaitanya Kulkarni 
92894a39d61SChaitanya Kulkarni static ssize_t nvmet_subsys_attr_cntlid_min_store(struct config_item *item,
92994a39d61SChaitanya Kulkarni 						  const char *page, size_t cnt)
93094a39d61SChaitanya Kulkarni {
93194a39d61SChaitanya Kulkarni 	u16 cntlid_min;
93294a39d61SChaitanya Kulkarni 
93394a39d61SChaitanya Kulkarni 	if (sscanf(page, "%hu\n", &cntlid_min) != 1)
93494a39d61SChaitanya Kulkarni 		return -EINVAL;
93594a39d61SChaitanya Kulkarni 
93694a39d61SChaitanya Kulkarni 	if (cntlid_min == 0)
93794a39d61SChaitanya Kulkarni 		return -EINVAL;
93894a39d61SChaitanya Kulkarni 
93994a39d61SChaitanya Kulkarni 	down_write(&nvmet_config_sem);
94094a39d61SChaitanya Kulkarni 	if (cntlid_min >= to_subsys(item)->cntlid_max)
94194a39d61SChaitanya Kulkarni 		goto out_unlock;
94294a39d61SChaitanya Kulkarni 	to_subsys(item)->cntlid_min = cntlid_min;
94394a39d61SChaitanya Kulkarni 	up_write(&nvmet_config_sem);
94494a39d61SChaitanya Kulkarni 	return cnt;
94594a39d61SChaitanya Kulkarni 
94694a39d61SChaitanya Kulkarni out_unlock:
94794a39d61SChaitanya Kulkarni 	up_write(&nvmet_config_sem);
94894a39d61SChaitanya Kulkarni 	return -EINVAL;
94994a39d61SChaitanya Kulkarni }
95094a39d61SChaitanya Kulkarni CONFIGFS_ATTR(nvmet_subsys_, attr_cntlid_min);
95194a39d61SChaitanya Kulkarni 
95294a39d61SChaitanya Kulkarni static ssize_t nvmet_subsys_attr_cntlid_max_show(struct config_item *item,
95394a39d61SChaitanya Kulkarni 						 char *page)
95494a39d61SChaitanya Kulkarni {
95594a39d61SChaitanya Kulkarni 	return snprintf(page, PAGE_SIZE, "%u\n", to_subsys(item)->cntlid_max);
95694a39d61SChaitanya Kulkarni }
95794a39d61SChaitanya Kulkarni 
95894a39d61SChaitanya Kulkarni static ssize_t nvmet_subsys_attr_cntlid_max_store(struct config_item *item,
95994a39d61SChaitanya Kulkarni 						  const char *page, size_t cnt)
96094a39d61SChaitanya Kulkarni {
96194a39d61SChaitanya Kulkarni 	u16 cntlid_max;
96294a39d61SChaitanya Kulkarni 
96394a39d61SChaitanya Kulkarni 	if (sscanf(page, "%hu\n", &cntlid_max) != 1)
96494a39d61SChaitanya Kulkarni 		return -EINVAL;
96594a39d61SChaitanya Kulkarni 
96694a39d61SChaitanya Kulkarni 	if (cntlid_max == 0)
96794a39d61SChaitanya Kulkarni 		return -EINVAL;
96894a39d61SChaitanya Kulkarni 
96994a39d61SChaitanya Kulkarni 	down_write(&nvmet_config_sem);
97094a39d61SChaitanya Kulkarni 	if (cntlid_max <= to_subsys(item)->cntlid_min)
97194a39d61SChaitanya Kulkarni 		goto out_unlock;
97294a39d61SChaitanya Kulkarni 	to_subsys(item)->cntlid_max = cntlid_max;
97394a39d61SChaitanya Kulkarni 	up_write(&nvmet_config_sem);
97494a39d61SChaitanya Kulkarni 	return cnt;
97594a39d61SChaitanya Kulkarni 
97694a39d61SChaitanya Kulkarni out_unlock:
97794a39d61SChaitanya Kulkarni 	up_write(&nvmet_config_sem);
97894a39d61SChaitanya Kulkarni 	return -EINVAL;
97994a39d61SChaitanya Kulkarni }
98094a39d61SChaitanya Kulkarni CONFIGFS_ATTR(nvmet_subsys_, attr_cntlid_max);
98194a39d61SChaitanya Kulkarni 
982013b7ebeSMark Ruijter static ssize_t nvmet_subsys_attr_model_show(struct config_item *item,
983013b7ebeSMark Ruijter 					    char *page)
984013b7ebeSMark Ruijter {
985013b7ebeSMark Ruijter 	struct nvmet_subsys *subsys = to_subsys(item);
986013b7ebeSMark Ruijter 	struct nvmet_subsys_model *subsys_model;
987013b7ebeSMark Ruijter 	char *model = NVMET_DEFAULT_CTRL_MODEL;
988013b7ebeSMark Ruijter 	int ret;
989013b7ebeSMark Ruijter 
990013b7ebeSMark Ruijter 	rcu_read_lock();
991013b7ebeSMark Ruijter 	subsys_model = rcu_dereference(subsys->model);
992013b7ebeSMark Ruijter 	if (subsys_model)
993013b7ebeSMark Ruijter 		model = subsys_model->number;
994013b7ebeSMark Ruijter 	ret = snprintf(page, PAGE_SIZE, "%s\n", model);
995013b7ebeSMark Ruijter 	rcu_read_unlock();
996013b7ebeSMark Ruijter 
997013b7ebeSMark Ruijter 	return ret;
998013b7ebeSMark Ruijter }
999013b7ebeSMark Ruijter 
1000013b7ebeSMark Ruijter /* See Section 1.5 of NVMe 1.4 */
1001013b7ebeSMark Ruijter static bool nvmet_is_ascii(const char c)
1002013b7ebeSMark Ruijter {
1003013b7ebeSMark Ruijter 	return c >= 0x20 && c <= 0x7e;
1004013b7ebeSMark Ruijter }
1005013b7ebeSMark Ruijter 
1006013b7ebeSMark Ruijter static ssize_t nvmet_subsys_attr_model_store(struct config_item *item,
1007013b7ebeSMark Ruijter 					     const char *page, size_t count)
1008013b7ebeSMark Ruijter {
1009013b7ebeSMark Ruijter 	struct nvmet_subsys *subsys = to_subsys(item);
1010013b7ebeSMark Ruijter 	struct nvmet_subsys_model *new_model;
1011013b7ebeSMark Ruijter 	char *new_model_number;
1012013b7ebeSMark Ruijter 	int pos = 0, len;
1013013b7ebeSMark Ruijter 
1014013b7ebeSMark Ruijter 	len = strcspn(page, "\n");
1015013b7ebeSMark Ruijter 	if (!len)
1016013b7ebeSMark Ruijter 		return -EINVAL;
1017013b7ebeSMark Ruijter 
1018013b7ebeSMark Ruijter 	for (pos = 0; pos < len; pos++) {
1019013b7ebeSMark Ruijter 		if (!nvmet_is_ascii(page[pos]))
1020013b7ebeSMark Ruijter 			return -EINVAL;
1021013b7ebeSMark Ruijter 	}
1022013b7ebeSMark Ruijter 
102309bb8986SChen Zhou 	new_model_number = kmemdup_nul(page, len, GFP_KERNEL);
1024013b7ebeSMark Ruijter 	if (!new_model_number)
1025013b7ebeSMark Ruijter 		return -ENOMEM;
1026013b7ebeSMark Ruijter 
1027013b7ebeSMark Ruijter 	new_model = kzalloc(sizeof(*new_model) + len + 1, GFP_KERNEL);
1028013b7ebeSMark Ruijter 	if (!new_model) {
1029013b7ebeSMark Ruijter 		kfree(new_model_number);
1030013b7ebeSMark Ruijter 		return -ENOMEM;
1031013b7ebeSMark Ruijter 	}
1032013b7ebeSMark Ruijter 	memcpy(new_model->number, new_model_number, len);
1033013b7ebeSMark Ruijter 
1034013b7ebeSMark Ruijter 	down_write(&nvmet_config_sem);
1035013b7ebeSMark Ruijter 	mutex_lock(&subsys->lock);
1036013b7ebeSMark Ruijter 	new_model = rcu_replace_pointer(subsys->model, new_model,
1037013b7ebeSMark Ruijter 					mutex_is_locked(&subsys->lock));
1038013b7ebeSMark Ruijter 	mutex_unlock(&subsys->lock);
1039013b7ebeSMark Ruijter 	up_write(&nvmet_config_sem);
1040013b7ebeSMark Ruijter 
1041013b7ebeSMark Ruijter 	kfree_rcu(new_model, rcuhead);
1042013b7ebeSMark Ruijter 
1043013b7ebeSMark Ruijter 	return count;
1044013b7ebeSMark Ruijter }
1045013b7ebeSMark Ruijter CONFIGFS_ATTR(nvmet_subsys_, attr_model);
1046013b7ebeSMark Ruijter 
1047ea52ac1cSIsrael Rukshin #ifdef CONFIG_BLK_DEV_INTEGRITY
1048ea52ac1cSIsrael Rukshin static ssize_t nvmet_subsys_attr_pi_enable_show(struct config_item *item,
1049ea52ac1cSIsrael Rukshin 						char *page)
1050ea52ac1cSIsrael Rukshin {
1051ea52ac1cSIsrael Rukshin 	return snprintf(page, PAGE_SIZE, "%d\n", to_subsys(item)->pi_support);
1052ea52ac1cSIsrael Rukshin }
1053ea52ac1cSIsrael Rukshin 
1054ea52ac1cSIsrael Rukshin static ssize_t nvmet_subsys_attr_pi_enable_store(struct config_item *item,
1055ea52ac1cSIsrael Rukshin 						 const char *page, size_t count)
1056ea52ac1cSIsrael Rukshin {
1057ea52ac1cSIsrael Rukshin 	struct nvmet_subsys *subsys = to_subsys(item);
1058ea52ac1cSIsrael Rukshin 	bool pi_enable;
1059ea52ac1cSIsrael Rukshin 
1060ea52ac1cSIsrael Rukshin 	if (strtobool(page, &pi_enable))
1061ea52ac1cSIsrael Rukshin 		return -EINVAL;
1062ea52ac1cSIsrael Rukshin 
1063ea52ac1cSIsrael Rukshin 	subsys->pi_support = pi_enable;
1064ea52ac1cSIsrael Rukshin 	return count;
1065ea52ac1cSIsrael Rukshin }
1066ea52ac1cSIsrael Rukshin CONFIGFS_ATTR(nvmet_subsys_, attr_pi_enable);
1067ea52ac1cSIsrael Rukshin #endif
1068ea52ac1cSIsrael Rukshin 
1069a07b4970SChristoph Hellwig static struct configfs_attribute *nvmet_subsys_attrs[] = {
1070a07b4970SChristoph Hellwig 	&nvmet_subsys_attr_attr_allow_any_host,
107141528f80SJohannes Thumshirn 	&nvmet_subsys_attr_attr_version,
1072fcbc5459SJohannes Thumshirn 	&nvmet_subsys_attr_attr_serial,
107394a39d61SChaitanya Kulkarni 	&nvmet_subsys_attr_attr_cntlid_min,
107494a39d61SChaitanya Kulkarni 	&nvmet_subsys_attr_attr_cntlid_max,
1075013b7ebeSMark Ruijter 	&nvmet_subsys_attr_attr_model,
1076ea52ac1cSIsrael Rukshin #ifdef CONFIG_BLK_DEV_INTEGRITY
1077ea52ac1cSIsrael Rukshin 	&nvmet_subsys_attr_attr_pi_enable,
1078ea52ac1cSIsrael Rukshin #endif
1079a07b4970SChristoph Hellwig 	NULL,
1080a07b4970SChristoph Hellwig };
1081a07b4970SChristoph Hellwig 
1082a07b4970SChristoph Hellwig /*
1083a07b4970SChristoph Hellwig  * Subsystem structures & folder operation functions below
1084a07b4970SChristoph Hellwig  */
1085a07b4970SChristoph Hellwig static void nvmet_subsys_release(struct config_item *item)
1086a07b4970SChristoph Hellwig {
1087a07b4970SChristoph Hellwig 	struct nvmet_subsys *subsys = to_subsys(item);
1088a07b4970SChristoph Hellwig 
1089344770b0SSagi Grimberg 	nvmet_subsys_del_ctrls(subsys);
1090a07b4970SChristoph Hellwig 	nvmet_subsys_put(subsys);
1091a07b4970SChristoph Hellwig }
1092a07b4970SChristoph Hellwig 
1093a07b4970SChristoph Hellwig static struct configfs_item_operations nvmet_subsys_item_ops = {
1094a07b4970SChristoph Hellwig 	.release		= nvmet_subsys_release,
1095a07b4970SChristoph Hellwig };
1096a07b4970SChristoph Hellwig 
109766603a31SBhumika Goyal static const struct config_item_type nvmet_subsys_type = {
1098a07b4970SChristoph Hellwig 	.ct_item_ops		= &nvmet_subsys_item_ops,
1099a07b4970SChristoph Hellwig 	.ct_attrs		= nvmet_subsys_attrs,
1100a07b4970SChristoph Hellwig 	.ct_owner		= THIS_MODULE,
1101a07b4970SChristoph Hellwig };
1102a07b4970SChristoph Hellwig 
1103a07b4970SChristoph Hellwig static struct config_group *nvmet_subsys_make(struct config_group *group,
1104a07b4970SChristoph Hellwig 		const char *name)
1105a07b4970SChristoph Hellwig {
1106a07b4970SChristoph Hellwig 	struct nvmet_subsys *subsys;
1107a07b4970SChristoph Hellwig 
1108a07b4970SChristoph Hellwig 	if (sysfs_streq(name, NVME_DISC_SUBSYS_NAME)) {
1109a07b4970SChristoph Hellwig 		pr_err("can't create discovery subsystem through configfs\n");
1110a07b4970SChristoph Hellwig 		return ERR_PTR(-EINVAL);
1111a07b4970SChristoph Hellwig 	}
1112a07b4970SChristoph Hellwig 
1113a07b4970SChristoph Hellwig 	subsys = nvmet_subsys_alloc(name, NVME_NQN_NVME);
11146b7e631bSMinwoo Im 	if (IS_ERR(subsys))
11156b7e631bSMinwoo Im 		return ERR_CAST(subsys);
1116a07b4970SChristoph Hellwig 
1117a07b4970SChristoph Hellwig 	config_group_init_type_name(&subsys->group, name, &nvmet_subsys_type);
1118a07b4970SChristoph Hellwig 
1119a07b4970SChristoph Hellwig 	config_group_init_type_name(&subsys->namespaces_group,
1120a07b4970SChristoph Hellwig 			"namespaces", &nvmet_namespaces_type);
1121a07b4970SChristoph Hellwig 	configfs_add_default_group(&subsys->namespaces_group, &subsys->group);
1122a07b4970SChristoph Hellwig 
1123a07b4970SChristoph Hellwig 	config_group_init_type_name(&subsys->allowed_hosts_group,
1124a07b4970SChristoph Hellwig 			"allowed_hosts", &nvmet_allowed_hosts_type);
1125a07b4970SChristoph Hellwig 	configfs_add_default_group(&subsys->allowed_hosts_group,
1126a07b4970SChristoph Hellwig 			&subsys->group);
1127a07b4970SChristoph Hellwig 
1128a07b4970SChristoph Hellwig 	return &subsys->group;
1129a07b4970SChristoph Hellwig }
1130a07b4970SChristoph Hellwig 
1131a07b4970SChristoph Hellwig static struct configfs_group_operations nvmet_subsystems_group_ops = {
1132a07b4970SChristoph Hellwig 	.make_group		= nvmet_subsys_make,
1133a07b4970SChristoph Hellwig };
1134a07b4970SChristoph Hellwig 
113566603a31SBhumika Goyal static const struct config_item_type nvmet_subsystems_type = {
1136a07b4970SChristoph Hellwig 	.ct_group_ops		= &nvmet_subsystems_group_ops,
1137a07b4970SChristoph Hellwig 	.ct_owner		= THIS_MODULE,
1138a07b4970SChristoph Hellwig };
1139a07b4970SChristoph Hellwig 
1140a07b4970SChristoph Hellwig static ssize_t nvmet_referral_enable_show(struct config_item *item,
1141a07b4970SChristoph Hellwig 		char *page)
1142a07b4970SChristoph Hellwig {
1143a07b4970SChristoph Hellwig 	return snprintf(page, PAGE_SIZE, "%d\n", to_nvmet_port(item)->enabled);
1144a07b4970SChristoph Hellwig }
1145a07b4970SChristoph Hellwig 
1146a07b4970SChristoph Hellwig static ssize_t nvmet_referral_enable_store(struct config_item *item,
1147a07b4970SChristoph Hellwig 		const char *page, size_t count)
1148a07b4970SChristoph Hellwig {
1149a07b4970SChristoph Hellwig 	struct nvmet_port *parent = to_nvmet_port(item->ci_parent->ci_parent);
1150a07b4970SChristoph Hellwig 	struct nvmet_port *port = to_nvmet_port(item);
1151a07b4970SChristoph Hellwig 	bool enable;
1152a07b4970SChristoph Hellwig 
1153a07b4970SChristoph Hellwig 	if (strtobool(page, &enable))
1154a07b4970SChristoph Hellwig 		goto inval;
1155a07b4970SChristoph Hellwig 
1156a07b4970SChristoph Hellwig 	if (enable)
1157a07b4970SChristoph Hellwig 		nvmet_referral_enable(parent, port);
1158a07b4970SChristoph Hellwig 	else
1159b662a078SJay Sternberg 		nvmet_referral_disable(parent, port);
1160a07b4970SChristoph Hellwig 
1161a07b4970SChristoph Hellwig 	return count;
1162a07b4970SChristoph Hellwig inval:
1163a07b4970SChristoph Hellwig 	pr_err("Invalid value '%s' for enable\n", page);
1164a07b4970SChristoph Hellwig 	return -EINVAL;
1165a07b4970SChristoph Hellwig }
1166a07b4970SChristoph Hellwig 
1167a07b4970SChristoph Hellwig CONFIGFS_ATTR(nvmet_referral_, enable);
1168a07b4970SChristoph Hellwig 
1169a07b4970SChristoph Hellwig /*
1170a07b4970SChristoph Hellwig  * Discovery Service subsystem definitions
1171a07b4970SChristoph Hellwig  */
1172a07b4970SChristoph Hellwig static struct configfs_attribute *nvmet_referral_attrs[] = {
1173a07b4970SChristoph Hellwig 	&nvmet_attr_addr_adrfam,
1174a07b4970SChristoph Hellwig 	&nvmet_attr_addr_portid,
1175a07b4970SChristoph Hellwig 	&nvmet_attr_addr_treq,
1176a07b4970SChristoph Hellwig 	&nvmet_attr_addr_traddr,
1177a07b4970SChristoph Hellwig 	&nvmet_attr_addr_trsvcid,
1178a07b4970SChristoph Hellwig 	&nvmet_attr_addr_trtype,
1179a07b4970SChristoph Hellwig 	&nvmet_referral_attr_enable,
1180a07b4970SChristoph Hellwig 	NULL,
1181a07b4970SChristoph Hellwig };
1182a07b4970SChristoph Hellwig 
1183f0e656e4SSagi Grimberg static void nvmet_referral_notify(struct config_group *group,
1184f0e656e4SSagi Grimberg 		struct config_item *item)
1185a07b4970SChristoph Hellwig {
1186b662a078SJay Sternberg 	struct nvmet_port *parent = to_nvmet_port(item->ci_parent->ci_parent);
1187a07b4970SChristoph Hellwig 	struct nvmet_port *port = to_nvmet_port(item);
1188a07b4970SChristoph Hellwig 
1189b662a078SJay Sternberg 	nvmet_referral_disable(parent, port);
1190f0e656e4SSagi Grimberg }
1191f0e656e4SSagi Grimberg 
1192f0e656e4SSagi Grimberg static void nvmet_referral_release(struct config_item *item)
1193f0e656e4SSagi Grimberg {
1194f0e656e4SSagi Grimberg 	struct nvmet_port *port = to_nvmet_port(item);
1195f0e656e4SSagi Grimberg 
1196a07b4970SChristoph Hellwig 	kfree(port);
1197a07b4970SChristoph Hellwig }
1198a07b4970SChristoph Hellwig 
1199a07b4970SChristoph Hellwig static struct configfs_item_operations nvmet_referral_item_ops = {
1200a07b4970SChristoph Hellwig 	.release	= nvmet_referral_release,
1201a07b4970SChristoph Hellwig };
1202a07b4970SChristoph Hellwig 
120366603a31SBhumika Goyal static const struct config_item_type nvmet_referral_type = {
1204a07b4970SChristoph Hellwig 	.ct_owner	= THIS_MODULE,
1205a07b4970SChristoph Hellwig 	.ct_attrs	= nvmet_referral_attrs,
1206a07b4970SChristoph Hellwig 	.ct_item_ops	= &nvmet_referral_item_ops,
1207a07b4970SChristoph Hellwig };
1208a07b4970SChristoph Hellwig 
1209a07b4970SChristoph Hellwig static struct config_group *nvmet_referral_make(
1210a07b4970SChristoph Hellwig 		struct config_group *group, const char *name)
1211a07b4970SChristoph Hellwig {
1212a07b4970SChristoph Hellwig 	struct nvmet_port *port;
1213a07b4970SChristoph Hellwig 
1214a07b4970SChristoph Hellwig 	port = kzalloc(sizeof(*port), GFP_KERNEL);
1215a07b4970SChristoph Hellwig 	if (!port)
1216f98d9ca1SDan Carpenter 		return ERR_PTR(-ENOMEM);
1217a07b4970SChristoph Hellwig 
1218a07b4970SChristoph Hellwig 	INIT_LIST_HEAD(&port->entry);
1219a07b4970SChristoph Hellwig 	config_group_init_type_name(&port->group, name, &nvmet_referral_type);
1220a07b4970SChristoph Hellwig 
1221a07b4970SChristoph Hellwig 	return &port->group;
1222a07b4970SChristoph Hellwig }
1223a07b4970SChristoph Hellwig 
1224a07b4970SChristoph Hellwig static struct configfs_group_operations nvmet_referral_group_ops = {
1225a07b4970SChristoph Hellwig 	.make_group		= nvmet_referral_make,
1226f0e656e4SSagi Grimberg 	.disconnect_notify	= nvmet_referral_notify,
1227a07b4970SChristoph Hellwig };
1228a07b4970SChristoph Hellwig 
122966603a31SBhumika Goyal static const struct config_item_type nvmet_referrals_type = {
1230a07b4970SChristoph Hellwig 	.ct_owner	= THIS_MODULE,
1231a07b4970SChristoph Hellwig 	.ct_group_ops	= &nvmet_referral_group_ops,
1232a07b4970SChristoph Hellwig };
1233a07b4970SChristoph Hellwig 
123474255969SChristoph Hellwig static struct nvmet_type_name_map nvmet_ana_state[] = {
123562ac0d32SChristoph Hellwig 	{ NVME_ANA_OPTIMIZED,		"optimized" },
123662ac0d32SChristoph Hellwig 	{ NVME_ANA_NONOPTIMIZED,	"non-optimized" },
123762ac0d32SChristoph Hellwig 	{ NVME_ANA_INACCESSIBLE,	"inaccessible" },
123862ac0d32SChristoph Hellwig 	{ NVME_ANA_PERSISTENT_LOSS,	"persistent-loss" },
123962ac0d32SChristoph Hellwig 	{ NVME_ANA_CHANGE,		"change" },
124062ac0d32SChristoph Hellwig };
124162ac0d32SChristoph Hellwig 
124262ac0d32SChristoph Hellwig static ssize_t nvmet_ana_group_ana_state_show(struct config_item *item,
124362ac0d32SChristoph Hellwig 		char *page)
124462ac0d32SChristoph Hellwig {
124562ac0d32SChristoph Hellwig 	struct nvmet_ana_group *grp = to_ana_group(item);
124662ac0d32SChristoph Hellwig 	enum nvme_ana_state state = grp->port->ana_state[grp->grpid];
124762ac0d32SChristoph Hellwig 	int i;
124862ac0d32SChristoph Hellwig 
124984b8d0d7SChaitanya Kulkarni 	for (i = 0; i < ARRAY_SIZE(nvmet_ana_state); i++) {
125084b8d0d7SChaitanya Kulkarni 		if (state == nvmet_ana_state[i].type)
125184b8d0d7SChaitanya Kulkarni 			return sprintf(page, "%s\n", nvmet_ana_state[i].name);
125262ac0d32SChristoph Hellwig 	}
125362ac0d32SChristoph Hellwig 
125462ac0d32SChristoph Hellwig 	return sprintf(page, "\n");
125562ac0d32SChristoph Hellwig }
125662ac0d32SChristoph Hellwig 
125762ac0d32SChristoph Hellwig static ssize_t nvmet_ana_group_ana_state_store(struct config_item *item,
125862ac0d32SChristoph Hellwig 		const char *page, size_t count)
125962ac0d32SChristoph Hellwig {
126062ac0d32SChristoph Hellwig 	struct nvmet_ana_group *grp = to_ana_group(item);
126184b8d0d7SChaitanya Kulkarni 	enum nvme_ana_state *ana_state = grp->port->ana_state;
126262ac0d32SChristoph Hellwig 	int i;
126362ac0d32SChristoph Hellwig 
126484b8d0d7SChaitanya Kulkarni 	for (i = 0; i < ARRAY_SIZE(nvmet_ana_state); i++) {
126584b8d0d7SChaitanya Kulkarni 		if (sysfs_streq(page, nvmet_ana_state[i].name))
126662ac0d32SChristoph Hellwig 			goto found;
126762ac0d32SChristoph Hellwig 	}
126862ac0d32SChristoph Hellwig 
126962ac0d32SChristoph Hellwig 	pr_err("Invalid value '%s' for ana_state\n", page);
127062ac0d32SChristoph Hellwig 	return -EINVAL;
127162ac0d32SChristoph Hellwig 
127262ac0d32SChristoph Hellwig found:
127362ac0d32SChristoph Hellwig 	down_write(&nvmet_ana_sem);
127484b8d0d7SChaitanya Kulkarni 	ana_state[grp->grpid] = (enum nvme_ana_state) nvmet_ana_state[i].type;
127562ac0d32SChristoph Hellwig 	nvmet_ana_chgcnt++;
127662ac0d32SChristoph Hellwig 	up_write(&nvmet_ana_sem);
127762ac0d32SChristoph Hellwig 	nvmet_port_send_ana_event(grp->port);
127862ac0d32SChristoph Hellwig 	return count;
127962ac0d32SChristoph Hellwig }
128062ac0d32SChristoph Hellwig 
128162ac0d32SChristoph Hellwig CONFIGFS_ATTR(nvmet_ana_group_, ana_state);
128262ac0d32SChristoph Hellwig 
128362ac0d32SChristoph Hellwig static struct configfs_attribute *nvmet_ana_group_attrs[] = {
128462ac0d32SChristoph Hellwig 	&nvmet_ana_group_attr_ana_state,
128562ac0d32SChristoph Hellwig 	NULL,
128662ac0d32SChristoph Hellwig };
128762ac0d32SChristoph Hellwig 
128862ac0d32SChristoph Hellwig static void nvmet_ana_group_release(struct config_item *item)
128962ac0d32SChristoph Hellwig {
129062ac0d32SChristoph Hellwig 	struct nvmet_ana_group *grp = to_ana_group(item);
129162ac0d32SChristoph Hellwig 
129262ac0d32SChristoph Hellwig 	if (grp == &grp->port->ana_default_group)
129362ac0d32SChristoph Hellwig 		return;
129462ac0d32SChristoph Hellwig 
129562ac0d32SChristoph Hellwig 	down_write(&nvmet_ana_sem);
129662ac0d32SChristoph Hellwig 	grp->port->ana_state[grp->grpid] = NVME_ANA_INACCESSIBLE;
129762ac0d32SChristoph Hellwig 	nvmet_ana_group_enabled[grp->grpid]--;
129862ac0d32SChristoph Hellwig 	up_write(&nvmet_ana_sem);
129962ac0d32SChristoph Hellwig 
130062ac0d32SChristoph Hellwig 	nvmet_port_send_ana_event(grp->port);
130162ac0d32SChristoph Hellwig 	kfree(grp);
130262ac0d32SChristoph Hellwig }
130362ac0d32SChristoph Hellwig 
130462ac0d32SChristoph Hellwig static struct configfs_item_operations nvmet_ana_group_item_ops = {
130562ac0d32SChristoph Hellwig 	.release		= nvmet_ana_group_release,
130662ac0d32SChristoph Hellwig };
130762ac0d32SChristoph Hellwig 
130862ac0d32SChristoph Hellwig static const struct config_item_type nvmet_ana_group_type = {
130962ac0d32SChristoph Hellwig 	.ct_item_ops		= &nvmet_ana_group_item_ops,
131062ac0d32SChristoph Hellwig 	.ct_attrs		= nvmet_ana_group_attrs,
131162ac0d32SChristoph Hellwig 	.ct_owner		= THIS_MODULE,
131262ac0d32SChristoph Hellwig };
131362ac0d32SChristoph Hellwig 
131462ac0d32SChristoph Hellwig static struct config_group *nvmet_ana_groups_make_group(
131562ac0d32SChristoph Hellwig 		struct config_group *group, const char *name)
131662ac0d32SChristoph Hellwig {
131762ac0d32SChristoph Hellwig 	struct nvmet_port *port = ana_groups_to_port(&group->cg_item);
131862ac0d32SChristoph Hellwig 	struct nvmet_ana_group *grp;
131962ac0d32SChristoph Hellwig 	u32 grpid;
132062ac0d32SChristoph Hellwig 	int ret;
132162ac0d32SChristoph Hellwig 
132262ac0d32SChristoph Hellwig 	ret = kstrtou32(name, 0, &grpid);
132362ac0d32SChristoph Hellwig 	if (ret)
132462ac0d32SChristoph Hellwig 		goto out;
132562ac0d32SChristoph Hellwig 
132662ac0d32SChristoph Hellwig 	ret = -EINVAL;
132762ac0d32SChristoph Hellwig 	if (grpid <= 1 || grpid > NVMET_MAX_ANAGRPS)
132862ac0d32SChristoph Hellwig 		goto out;
132962ac0d32SChristoph Hellwig 
133062ac0d32SChristoph Hellwig 	ret = -ENOMEM;
133162ac0d32SChristoph Hellwig 	grp = kzalloc(sizeof(*grp), GFP_KERNEL);
133262ac0d32SChristoph Hellwig 	if (!grp)
133362ac0d32SChristoph Hellwig 		goto out;
133462ac0d32SChristoph Hellwig 	grp->port = port;
133562ac0d32SChristoph Hellwig 	grp->grpid = grpid;
133662ac0d32SChristoph Hellwig 
133762ac0d32SChristoph Hellwig 	down_write(&nvmet_ana_sem);
133862ac0d32SChristoph Hellwig 	nvmet_ana_group_enabled[grpid]++;
133962ac0d32SChristoph Hellwig 	up_write(&nvmet_ana_sem);
134062ac0d32SChristoph Hellwig 
134162ac0d32SChristoph Hellwig 	nvmet_port_send_ana_event(grp->port);
134262ac0d32SChristoph Hellwig 
134362ac0d32SChristoph Hellwig 	config_group_init_type_name(&grp->group, name, &nvmet_ana_group_type);
134462ac0d32SChristoph Hellwig 	return &grp->group;
134562ac0d32SChristoph Hellwig out:
134662ac0d32SChristoph Hellwig 	return ERR_PTR(ret);
134762ac0d32SChristoph Hellwig }
134862ac0d32SChristoph Hellwig 
134962ac0d32SChristoph Hellwig static struct configfs_group_operations nvmet_ana_groups_group_ops = {
135062ac0d32SChristoph Hellwig 	.make_group		= nvmet_ana_groups_make_group,
135162ac0d32SChristoph Hellwig };
135262ac0d32SChristoph Hellwig 
135362ac0d32SChristoph Hellwig static const struct config_item_type nvmet_ana_groups_type = {
135462ac0d32SChristoph Hellwig 	.ct_group_ops		= &nvmet_ana_groups_group_ops,
135562ac0d32SChristoph Hellwig 	.ct_owner		= THIS_MODULE,
135662ac0d32SChristoph Hellwig };
135762ac0d32SChristoph Hellwig 
1358a07b4970SChristoph Hellwig /*
1359a07b4970SChristoph Hellwig  * Ports definitions.
1360a07b4970SChristoph Hellwig  */
1361a07b4970SChristoph Hellwig static void nvmet_port_release(struct config_item *item)
1362a07b4970SChristoph Hellwig {
1363a07b4970SChristoph Hellwig 	struct nvmet_port *port = to_nvmet_port(item);
1364a07b4970SChristoph Hellwig 
1365b662a078SJay Sternberg 	list_del(&port->global_entry);
1366b662a078SJay Sternberg 
136772efd25dSChristoph Hellwig 	kfree(port->ana_state);
1368a07b4970SChristoph Hellwig 	kfree(port);
1369a07b4970SChristoph Hellwig }
1370a07b4970SChristoph Hellwig 
1371a07b4970SChristoph Hellwig static struct configfs_attribute *nvmet_port_attrs[] = {
1372a07b4970SChristoph Hellwig 	&nvmet_attr_addr_adrfam,
1373a07b4970SChristoph Hellwig 	&nvmet_attr_addr_treq,
1374a07b4970SChristoph Hellwig 	&nvmet_attr_addr_traddr,
1375a07b4970SChristoph Hellwig 	&nvmet_attr_addr_trsvcid,
1376a07b4970SChristoph Hellwig 	&nvmet_attr_addr_trtype,
13770d5ee2b2SSteve Wise 	&nvmet_attr_param_inline_data_size,
1378ea52ac1cSIsrael Rukshin #ifdef CONFIG_BLK_DEV_INTEGRITY
1379ea52ac1cSIsrael Rukshin 	&nvmet_attr_param_pi_enable,
1380ea52ac1cSIsrael Rukshin #endif
1381a07b4970SChristoph Hellwig 	NULL,
1382a07b4970SChristoph Hellwig };
1383a07b4970SChristoph Hellwig 
1384a07b4970SChristoph Hellwig static struct configfs_item_operations nvmet_port_item_ops = {
1385a07b4970SChristoph Hellwig 	.release		= nvmet_port_release,
1386a07b4970SChristoph Hellwig };
1387a07b4970SChristoph Hellwig 
138866603a31SBhumika Goyal static const struct config_item_type nvmet_port_type = {
1389a07b4970SChristoph Hellwig 	.ct_attrs		= nvmet_port_attrs,
1390a07b4970SChristoph Hellwig 	.ct_item_ops		= &nvmet_port_item_ops,
1391a07b4970SChristoph Hellwig 	.ct_owner		= THIS_MODULE,
1392a07b4970SChristoph Hellwig };
1393a07b4970SChristoph Hellwig 
1394a07b4970SChristoph Hellwig static struct config_group *nvmet_ports_make(struct config_group *group,
1395a07b4970SChristoph Hellwig 		const char *name)
1396a07b4970SChristoph Hellwig {
1397a07b4970SChristoph Hellwig 	struct nvmet_port *port;
1398a07b4970SChristoph Hellwig 	u16 portid;
139962ac0d32SChristoph Hellwig 	u32 i;
1400a07b4970SChristoph Hellwig 
1401a07b4970SChristoph Hellwig 	if (kstrtou16(name, 0, &portid))
1402a07b4970SChristoph Hellwig 		return ERR_PTR(-EINVAL);
1403a07b4970SChristoph Hellwig 
1404a07b4970SChristoph Hellwig 	port = kzalloc(sizeof(*port), GFP_KERNEL);
1405a07b4970SChristoph Hellwig 	if (!port)
1406f98d9ca1SDan Carpenter 		return ERR_PTR(-ENOMEM);
1407a07b4970SChristoph Hellwig 
140872efd25dSChristoph Hellwig 	port->ana_state = kcalloc(NVMET_MAX_ANAGRPS + 1,
140972efd25dSChristoph Hellwig 			sizeof(*port->ana_state), GFP_KERNEL);
141072efd25dSChristoph Hellwig 	if (!port->ana_state) {
141172efd25dSChristoph Hellwig 		kfree(port);
141272efd25dSChristoph Hellwig 		return ERR_PTR(-ENOMEM);
141372efd25dSChristoph Hellwig 	}
141472efd25dSChristoph Hellwig 
141562ac0d32SChristoph Hellwig 	for (i = 1; i <= NVMET_MAX_ANAGRPS; i++) {
141662ac0d32SChristoph Hellwig 		if (i == NVMET_DEFAULT_ANA_GRPID)
141762ac0d32SChristoph Hellwig 			port->ana_state[1] = NVME_ANA_OPTIMIZED;
141862ac0d32SChristoph Hellwig 		else
141962ac0d32SChristoph Hellwig 			port->ana_state[i] = NVME_ANA_INACCESSIBLE;
142062ac0d32SChristoph Hellwig 	}
142172efd25dSChristoph Hellwig 
1422b662a078SJay Sternberg 	list_add(&port->global_entry, &nvmet_ports_list);
1423b662a078SJay Sternberg 
1424a07b4970SChristoph Hellwig 	INIT_LIST_HEAD(&port->entry);
1425a07b4970SChristoph Hellwig 	INIT_LIST_HEAD(&port->subsystems);
1426a07b4970SChristoph Hellwig 	INIT_LIST_HEAD(&port->referrals);
14270d5ee2b2SSteve Wise 	port->inline_data_size = -1;	/* < 0 == let the transport choose */
1428a07b4970SChristoph Hellwig 
1429a07b4970SChristoph Hellwig 	port->disc_addr.portid = cpu_to_le16(portid);
1430d02abd19SChaitanya Kulkarni 	port->disc_addr.adrfam = NVMF_ADDR_FAMILY_MAX;
14319b95d2fbSSagi Grimberg 	port->disc_addr.treq = NVMF_TREQ_DISABLE_SQFLOW;
1432a07b4970SChristoph Hellwig 	config_group_init_type_name(&port->group, name, &nvmet_port_type);
1433a07b4970SChristoph Hellwig 
1434a07b4970SChristoph Hellwig 	config_group_init_type_name(&port->subsys_group,
1435a07b4970SChristoph Hellwig 			"subsystems", &nvmet_port_subsys_type);
1436a07b4970SChristoph Hellwig 	configfs_add_default_group(&port->subsys_group, &port->group);
1437a07b4970SChristoph Hellwig 
1438a07b4970SChristoph Hellwig 	config_group_init_type_name(&port->referrals_group,
1439a07b4970SChristoph Hellwig 			"referrals", &nvmet_referrals_type);
1440a07b4970SChristoph Hellwig 	configfs_add_default_group(&port->referrals_group, &port->group);
1441a07b4970SChristoph Hellwig 
144262ac0d32SChristoph Hellwig 	config_group_init_type_name(&port->ana_groups_group,
144362ac0d32SChristoph Hellwig 			"ana_groups", &nvmet_ana_groups_type);
144462ac0d32SChristoph Hellwig 	configfs_add_default_group(&port->ana_groups_group, &port->group);
144562ac0d32SChristoph Hellwig 
144662ac0d32SChristoph Hellwig 	port->ana_default_group.port = port;
144762ac0d32SChristoph Hellwig 	port->ana_default_group.grpid = NVMET_DEFAULT_ANA_GRPID;
144862ac0d32SChristoph Hellwig 	config_group_init_type_name(&port->ana_default_group.group,
144962ac0d32SChristoph Hellwig 			__stringify(NVMET_DEFAULT_ANA_GRPID),
145062ac0d32SChristoph Hellwig 			&nvmet_ana_group_type);
145162ac0d32SChristoph Hellwig 	configfs_add_default_group(&port->ana_default_group.group,
145262ac0d32SChristoph Hellwig 			&port->ana_groups_group);
145362ac0d32SChristoph Hellwig 
1454a07b4970SChristoph Hellwig 	return &port->group;
1455a07b4970SChristoph Hellwig }
1456a07b4970SChristoph Hellwig 
1457a07b4970SChristoph Hellwig static struct configfs_group_operations nvmet_ports_group_ops = {
1458a07b4970SChristoph Hellwig 	.make_group		= nvmet_ports_make,
1459a07b4970SChristoph Hellwig };
1460a07b4970SChristoph Hellwig 
146166603a31SBhumika Goyal static const struct config_item_type nvmet_ports_type = {
1462a07b4970SChristoph Hellwig 	.ct_group_ops		= &nvmet_ports_group_ops,
1463a07b4970SChristoph Hellwig 	.ct_owner		= THIS_MODULE,
1464a07b4970SChristoph Hellwig };
1465a07b4970SChristoph Hellwig 
1466a07b4970SChristoph Hellwig static struct config_group nvmet_subsystems_group;
1467a07b4970SChristoph Hellwig static struct config_group nvmet_ports_group;
1468a07b4970SChristoph Hellwig 
1469a07b4970SChristoph Hellwig static void nvmet_host_release(struct config_item *item)
1470a07b4970SChristoph Hellwig {
1471a07b4970SChristoph Hellwig 	struct nvmet_host *host = to_host(item);
1472a07b4970SChristoph Hellwig 
1473a07b4970SChristoph Hellwig 	kfree(host);
1474a07b4970SChristoph Hellwig }
1475a07b4970SChristoph Hellwig 
1476a07b4970SChristoph Hellwig static struct configfs_item_operations nvmet_host_item_ops = {
1477a07b4970SChristoph Hellwig 	.release		= nvmet_host_release,
1478a07b4970SChristoph Hellwig };
1479a07b4970SChristoph Hellwig 
148066603a31SBhumika Goyal static const struct config_item_type nvmet_host_type = {
1481a07b4970SChristoph Hellwig 	.ct_item_ops		= &nvmet_host_item_ops,
1482a07b4970SChristoph Hellwig 	.ct_owner		= THIS_MODULE,
1483a07b4970SChristoph Hellwig };
1484a07b4970SChristoph Hellwig 
1485a07b4970SChristoph Hellwig static struct config_group *nvmet_hosts_make_group(struct config_group *group,
1486a07b4970SChristoph Hellwig 		const char *name)
1487a07b4970SChristoph Hellwig {
1488a07b4970SChristoph Hellwig 	struct nvmet_host *host;
1489a07b4970SChristoph Hellwig 
1490a07b4970SChristoph Hellwig 	host = kzalloc(sizeof(*host), GFP_KERNEL);
1491a07b4970SChristoph Hellwig 	if (!host)
1492a07b4970SChristoph Hellwig 		return ERR_PTR(-ENOMEM);
1493a07b4970SChristoph Hellwig 
1494a07b4970SChristoph Hellwig 	config_group_init_type_name(&host->group, name, &nvmet_host_type);
1495a07b4970SChristoph Hellwig 
1496a07b4970SChristoph Hellwig 	return &host->group;
1497a07b4970SChristoph Hellwig }
1498a07b4970SChristoph Hellwig 
1499a07b4970SChristoph Hellwig static struct configfs_group_operations nvmet_hosts_group_ops = {
1500a07b4970SChristoph Hellwig 	.make_group		= nvmet_hosts_make_group,
1501a07b4970SChristoph Hellwig };
1502a07b4970SChristoph Hellwig 
150366603a31SBhumika Goyal static const struct config_item_type nvmet_hosts_type = {
1504a07b4970SChristoph Hellwig 	.ct_group_ops		= &nvmet_hosts_group_ops,
1505a07b4970SChristoph Hellwig 	.ct_owner		= THIS_MODULE,
1506a07b4970SChristoph Hellwig };
1507a07b4970SChristoph Hellwig 
1508a07b4970SChristoph Hellwig static struct config_group nvmet_hosts_group;
1509a07b4970SChristoph Hellwig 
151066603a31SBhumika Goyal static const struct config_item_type nvmet_root_type = {
1511a07b4970SChristoph Hellwig 	.ct_owner		= THIS_MODULE,
1512a07b4970SChristoph Hellwig };
1513a07b4970SChristoph Hellwig 
1514a07b4970SChristoph Hellwig static struct configfs_subsystem nvmet_configfs_subsystem = {
1515a07b4970SChristoph Hellwig 	.su_group = {
1516a07b4970SChristoph Hellwig 		.cg_item = {
1517a07b4970SChristoph Hellwig 			.ci_namebuf	= "nvmet",
1518a07b4970SChristoph Hellwig 			.ci_type	= &nvmet_root_type,
1519a07b4970SChristoph Hellwig 		},
1520a07b4970SChristoph Hellwig 	},
1521a07b4970SChristoph Hellwig };
1522a07b4970SChristoph Hellwig 
1523a07b4970SChristoph Hellwig int __init nvmet_init_configfs(void)
1524a07b4970SChristoph Hellwig {
1525a07b4970SChristoph Hellwig 	int ret;
1526a07b4970SChristoph Hellwig 
1527a07b4970SChristoph Hellwig 	config_group_init(&nvmet_configfs_subsystem.su_group);
1528a07b4970SChristoph Hellwig 	mutex_init(&nvmet_configfs_subsystem.su_mutex);
1529a07b4970SChristoph Hellwig 
1530a07b4970SChristoph Hellwig 	config_group_init_type_name(&nvmet_subsystems_group,
1531a07b4970SChristoph Hellwig 			"subsystems", &nvmet_subsystems_type);
1532a07b4970SChristoph Hellwig 	configfs_add_default_group(&nvmet_subsystems_group,
1533a07b4970SChristoph Hellwig 			&nvmet_configfs_subsystem.su_group);
1534a07b4970SChristoph Hellwig 
1535a07b4970SChristoph Hellwig 	config_group_init_type_name(&nvmet_ports_group,
1536a07b4970SChristoph Hellwig 			"ports", &nvmet_ports_type);
1537a07b4970SChristoph Hellwig 	configfs_add_default_group(&nvmet_ports_group,
1538a07b4970SChristoph Hellwig 			&nvmet_configfs_subsystem.su_group);
1539a07b4970SChristoph Hellwig 
1540a07b4970SChristoph Hellwig 	config_group_init_type_name(&nvmet_hosts_group,
1541a07b4970SChristoph Hellwig 			"hosts", &nvmet_hosts_type);
1542a07b4970SChristoph Hellwig 	configfs_add_default_group(&nvmet_hosts_group,
1543a07b4970SChristoph Hellwig 			&nvmet_configfs_subsystem.su_group);
1544a07b4970SChristoph Hellwig 
1545a07b4970SChristoph Hellwig 	ret = configfs_register_subsystem(&nvmet_configfs_subsystem);
1546a07b4970SChristoph Hellwig 	if (ret) {
1547a07b4970SChristoph Hellwig 		pr_err("configfs_register_subsystem: %d\n", ret);
1548a07b4970SChristoph Hellwig 		return ret;
1549a07b4970SChristoph Hellwig 	}
1550a07b4970SChristoph Hellwig 
1551a07b4970SChristoph Hellwig 	return 0;
1552a07b4970SChristoph Hellwig }
1553a07b4970SChristoph Hellwig 
1554a07b4970SChristoph Hellwig void __exit nvmet_exit_configfs(void)
1555a07b4970SChristoph Hellwig {
1556a07b4970SChristoph Hellwig 	configfs_unregister_subsystem(&nvmet_configfs_subsystem);
1557a07b4970SChristoph Hellwig }
1558