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