xref: /linux/net/sunrpc/sysfs.c (revision 7c6c4ed80b874f721bc7c2c937e098c56e37d2f0)
1 // SPDX-License-Identifier: GPL-2.0
2 /*
3  * Copyright (c) 2020 Anna Schumaker <Anna.Schumaker@Netapp.com>
4  */
5 #include <linux/sunrpc/clnt.h>
6 #include <linux/kobject.h>
7 #include <linux/sunrpc/addr.h>
8 #include <linux/sunrpc/xprtsock.h>
9 #include <net/net_namespace.h>
10 
11 #include "sysfs.h"
12 
13 struct xprt_addr {
14 	const char *addr;
15 	struct rcu_head rcu;
16 };
17 
free_xprt_addr(struct rcu_head * head)18 static void free_xprt_addr(struct rcu_head *head)
19 {
20 	struct xprt_addr *addr = container_of(head, struct xprt_addr, rcu);
21 
22 	kfree(addr->addr);
23 	kfree(addr);
24 }
25 
26 static struct kset *rpc_sunrpc_kset;
27 static struct kobject *rpc_sunrpc_client_kobj, *rpc_sunrpc_xprt_switch_kobj;
28 
rpc_sysfs_object_release(struct kobject * kobj)29 static void rpc_sysfs_object_release(struct kobject *kobj)
30 {
31 	kfree(kobj);
32 }
33 
34 static const struct kobj_ns_type_operations *
rpc_sysfs_object_child_ns_type(const struct kobject * kobj)35 rpc_sysfs_object_child_ns_type(const struct kobject *kobj)
36 {
37 	return &net_ns_type_operations;
38 }
39 
40 static const struct kobj_type rpc_sysfs_object_type = {
41 	.release = rpc_sysfs_object_release,
42 	.sysfs_ops = &kobj_sysfs_ops,
43 	.child_ns_type = rpc_sysfs_object_child_ns_type,
44 };
45 
rpc_sysfs_object_alloc(const char * name,struct kset * kset,struct kobject * parent)46 static struct kobject *rpc_sysfs_object_alloc(const char *name,
47 					      struct kset *kset,
48 					      struct kobject *parent)
49 {
50 	struct kobject *kobj;
51 
52 	kobj = kzalloc_obj(*kobj);
53 	if (kobj) {
54 		kobj->kset = kset;
55 		if (kobject_init_and_add(kobj, &rpc_sysfs_object_type,
56 					 parent, "%s", name) == 0)
57 			return kobj;
58 		kobject_put(kobj);
59 	}
60 	return NULL;
61 }
62 
63 static inline struct rpc_clnt *
rpc_sysfs_client_kobj_get_clnt(struct kobject * kobj)64 rpc_sysfs_client_kobj_get_clnt(struct kobject *kobj)
65 {
66 	struct rpc_sysfs_client *c = container_of(kobj,
67 		struct rpc_sysfs_client, kobject);
68 	struct rpc_clnt *ret = c->clnt;
69 
70 	return refcount_inc_not_zero(&ret->cl_count) ? ret : NULL;
71 }
72 
73 static inline struct rpc_xprt *
rpc_sysfs_xprt_kobj_get_xprt(struct kobject * kobj)74 rpc_sysfs_xprt_kobj_get_xprt(struct kobject *kobj)
75 {
76 	struct rpc_sysfs_xprt *x = container_of(kobj,
77 		struct rpc_sysfs_xprt, kobject);
78 
79 	return xprt_get(x->xprt);
80 }
81 
82 static inline struct rpc_xprt_switch *
rpc_sysfs_xprt_kobj_get_xprt_switch(struct kobject * kobj)83 rpc_sysfs_xprt_kobj_get_xprt_switch(struct kobject *kobj)
84 {
85 	struct rpc_sysfs_xprt *x = container_of(kobj,
86 		struct rpc_sysfs_xprt, kobject);
87 
88 	return xprt_switch_get(x->xprt_switch);
89 }
90 
91 static inline struct rpc_xprt_switch *
rpc_sysfs_xprt_switch_kobj_get_xprt(struct kobject * kobj)92 rpc_sysfs_xprt_switch_kobj_get_xprt(struct kobject *kobj)
93 {
94 	struct rpc_sysfs_xprt_switch *x = container_of(kobj,
95 		struct rpc_sysfs_xprt_switch, kobject);
96 
97 	return xprt_switch_get(x->xprt_switch);
98 }
99 
rpc_sysfs_clnt_version_show(struct kobject * kobj,struct kobj_attribute * attr,char * buf)100 static ssize_t rpc_sysfs_clnt_version_show(struct kobject *kobj,
101 					   struct kobj_attribute *attr,
102 					   char *buf)
103 {
104 	struct rpc_clnt *clnt = rpc_sysfs_client_kobj_get_clnt(kobj);
105 	ssize_t ret;
106 
107 	if (!clnt)
108 		return sprintf(buf, "<closed>\n");
109 
110 	ret = sprintf(buf, "%u", clnt->cl_vers);
111 	refcount_dec(&clnt->cl_count);
112 	return ret;
113 }
114 
rpc_sysfs_clnt_program_show(struct kobject * kobj,struct kobj_attribute * attr,char * buf)115 static ssize_t rpc_sysfs_clnt_program_show(struct kobject *kobj,
116 					   struct kobj_attribute *attr,
117 					   char *buf)
118 {
119 	struct rpc_clnt *clnt = rpc_sysfs_client_kobj_get_clnt(kobj);
120 	ssize_t ret;
121 
122 	if (!clnt)
123 		return sprintf(buf, "<closed>\n");
124 
125 	ret = sprintf(buf, "%s", clnt->cl_program->name);
126 	refcount_dec(&clnt->cl_count);
127 	return ret;
128 }
129 
rpc_sysfs_clnt_max_connect_show(struct kobject * kobj,struct kobj_attribute * attr,char * buf)130 static ssize_t rpc_sysfs_clnt_max_connect_show(struct kobject *kobj,
131 					       struct kobj_attribute *attr,
132 					       char *buf)
133 {
134 	struct rpc_clnt *clnt = rpc_sysfs_client_kobj_get_clnt(kobj);
135 	ssize_t ret;
136 
137 	if (!clnt)
138 		return sprintf(buf, "<closed>\n");
139 
140 	ret = sprintf(buf, "%u\n", clnt->cl_max_connect);
141 	refcount_dec(&clnt->cl_count);
142 	return ret;
143 }
144 
rpc_sysfs_xprt_dstaddr_show(struct kobject * kobj,struct kobj_attribute * attr,char * buf)145 static ssize_t rpc_sysfs_xprt_dstaddr_show(struct kobject *kobj,
146 					   struct kobj_attribute *attr,
147 					   char *buf)
148 {
149 	struct rpc_xprt *xprt = rpc_sysfs_xprt_kobj_get_xprt(kobj);
150 	ssize_t ret;
151 
152 	if (!xprt) {
153 		ret = sprintf(buf, "<closed>\n");
154 		goto out;
155 	}
156 	ret = sprintf(buf, "%s\n", xprt->address_strings[RPC_DISPLAY_ADDR]);
157 	xprt_put(xprt);
158 out:
159 	return ret;
160 }
161 
rpc_sysfs_xprt_srcaddr_show(struct kobject * kobj,struct kobj_attribute * attr,char * buf)162 static ssize_t rpc_sysfs_xprt_srcaddr_show(struct kobject *kobj,
163 					   struct kobj_attribute *attr,
164 					   char *buf)
165 {
166 	struct rpc_xprt *xprt = rpc_sysfs_xprt_kobj_get_xprt(kobj);
167 	size_t buflen = PAGE_SIZE;
168 	ssize_t ret;
169 
170 	if (!xprt || !xprt_connected(xprt)) {
171 		ret = sprintf(buf, "<closed>\n");
172 	} else if (xprt->ops->get_srcaddr) {
173 		ret = xprt->ops->get_srcaddr(xprt, buf, buflen);
174 		if (ret > 0) {
175 			if (ret < buflen - 1) {
176 				buf[ret] = '\n';
177 				ret++;
178 				buf[ret] = '\0';
179 			}
180 		} else
181 			ret = sprintf(buf, "<closed>\n");
182 	} else
183 		ret = sprintf(buf, "<not a socket>\n");
184 	xprt_put(xprt);
185 	return ret;
186 }
187 
188 static const char *xprtsec_strings[] = {
189 	[RPC_XPRTSEC_NONE] = "none",
190 	[RPC_XPRTSEC_TLS_ANON] = "tls-anon",
191 	[RPC_XPRTSEC_TLS_X509] = "tls-x509",
192 };
193 
rpc_sysfs_xprt_xprtsec_show(struct kobject * kobj,struct kobj_attribute * attr,char * buf)194 static ssize_t rpc_sysfs_xprt_xprtsec_show(struct kobject *kobj,
195 					   struct kobj_attribute *attr,
196 					   char *buf)
197 {
198 	struct rpc_xprt *xprt = rpc_sysfs_xprt_kobj_get_xprt(kobj);
199 	ssize_t ret;
200 
201 	if (!xprt) {
202 		ret = sprintf(buf, "<closed>\n");
203 		goto out;
204 	}
205 
206 	ret = sprintf(buf, "%s\n", xprtsec_strings[xprt->xprtsec.policy]);
207 	xprt_put(xprt);
208 out:
209 	return ret;
210 
211 }
212 
rpc_sysfs_xprt_info_show(struct kobject * kobj,struct kobj_attribute * attr,char * buf)213 static ssize_t rpc_sysfs_xprt_info_show(struct kobject *kobj,
214 					struct kobj_attribute *attr, char *buf)
215 {
216 	struct rpc_xprt *xprt = rpc_sysfs_xprt_kobj_get_xprt(kobj);
217 	unsigned short srcport = 0;
218 	size_t buflen = PAGE_SIZE;
219 	ssize_t ret;
220 
221 	if (!xprt || !xprt_connected(xprt)) {
222 		ret = sprintf(buf, "<closed>\n");
223 		goto out;
224 	}
225 
226 	if (xprt->ops->get_srcport)
227 		srcport = xprt->ops->get_srcport(xprt);
228 
229 	ret = snprintf(buf, buflen,
230 		       "last_used=%lu\ncur_cong=%lu\ncong_win=%lu\n"
231 		       "max_num_slots=%u\nmin_num_slots=%u\nnum_reqs=%u\n"
232 		       "binding_q_len=%u\nsending_q_len=%u\npending_q_len=%u\n"
233 		       "backlog_q_len=%u\nmain_xprt=%d\nsrc_port=%u\n"
234 		       "tasks_queuelen=%ld\ndst_port=%s\n",
235 		       xprt->last_used, xprt->cong, xprt->cwnd, xprt->max_reqs,
236 		       xprt->min_reqs, xprt->num_reqs, xprt->binding.qlen,
237 		       xprt->sending.qlen, xprt->pending.qlen,
238 		       xprt->backlog.qlen, xprt->main, srcport,
239 		       atomic_long_read(&xprt->queuelen),
240 		       xprt->address_strings[RPC_DISPLAY_PORT]);
241 out:
242 	xprt_put(xprt);
243 	return ret;
244 }
245 
rpc_sysfs_xprt_state_show(struct kobject * kobj,struct kobj_attribute * attr,char * buf)246 static ssize_t rpc_sysfs_xprt_state_show(struct kobject *kobj,
247 					 struct kobj_attribute *attr,
248 					 char *buf)
249 {
250 	struct rpc_xprt *xprt = rpc_sysfs_xprt_kobj_get_xprt(kobj);
251 	ssize_t ret;
252 	int locked, connected, connecting, close_wait, bound, binding,
253 	    closing, congested, cwnd_wait, write_space, offline, remove;
254 
255 	if (!(xprt && xprt->state)) {
256 		ret = sprintf(buf, "state=CLOSED\n");
257 	} else {
258 		locked = test_bit(XPRT_LOCKED, &xprt->state);
259 		connected = test_bit(XPRT_CONNECTED, &xprt->state);
260 		connecting = test_bit(XPRT_CONNECTING, &xprt->state);
261 		close_wait = test_bit(XPRT_CLOSE_WAIT, &xprt->state);
262 		bound = test_bit(XPRT_BOUND, &xprt->state);
263 		binding = test_bit(XPRT_BINDING, &xprt->state);
264 		closing = test_bit(XPRT_CLOSING, &xprt->state);
265 		congested = test_bit(XPRT_CONGESTED, &xprt->state);
266 		cwnd_wait = test_bit(XPRT_CWND_WAIT, &xprt->state);
267 		write_space = test_bit(XPRT_WRITE_SPACE, &xprt->state);
268 		offline = test_bit(XPRT_OFFLINE, &xprt->state);
269 		remove = test_bit(XPRT_REMOVE, &xprt->state);
270 
271 		ret = sprintf(buf, "state=%s %s %s %s %s %s %s %s %s %s %s %s\n",
272 			      locked ? "LOCKED" : "",
273 			      connected ? "CONNECTED" : "",
274 			      connecting ? "CONNECTING" : "",
275 			      close_wait ? "CLOSE_WAIT" : "",
276 			      bound ? "BOUND" : "",
277 			      binding ? "BOUNDING" : "",
278 			      closing ? "CLOSING" : "",
279 			      congested ? "CONGESTED" : "",
280 			      cwnd_wait ? "CWND_WAIT" : "",
281 			      write_space ? "WRITE_SPACE" : "",
282 			      offline ? "OFFLINE" : "",
283 			      remove ? "REMOVE" : "");
284 	}
285 
286 	xprt_put(xprt);
287 	return ret;
288 }
289 
rpc_sysfs_xprt_del_xprt_show(struct kobject * kobj,struct kobj_attribute * attr,char * buf)290 static ssize_t rpc_sysfs_xprt_del_xprt_show(struct kobject *kobj,
291 					    struct kobj_attribute *attr,
292 					    char *buf)
293 {
294 	return sprintf(buf, "# delete this xprt\n");
295 }
296 
297 
rpc_sysfs_xprt_switch_info_show(struct kobject * kobj,struct kobj_attribute * attr,char * buf)298 static ssize_t rpc_sysfs_xprt_switch_info_show(struct kobject *kobj,
299 					       struct kobj_attribute *attr,
300 					       char *buf)
301 {
302 	struct rpc_xprt_switch *xprt_switch =
303 		rpc_sysfs_xprt_switch_kobj_get_xprt(kobj);
304 	ssize_t ret;
305 
306 	if (!xprt_switch)
307 		return 0;
308 	ret = sprintf(buf, "num_xprts=%u\nnum_active=%u\n"
309 		      "num_unique_destaddr=%u\nqueue_len=%ld\n",
310 		      xprt_switch->xps_nxprts, xprt_switch->xps_nactive,
311 		      xprt_switch->xps_nunique_destaddr_xprts,
312 		      atomic_long_read(&xprt_switch->xps_queuelen));
313 	xprt_switch_put(xprt_switch);
314 	return ret;
315 }
316 
rpc_sysfs_xprt_switch_add_xprt_show(struct kobject * kobj,struct kobj_attribute * attr,char * buf)317 static ssize_t rpc_sysfs_xprt_switch_add_xprt_show(struct kobject *kobj,
318 						   struct kobj_attribute *attr,
319 						   char *buf)
320 {
321 	return sprintf(buf, "# add one xprt to this xprt_switch\n");
322 }
323 
rpc_sysfs_xprt_switch_add_xprt_store(struct kobject * kobj,struct kobj_attribute * attr,const char * buf,size_t count)324 static ssize_t rpc_sysfs_xprt_switch_add_xprt_store(struct kobject *kobj,
325 						    struct kobj_attribute *attr,
326 						    const char *buf, size_t count)
327 {
328 	struct rpc_xprt_switch *xprt_switch =
329 		rpc_sysfs_xprt_switch_kobj_get_xprt(kobj);
330 	struct xprt_create xprt_create_args;
331 	struct rpc_xprt *xprt, *new;
332 
333 	if (!xprt_switch)
334 		return 0;
335 
336 	xprt = rpc_xprt_switch_get_main_xprt(xprt_switch);
337 	if (!xprt)
338 		goto out;
339 
340 	xprt_create_args.ident = xprt->xprt_class->ident;
341 	xprt_create_args.net = xprt->xprt_net;
342 	xprt_create_args.dstaddr = (struct sockaddr *)&xprt->addr;
343 	xprt_create_args.addrlen = xprt->addrlen;
344 	xprt_create_args.servername = xprt->servername;
345 	xprt_create_args.bc_xprt = xprt->bc_xprt;
346 	xprt_create_args.xprtsec = xprt->xprtsec;
347 	xprt_create_args.connect_timeout = xprt->connect_timeout;
348 	xprt_create_args.reconnect_timeout = xprt->max_reconnect_timeout;
349 
350 	new = xprt_create_transport(&xprt_create_args);
351 	if (IS_ERR_OR_NULL(new)) {
352 		count = PTR_ERR(new);
353 		goto out_put_xprt;
354 	}
355 
356 	rpc_xprt_switch_add_xprt(xprt_switch, new);
357 	xprt_put(new);
358 
359 out_put_xprt:
360 	xprt_put(xprt);
361 out:
362 	xprt_switch_put(xprt_switch);
363 	return count;
364 }
365 
rpc_sysfs_xprt_dstaddr_store(struct kobject * kobj,struct kobj_attribute * attr,const char * buf,size_t count)366 static ssize_t rpc_sysfs_xprt_dstaddr_store(struct kobject *kobj,
367 					    struct kobj_attribute *attr,
368 					    const char *buf, size_t count)
369 {
370 	struct rpc_xprt *xprt = rpc_sysfs_xprt_kobj_get_xprt(kobj);
371 	struct sockaddr *saddr;
372 	char *dst_addr;
373 	int port;
374 	struct xprt_addr *saved_addr;
375 	size_t buf_len;
376 
377 	if (!xprt)
378 		return 0;
379 	if (!(xprt->xprt_class->ident == XPRT_TRANSPORT_TCP ||
380 	      xprt->xprt_class->ident == XPRT_TRANSPORT_TCP_TLS ||
381 	      xprt->xprt_class->ident == XPRT_TRANSPORT_RDMA)) {
382 		xprt_put(xprt);
383 		return -EOPNOTSUPP;
384 	}
385 
386 	if (wait_on_bit_lock(&xprt->state, XPRT_LOCKED, TASK_KILLABLE)) {
387 		count = -EINTR;
388 		goto out_put;
389 	}
390 	saddr = (struct sockaddr *)&xprt->addr;
391 	port = rpc_get_port(saddr);
392 
393 	/* buf_len is the len until the first occurrence of either
394 	 * '\n' or '\0'
395 	 */
396 	buf_len = strcspn(buf, "\n");
397 
398 	dst_addr = kstrndup(buf, buf_len, GFP_KERNEL);
399 	if (!dst_addr)
400 		goto out_err;
401 	saved_addr = kzalloc_obj(*saved_addr);
402 	if (!saved_addr)
403 		goto out_err_free;
404 	saved_addr->addr =
405 		rcu_dereference_raw(xprt->address_strings[RPC_DISPLAY_ADDR]);
406 	rcu_assign_pointer(xprt->address_strings[RPC_DISPLAY_ADDR], dst_addr);
407 	call_rcu(&saved_addr->rcu, free_xprt_addr);
408 	xprt->addrlen = rpc_pton(xprt->xprt_net, buf, buf_len, saddr,
409 				 sizeof(*saddr));
410 	rpc_set_port(saddr, port);
411 
412 	xprt_force_disconnect(xprt);
413 out:
414 	xprt_release_write(xprt, NULL);
415 out_put:
416 	xprt_put(xprt);
417 	return count;
418 out_err_free:
419 	kfree(dst_addr);
420 out_err:
421 	count = -ENOMEM;
422 	goto out;
423 }
424 
rpc_sysfs_xprt_state_change(struct kobject * kobj,struct kobj_attribute * attr,const char * buf,size_t count)425 static ssize_t rpc_sysfs_xprt_state_change(struct kobject *kobj,
426 					   struct kobj_attribute *attr,
427 					   const char *buf, size_t count)
428 {
429 	struct rpc_xprt *xprt = rpc_sysfs_xprt_kobj_get_xprt(kobj);
430 	int offline = 0, online = 0, remove = 0;
431 	struct rpc_xprt_switch *xps = rpc_sysfs_xprt_kobj_get_xprt_switch(kobj);
432 
433 	if (!xprt || !xps) {
434 		count = 0;
435 		goto out_put;
436 	}
437 
438 	if (!strncmp(buf, "offline", 7))
439 		offline = 1;
440 	else if (!strncmp(buf, "online", 6))
441 		online = 1;
442 	else if (!strncmp(buf, "remove", 6))
443 		remove = 1;
444 	else {
445 		count = -EINVAL;
446 		goto out_put;
447 	}
448 
449 	if (wait_on_bit_lock(&xprt->state, XPRT_LOCKED, TASK_KILLABLE)) {
450 		count = -EINTR;
451 		goto out_put;
452 	}
453 	if (xprt->main) {
454 		count = -EINVAL;
455 		goto release_tasks;
456 	}
457 	if (offline) {
458 		xprt_set_offline_locked(xprt, xps);
459 	} else if (online) {
460 		xprt_set_online_locked(xprt, xps);
461 	} else if (remove) {
462 		if (test_bit(XPRT_OFFLINE, &xprt->state))
463 			xprt_delete_locked(xprt, xps);
464 		else
465 			count = -EINVAL;
466 	}
467 
468 release_tasks:
469 	xprt_release_write(xprt, NULL);
470 out_put:
471 	xprt_put(xprt);
472 	xprt_switch_put(xps);
473 	return count;
474 }
475 
rpc_sysfs_xprt_del_xprt(struct kobject * kobj,struct kobj_attribute * attr,const char * buf,size_t count)476 static ssize_t rpc_sysfs_xprt_del_xprt(struct kobject *kobj,
477 				       struct kobj_attribute *attr,
478 				       const char *buf, size_t count)
479 {
480 	struct rpc_xprt *xprt = rpc_sysfs_xprt_kobj_get_xprt(kobj);
481 	struct rpc_xprt_switch *xps = rpc_sysfs_xprt_kobj_get_xprt_switch(kobj);
482 
483 	if (!xprt || !xps) {
484 		count = 0;
485 		goto out;
486 	}
487 
488 	if (xprt->main) {
489 		count = -EINVAL;
490 		goto release_tasks;
491 	}
492 
493 	if (wait_on_bit_lock(&xprt->state, XPRT_LOCKED, TASK_KILLABLE)) {
494 		count = -EINTR;
495 		goto out_put;
496 	}
497 
498 	xprt_set_offline_locked(xprt, xps);
499 	xprt_delete_locked(xprt, xps);
500 
501 release_tasks:
502 	xprt_release_write(xprt, NULL);
503 out_put:
504 	xprt_put(xprt);
505 	xprt_switch_put(xps);
506 out:
507 	return count;
508 }
509 
rpc_sysfs_init(void)510 int rpc_sysfs_init(void)
511 {
512 	rpc_sunrpc_kset = kset_create_and_add("sunrpc", NULL, kernel_kobj);
513 	if (!rpc_sunrpc_kset)
514 		return -ENOMEM;
515 	rpc_sunrpc_client_kobj =
516 		rpc_sysfs_object_alloc("rpc-clients", rpc_sunrpc_kset, NULL);
517 	if (!rpc_sunrpc_client_kobj)
518 		goto err_client;
519 	rpc_sunrpc_xprt_switch_kobj =
520 		rpc_sysfs_object_alloc("xprt-switches", rpc_sunrpc_kset, NULL);
521 	if (!rpc_sunrpc_xprt_switch_kobj)
522 		goto err_switch;
523 	return 0;
524 err_switch:
525 	kobject_put(rpc_sunrpc_client_kobj);
526 	rpc_sunrpc_client_kobj = NULL;
527 err_client:
528 	kset_unregister(rpc_sunrpc_kset);
529 	rpc_sunrpc_kset = NULL;
530 	return -ENOMEM;
531 }
532 
rpc_sysfs_client_release(struct kobject * kobj)533 static void rpc_sysfs_client_release(struct kobject *kobj)
534 {
535 	struct rpc_sysfs_client *c;
536 
537 	c = container_of(kobj, struct rpc_sysfs_client, kobject);
538 	kfree(c);
539 }
540 
rpc_sysfs_xprt_switch_release(struct kobject * kobj)541 static void rpc_sysfs_xprt_switch_release(struct kobject *kobj)
542 {
543 	struct rpc_sysfs_xprt_switch *xprt_switch;
544 
545 	xprt_switch = container_of(kobj, struct rpc_sysfs_xprt_switch, kobject);
546 	kfree(xprt_switch);
547 }
548 
rpc_sysfs_xprt_release(struct kobject * kobj)549 static void rpc_sysfs_xprt_release(struct kobject *kobj)
550 {
551 	struct rpc_sysfs_xprt *xprt;
552 
553 	xprt = container_of(kobj, struct rpc_sysfs_xprt, kobject);
554 	kfree(xprt);
555 }
556 
rpc_sysfs_client_namespace(const struct kobject * kobj)557 static const struct ns_common *rpc_sysfs_client_namespace(const struct kobject *kobj)
558 {
559 	return to_ns_common(container_of(kobj, struct rpc_sysfs_client,
560 					 kobject)->net);
561 }
562 
rpc_sysfs_xprt_switch_namespace(const struct kobject * kobj)563 static const struct ns_common *rpc_sysfs_xprt_switch_namespace(const struct kobject *kobj)
564 {
565 	return to_ns_common(container_of(kobj, struct rpc_sysfs_xprt_switch,
566 					 kobject)->net);
567 }
568 
rpc_sysfs_xprt_namespace(const struct kobject * kobj)569 static const struct ns_common *rpc_sysfs_xprt_namespace(const struct kobject *kobj)
570 {
571 	return to_ns_common(container_of(kobj, struct rpc_sysfs_xprt,
572 					 kobject)->xprt->xprt_net);
573 }
574 
575 static struct kobj_attribute rpc_sysfs_clnt_version = __ATTR(rpc_version,
576 	0444, rpc_sysfs_clnt_version_show, NULL);
577 
578 static struct kobj_attribute rpc_sysfs_clnt_program = __ATTR(program,
579 	0444, rpc_sysfs_clnt_program_show, NULL);
580 
581 static struct kobj_attribute rpc_sysfs_clnt_max_connect = __ATTR(max_connect,
582 	0444, rpc_sysfs_clnt_max_connect_show, NULL);
583 
584 static struct attribute *rpc_sysfs_rpc_clnt_attrs[] = {
585 	&rpc_sysfs_clnt_version.attr,
586 	&rpc_sysfs_clnt_program.attr,
587 	&rpc_sysfs_clnt_max_connect.attr,
588 	NULL,
589 };
590 ATTRIBUTE_GROUPS(rpc_sysfs_rpc_clnt);
591 
592 static struct kobj_attribute rpc_sysfs_xprt_dstaddr = __ATTR(dstaddr,
593 	0644, rpc_sysfs_xprt_dstaddr_show, rpc_sysfs_xprt_dstaddr_store);
594 
595 static struct kobj_attribute rpc_sysfs_xprt_srcaddr = __ATTR(srcaddr,
596 	0644, rpc_sysfs_xprt_srcaddr_show, NULL);
597 
598 static struct kobj_attribute rpc_sysfs_xprt_xprtsec = __ATTR(xprtsec,
599 	0644, rpc_sysfs_xprt_xprtsec_show, NULL);
600 
601 static struct kobj_attribute rpc_sysfs_xprt_info = __ATTR(xprt_info,
602 	0444, rpc_sysfs_xprt_info_show, NULL);
603 
604 static struct kobj_attribute rpc_sysfs_xprt_change_state = __ATTR(xprt_state,
605 	0644, rpc_sysfs_xprt_state_show, rpc_sysfs_xprt_state_change);
606 
607 static struct kobj_attribute rpc_sysfs_xprt_del = __ATTR(del_xprt,
608 	0644, rpc_sysfs_xprt_del_xprt_show, rpc_sysfs_xprt_del_xprt);
609 
610 static struct attribute *rpc_sysfs_xprt_attrs[] = {
611 	&rpc_sysfs_xprt_dstaddr.attr,
612 	&rpc_sysfs_xprt_srcaddr.attr,
613 	&rpc_sysfs_xprt_xprtsec.attr,
614 	&rpc_sysfs_xprt_info.attr,
615 	&rpc_sysfs_xprt_change_state.attr,
616 	&rpc_sysfs_xprt_del.attr,
617 	NULL,
618 };
619 ATTRIBUTE_GROUPS(rpc_sysfs_xprt);
620 
621 static struct kobj_attribute rpc_sysfs_xprt_switch_info =
622 	__ATTR(xprt_switch_info, 0444, rpc_sysfs_xprt_switch_info_show, NULL);
623 
624 static struct kobj_attribute rpc_sysfs_xprt_switch_add_xprt =
625 	__ATTR(add_xprt, 0644, rpc_sysfs_xprt_switch_add_xprt_show,
626 		rpc_sysfs_xprt_switch_add_xprt_store);
627 
628 static struct attribute *rpc_sysfs_xprt_switch_attrs[] = {
629 	&rpc_sysfs_xprt_switch_info.attr,
630 	&rpc_sysfs_xprt_switch_add_xprt.attr,
631 	NULL,
632 };
633 ATTRIBUTE_GROUPS(rpc_sysfs_xprt_switch);
634 
635 static const struct kobj_type rpc_sysfs_client_type = {
636 	.release = rpc_sysfs_client_release,
637 	.default_groups = rpc_sysfs_rpc_clnt_groups,
638 	.sysfs_ops = &kobj_sysfs_ops,
639 	.namespace = rpc_sysfs_client_namespace,
640 };
641 
642 static const struct kobj_type rpc_sysfs_xprt_switch_type = {
643 	.release = rpc_sysfs_xprt_switch_release,
644 	.default_groups = rpc_sysfs_xprt_switch_groups,
645 	.sysfs_ops = &kobj_sysfs_ops,
646 	.namespace = rpc_sysfs_xprt_switch_namespace,
647 };
648 
649 static const struct kobj_type rpc_sysfs_xprt_type = {
650 	.release = rpc_sysfs_xprt_release,
651 	.default_groups = rpc_sysfs_xprt_groups,
652 	.sysfs_ops = &kobj_sysfs_ops,
653 	.namespace = rpc_sysfs_xprt_namespace,
654 };
655 
rpc_sysfs_exit(void)656 void rpc_sysfs_exit(void)
657 {
658 	kobject_put(rpc_sunrpc_client_kobj);
659 	kobject_put(rpc_sunrpc_xprt_switch_kobj);
660 	kset_unregister(rpc_sunrpc_kset);
661 }
662 
rpc_sysfs_client_alloc(struct kobject * parent,struct net * net,int clid)663 static struct rpc_sysfs_client *rpc_sysfs_client_alloc(struct kobject *parent,
664 						       struct net *net,
665 						       int clid)
666 {
667 	struct rpc_sysfs_client *p;
668 
669 	p = kzalloc_obj(*p);
670 	if (p) {
671 		p->net = net;
672 		p->kobject.kset = rpc_sunrpc_kset;
673 		if (kobject_init_and_add(&p->kobject, &rpc_sysfs_client_type,
674 					 parent, "clnt-%d", clid) == 0)
675 			return p;
676 		kobject_put(&p->kobject);
677 	}
678 	return NULL;
679 }
680 
681 static struct rpc_sysfs_xprt_switch *
rpc_sysfs_xprt_switch_alloc(struct kobject * parent,struct rpc_xprt_switch * xprt_switch,struct net * net,gfp_t gfp_flags)682 rpc_sysfs_xprt_switch_alloc(struct kobject *parent,
683 			    struct rpc_xprt_switch *xprt_switch,
684 			    struct net *net,
685 			    gfp_t gfp_flags)
686 {
687 	struct rpc_sysfs_xprt_switch *p;
688 
689 	p = kzalloc_obj(*p, gfp_flags);
690 	if (p) {
691 		p->net = net;
692 		p->kobject.kset = rpc_sunrpc_kset;
693 		if (kobject_init_and_add(&p->kobject,
694 					 &rpc_sysfs_xprt_switch_type,
695 					 parent, "switch-%d",
696 					 xprt_switch->xps_id) == 0)
697 			return p;
698 		kobject_put(&p->kobject);
699 	}
700 	return NULL;
701 }
702 
rpc_sysfs_xprt_alloc(struct kobject * parent,struct rpc_xprt * xprt,gfp_t gfp_flags)703 static struct rpc_sysfs_xprt *rpc_sysfs_xprt_alloc(struct kobject *parent,
704 						   struct rpc_xprt *xprt,
705 						   gfp_t gfp_flags)
706 {
707 	struct rpc_sysfs_xprt *p;
708 
709 	p = kzalloc_obj(*p, gfp_flags);
710 	if (!p)
711 		goto out;
712 	p->kobject.kset = rpc_sunrpc_kset;
713 	if (kobject_init_and_add(&p->kobject, &rpc_sysfs_xprt_type,
714 				 parent, "xprt-%d-%s", xprt->id,
715 				 xprt->address_strings[RPC_DISPLAY_PROTO]) == 0)
716 		return p;
717 	kobject_put(&p->kobject);
718 out:
719 	return NULL;
720 }
721 
rpc_sysfs_client_setup(struct rpc_clnt * clnt,struct rpc_xprt_switch * xprt_switch,struct net * net)722 void rpc_sysfs_client_setup(struct rpc_clnt *clnt,
723 			    struct rpc_xprt_switch *xprt_switch,
724 			    struct net *net)
725 {
726 	struct rpc_sysfs_client *rpc_client;
727 	struct rpc_sysfs_xprt_switch *xswitch =
728 		(struct rpc_sysfs_xprt_switch *)xprt_switch->xps_sysfs;
729 
730 	if (!xswitch)
731 		return;
732 
733 	rpc_client = rpc_sysfs_client_alloc(rpc_sunrpc_client_kobj,
734 					    net, clnt->cl_clid);
735 	if (rpc_client) {
736 		char name[] = "switch";
737 		int ret;
738 
739 		clnt->cl_sysfs = rpc_client;
740 		rpc_client->clnt = clnt;
741 		rpc_client->xprt_switch = xprt_switch;
742 		kobject_uevent(&rpc_client->kobject, KOBJ_ADD);
743 		ret = sysfs_create_link_nowarn(&rpc_client->kobject,
744 					       &xswitch->kobject, name);
745 		if (ret)
746 			pr_warn("can't create link to %s in sysfs (%d)\n",
747 				name, ret);
748 	}
749 }
750 
rpc_sysfs_xprt_switch_setup(struct rpc_xprt_switch * xprt_switch,struct rpc_xprt * xprt,gfp_t gfp_flags)751 void rpc_sysfs_xprt_switch_setup(struct rpc_xprt_switch *xprt_switch,
752 				 struct rpc_xprt *xprt,
753 				 gfp_t gfp_flags)
754 {
755 	struct rpc_sysfs_xprt_switch *rpc_xprt_switch;
756 	struct net *net;
757 
758 	if (xprt_switch->xps_net)
759 		net = xprt_switch->xps_net;
760 	else
761 		net = xprt->xprt_net;
762 	rpc_xprt_switch =
763 		rpc_sysfs_xprt_switch_alloc(rpc_sunrpc_xprt_switch_kobj,
764 					    xprt_switch, net, gfp_flags);
765 	if (rpc_xprt_switch) {
766 		xprt_switch->xps_sysfs = rpc_xprt_switch;
767 		rpc_xprt_switch->xprt_switch = xprt_switch;
768 		rpc_xprt_switch->xprt = xprt;
769 		kobject_uevent(&rpc_xprt_switch->kobject, KOBJ_ADD);
770 	} else {
771 		xprt_switch->xps_sysfs = NULL;
772 	}
773 }
774 
rpc_sysfs_xprt_setup(struct rpc_xprt_switch * xprt_switch,struct rpc_xprt * xprt,gfp_t gfp_flags)775 void rpc_sysfs_xprt_setup(struct rpc_xprt_switch *xprt_switch,
776 			  struct rpc_xprt *xprt,
777 			  gfp_t gfp_flags)
778 {
779 	struct rpc_sysfs_xprt *rpc_xprt;
780 	struct rpc_sysfs_xprt_switch *switch_obj =
781 		(struct rpc_sysfs_xprt_switch *)xprt_switch->xps_sysfs;
782 
783 	if (!switch_obj)
784 		return;
785 
786 	rpc_xprt = rpc_sysfs_xprt_alloc(&switch_obj->kobject, xprt, gfp_flags);
787 	if (rpc_xprt) {
788 		xprt->xprt_sysfs = rpc_xprt;
789 		rpc_xprt->xprt = xprt;
790 		rpc_xprt->xprt_switch = xprt_switch;
791 		kobject_uevent(&rpc_xprt->kobject, KOBJ_ADD);
792 	}
793 }
794 
rpc_sysfs_client_destroy(struct rpc_clnt * clnt)795 void rpc_sysfs_client_destroy(struct rpc_clnt *clnt)
796 {
797 	struct rpc_sysfs_client *rpc_client = clnt->cl_sysfs;
798 
799 	if (rpc_client) {
800 		char name[] = "switch";
801 
802 		sysfs_remove_link(&rpc_client->kobject, name);
803 		kobject_uevent(&rpc_client->kobject, KOBJ_REMOVE);
804 		kobject_del(&rpc_client->kobject);
805 		kobject_put(&rpc_client->kobject);
806 		clnt->cl_sysfs = NULL;
807 	}
808 }
809 
rpc_sysfs_xprt_switch_destroy(struct rpc_xprt_switch * xprt_switch)810 void rpc_sysfs_xprt_switch_destroy(struct rpc_xprt_switch *xprt_switch)
811 {
812 	struct rpc_sysfs_xprt_switch *rpc_xprt_switch = xprt_switch->xps_sysfs;
813 
814 	if (rpc_xprt_switch) {
815 		kobject_uevent(&rpc_xprt_switch->kobject, KOBJ_REMOVE);
816 		kobject_del(&rpc_xprt_switch->kobject);
817 		kobject_put(&rpc_xprt_switch->kobject);
818 		xprt_switch->xps_sysfs = NULL;
819 	}
820 }
821 
rpc_sysfs_xprt_destroy(struct rpc_xprt * xprt)822 void rpc_sysfs_xprt_destroy(struct rpc_xprt *xprt)
823 {
824 	struct rpc_sysfs_xprt *rpc_xprt = xprt->xprt_sysfs;
825 
826 	if (rpc_xprt) {
827 		kobject_uevent(&rpc_xprt->kobject, KOBJ_REMOVE);
828 		kobject_del(&rpc_xprt->kobject);
829 		kobject_put(&rpc_xprt->kobject);
830 		xprt->xprt_sysfs = NULL;
831 	}
832 }
833