sysfs.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Copyright (c) 2019 Hammerspace Inc
  4. */
  5. #include <linux/module.h>
  6. #include <linux/kobject.h>
  7. #include <linux/sysfs.h>
  8. #include <linux/fs.h>
  9. #include <linux/slab.h>
  10. #include <linux/netdevice.h>
  11. #include <linux/string.h>
  12. #include <linux/nfs_fs.h>
  13. #include <net/net_namespace.h>
  14. #include <linux/rcupdate.h>
  15. #include <linux/lockd/lockd.h>
  16. #include "internal.h"
  17. #include "nfs4_fs.h"
  18. #include "netns.h"
  19. #include "sysfs.h"
  20. static struct kset *nfs_kset;
  21. static void nfs_kset_release(struct kobject *kobj)
  22. {
  23. struct kset *kset = container_of(kobj, struct kset, kobj);
  24. kfree(kset);
  25. }
  26. static const struct kobj_ns_type_operations *nfs_netns_object_child_ns_type(
  27. const struct kobject *kobj)
  28. {
  29. return &net_ns_type_operations;
  30. }
  31. static struct kobj_type nfs_kset_type = {
  32. .release = nfs_kset_release,
  33. .sysfs_ops = &kobj_sysfs_ops,
  34. .child_ns_type = nfs_netns_object_child_ns_type,
  35. };
  36. int nfs_sysfs_init(void)
  37. {
  38. int ret;
  39. nfs_kset = kzalloc_obj(*nfs_kset);
  40. if (!nfs_kset)
  41. return -ENOMEM;
  42. ret = kobject_set_name(&nfs_kset->kobj, "nfs");
  43. if (ret) {
  44. kfree(nfs_kset);
  45. return ret;
  46. }
  47. nfs_kset->kobj.parent = fs_kobj;
  48. nfs_kset->kobj.ktype = &nfs_kset_type;
  49. nfs_kset->kobj.kset = NULL;
  50. ret = kset_register(nfs_kset);
  51. if (ret) {
  52. kfree(nfs_kset);
  53. return ret;
  54. }
  55. return 0;
  56. }
  57. void nfs_sysfs_exit(void)
  58. {
  59. kset_unregister(nfs_kset);
  60. }
  61. static ssize_t nfs_netns_identifier_show(struct kobject *kobj,
  62. struct kobj_attribute *attr, char *buf)
  63. {
  64. struct nfs_netns_client *c = container_of(kobj,
  65. struct nfs_netns_client,
  66. kobject);
  67. ssize_t ret;
  68. rcu_read_lock();
  69. ret = sysfs_emit(buf, "%s\n", rcu_dereference(c->identifier));
  70. rcu_read_unlock();
  71. return ret;
  72. }
  73. /* Strip trailing '\n' */
  74. static size_t nfs_string_strip(const char *c, size_t len)
  75. {
  76. while (len > 0 && c[len-1] == '\n')
  77. --len;
  78. return len;
  79. }
  80. static ssize_t nfs_netns_identifier_store(struct kobject *kobj,
  81. struct kobj_attribute *attr,
  82. const char *buf, size_t count)
  83. {
  84. struct nfs_netns_client *c = container_of(kobj,
  85. struct nfs_netns_client,
  86. kobject);
  87. const char *old;
  88. char *p;
  89. size_t len;
  90. len = nfs_string_strip(buf, min_t(size_t, count, CONTAINER_ID_MAXLEN));
  91. if (!len)
  92. return 0;
  93. p = kmemdup_nul(buf, len, GFP_KERNEL);
  94. if (!p)
  95. return -ENOMEM;
  96. old = rcu_dereference_protected(xchg(&c->identifier, (char __rcu *)p), 1);
  97. if (old) {
  98. synchronize_rcu();
  99. kfree(old);
  100. }
  101. return count;
  102. }
  103. static void nfs_netns_client_release(struct kobject *kobj)
  104. {
  105. struct nfs_netns_client *c = container_of(kobj,
  106. struct nfs_netns_client,
  107. kobject);
  108. kfree(rcu_dereference_raw(c->identifier));
  109. }
  110. static const struct ns_common *nfs_netns_client_namespace(const struct kobject *kobj)
  111. {
  112. return to_ns_common(container_of(kobj, struct nfs_netns_client,
  113. kobject)->net);
  114. }
  115. static struct kobj_attribute nfs_netns_client_id = __ATTR(identifier,
  116. 0644, nfs_netns_identifier_show, nfs_netns_identifier_store);
  117. static struct attribute *nfs_netns_client_attrs[] = {
  118. &nfs_netns_client_id.attr,
  119. NULL,
  120. };
  121. ATTRIBUTE_GROUPS(nfs_netns_client);
  122. static struct kobj_type nfs_netns_client_type = {
  123. .release = nfs_netns_client_release,
  124. .default_groups = nfs_netns_client_groups,
  125. .sysfs_ops = &kobj_sysfs_ops,
  126. .namespace = nfs_netns_client_namespace,
  127. };
  128. static void nfs_netns_object_release(struct kobject *kobj)
  129. {
  130. struct nfs_netns_client *c = container_of(kobj,
  131. struct nfs_netns_client,
  132. nfs_net_kobj);
  133. kfree(c);
  134. }
  135. static const struct ns_common *nfs_netns_namespace(const struct kobject *kobj)
  136. {
  137. return to_ns_common(container_of(kobj, struct nfs_netns_client,
  138. nfs_net_kobj)->net);
  139. }
  140. static struct kobj_type nfs_netns_object_type = {
  141. .release = nfs_netns_object_release,
  142. .sysfs_ops = &kobj_sysfs_ops,
  143. .namespace = nfs_netns_namespace,
  144. };
  145. static struct nfs_netns_client *nfs_netns_client_alloc(struct kobject *parent,
  146. struct net *net)
  147. {
  148. struct nfs_netns_client *p;
  149. p = kzalloc_obj(*p);
  150. if (p) {
  151. p->net = net;
  152. p->kobject.kset = nfs_kset;
  153. p->nfs_net_kobj.kset = nfs_kset;
  154. if (kobject_init_and_add(&p->nfs_net_kobj, &nfs_netns_object_type,
  155. parent, "net") != 0) {
  156. kobject_put(&p->nfs_net_kobj);
  157. return NULL;
  158. }
  159. if (kobject_init_and_add(&p->kobject, &nfs_netns_client_type,
  160. &p->nfs_net_kobj, "nfs_client") == 0)
  161. return p;
  162. kobject_put(&p->kobject);
  163. kobject_put(&p->nfs_net_kobj);
  164. }
  165. return NULL;
  166. }
  167. void nfs_netns_sysfs_setup(struct nfs_net *netns, struct net *net)
  168. {
  169. struct nfs_netns_client *clp;
  170. clp = nfs_netns_client_alloc(&nfs_kset->kobj, net);
  171. if (clp) {
  172. netns->nfs_client = clp;
  173. kobject_uevent(&clp->kobject, KOBJ_ADD);
  174. }
  175. }
  176. void nfs_netns_sysfs_destroy(struct nfs_net *netns)
  177. {
  178. struct nfs_netns_client *clp = netns->nfs_client;
  179. if (clp) {
  180. kobject_uevent(&clp->kobject, KOBJ_REMOVE);
  181. kobject_del(&clp->kobject);
  182. kobject_put(&clp->kobject);
  183. kobject_del(&clp->nfs_net_kobj);
  184. kobject_put(&clp->nfs_net_kobj);
  185. netns->nfs_client = NULL;
  186. }
  187. }
  188. static bool shutdown_match_client(const struct rpc_task *task, const void *data)
  189. {
  190. return true;
  191. }
  192. static void shutdown_client(struct rpc_clnt *clnt)
  193. {
  194. clnt->cl_shutdown = 1;
  195. rpc_cancel_tasks(clnt, -EIO, shutdown_match_client, NULL);
  196. }
  197. /*
  198. * Shut down the nfs_client only once all the superblocks
  199. * have been shut down.
  200. */
  201. static void shutdown_nfs_client(struct nfs_client *clp)
  202. {
  203. struct nfs_server *server;
  204. rcu_read_lock();
  205. list_for_each_entry_rcu(server, &clp->cl_superblocks, client_link) {
  206. if (!(server->flags & NFS_MOUNT_SHUTDOWN)) {
  207. rcu_read_unlock();
  208. return;
  209. }
  210. }
  211. rcu_read_unlock();
  212. nfs_mark_client_ready(clp, -EIO);
  213. shutdown_client(clp->cl_rpcclient);
  214. }
  215. static ssize_t
  216. shutdown_show(struct kobject *kobj, struct kobj_attribute *attr,
  217. char *buf)
  218. {
  219. struct nfs_server *server = container_of(kobj, struct nfs_server, kobj);
  220. bool shutdown = server->flags & NFS_MOUNT_SHUTDOWN;
  221. return sysfs_emit(buf, "%d\n", shutdown);
  222. }
  223. static ssize_t
  224. shutdown_store(struct kobject *kobj, struct kobj_attribute *attr,
  225. const char *buf, size_t count)
  226. {
  227. struct nfs_server *server;
  228. int ret, val;
  229. server = container_of(kobj, struct nfs_server, kobj);
  230. ret = kstrtoint(buf, 0, &val);
  231. if (ret)
  232. return ret;
  233. if (val != 1)
  234. return -EINVAL;
  235. /* already shut down? */
  236. if (server->flags & NFS_MOUNT_SHUTDOWN)
  237. goto out;
  238. server->flags |= NFS_MOUNT_SHUTDOWN;
  239. shutdown_client(server->client);
  240. if (!IS_ERR(server->client_acl))
  241. shutdown_client(server->client_acl);
  242. if (server->nlm_host)
  243. shutdown_client(server->nlm_host->h_rpcclnt);
  244. out:
  245. shutdown_nfs_client(server->nfs_client);
  246. return count;
  247. }
  248. static struct kobj_attribute nfs_sysfs_attr_shutdown = __ATTR_RW(shutdown);
  249. #if IS_ENABLED(CONFIG_NFS_V4)
  250. static ssize_t
  251. implid_domain_show(struct kobject *kobj, struct kobj_attribute *attr,
  252. char *buf)
  253. {
  254. struct nfs_server *server = container_of(kobj, struct nfs_server, kobj);
  255. struct nfs41_impl_id *impl_id = server->nfs_client->cl_implid;
  256. if (!impl_id || strlen(impl_id->domain) == 0)
  257. return 0; //sysfs_emit(buf, "");
  258. return sysfs_emit(buf, "%s\n", impl_id->domain);
  259. }
  260. static struct kobj_attribute nfs_sysfs_attr_implid_domain = __ATTR_RO(implid_domain);
  261. static ssize_t
  262. implid_name_show(struct kobject *kobj, struct kobj_attribute *attr,
  263. char *buf)
  264. {
  265. struct nfs_server *server = container_of(kobj, struct nfs_server, kobj);
  266. struct nfs41_impl_id *impl_id = server->nfs_client->cl_implid;
  267. if (!impl_id || strlen(impl_id->name) == 0)
  268. return 0; //sysfs_emit(buf, "");
  269. return sysfs_emit(buf, "%s\n", impl_id->name);
  270. }
  271. static struct kobj_attribute nfs_sysfs_attr_implid_name = __ATTR_RO(implid_name);
  272. #endif /* IS_ENABLED(CONFIG_NFS_V4) */
  273. #define RPC_CLIENT_NAME_SIZE 64
  274. void nfs_sysfs_link_rpc_client(struct nfs_server *server,
  275. struct rpc_clnt *clnt, const char *uniq)
  276. {
  277. char name[RPC_CLIENT_NAME_SIZE];
  278. int ret;
  279. strscpy(name, clnt->cl_program->name, sizeof(name));
  280. strncat(name, uniq ? uniq : "", sizeof(name) - strlen(name) - 1);
  281. strncat(name, "_client", sizeof(name) - strlen(name) - 1);
  282. ret = sysfs_create_link_nowarn(&server->kobj,
  283. &clnt->cl_sysfs->kobject, name);
  284. if (ret < 0)
  285. pr_warn("NFS: can't create link to %s in sysfs (%d)\n",
  286. name, ret);
  287. }
  288. EXPORT_SYMBOL_GPL(nfs_sysfs_link_rpc_client);
  289. static void nfs_sysfs_sb_release(struct kobject *kobj)
  290. {
  291. /* no-op: why? see lib/kobject.c kobject_cleanup() */
  292. }
  293. static const struct ns_common *nfs_netns_server_namespace(const struct kobject *kobj)
  294. {
  295. return to_ns_common(container_of(kobj, struct nfs_server,
  296. kobj)->nfs_client->cl_net);
  297. }
  298. static struct kobj_type nfs_sb_ktype = {
  299. .release = nfs_sysfs_sb_release,
  300. .sysfs_ops = &kobj_sysfs_ops,
  301. .namespace = nfs_netns_server_namespace,
  302. .child_ns_type = nfs_netns_object_child_ns_type,
  303. };
  304. #if IS_ENABLED(CONFIG_NFS_V4)
  305. static void nfs_sysfs_add_nfsv41_server(struct nfs_server *server)
  306. {
  307. int ret;
  308. if (!server->nfs_client->cl_implid)
  309. return;
  310. ret = sysfs_create_file_ns(&server->kobj, &nfs_sysfs_attr_implid_domain.attr,
  311. nfs_netns_server_namespace(&server->kobj));
  312. if (ret < 0)
  313. pr_warn("NFS: sysfs_create_file_ns for server-%d failed (%d)\n",
  314. server->s_sysfs_id, ret);
  315. ret = sysfs_create_file_ns(&server->kobj, &nfs_sysfs_attr_implid_name.attr,
  316. nfs_netns_server_namespace(&server->kobj));
  317. if (ret < 0)
  318. pr_warn("NFS: sysfs_create_file_ns for server-%d failed (%d)\n",
  319. server->s_sysfs_id, ret);
  320. }
  321. #else /* CONFIG_NFS_V4 */
  322. static inline void nfs_sysfs_add_nfsv41_server(struct nfs_server *server)
  323. {
  324. }
  325. #endif /* CONFIG_NFS_V4 */
  326. #if IS_ENABLED(CONFIG_NFS_LOCALIO)
  327. static ssize_t
  328. localio_show(struct kobject *kobj, struct kobj_attribute *attr,
  329. char *buf)
  330. {
  331. struct nfs_server *server = container_of(kobj, struct nfs_server, kobj);
  332. bool localio = nfs_server_is_local(server->nfs_client);
  333. return sysfs_emit(buf, "%d\n", localio);
  334. }
  335. static struct kobj_attribute nfs_sysfs_attr_localio = __ATTR_RO(localio);
  336. static void nfs_sysfs_add_nfs_localio_server(struct nfs_server *server)
  337. {
  338. int ret = sysfs_create_file_ns(&server->kobj, &nfs_sysfs_attr_localio.attr,
  339. nfs_netns_server_namespace(&server->kobj));
  340. if (ret < 0)
  341. pr_warn("NFS: sysfs_create_file_ns for server-%d failed (%d)\n",
  342. server->s_sysfs_id, ret);
  343. }
  344. #else
  345. static inline void nfs_sysfs_add_nfs_localio_server(struct nfs_server *server)
  346. {
  347. }
  348. #endif /* IS_ENABLED(CONFIG_NFS_LOCALIO) */
  349. void nfs_sysfs_add_server(struct nfs_server *server)
  350. {
  351. int ret;
  352. ret = kobject_init_and_add(&server->kobj, &nfs_sb_ktype,
  353. &nfs_kset->kobj, "server-%d", server->s_sysfs_id);
  354. if (ret < 0) {
  355. pr_warn("NFS: nfs sysfs add server-%d failed (%d)\n",
  356. server->s_sysfs_id, ret);
  357. return;
  358. }
  359. ret = sysfs_create_file_ns(&server->kobj, &nfs_sysfs_attr_shutdown.attr,
  360. nfs_netns_server_namespace(&server->kobj));
  361. if (ret < 0)
  362. pr_warn("NFS: sysfs_create_file_ns for server-%d failed (%d)\n",
  363. server->s_sysfs_id, ret);
  364. nfs_sysfs_add_nfsv41_server(server);
  365. nfs_sysfs_add_nfs_localio_server(server);
  366. }
  367. EXPORT_SYMBOL_GPL(nfs_sysfs_add_server);
  368. void nfs_sysfs_move_server_to_sb(struct super_block *s)
  369. {
  370. struct nfs_server *server = s->s_fs_info;
  371. int ret;
  372. ret = kobject_rename(&server->kobj, s->s_id);
  373. if (ret < 0)
  374. pr_warn("NFS: rename sysfs %s failed (%d)\n",
  375. server->kobj.name, ret);
  376. }
  377. void nfs_sysfs_move_sb_to_server(struct nfs_server *server)
  378. {
  379. const char *s;
  380. int ret = -ENOMEM;
  381. s = kasprintf(GFP_KERNEL, "server-%d", server->s_sysfs_id);
  382. if (s) {
  383. ret = kobject_rename(&server->kobj, s);
  384. kfree(s);
  385. }
  386. if (ret < 0)
  387. pr_warn("NFS: rename sysfs %s failed (%d)\n",
  388. server->kobj.name, ret);
  389. }
  390. /* unlink, not dec-ref */
  391. void nfs_sysfs_remove_server(struct nfs_server *server)
  392. {
  393. kobject_del(&server->kobj);
  394. }