nfs4client.c 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (C) 2006 Red Hat, Inc. All Rights Reserved.
  4. * Written by David Howells (dhowells@redhat.com)
  5. */
  6. #include <linux/module.h>
  7. #include <linux/nfs_fs.h>
  8. #include <linux/nfs_mount.h>
  9. #include <linux/sunrpc/addr.h>
  10. #include <linux/sunrpc/auth.h>
  11. #include <linux/sunrpc/xprt.h>
  12. #include <linux/sunrpc/bc_xprt.h>
  13. #include <linux/sunrpc/rpc_pipe_fs.h>
  14. #include <net/handshake.h>
  15. #include "internal.h"
  16. #include "callback.h"
  17. #include "delegation.h"
  18. #include "nfs4session.h"
  19. #include "nfs4idmap.h"
  20. #include "pnfs.h"
  21. #include "netns.h"
  22. #include "sysfs.h"
  23. #define NFSDBG_FACILITY NFSDBG_CLIENT
  24. /*
  25. * Get a unique NFSv4.0 callback identifier which will be used
  26. * by the V4.0 callback service to lookup the nfs_client struct
  27. */
  28. static int nfs_get_cb_ident_idr(struct nfs_client *clp, int minorversion)
  29. {
  30. int ret = 0;
  31. struct nfs_net *nn = net_generic(clp->cl_net, nfs_net_id);
  32. if (clp->rpc_ops->version != 4 || minorversion != 0)
  33. return ret;
  34. idr_preload(GFP_KERNEL);
  35. spin_lock(&nn->nfs_client_lock);
  36. ret = idr_alloc(&nn->cb_ident_idr, clp, 1, 0, GFP_NOWAIT);
  37. if (ret >= 0)
  38. clp->cl_cb_ident = ret;
  39. spin_unlock(&nn->nfs_client_lock);
  40. idr_preload_end();
  41. return ret < 0 ? ret : 0;
  42. }
  43. /*
  44. * Per auth flavor data server rpc clients
  45. */
  46. struct nfs4_ds_server {
  47. struct list_head list; /* ds_clp->cl_ds_clients */
  48. struct rpc_clnt *rpc_clnt;
  49. };
  50. /**
  51. * nfs4_find_ds_client - Common lookup case for DS I/O
  52. * @ds_clp: pointer to the DS's nfs_client
  53. * @flavor: rpc auth flavour to match
  54. */
  55. static struct nfs4_ds_server *
  56. nfs4_find_ds_client(struct nfs_client *ds_clp, rpc_authflavor_t flavor)
  57. {
  58. struct nfs4_ds_server *dss;
  59. rcu_read_lock();
  60. list_for_each_entry_rcu(dss, &ds_clp->cl_ds_clients, list) {
  61. if (dss->rpc_clnt->cl_auth->au_flavor != flavor)
  62. continue;
  63. goto out;
  64. }
  65. dss = NULL;
  66. out:
  67. rcu_read_unlock();
  68. return dss;
  69. }
  70. static struct nfs4_ds_server *
  71. nfs4_add_ds_client(struct nfs_client *ds_clp, rpc_authflavor_t flavor,
  72. struct nfs4_ds_server *new)
  73. {
  74. struct nfs4_ds_server *dss;
  75. spin_lock(&ds_clp->cl_lock);
  76. list_for_each_entry(dss, &ds_clp->cl_ds_clients, list) {
  77. if (dss->rpc_clnt->cl_auth->au_flavor != flavor)
  78. continue;
  79. goto out;
  80. }
  81. if (new)
  82. list_add_rcu(&new->list, &ds_clp->cl_ds_clients);
  83. dss = new;
  84. out:
  85. spin_unlock(&ds_clp->cl_lock); /* need some lock to protect list */
  86. return dss;
  87. }
  88. static struct nfs4_ds_server *
  89. nfs4_alloc_ds_server(struct nfs_client *ds_clp, rpc_authflavor_t flavor)
  90. {
  91. struct nfs4_ds_server *dss;
  92. dss = kmalloc_obj(*dss, GFP_NOFS);
  93. if (dss == NULL)
  94. return ERR_PTR(-ENOMEM);
  95. dss->rpc_clnt = rpc_clone_client_set_auth(ds_clp->cl_rpcclient, flavor);
  96. if (IS_ERR(dss->rpc_clnt)) {
  97. int err = PTR_ERR(dss->rpc_clnt);
  98. kfree (dss);
  99. return ERR_PTR(err);
  100. }
  101. INIT_LIST_HEAD(&dss->list);
  102. return dss;
  103. }
  104. static void
  105. nfs4_free_ds_server(struct nfs4_ds_server *dss)
  106. {
  107. rpc_release_client(dss->rpc_clnt);
  108. kfree(dss);
  109. }
  110. /**
  111. * nfs4_find_or_create_ds_client - Find or create a DS rpc client
  112. * @ds_clp: pointer to the DS's nfs_client
  113. * @inode: pointer to the inode
  114. *
  115. * Find or create a DS rpc client with th MDS server rpc client auth flavor
  116. * in the nfs_client cl_ds_clients list.
  117. */
  118. struct rpc_clnt *
  119. nfs4_find_or_create_ds_client(struct nfs_client *ds_clp, struct inode *inode)
  120. {
  121. struct nfs4_ds_server *dss, *new;
  122. rpc_authflavor_t flavor = NFS_SERVER(inode)->client->cl_auth->au_flavor;
  123. dss = nfs4_find_ds_client(ds_clp, flavor);
  124. if (dss != NULL)
  125. goto out;
  126. new = nfs4_alloc_ds_server(ds_clp, flavor);
  127. if (IS_ERR(new))
  128. return ERR_CAST(new);
  129. dss = nfs4_add_ds_client(ds_clp, flavor, new);
  130. if (dss != new)
  131. nfs4_free_ds_server(new);
  132. out:
  133. return dss->rpc_clnt;
  134. }
  135. EXPORT_SYMBOL_GPL(nfs4_find_or_create_ds_client);
  136. static void
  137. nfs4_shutdown_ds_clients(struct nfs_client *clp)
  138. {
  139. struct nfs4_ds_server *dss;
  140. while (!list_empty(&clp->cl_ds_clients)) {
  141. dss = list_entry(clp->cl_ds_clients.next,
  142. struct nfs4_ds_server, list);
  143. list_del(&dss->list);
  144. rpc_shutdown_client(dss->rpc_clnt);
  145. kfree (dss);
  146. }
  147. }
  148. static void
  149. nfs4_cleanup_callback(struct nfs_client *clp)
  150. {
  151. struct nfs4_copy_state *cp_state;
  152. while (!list_empty(&clp->pending_cb_stateids)) {
  153. cp_state = list_entry(clp->pending_cb_stateids.next,
  154. struct nfs4_copy_state, copies);
  155. list_del(&cp_state->copies);
  156. kfree(cp_state);
  157. }
  158. }
  159. void nfs41_shutdown_client(struct nfs_client *clp)
  160. {
  161. if (nfs4_has_session(clp)) {
  162. nfs4_cleanup_callback(clp);
  163. nfs4_shutdown_ds_clients(clp);
  164. nfs4_destroy_session(clp->cl_session);
  165. nfs4_destroy_clientid(clp);
  166. }
  167. }
  168. struct nfs_client *nfs4_alloc_client(const struct nfs_client_initdata *cl_init)
  169. {
  170. char buf[INET6_ADDRSTRLEN + 1];
  171. const char *ip_addr = cl_init->ip_addr;
  172. struct nfs_client *clp = nfs_alloc_client(cl_init);
  173. int err;
  174. if (IS_ERR(clp))
  175. return clp;
  176. err = nfs_get_cb_ident_idr(clp, cl_init->minorversion);
  177. if (err)
  178. goto error;
  179. if (cl_init->minorversion < NFS4_MIN_MINOR_VERSION ||
  180. cl_init->minorversion > NFS4_MAX_MINOR_VERSION) {
  181. err = -EINVAL;
  182. goto error;
  183. }
  184. spin_lock_init(&clp->cl_lock);
  185. INIT_DELAYED_WORK(&clp->cl_renewd, nfs4_renew_state);
  186. INIT_LIST_HEAD(&clp->cl_ds_clients);
  187. rpc_init_wait_queue(&clp->cl_rpcwaitq, "NFS client");
  188. clp->cl_state = 1 << NFS4CLNT_LEASE_EXPIRED;
  189. clp->cl_mvops = nfs_v4_minor_ops[cl_init->minorversion];
  190. clp->cl_mig_gen = 1;
  191. clp->cl_last_renewal = jiffies;
  192. init_waitqueue_head(&clp->cl_lock_waitq);
  193. INIT_LIST_HEAD(&clp->pending_cb_stateids);
  194. if (cl_init->minorversion != 0)
  195. __set_bit(NFS_CS_INFINITE_SLOTS, &clp->cl_flags);
  196. __set_bit(NFS_CS_DISCRTRY, &clp->cl_flags);
  197. __set_bit(NFS_CS_NO_RETRANS_TIMEOUT, &clp->cl_flags);
  198. if (test_bit(NFS_CS_PNFS, &cl_init->init_flags))
  199. __set_bit(NFS_CS_PNFS, &clp->cl_flags);
  200. if (test_bit(NFS_CS_NETUNREACH_FATAL, &cl_init->init_flags))
  201. __set_bit(NFS_CS_NETUNREACH_FATAL, &clp->cl_flags);
  202. /*
  203. * Set up the connection to the server before we add add to the
  204. * global list.
  205. */
  206. err = nfs_create_rpc_client(clp, cl_init, RPC_AUTH_GSS_KRB5I);
  207. if (err == -EINVAL)
  208. err = nfs_create_rpc_client(clp, cl_init, RPC_AUTH_UNIX);
  209. if (err < 0)
  210. goto error;
  211. /* If no clientaddr= option was specified, find a usable cb address */
  212. if (ip_addr == NULL) {
  213. struct sockaddr_storage cb_addr;
  214. struct sockaddr *sap = (struct sockaddr *)&cb_addr;
  215. err = rpc_localaddr(clp->cl_rpcclient, sap, sizeof(cb_addr));
  216. if (err < 0)
  217. goto error;
  218. err = rpc_ntop(sap, buf, sizeof(buf));
  219. if (err < 0)
  220. goto error;
  221. ip_addr = (const char *)buf;
  222. }
  223. strscpy(clp->cl_ipaddr, ip_addr, sizeof(clp->cl_ipaddr));
  224. err = nfs_idmap_new(clp);
  225. if (err < 0) {
  226. dprintk("%s: failed to create idmapper. Error = %d\n",
  227. __func__, err);
  228. goto error;
  229. }
  230. __set_bit(NFS_CS_IDMAP, &clp->cl_res_state);
  231. return clp;
  232. error:
  233. nfs_free_client(clp);
  234. return ERR_PTR(err);
  235. }
  236. /*
  237. * Destroy the NFS4 callback service
  238. */
  239. static void nfs4_destroy_callback(struct nfs_client *clp)
  240. {
  241. if (__test_and_clear_bit(NFS_CS_CALLBACK, &clp->cl_res_state)) {
  242. struct rpc_xprt *xprt;
  243. xprt = rcu_dereference_raw(clp->cl_rpcclient->cl_xprt);
  244. nfs_callback_down(clp->cl_mvops->minor_version, clp->cl_net,
  245. xprt);
  246. }
  247. }
  248. static void nfs4_shutdown_client(struct nfs_client *clp)
  249. {
  250. if (__test_and_clear_bit(NFS_CS_RENEWD, &clp->cl_res_state))
  251. nfs4_kill_renewd(clp);
  252. clp->cl_mvops->shutdown_client(clp);
  253. nfs4_destroy_callback(clp);
  254. if (__test_and_clear_bit(NFS_CS_IDMAP, &clp->cl_res_state))
  255. nfs_idmap_delete(clp);
  256. rpc_destroy_wait_queue(&clp->cl_rpcwaitq);
  257. kfree(clp->cl_serverowner);
  258. kfree(clp->cl_serverscope);
  259. kfree(clp->cl_implid);
  260. kfree(clp->cl_owner_id);
  261. }
  262. void nfs4_free_client(struct nfs_client *clp)
  263. {
  264. nfs4_shutdown_client(clp);
  265. nfs_free_client(clp);
  266. }
  267. /*
  268. * Initialize the NFS4 callback service
  269. */
  270. static int nfs4_init_callback(struct nfs_client *clp)
  271. {
  272. struct rpc_xprt *xprt;
  273. int error;
  274. xprt = rcu_dereference_raw(clp->cl_rpcclient->cl_xprt);
  275. if (nfs4_has_session(clp)) {
  276. error = xprt_setup_backchannel(xprt, NFS41_BC_MIN_CALLBACKS);
  277. if (error < 0)
  278. return error;
  279. }
  280. error = nfs_callback_up(clp->cl_mvops->minor_version, xprt);
  281. if (error < 0) {
  282. dprintk("%s: failed to start callback. Error = %d\n",
  283. __func__, error);
  284. return error;
  285. }
  286. __set_bit(NFS_CS_CALLBACK, &clp->cl_res_state);
  287. return 0;
  288. }
  289. /**
  290. * nfs41_init_client - nfs_client initialization tasks for NFSv4.1+
  291. * @clp: nfs_client to initialize
  292. *
  293. * Returns zero on success, or a negative errno if some error occurred.
  294. */
  295. int nfs41_init_client(struct nfs_client *clp)
  296. {
  297. struct nfs4_session *session = NULL;
  298. /*
  299. * Create the session and mark it expired.
  300. * When a SEQUENCE operation encounters the expired session
  301. * it will do session recovery to initialize it.
  302. */
  303. session = nfs4_alloc_session(clp);
  304. if (!session)
  305. return -ENOMEM;
  306. clp->cl_session = session;
  307. /*
  308. * The create session reply races with the server back
  309. * channel probe. Mark the client NFS_CS_SESSION_INITING
  310. * so that the client back channel can find the
  311. * nfs_client struct
  312. */
  313. nfs_mark_client_ready(clp, NFS_CS_SESSION_INITING);
  314. return 0;
  315. }
  316. /*
  317. * Initialize the minor version specific parts of an NFS4 client record
  318. */
  319. static int nfs4_init_client_minor_version(struct nfs_client *clp)
  320. {
  321. int ret;
  322. ret = clp->cl_mvops->init_client(clp);
  323. if (ret)
  324. return ret;
  325. return nfs4_init_callback(clp);
  326. }
  327. static void nfs4_add_trunk(struct nfs_client *clp, struct nfs_client *old)
  328. {
  329. struct sockaddr_storage clp_addr, old_addr;
  330. struct sockaddr *clp_sap = (struct sockaddr *)&clp_addr;
  331. struct sockaddr *old_sap = (struct sockaddr *)&old_addr;
  332. size_t clp_salen;
  333. struct xprt_create xprt_args = {
  334. .ident = old->cl_proto,
  335. .net = old->cl_net,
  336. .servername = old->cl_hostname,
  337. };
  338. int max_connect = test_bit(NFS_CS_PNFS, &clp->cl_flags) ?
  339. clp->cl_max_connect : old->cl_max_connect;
  340. if (clp->cl_proto != old->cl_proto)
  341. return;
  342. clp_salen = rpc_peeraddr(clp->cl_rpcclient, clp_sap, sizeof(clp_addr));
  343. rpc_peeraddr(old->cl_rpcclient, old_sap, sizeof(old_addr));
  344. if (clp_addr.ss_family != old_addr.ss_family)
  345. return;
  346. xprt_args.dstaddr = clp_sap;
  347. xprt_args.addrlen = clp_salen;
  348. rpc_clnt_add_xprt(old->cl_rpcclient, &xprt_args,
  349. rpc_clnt_test_and_add_xprt, &max_connect);
  350. }
  351. /**
  352. * nfs4_init_client - Initialise an NFS4 client record
  353. *
  354. * @clp: nfs_client to initialise
  355. * @cl_init: pointer to nfs_client_initdata
  356. *
  357. * Returns pointer to an NFS client, or an ERR_PTR value.
  358. */
  359. struct nfs_client *nfs4_init_client(struct nfs_client *clp,
  360. const struct nfs_client_initdata *cl_init)
  361. {
  362. struct nfs_client *old;
  363. int error;
  364. if (clp->cl_cons_state == NFS_CS_READY)
  365. /* the client is initialised already */
  366. return clp;
  367. error = nfs4_init_client_minor_version(clp);
  368. if (error < 0)
  369. goto error;
  370. error = nfs4_discover_server_trunking(clp, &old);
  371. if (error < 0)
  372. goto error;
  373. if (clp != old) {
  374. clp->cl_preserve_clid = true;
  375. /*
  376. * Mark the client as having failed initialization so other
  377. * processes walking the nfs_client_list in nfs_match_client()
  378. * won't try to use it.
  379. */
  380. nfs_mark_client_ready(clp, -EPERM);
  381. if (old->cl_mvops->session_trunk)
  382. nfs4_add_trunk(clp, old);
  383. }
  384. clear_bit(NFS_CS_TSM_POSSIBLE, &clp->cl_flags);
  385. nfs_put_client(clp);
  386. return old;
  387. error:
  388. nfs_mark_client_ready(clp, error);
  389. nfs_put_client(clp);
  390. return ERR_PTR(error);
  391. }
  392. static bool nfs4_match_client_owner_id(const struct nfs_client *clp1,
  393. const struct nfs_client *clp2)
  394. {
  395. if (clp1->cl_owner_id == NULL || clp2->cl_owner_id == NULL)
  396. return true;
  397. return strcmp(clp1->cl_owner_id, clp2->cl_owner_id) == 0;
  398. }
  399. int nfs4_match_client(struct nfs_client *pos, struct nfs_client *new,
  400. struct nfs_client **prev, struct nfs_net *nn)
  401. {
  402. int status;
  403. if (pos->rpc_ops != new->rpc_ops)
  404. return 1;
  405. if (pos->cl_minorversion != new->cl_minorversion)
  406. return 1;
  407. /* If "pos" isn't marked ready, we can't trust the
  408. * remaining fields in "pos", especially the client
  409. * ID and serverowner fields. Wait for CREATE_SESSION
  410. * to finish. */
  411. if (pos->cl_cons_state > NFS_CS_READY) {
  412. refcount_inc(&pos->cl_count);
  413. spin_unlock(&nn->nfs_client_lock);
  414. nfs_put_client(*prev);
  415. *prev = pos;
  416. status = nfs_wait_client_init_complete(pos);
  417. spin_lock(&nn->nfs_client_lock);
  418. if (status < 0)
  419. return status;
  420. }
  421. if (pos->cl_cons_state != NFS_CS_READY)
  422. return 1;
  423. if (pos->cl_clientid != new->cl_clientid)
  424. return 1;
  425. /* NFSv4.1 always uses the uniform string, however someone
  426. * might switch the uniquifier string on us.
  427. */
  428. if (!nfs4_match_client_owner_id(pos, new))
  429. return 1;
  430. return 0;
  431. }
  432. /*
  433. * Returns true if the server major ids match
  434. */
  435. bool
  436. nfs4_check_serverowner_major_id(struct nfs41_server_owner *o1,
  437. struct nfs41_server_owner *o2)
  438. {
  439. if (o1->major_id_sz != o2->major_id_sz)
  440. return false;
  441. return memcmp(o1->major_id, o2->major_id, o1->major_id_sz) == 0;
  442. }
  443. /*
  444. * Returns true if the server scopes match
  445. */
  446. static bool
  447. nfs4_check_server_scope(struct nfs41_server_scope *s1,
  448. struct nfs41_server_scope *s2)
  449. {
  450. if (s1->server_scope_sz != s2->server_scope_sz)
  451. return false;
  452. return memcmp(s1->server_scope, s2->server_scope,
  453. s1->server_scope_sz) == 0;
  454. }
  455. /**
  456. * nfs4_detect_session_trunking - Checks for session trunking.
  457. * @clp: original mount nfs_client
  458. * @res: result structure from an exchange_id using the original mount
  459. * nfs_client with a new multi_addr transport
  460. * @xprt: pointer to the transport to add.
  461. *
  462. * Called after a successful EXCHANGE_ID on a multi-addr connection.
  463. * Upon success, add the transport.
  464. *
  465. * Returns zero on success, otherwise -EINVAL
  466. *
  467. * Note: since the exchange_id for the new multi_addr transport uses the
  468. * same nfs_client from the original mount, the cl_owner_id is reused,
  469. * so eir_clientowner is the same.
  470. */
  471. int nfs4_detect_session_trunking(struct nfs_client *clp,
  472. struct nfs41_exchange_id_res *res,
  473. struct rpc_xprt *xprt)
  474. {
  475. /* Check eir_clientid */
  476. if (clp->cl_clientid != res->clientid)
  477. goto out_err;
  478. /* Check eir_server_owner so_major_id */
  479. if (!nfs4_check_serverowner_major_id(clp->cl_serverowner,
  480. res->server_owner))
  481. goto out_err;
  482. /* Check eir_server_owner so_minor_id */
  483. if (clp->cl_serverowner->minor_id != res->server_owner->minor_id)
  484. goto out_err;
  485. /* Check eir_server_scope */
  486. if (!nfs4_check_server_scope(clp->cl_serverscope, res->server_scope))
  487. goto out_err;
  488. pr_info("NFS: %s: Session trunking succeeded for %s\n",
  489. clp->cl_hostname,
  490. xprt->address_strings[RPC_DISPLAY_ADDR]);
  491. return 0;
  492. out_err:
  493. pr_info("NFS: %s: Session trunking failed for %s\n", clp->cl_hostname,
  494. xprt->address_strings[RPC_DISPLAY_ADDR]);
  495. return -EINVAL;
  496. }
  497. /**
  498. * nfs41_walk_client_list - Find nfs_client that matches a client/server owner
  499. *
  500. * @new: nfs_client with client ID to test
  501. * @result: OUT: found nfs_client, or new
  502. * @cred: credential to use for trunking test
  503. *
  504. * Returns zero, a negative errno, or a negative NFS4ERR status.
  505. * If zero is returned, an nfs_client pointer is planted in "result."
  506. *
  507. * NB: nfs41_walk_client_list() relies on the new nfs_client being
  508. * the last nfs_client on the list.
  509. */
  510. int nfs41_walk_client_list(struct nfs_client *new,
  511. struct nfs_client **result,
  512. const struct cred *cred)
  513. {
  514. struct nfs_net *nn = net_generic(new->cl_net, nfs_net_id);
  515. struct nfs_client *pos, *prev = NULL;
  516. int status = -NFS4ERR_STALE_CLIENTID;
  517. spin_lock(&nn->nfs_client_lock);
  518. list_for_each_entry(pos, &nn->nfs_client_list, cl_share_link) {
  519. if (pos == new)
  520. goto found;
  521. status = nfs4_match_client(pos, new, &prev, nn);
  522. if (status < 0)
  523. goto out;
  524. if (status != 0)
  525. continue;
  526. /*
  527. * Note that session trunking is just a special subcase of
  528. * client id trunking. In either case, we want to fall back
  529. * to using the existing nfs_client.
  530. */
  531. if (!nfs4_check_serverowner_major_id(pos->cl_serverowner,
  532. new->cl_serverowner))
  533. continue;
  534. found:
  535. refcount_inc(&pos->cl_count);
  536. *result = pos;
  537. status = 0;
  538. break;
  539. }
  540. out:
  541. spin_unlock(&nn->nfs_client_lock);
  542. nfs_put_client(prev);
  543. return status;
  544. }
  545. static void nfs4_destroy_server(struct nfs_server *server)
  546. {
  547. LIST_HEAD(freeme);
  548. nfs_server_return_all_delegations(server);
  549. unset_pnfs_layoutdriver(server);
  550. nfs4_purge_state_owners(server, &freeme);
  551. nfs4_free_state_owners(&freeme);
  552. kfree(server->delegation_hash_table);
  553. }
  554. /*
  555. * NFSv4.0 callback thread helper
  556. *
  557. * Find a client by callback identifier
  558. */
  559. struct nfs_client *
  560. nfs4_find_client_ident(struct net *net, int cb_ident)
  561. {
  562. struct nfs_client *clp;
  563. struct nfs_net *nn = net_generic(net, nfs_net_id);
  564. spin_lock(&nn->nfs_client_lock);
  565. clp = idr_find(&nn->cb_ident_idr, cb_ident);
  566. if (clp)
  567. refcount_inc(&clp->cl_count);
  568. spin_unlock(&nn->nfs_client_lock);
  569. return clp;
  570. }
  571. /* Common match routine for v4.0 and v4.1 callback services */
  572. static bool nfs4_cb_match_client(const struct sockaddr *addr,
  573. struct nfs_client *clp, u32 minorversion)
  574. {
  575. struct sockaddr *clap = (struct sockaddr *)&clp->cl_addr;
  576. /* Don't match clients that failed to initialise */
  577. if (!(clp->cl_cons_state == NFS_CS_READY ||
  578. clp->cl_cons_state == NFS_CS_SESSION_INITING))
  579. return false;
  580. smp_rmb();
  581. /* Match the version and minorversion */
  582. if (clp->rpc_ops->version != 4 ||
  583. clp->cl_minorversion != minorversion)
  584. return false;
  585. /* Match only the IP address, not the port number */
  586. return rpc_cmp_addr(addr, clap);
  587. }
  588. /*
  589. * NFSv4.1 callback thread helper
  590. * For CB_COMPOUND calls, find a client by IP address, protocol version,
  591. * minorversion, and sessionID
  592. *
  593. * Returns NULL if no such client
  594. */
  595. struct nfs_client *
  596. nfs4_find_client_sessionid(struct net *net, const struct sockaddr *addr,
  597. struct nfs4_sessionid *sid, u32 minorversion)
  598. {
  599. struct nfs_client *clp;
  600. struct nfs_net *nn = net_generic(net, nfs_net_id);
  601. spin_lock(&nn->nfs_client_lock);
  602. list_for_each_entry(clp, &nn->nfs_client_list, cl_share_link) {
  603. if (!nfs4_cb_match_client(addr, clp, minorversion))
  604. continue;
  605. if (!nfs4_has_session(clp))
  606. continue;
  607. /* Match sessionid*/
  608. if (memcmp(clp->cl_session->sess_id.data,
  609. sid->data, NFS4_MAX_SESSIONID_LEN) != 0)
  610. continue;
  611. refcount_inc(&clp->cl_count);
  612. spin_unlock(&nn->nfs_client_lock);
  613. return clp;
  614. }
  615. spin_unlock(&nn->nfs_client_lock);
  616. return NULL;
  617. }
  618. /*
  619. * Set up an NFS4 client
  620. */
  621. static int nfs4_set_client(struct nfs_server *server,
  622. struct nfs_client_initdata *cl_init)
  623. {
  624. struct nfs_client *clp;
  625. cl_init->nfs_mod = &nfs_v4;
  626. cl_init->cred = server->cred;
  627. if (cl_init->minorversion == 0) {
  628. __set_bit(NFS_CS_REUSEPORT, &cl_init->init_flags);
  629. cl_init->max_connect = 0;
  630. }
  631. switch (cl_init->proto) {
  632. case XPRT_TRANSPORT_RDMA:
  633. case XPRT_TRANSPORT_TCP:
  634. case XPRT_TRANSPORT_TCP_TLS:
  635. break;
  636. default:
  637. cl_init->nconnect = 0;
  638. }
  639. if (server->flags & NFS_MOUNT_NORESVPORT)
  640. __set_bit(NFS_CS_NORESVPORT, &cl_init->init_flags);
  641. if (server->options & NFS_OPTION_MIGRATION)
  642. __set_bit(NFS_CS_MIGRATION, &cl_init->init_flags);
  643. if (test_bit(NFS_MIG_TSM_POSSIBLE, &server->mig_status))
  644. __set_bit(NFS_CS_TSM_POSSIBLE, &cl_init->init_flags);
  645. server->port = rpc_get_port((struct sockaddr *)cl_init->addr);
  646. if (server->flags & NFS_MOUNT_NETUNREACH_FATAL)
  647. __set_bit(NFS_CS_NETUNREACH_FATAL, &cl_init->init_flags);
  648. /* Allocate or find a client reference we can use */
  649. clp = nfs_get_client(cl_init);
  650. if (IS_ERR(clp))
  651. return PTR_ERR(clp);
  652. if (server->nfs_client == clp) {
  653. nfs_put_client(clp);
  654. return -ELOOP;
  655. }
  656. /*
  657. * Query for the lease time on clientid setup or renewal
  658. *
  659. * Note that this will be set on nfs_clients that were created
  660. * only for the DS role and did not set this bit, but now will
  661. * serve a dual role.
  662. */
  663. set_bit(NFS_CS_CHECK_LEASE_TIME, &clp->cl_res_state);
  664. server->nfs_client = clp;
  665. nfs_sysfs_add_server(server);
  666. nfs_sysfs_link_rpc_client(server, clp->cl_rpcclient, "_state");
  667. return 0;
  668. }
  669. /*
  670. * Set up a pNFS Data Server client.
  671. *
  672. * Return any existing nfs_client that matches server address,port,version
  673. * and minorversion.
  674. *
  675. * For a new nfs_client, use a soft mount (default), a low retrans and a
  676. * low timeout interval so that if a connection is lost, we retry through
  677. * the MDS.
  678. */
  679. struct nfs_client *nfs4_set_ds_client(struct nfs_server *mds_srv,
  680. const struct sockaddr_storage *ds_addr, int ds_addrlen,
  681. int ds_proto, unsigned int ds_timeo, unsigned int ds_retrans,
  682. u32 minor_version)
  683. {
  684. struct rpc_timeout ds_timeout;
  685. struct nfs_client *mds_clp = mds_srv->nfs_client;
  686. struct nfs_client_initdata cl_init = {
  687. .addr = ds_addr,
  688. .addrlen = ds_addrlen,
  689. .nodename = mds_clp->cl_rpcclient->cl_nodename,
  690. .ip_addr = mds_clp->cl_ipaddr,
  691. .nfs_mod = &nfs_v4,
  692. .proto = ds_proto,
  693. .minorversion = minor_version,
  694. .net = mds_clp->cl_net,
  695. .timeparms = &ds_timeout,
  696. .cred = mds_srv->cred,
  697. .xprtsec = {
  698. .policy = RPC_XPRTSEC_NONE,
  699. .cert_serial = TLS_NO_CERT,
  700. .privkey_serial = TLS_NO_PRIVKEY,
  701. },
  702. };
  703. char buf[INET6_ADDRSTRLEN + 1];
  704. if (rpc_ntop((struct sockaddr *)ds_addr, buf, sizeof(buf)) <= 0)
  705. return ERR_PTR(-EINVAL);
  706. cl_init.hostname = buf;
  707. switch (ds_proto) {
  708. case XPRT_TRANSPORT_TCP_TLS:
  709. if (mds_srv->nfs_client->cl_xprtsec.policy != RPC_XPRTSEC_NONE)
  710. cl_init.xprtsec = mds_srv->nfs_client->cl_xprtsec;
  711. else
  712. ds_proto = XPRT_TRANSPORT_TCP;
  713. fallthrough;
  714. case XPRT_TRANSPORT_RDMA:
  715. case XPRT_TRANSPORT_TCP:
  716. if (mds_clp->cl_nconnect > 1) {
  717. cl_init.nconnect = mds_clp->cl_nconnect;
  718. cl_init.max_connect = NFS_MAX_TRANSPORTS;
  719. }
  720. }
  721. if (mds_srv->flags & NFS_MOUNT_NORESVPORT)
  722. __set_bit(NFS_CS_NORESVPORT, &cl_init.init_flags);
  723. if (test_bit(NFS_CS_NETUNREACH_FATAL, &mds_clp->cl_flags))
  724. __set_bit(NFS_CS_NETUNREACH_FATAL, &cl_init.init_flags);
  725. __set_bit(NFS_CS_PNFS, &cl_init.init_flags);
  726. cl_init.max_connect = NFS_MAX_TRANSPORTS;
  727. /*
  728. * Set an authflavor equual to the MDS value. Use the MDS nfs_client
  729. * cl_ipaddr so as to use the same EXCHANGE_ID co_ownerid as the MDS
  730. * (section 13.1 RFC 5661).
  731. */
  732. nfs_init_timeout_values(&ds_timeout, ds_proto, ds_timeo, ds_retrans);
  733. return nfs_get_client(&cl_init);
  734. }
  735. EXPORT_SYMBOL_GPL(nfs4_set_ds_client);
  736. /*
  737. * Session has been established, and the client marked ready.
  738. * Limit the mount rsize, wsize and dtsize using negotiated fore
  739. * channel attributes.
  740. */
  741. static void nfs4_session_limit_rwsize(struct nfs_server *server)
  742. {
  743. struct nfs4_session *sess;
  744. u32 server_resp_sz;
  745. u32 server_rqst_sz;
  746. if (!nfs4_has_session(server->nfs_client))
  747. return;
  748. sess = server->nfs_client->cl_session;
  749. server_resp_sz = sess->fc_attrs.max_resp_sz - nfs41_maxread_overhead;
  750. server_rqst_sz = sess->fc_attrs.max_rqst_sz - nfs41_maxwrite_overhead;
  751. if (server->dtsize > server_resp_sz)
  752. server->dtsize = server_resp_sz;
  753. if (server->rsize > server_resp_sz)
  754. server->rsize = server_resp_sz;
  755. if (server->wsize > server_rqst_sz)
  756. server->wsize = server_rqst_sz;
  757. }
  758. /*
  759. * Limit xattr sizes using the channel attributes.
  760. */
  761. static void nfs4_session_limit_xasize(struct nfs_server *server)
  762. {
  763. #ifdef CONFIG_NFS_V4_2
  764. struct nfs4_session *sess;
  765. u32 server_gxa_sz;
  766. u32 server_sxa_sz;
  767. u32 server_lxa_sz;
  768. if (!nfs4_has_session(server->nfs_client))
  769. return;
  770. sess = server->nfs_client->cl_session;
  771. server_gxa_sz = sess->fc_attrs.max_resp_sz - nfs42_maxgetxattr_overhead;
  772. server_sxa_sz = sess->fc_attrs.max_rqst_sz - nfs42_maxsetxattr_overhead;
  773. server_lxa_sz = sess->fc_attrs.max_resp_sz -
  774. nfs42_maxlistxattrs_overhead;
  775. if (server->gxasize > server_gxa_sz)
  776. server->gxasize = server_gxa_sz;
  777. if (server->sxasize > server_sxa_sz)
  778. server->sxasize = server_sxa_sz;
  779. if (server->lxasize > server_lxa_sz)
  780. server->lxasize = server_lxa_sz;
  781. #endif
  782. }
  783. static int nfs4_server_common_setup(struct nfs_server *server,
  784. struct nfs_fh *mntfh, bool auth_probe)
  785. {
  786. int error;
  787. error = nfs4_delegation_hash_alloc(server);
  788. if (error)
  789. return error;
  790. /* data servers support only a subset of NFSv4.1 */
  791. if (is_ds_only_client(server->nfs_client))
  792. return -EPROTONOSUPPORT;
  793. /* We must ensure the session is initialised first */
  794. error = nfs4_init_session(server->nfs_client);
  795. if (error < 0)
  796. return error;
  797. nfs_server_set_init_caps(server);
  798. /* Probe the root fh to retrieve its FSID and filehandle */
  799. error = nfs4_get_rootfh(server, mntfh, auth_probe);
  800. if (error < 0)
  801. return error;
  802. dprintk("Server FSID: %llx:%llx\n",
  803. (unsigned long long) server->fsid.major,
  804. (unsigned long long) server->fsid.minor);
  805. nfs_display_fhandle(mntfh, "Pseudo-fs root FH");
  806. error = nfs_probe_server(server, mntfh);
  807. if (error < 0)
  808. return error;
  809. nfs4_session_limit_rwsize(server);
  810. nfs4_session_limit_xasize(server);
  811. if (server->namelen == 0 || server->namelen > NFS4_MAXNAMLEN)
  812. server->namelen = NFS4_MAXNAMLEN;
  813. nfs_server_insert_lists(server);
  814. server->mount_time = jiffies;
  815. server->destroy = nfs4_destroy_server;
  816. return 0;
  817. }
  818. /*
  819. * Create a version 4 volume record
  820. */
  821. static int nfs4_init_server(struct nfs_server *server, struct fs_context *fc)
  822. {
  823. struct nfs_fs_context *ctx = nfs_fc2context(fc);
  824. struct rpc_timeout timeparms;
  825. struct nfs_client_initdata cl_init = {
  826. .hostname = ctx->nfs_server.hostname,
  827. .addr = &ctx->nfs_server._address,
  828. .addrlen = ctx->nfs_server.addrlen,
  829. .ip_addr = ctx->client_address,
  830. .proto = ctx->nfs_server.protocol,
  831. .minorversion = ctx->minorversion,
  832. .net = fc->net_ns,
  833. .timeparms = &timeparms,
  834. .xprtsec = ctx->xprtsec,
  835. .nconnect = ctx->nfs_server.nconnect,
  836. .max_connect = ctx->nfs_server.max_connect,
  837. };
  838. int error;
  839. nfs_init_timeout_values(&timeparms, ctx->nfs_server.protocol,
  840. ctx->timeo, ctx->retrans);
  841. /* Initialise the client representation from the mount data */
  842. server->flags = ctx->flags;
  843. server->options = ctx->options;
  844. server->auth_info = ctx->auth_info;
  845. /* Use the first specified auth flavor. If this flavor isn't
  846. * allowed by the server, use the SECINFO path to try the
  847. * other specified flavors */
  848. if (ctx->auth_info.flavor_len >= 1)
  849. ctx->selected_flavor = ctx->auth_info.flavors[0];
  850. else
  851. ctx->selected_flavor = RPC_AUTH_UNIX;
  852. /* Get a client record */
  853. error = nfs4_set_client(server, &cl_init);
  854. if (error < 0)
  855. return error;
  856. if (ctx->bsize) {
  857. server->bsize = ctx->bsize;
  858. server->automount_inherit |= NFS_AUTOMOUNT_INHERIT_BSIZE;
  859. }
  860. if (ctx->rsize) {
  861. server->rsize =
  862. nfs_io_size(ctx->rsize, server->nfs_client->cl_proto);
  863. server->automount_inherit |= NFS_AUTOMOUNT_INHERIT_RSIZE;
  864. }
  865. if (ctx->wsize) {
  866. server->wsize =
  867. nfs_io_size(ctx->wsize, server->nfs_client->cl_proto);
  868. server->automount_inherit |= NFS_AUTOMOUNT_INHERIT_WSIZE;
  869. }
  870. server->acregmin = ctx->acregmin * HZ;
  871. server->acregmax = ctx->acregmax * HZ;
  872. server->acdirmin = ctx->acdirmin * HZ;
  873. server->acdirmax = ctx->acdirmax * HZ;
  874. server->port = ctx->nfs_server.port;
  875. return nfs_init_server_rpcclient(server, &timeparms,
  876. ctx->selected_flavor);
  877. }
  878. /*
  879. * Create a version 4 volume record
  880. * - keyed on server and FSID
  881. */
  882. struct nfs_server *nfs4_create_server(struct fs_context *fc)
  883. {
  884. struct nfs_fs_context *ctx = nfs_fc2context(fc);
  885. struct nfs_server *server;
  886. bool auth_probe;
  887. int error;
  888. server = nfs_alloc_server();
  889. if (!server)
  890. return ERR_PTR(-ENOMEM);
  891. server->cred = get_cred(fc->cred);
  892. auth_probe = ctx->auth_info.flavor_len < 1;
  893. /* set up the general RPC client */
  894. error = nfs4_init_server(server, fc);
  895. if (error < 0)
  896. goto error;
  897. error = nfs4_server_common_setup(server, ctx->mntfh, auth_probe);
  898. if (error < 0)
  899. goto error;
  900. return server;
  901. error:
  902. nfs_free_server(server);
  903. return ERR_PTR(error);
  904. }
  905. /*
  906. * Create an NFS4 referral server record
  907. */
  908. struct nfs_server *nfs4_create_referral_server(struct fs_context *fc)
  909. {
  910. struct nfs_fs_context *ctx = nfs_fc2context(fc);
  911. struct nfs_server *parent_server = NFS_SB(ctx->clone_data.sb);
  912. struct nfs_client *parent_client = parent_server->nfs_client;
  913. struct nfs_client_initdata cl_init = {
  914. .hostname = ctx->nfs_server.hostname,
  915. .addr = &ctx->nfs_server._address,
  916. .addrlen = ctx->nfs_server.addrlen,
  917. .ip_addr = parent_client->cl_ipaddr,
  918. .minorversion = parent_client->cl_mvops->minor_version,
  919. .net = parent_client->cl_net,
  920. .timeparms = parent_server->client->cl_timeout,
  921. .xprtsec = parent_client->cl_xprtsec,
  922. .nconnect = parent_client->cl_nconnect,
  923. .max_connect = parent_client->cl_max_connect,
  924. };
  925. struct nfs_server *server;
  926. bool auth_probe;
  927. int error;
  928. server = nfs_alloc_server();
  929. if (!server)
  930. return ERR_PTR(-ENOMEM);
  931. server->cred = get_cred(parent_server->cred);
  932. /* Initialise the client representation from the parent server */
  933. nfs_server_copy_userdata(server, parent_server);
  934. /* Get a client representation */
  935. #if IS_ENABLED(CONFIG_SUNRPC_XPRT_RDMA)
  936. rpc_set_port(&ctx->nfs_server.address, NFS_RDMA_PORT);
  937. cl_init.proto = XPRT_TRANSPORT_RDMA;
  938. error = nfs4_set_client(server, &cl_init);
  939. if (!error)
  940. goto init_server;
  941. #endif /* IS_ENABLED(CONFIG_SUNRPC_XPRT_RDMA) */
  942. cl_init.proto = XPRT_TRANSPORT_TCP;
  943. if (parent_client->cl_xprtsec.policy != RPC_XPRTSEC_NONE)
  944. cl_init.proto = XPRT_TRANSPORT_TCP_TLS;
  945. rpc_set_port(&ctx->nfs_server.address, NFS_PORT);
  946. error = nfs4_set_client(server, &cl_init);
  947. if (error < 0)
  948. goto error;
  949. #if IS_ENABLED(CONFIG_SUNRPC_XPRT_RDMA)
  950. init_server:
  951. #endif
  952. error = nfs_init_server_rpcclient(server, parent_server->client->cl_timeout,
  953. ctx->selected_flavor);
  954. if (error < 0)
  955. goto error;
  956. auth_probe = parent_server->auth_info.flavor_len < 1;
  957. error = nfs4_server_common_setup(server, ctx->mntfh, auth_probe);
  958. if (error < 0)
  959. goto error;
  960. return server;
  961. error:
  962. nfs_free_server(server);
  963. return ERR_PTR(error);
  964. }
  965. /**
  966. * nfs4_update_server - Move an nfs_server to a different nfs_client
  967. *
  968. * @server: represents FSID to be moved
  969. * @hostname: new end-point's hostname
  970. * @sap: new end-point's socket address
  971. * @salen: size of "sap"
  972. * @net: net namespace
  973. *
  974. * The nfs_server must be quiescent before this function is invoked.
  975. * Either its session is drained (NFSv4.1+), or its transport is
  976. * plugged and drained (NFSv4.0).
  977. *
  978. * Returns zero on success, or a negative errno value.
  979. */
  980. int nfs4_update_server(struct nfs_server *server, const char *hostname,
  981. struct sockaddr_storage *sap, size_t salen, struct net *net)
  982. {
  983. struct nfs_client *clp = server->nfs_client;
  984. struct rpc_clnt *clnt = server->client;
  985. struct xprt_create xargs = {
  986. .ident = clp->cl_proto,
  987. .net = net,
  988. .dstaddr = (struct sockaddr *)sap,
  989. .addrlen = salen,
  990. .servername = hostname,
  991. /* cel: bleh. We might need to pass TLS parameters here */
  992. };
  993. char buf[INET6_ADDRSTRLEN + 1];
  994. struct sockaddr_storage address;
  995. struct sockaddr *localaddr = (struct sockaddr *)&address;
  996. struct nfs_client_initdata cl_init = {
  997. .hostname = hostname,
  998. .addr = sap,
  999. .addrlen = salen,
  1000. .ip_addr = buf,
  1001. .proto = clp->cl_proto,
  1002. .minorversion = clp->cl_minorversion,
  1003. .net = net,
  1004. .timeparms = clnt->cl_timeout,
  1005. .xprtsec = clp->cl_xprtsec,
  1006. .nconnect = clp->cl_nconnect,
  1007. .max_connect = clp->cl_max_connect,
  1008. };
  1009. int error;
  1010. error = rpc_switch_client_transport(clnt, &xargs, clnt->cl_timeout);
  1011. if (error != 0)
  1012. return error;
  1013. error = rpc_localaddr(clnt, localaddr, sizeof(address));
  1014. if (error != 0)
  1015. return error;
  1016. if (rpc_ntop(localaddr, buf, sizeof(buf)) == 0)
  1017. return -EAFNOSUPPORT;
  1018. nfs_server_remove_lists(server);
  1019. set_bit(NFS_MIG_TSM_POSSIBLE, &server->mig_status);
  1020. error = nfs4_set_client(server, &cl_init);
  1021. clear_bit(NFS_MIG_TSM_POSSIBLE, &server->mig_status);
  1022. if (error != 0) {
  1023. nfs_server_insert_lists(server);
  1024. return error;
  1025. }
  1026. nfs_put_client(clp);
  1027. if (server->nfs_client->cl_hostname == NULL) {
  1028. server->nfs_client->cl_hostname = kstrdup(hostname, GFP_KERNEL);
  1029. if (server->nfs_client->cl_hostname == NULL)
  1030. return -ENOMEM;
  1031. }
  1032. nfs_server_insert_lists(server);
  1033. return nfs_probe_server(server, NFS_FH(d_inode(server->super->s_root)));
  1034. }