xref: /openbmc/linux/fs/nfs/callback.c (revision 1760371b277718062211fc7eb6f3042c5051c1a5)
1b2441318SGreg Kroah-Hartman // SPDX-License-Identifier: GPL-2.0
21da177e4SLinus Torvalds /*
31da177e4SLinus Torvalds  * linux/fs/nfs/callback.c
41da177e4SLinus Torvalds  *
51da177e4SLinus Torvalds  * Copyright (C) 2004 Trond Myklebust
61da177e4SLinus Torvalds  *
71da177e4SLinus Torvalds  * NFSv4 callback handling
81da177e4SLinus Torvalds  */
91da177e4SLinus Torvalds 
101da177e4SLinus Torvalds #include <linux/completion.h>
111da177e4SLinus Torvalds #include <linux/ip.h>
121da177e4SLinus Torvalds #include <linux/module.h>
133f07c014SIngo Molnar #include <linux/sched/signal.h>
141da177e4SLinus Torvalds #include <linux/sunrpc/svc.h>
151da177e4SLinus Torvalds #include <linux/sunrpc/svcsock.h>
161da177e4SLinus Torvalds #include <linux/nfs_fs.h>
17758201e2STrond Myklebust #include <linux/errno.h>
18353ab6e9SIngo Molnar #include <linux/mutex.h>
1983144186SRafael J. Wysocki #include <linux/freezer.h>
20945b34a7SOlga Kornievskaia #include <linux/sunrpc/svcauth_gss.h>
21a43cde94SRicardo Labiaga #include <linux/sunrpc/bc_xprt.h>
2214c85021SArnaldo Carvalho de Melo 
2314c85021SArnaldo Carvalho de Melo #include <net/inet_sock.h>
2414c85021SArnaldo Carvalho de Melo 
254ce79717STrond Myklebust #include "nfs4_fs.h"
261da177e4SLinus Torvalds #include "callback.h"
2724c8dbbbSDavid Howells #include "internal.h"
28bbe0a3aaSStanislav Kinsbursky #include "netns.h"
291da177e4SLinus Torvalds 
301da177e4SLinus Torvalds #define NFSDBG_FACILITY NFSDBG_CALLBACK
311da177e4SLinus Torvalds 
321da177e4SLinus Torvalds struct nfs_callback_data {
331da177e4SLinus Torvalds 	unsigned int users;
34a43cde94SRicardo Labiaga 	struct svc_serv *serv;
351da177e4SLinus Torvalds };
361da177e4SLinus Torvalds 
37e82dc22dSAndy Adamson static struct nfs_callback_data nfs_callback_info[NFS4_MAX_MINOR_VERSION + 1];
38353ab6e9SIngo Molnar static DEFINE_MUTEX(nfs_callback_mutex);
391da177e4SLinus Torvalds static struct svc_program nfs4_callback_program;
401da177e4SLinus Torvalds 
nfs4_callback_up_net(struct svc_serv * serv,struct net * net)41c946556bSStanislav Kinsbursky static int nfs4_callback_up_net(struct svc_serv *serv, struct net *net)
42c946556bSStanislav Kinsbursky {
434df493a2STrond Myklebust 	const struct cred *cred = current_cred();
44c946556bSStanislav Kinsbursky 	int ret;
45bbe0a3aaSStanislav Kinsbursky 	struct nfs_net *nn = net_generic(net, nfs_net_id);
46c946556bSStanislav Kinsbursky 
47352ad314SChuck Lever 	ret = svc_xprt_create(serv, "tcp", net, PF_INET,
484df493a2STrond Myklebust 			      nfs_callback_set_tcpport, SVC_SOCK_ANONYMOUS,
494df493a2STrond Myklebust 			      cred);
50c946556bSStanislav Kinsbursky 	if (ret <= 0)
51c946556bSStanislav Kinsbursky 		goto out_err;
52bbe0a3aaSStanislav Kinsbursky 	nn->nfs_callback_tcpport = ret;
53e4949e4bSVasily Averin 	dprintk("NFS: Callback listener port = %u (af %u, net %x)\n",
54e4949e4bSVasily Averin 		nn->nfs_callback_tcpport, PF_INET, net->ns.inum);
55c946556bSStanislav Kinsbursky 
56352ad314SChuck Lever 	ret = svc_xprt_create(serv, "tcp", net, PF_INET6,
574df493a2STrond Myklebust 			      nfs_callback_set_tcpport, SVC_SOCK_ANONYMOUS,
584df493a2STrond Myklebust 			      cred);
59c946556bSStanislav Kinsbursky 	if (ret > 0) {
6029dcc16aSStanislav Kinsbursky 		nn->nfs_callback_tcpport6 = ret;
6191bd2ffaSVasily Averin 		dprintk("NFS: Callback listener port = %u (af %u, net %x)\n",
62e4949e4bSVasily Averin 			nn->nfs_callback_tcpport6, PF_INET6, net->ns.inum);
63c946556bSStanislav Kinsbursky 	} else if (ret != -EAFNOSUPPORT)
64c946556bSStanislav Kinsbursky 		goto out_err;
65c946556bSStanislav Kinsbursky 	return 0;
66c946556bSStanislav Kinsbursky 
67c946556bSStanislav Kinsbursky out_err:
68c946556bSStanislav Kinsbursky 	return (ret) ? ret : -ENOMEM;
69c946556bSStanislav Kinsbursky }
70c946556bSStanislav Kinsbursky 
711da177e4SLinus Torvalds /*
72e82dc22dSAndy Adamson  * This is the NFSv4 callback kernel thread.
731da177e4SLinus Torvalds  */
74a277e33cSJeff Layton static int
nfs4_callback_svc(void * vrqstp)7571468513SBenny Halevy nfs4_callback_svc(void *vrqstp)
761da177e4SLinus Torvalds {
77a277e33cSJeff Layton 	struct svc_rqst *rqstp = vrqstp;
781da177e4SLinus Torvalds 
7983144186SRafael J. Wysocki 	set_freezable();
801da177e4SLinus Torvalds 
817b719e2bSNeilBrown 	while (!kthread_freezable_should_stop(NULL))
82c743b425SNeilBrown 		svc_recv(rqstp);
83f49169c9SChuck Lever 
84ed6473ddSTrond Myklebust 	svc_exit_thread(rqstp);
85a277e33cSJeff Layton 	return 0;
861da177e4SLinus Torvalds }
871da177e4SLinus Torvalds 
88a43cde94SRicardo Labiaga #if defined(CONFIG_NFS_V4_1)
89a43cde94SRicardo Labiaga /*
90a43cde94SRicardo Labiaga  * The callback service for NFSv4.1 callbacks
91a43cde94SRicardo Labiaga  */
92a43cde94SRicardo Labiaga static int
nfs41_callback_svc(void * vrqstp)93a43cde94SRicardo Labiaga nfs41_callback_svc(void *vrqstp)
94a43cde94SRicardo Labiaga {
95a43cde94SRicardo Labiaga 	struct svc_rqst *rqstp = vrqstp;
96a43cde94SRicardo Labiaga 	struct svc_serv *serv = rqstp->rq_server;
97a43cde94SRicardo Labiaga 	struct rpc_rqst *req;
98a43cde94SRicardo Labiaga 	int error;
99a43cde94SRicardo Labiaga 	DEFINE_WAIT(wq);
100a43cde94SRicardo Labiaga 
101a43cde94SRicardo Labiaga 	set_freezable();
102a43cde94SRicardo Labiaga 
103ed6473ddSTrond Myklebust 	while (!kthread_freezable_should_stop(NULL)) {
10439039024SNeilBrown 		prepare_to_wait(&serv->sv_cb_waitq, &wq, TASK_IDLE);
105a43cde94SRicardo Labiaga 		spin_lock_bh(&serv->sv_cb_lock);
106a43cde94SRicardo Labiaga 		if (!list_empty(&serv->sv_cb_list)) {
107a43cde94SRicardo Labiaga 			req = list_first_entry(&serv->sv_cb_list,
108a43cde94SRicardo Labiaga 					struct rpc_rqst, rq_bc_list);
109a43cde94SRicardo Labiaga 			list_del(&req->rq_bc_list);
110a43cde94SRicardo Labiaga 			spin_unlock_bh(&serv->sv_cb_lock);
1116ffa30d3SJeff Layton 			finish_wait(&serv->sv_cb_waitq, &wq);
112a43cde94SRicardo Labiaga 			dprintk("Invoking bc_svc_process()\n");
113a43cde94SRicardo Labiaga 			error = bc_svc_process(serv, req, rqstp);
114a43cde94SRicardo Labiaga 			dprintk("bc_svc_process() returned w/ error code= %d\n",
115a43cde94SRicardo Labiaga 				error);
116a43cde94SRicardo Labiaga 		} else {
117a43cde94SRicardo Labiaga 			spin_unlock_bh(&serv->sv_cb_lock);
118ed6473ddSTrond Myklebust 			if (!kthread_should_stop())
1195d05e54aSJeff Layton 				schedule();
120a43cde94SRicardo Labiaga 			finish_wait(&serv->sv_cb_waitq, &wq);
121a43cde94SRicardo Labiaga 		}
1226ffa30d3SJeff Layton 	}
123f49169c9SChuck Lever 
124ed6473ddSTrond Myklebust 	svc_exit_thread(rqstp);
125a43cde94SRicardo Labiaga 	return 0;
126a43cde94SRicardo Labiaga }
127a43cde94SRicardo Labiaga 
nfs_callback_bc_serv(u32 minorversion,struct rpc_xprt * xprt,struct svc_serv * serv)128a43cde94SRicardo Labiaga static inline void nfs_callback_bc_serv(u32 minorversion, struct rpc_xprt *xprt,
129691c457aSStanislav Kinsbursky 		struct svc_serv *serv)
130a43cde94SRicardo Labiaga {
131a43cde94SRicardo Labiaga 	if (minorversion)
132691c457aSStanislav Kinsbursky 		/*
133691c457aSStanislav Kinsbursky 		 * Save the svc_serv in the transport so that it can
134691c457aSStanislav Kinsbursky 		 * be referenced when the session backchannel is initialized
135691c457aSStanislav Kinsbursky 		 */
136691c457aSStanislav Kinsbursky 		xprt->bc_serv = serv;
137a43cde94SRicardo Labiaga }
138a43cde94SRicardo Labiaga #else
nfs_callback_bc_serv(u32 minorversion,struct rpc_xprt * xprt,struct svc_serv * serv)139a43cde94SRicardo Labiaga static inline void nfs_callback_bc_serv(u32 minorversion, struct rpc_xprt *xprt,
140691c457aSStanislav Kinsbursky 		struct svc_serv *serv)
141a43cde94SRicardo Labiaga {
142a43cde94SRicardo Labiaga }
143a43cde94SRicardo Labiaga #endif /* CONFIG_NFS_V4_1 */
144a43cde94SRicardo Labiaga 
nfs_callback_start_svc(int minorversion,struct rpc_xprt * xprt,struct svc_serv * serv)1458e246144SStanislav Kinsbursky static int nfs_callback_start_svc(int minorversion, struct rpc_xprt *xprt,
1468e246144SStanislav Kinsbursky 				  struct svc_serv *serv)
1478e246144SStanislav Kinsbursky {
1485405fc44STrond Myklebust 	int nrservs = nfs_callback_nr_threads;
1498e246144SStanislav Kinsbursky 	int ret;
1508e246144SStanislav Kinsbursky 
1518e246144SStanislav Kinsbursky 	nfs_callback_bc_serv(minorversion, xprt, serv);
1528e246144SStanislav Kinsbursky 
1535405fc44STrond Myklebust 	if (nrservs < NFS4_MIN_NR_CALLBACK_THREADS)
1545405fc44STrond Myklebust 		nrservs = NFS4_MIN_NR_CALLBACK_THREADS;
1555405fc44STrond Myklebust 
156ec52361dSNeilBrown 	if (serv->sv_nrthreads == nrservs)
15723c20ecdSStanislav Kinsbursky 		return 0;
15823c20ecdSStanislav Kinsbursky 
1593ebdbe52SNeilBrown 	ret = svc_set_num_threads(serv, NULL, nrservs);
160bb6aeba7STrond Myklebust 	if (ret) {
1613ebdbe52SNeilBrown 		svc_set_num_threads(serv, NULL, 0);
162e9b7e917STrond Myklebust 		return ret;
1638e246144SStanislav Kinsbursky 	}
1648e246144SStanislav Kinsbursky 	dprintk("nfs_callback_up: service started\n");
1658e246144SStanislav Kinsbursky 	return 0;
1668e246144SStanislav Kinsbursky }
1678e246144SStanislav Kinsbursky 
nfs_callback_down_net(u32 minorversion,struct svc_serv * serv,struct net * net)168b3d19c51SStanislav Kinsbursky static void nfs_callback_down_net(u32 minorversion, struct svc_serv *serv, struct net *net)
169b3d19c51SStanislav Kinsbursky {
170b3d19c51SStanislav Kinsbursky 	struct nfs_net *nn = net_generic(net, nfs_net_id);
171b3d19c51SStanislav Kinsbursky 
172b3d19c51SStanislav Kinsbursky 	if (--nn->cb_users[minorversion])
173b3d19c51SStanislav Kinsbursky 		return;
174b3d19c51SStanislav Kinsbursky 
175e4949e4bSVasily Averin 	dprintk("NFS: destroy per-net callback data; net=%x\n", net->ns.inum);
176c7d7ec8fSChuck Lever 	svc_xprt_destroy_all(serv, net);
177b3d19c51SStanislav Kinsbursky }
178b3d19c51SStanislav Kinsbursky 
nfs_callback_up_net(int minorversion,struct svc_serv * serv,struct net * net,struct rpc_xprt * xprt)17976566773SChuck Lever static int nfs_callback_up_net(int minorversion, struct svc_serv *serv,
18076566773SChuck Lever 			       struct net *net, struct rpc_xprt *xprt)
181c946556bSStanislav Kinsbursky {
182b3d19c51SStanislav Kinsbursky 	struct nfs_net *nn = net_generic(net, nfs_net_id);
183c946556bSStanislav Kinsbursky 	int ret;
184c946556bSStanislav Kinsbursky 
185b3d19c51SStanislav Kinsbursky 	if (nn->cb_users[minorversion]++)
186b3d19c51SStanislav Kinsbursky 		return 0;
187b3d19c51SStanislav Kinsbursky 
188e4949e4bSVasily Averin 	dprintk("NFS: create per-net callback data; net=%x\n", net->ns.inum);
189c946556bSStanislav Kinsbursky 
190c946556bSStanislav Kinsbursky 	ret = svc_bind(serv, net);
191c946556bSStanislav Kinsbursky 	if (ret < 0) {
192c946556bSStanislav Kinsbursky 		printk(KERN_WARNING "NFS: bind callback service failed\n");
193c946556bSStanislav Kinsbursky 		goto err_bind;
194c946556bSStanislav Kinsbursky 	}
195c946556bSStanislav Kinsbursky 
196a289ce53SVasily Averin 	ret = 0;
197d55b352bSArnd Bergmann 	if (!IS_ENABLED(CONFIG_NFS_V4_1) || minorversion == 0)
198c946556bSStanislav Kinsbursky 		ret = nfs4_callback_up_net(serv, net);
199a289ce53SVasily Averin 	else if (xprt->ops->bc_setup)
2000ad30ff6SVasily Averin 		set_bc_enabled(serv);
201a289ce53SVasily Averin 	else
202a289ce53SVasily Averin 		ret = -EPROTONOSUPPORT;
203c946556bSStanislav Kinsbursky 
204c946556bSStanislav Kinsbursky 	if (ret < 0) {
205c946556bSStanislav Kinsbursky 		printk(KERN_ERR "NFS: callback service start failed\n");
206c946556bSStanislav Kinsbursky 		goto err_socks;
207c946556bSStanislav Kinsbursky 	}
208c946556bSStanislav Kinsbursky 	return 0;
209c946556bSStanislav Kinsbursky 
210c946556bSStanislav Kinsbursky err_socks:
211c946556bSStanislav Kinsbursky 	svc_rpcb_cleanup(serv, net);
212c946556bSStanislav Kinsbursky err_bind:
21398b0f80cSTrond Myklebust 	nn->cb_users[minorversion]--;
21423c20ecdSStanislav Kinsbursky 	dprintk("NFS: Couldn't create callback socket: err = %d; "
215e4949e4bSVasily Averin 			"net = %x\n", ret, net->ns.inum);
216c946556bSStanislav Kinsbursky 	return ret;
217c946556bSStanislav Kinsbursky }
218c946556bSStanislav Kinsbursky 
nfs_callback_create_svc(int minorversion)219dd018428SStanislav Kinsbursky static struct svc_serv *nfs_callback_create_svc(int minorversion)
220dd018428SStanislav Kinsbursky {
221dd018428SStanislav Kinsbursky 	struct nfs_callback_data *cb_info = &nfs_callback_info[minorversion];
22237902c63SChuck Lever 	int (*threadfn)(void *data);
223dd018428SStanislav Kinsbursky 	struct svc_serv *serv;
224dd018428SStanislav Kinsbursky 
225dd018428SStanislav Kinsbursky 	/*
226dd018428SStanislav Kinsbursky 	 * Check whether we're already up and running.
227dd018428SStanislav Kinsbursky 	 */
228df5e49c8SNeilBrown 	if (cb_info->serv)
229df5e49c8SNeilBrown 		return svc_get(cb_info->serv);
230dd018428SStanislav Kinsbursky 
231dd018428SStanislav Kinsbursky 	/*
232dd018428SStanislav Kinsbursky 	 * Sanity check: if there's no task,
233dd018428SStanislav Kinsbursky 	 * we should be the first user ...
234dd018428SStanislav Kinsbursky 	 */
235dd018428SStanislav Kinsbursky 	if (cb_info->users)
236dd018428SStanislav Kinsbursky 		printk(KERN_WARNING "nfs_callback_create_svc: no kthread, %d users??\n",
237dd018428SStanislav Kinsbursky 			cb_info->users);
238dd018428SStanislav Kinsbursky 
23937902c63SChuck Lever 	threadfn = nfs4_callback_svc;
24037902c63SChuck Lever #if defined(CONFIG_NFS_V4_1)
24137902c63SChuck Lever 	if (minorversion)
24237902c63SChuck Lever 		threadfn = nfs41_callback_svc;
24337902c63SChuck Lever #else
24437902c63SChuck Lever 	if (minorversion)
24537902c63SChuck Lever 		return ERR_PTR(-ENOTSUPP);
24637902c63SChuck Lever #endif
24737902c63SChuck Lever 	serv = svc_create(&nfs4_callback_program, NFS4_CALLBACK_BUFSIZE,
24837902c63SChuck Lever 			  threadfn);
249dd018428SStanislav Kinsbursky 	if (!serv) {
250dd018428SStanislav Kinsbursky 		printk(KERN_ERR "nfs_callback_create_svc: create service failed\n");
251dd018428SStanislav Kinsbursky 		return ERR_PTR(-ENOMEM);
252dd018428SStanislav Kinsbursky 	}
2533b01c11eSTrond Myklebust 	cb_info->serv = serv;
254dd018428SStanislav Kinsbursky 	/* As there is only one thread we need to over-ride the
255dd018428SStanislav Kinsbursky 	 * default maximum of 80 connections
256dd018428SStanislav Kinsbursky 	 */
257dd018428SStanislav Kinsbursky 	serv->sv_maxconn = 1024;
258dd018428SStanislav Kinsbursky 	dprintk("nfs_callback_create_svc: service created\n");
259dd018428SStanislav Kinsbursky 	return serv;
260dd018428SStanislav Kinsbursky }
261dd018428SStanislav Kinsbursky 
26271468513SBenny Halevy /*
26371468513SBenny Halevy  * Bring up the callback thread if it is not already up.
26471468513SBenny Halevy  */
nfs_callback_up(u32 minorversion,struct rpc_xprt * xprt)26571468513SBenny Halevy int nfs_callback_up(u32 minorversion, struct rpc_xprt *xprt)
26671468513SBenny Halevy {
267dd018428SStanislav Kinsbursky 	struct svc_serv *serv;
268e82dc22dSAndy Adamson 	struct nfs_callback_data *cb_info = &nfs_callback_info[minorversion];
26923c20ecdSStanislav Kinsbursky 	int ret;
270c946556bSStanislav Kinsbursky 	struct net *net = xprt->xprt_net;
27171468513SBenny Halevy 
27271468513SBenny Halevy 	mutex_lock(&nfs_callback_mutex);
273dd018428SStanislav Kinsbursky 
274dd018428SStanislav Kinsbursky 	serv = nfs_callback_create_svc(minorversion);
275dd018428SStanislav Kinsbursky 	if (IS_ERR(serv)) {
276dd018428SStanislav Kinsbursky 		ret = PTR_ERR(serv);
277dd018428SStanislav Kinsbursky 		goto err_create;
278dd018428SStanislav Kinsbursky 	}
279dd018428SStanislav Kinsbursky 
28076566773SChuck Lever 	ret = nfs_callback_up_net(minorversion, serv, net, xprt);
281c946556bSStanislav Kinsbursky 	if (ret < 0)
282c946556bSStanislav Kinsbursky 		goto err_net;
2839793f7c8SStanislav Kinsbursky 
2848e246144SStanislav Kinsbursky 	ret = nfs_callback_start_svc(minorversion, xprt, serv);
2858e246144SStanislav Kinsbursky 	if (ret < 0)
2868e246144SStanislav Kinsbursky 		goto err_start;
287691c457aSStanislav Kinsbursky 
28823c20ecdSStanislav Kinsbursky 	cb_info->users++;
28923c20ecdSStanislav Kinsbursky err_net:
2903b01c11eSTrond Myklebust 	if (!cb_info->users)
2913b01c11eSTrond Myklebust 		cb_info->serv = NULL;
2928c62d127SNeilBrown 	svc_put(serv);
293dd018428SStanislav Kinsbursky err_create:
294353ab6e9SIngo Molnar 	mutex_unlock(&nfs_callback_mutex);
2951da177e4SLinus Torvalds 	return ret;
2968e246144SStanislav Kinsbursky 
2978e246144SStanislav Kinsbursky err_start:
298b3d19c51SStanislav Kinsbursky 	nfs_callback_down_net(minorversion, serv, net);
29923c20ecdSStanislav Kinsbursky 	dprintk("NFS: Couldn't create server thread; err = %d\n", ret);
30023c20ecdSStanislav Kinsbursky 	goto err_net;
3011da177e4SLinus Torvalds }
3021da177e4SLinus Torvalds 
3031da177e4SLinus Torvalds /*
3045afc597cSJeff Layton  * Kill the callback thread if it's no longer being used.
3051da177e4SLinus Torvalds  */
nfs_callback_down(int minorversion,struct net * net)306c8ceb412SStanislav Kinsbursky void nfs_callback_down(int minorversion, struct net *net)
3071da177e4SLinus Torvalds {
308e82dc22dSAndy Adamson 	struct nfs_callback_data *cb_info = &nfs_callback_info[minorversion];
309bb6aeba7STrond Myklebust 	struct svc_serv *serv;
310e82dc22dSAndy Adamson 
311353ab6e9SIngo Molnar 	mutex_lock(&nfs_callback_mutex);
312bb6aeba7STrond Myklebust 	serv = cb_info->serv;
313bb6aeba7STrond Myklebust 	nfs_callback_down_net(minorversion, serv, net);
314e82dc22dSAndy Adamson 	cb_info->users--;
3153b01c11eSTrond Myklebust 	if (cb_info->users == 0) {
316bb6aeba7STrond Myklebust 		svc_get(serv);
3173ebdbe52SNeilBrown 		svc_set_num_threads(serv, NULL, 0);
3188c62d127SNeilBrown 		svc_put(serv);
3191dc42e04SStanislav Kinsbursky 		dprintk("nfs_callback_down: service destroyed\n");
320e82dc22dSAndy Adamson 		cb_info->serv = NULL;
3215afc597cSJeff Layton 	}
322353ab6e9SIngo Molnar 	mutex_unlock(&nfs_callback_mutex);
3231da177e4SLinus Torvalds }
3241da177e4SLinus Torvalds 
325778be232SAndy Adamson /* Boolean check of RPC_AUTH_GSS principal */
326778be232SAndy Adamson int
check_gss_callback_principal(struct nfs_client * clp,struct svc_rqst * rqstp)327778be232SAndy Adamson check_gss_callback_principal(struct nfs_client *clp, struct svc_rqst *rqstp)
328945b34a7SOlga Kornievskaia {
32903a4e1f6SJ. Bruce Fields 	char *p = rqstp->rq_cred.cr_principal;
330945b34a7SOlga Kornievskaia 
331778be232SAndy Adamson 	if (rqstp->rq_authop->flavour != RPC_AUTH_GSS)
332778be232SAndy Adamson 		return 1;
333778be232SAndy Adamson 
334ece0de63SAndy Adamson 	/* No RPC_AUTH_GSS on NFSv4.1 back channel yet */
335ece0de63SAndy Adamson 	if (clp->cl_minorversion != 0)
336778be232SAndy Adamson 		return 0;
337945b34a7SOlga Kornievskaia 	/*
338945b34a7SOlga Kornievskaia 	 * It might just be a normal user principal, in which case
339945b34a7SOlga Kornievskaia 	 * userspace won't bother to tell us the name at all.
340945b34a7SOlga Kornievskaia 	 */
341945b34a7SOlga Kornievskaia 	if (p == NULL)
342778be232SAndy Adamson 		return 0;
343945b34a7SOlga Kornievskaia 
344f11b2a1cSJeff Layton 	/*
345f11b2a1cSJeff Layton 	 * Did we get the acceptor from userland during the SETCLIENID
346f11b2a1cSJeff Layton 	 * negotiation?
347f11b2a1cSJeff Layton 	 */
348f11b2a1cSJeff Layton 	if (clp->cl_acceptor)
349f11b2a1cSJeff Layton 		return !strcmp(p, clp->cl_acceptor);
350f11b2a1cSJeff Layton 
351f11b2a1cSJeff Layton 	/*
352f11b2a1cSJeff Layton 	 * Otherwise try to verify it using the cl_hostname. Note that this
353f11b2a1cSJeff Layton 	 * doesn't work if a non-canonical hostname was used in the devname.
354f11b2a1cSJeff Layton 	 */
355f11b2a1cSJeff Layton 
356945b34a7SOlga Kornievskaia 	/* Expect a GSS_C_NT_HOSTBASED_NAME like "nfs@serverhostname" */
357945b34a7SOlga Kornievskaia 
358945b34a7SOlga Kornievskaia 	if (memcmp(p, "nfs@", 4) != 0)
359778be232SAndy Adamson 		return 0;
360945b34a7SOlga Kornievskaia 	p += 4;
3614e0038b6STrond Myklebust 	if (strcmp(p, clp->cl_hostname) != 0)
362778be232SAndy Adamson 		return 0;
363778be232SAndy Adamson 	return 1;
364945b34a7SOlga Kornievskaia }
365945b34a7SOlga Kornievskaia 
366778be232SAndy Adamson /*
367778be232SAndy Adamson  * pg_authenticate method for nfsv4 callback threads.
368778be232SAndy Adamson  *
369778be232SAndy Adamson  * The authflavor has been negotiated, so an incorrect flavor is a server
3706f02dc88SJeff Layton  * bug. Deny packets with incorrect authflavor.
371778be232SAndy Adamson  *
372778be232SAndy Adamson  * All other checking done after NFS decoding where the nfs_client can be
373778be232SAndy Adamson  * found in nfs4_callback_compound
374778be232SAndy Adamson  */
nfs_callback_authenticate(struct svc_rqst * rqstp)375*78c542f9SChuck Lever static enum svc_auth_status nfs_callback_authenticate(struct svc_rqst *rqstp)
3761da177e4SLinus Torvalds {
3775c2465dfSChuck Lever 	rqstp->rq_auth_stat = rpc_autherr_badcred;
3785c2465dfSChuck Lever 
3791da177e4SLinus Torvalds 	switch (rqstp->rq_authop->flavour) {
3801da177e4SLinus Torvalds 	case RPC_AUTH_NULL:
3811da177e4SLinus Torvalds 		if (rqstp->rq_proc != CB_NULL)
3826f02dc88SJeff Layton 			return SVC_DENIED;
3831da177e4SLinus Torvalds 		break;
3841da177e4SLinus Torvalds 	case RPC_AUTH_GSS:
385778be232SAndy Adamson 		/* No RPC_AUTH_GSS support yet in NFSv4.1 */
386778be232SAndy Adamson 		 if (svc_is_backchannel(rqstp))
3876f02dc88SJeff Layton 			return SVC_DENIED;
3881da177e4SLinus Torvalds 	}
3895c2465dfSChuck Lever 
3905c2465dfSChuck Lever 	rqstp->rq_auth_stat = rpc_auth_ok;
391778be232SAndy Adamson 	return SVC_OK;
3921da177e4SLinus Torvalds }
3931da177e4SLinus Torvalds 
3941da177e4SLinus Torvalds /*
3951da177e4SLinus Torvalds  * Define NFS4 callback program
3961da177e4SLinus Torvalds  */
397e9679189SChristoph Hellwig static const struct svc_version *nfs4_callback_version[] = {
3981da177e4SLinus Torvalds 	[1] = &nfs4_callback_version1,
39907bccc2dSAlexandros Batsakis 	[4] = &nfs4_callback_version4,
4001da177e4SLinus Torvalds };
4011da177e4SLinus Torvalds 
4021da177e4SLinus Torvalds static struct svc_program nfs4_callback_program = {
4031da177e4SLinus Torvalds 	.pg_prog = NFS4_CALLBACK,			/* RPC service number */
4041da177e4SLinus Torvalds 	.pg_nvers = ARRAY_SIZE(nfs4_callback_version),	/* Number of entries */
4051da177e4SLinus Torvalds 	.pg_vers = nfs4_callback_version,		/* version table */
4061da177e4SLinus Torvalds 	.pg_name = "NFSv4 callback",			/* service name */
4071da177e4SLinus Torvalds 	.pg_class = "nfs",				/* authentication class */
4081da177e4SLinus Torvalds 	.pg_authenticate = nfs_callback_authenticate,
4098e5b6773STrond Myklebust 	.pg_init_request = svc_generic_init_request,
410642ee6b2STrond Myklebust 	.pg_rpcbind_set	= svc_generic_rpcbind_set,
4111da177e4SLinus Torvalds };
412