From: Chuck Lever <chuck.lever@oracle.com>
To: Olaf Kirch <okir@suse.de>
Cc: linux-nfs@vger.kernel.org, nfsv4@linux-nfs.org,
Steve Dickson <SteveD@redhat.com>
Subject: Re: [PATCH 0/2] Make libtirpc work with old style portmapper
Date: Tue, 7 Sep 2010 11:36:58 -0400 [thread overview]
Message-ID: <F1D592E6-35AD-45DA-B69E-044D59500DFB@oracle.com> (raw)
In-Reply-To: <201009071327.07291.okir@suse.de>
On Sep 7, 2010, at 7:27 AM, Olaf Kirch wrote:
> On Tuesday 31 August 2010 01:48:02 Chuck Lever wrote:
>> On Aug 30, 2010, at 12:19 PM, Olaf Kirch wrote:
>>> On Monday 30 August 2010 17:59:18 Chuck Lever wrote:
>>>> Another minor problem I think I remember is that if libtirpc is used on
>>>> a system (perhaps because it is statically linked with said ISV
>>>> RPC-enabled application) that does not have /etc/netconfig installed,
>>>> the transport creation logic in rpcb_clnt.c simply doesn't work.
>>>
>>> Well, but that's something that's fixed easily - we can always tell
>>> such customer to install an /etc/netconfig on their system.
>>
>> Agreed, but for various reasons, the symptoms don't necessarily immediately
>> imply what is needed to correct the problem.
>
> I've updated the second patch, which now modifies pmap_set/unset
> directly, without messing with the functionality of rpbc_set/unset.
Probably not a bad idea. But I thought this was a problem for applications that are calling rpcb_{un,}set(3t), not apps that are invoking the legacy variants. How does this new patch address the problem?
> With this change, the compatibility functions (pmap_{,un}set) will
> first try the old API and then fall through to the new rpcbind
> interface.
What happens if you try the other way around: try the AF_UNIX mechanism first, then the legacy mechanism? It's arguable that any application calling the pmap_{un,}set(3t) interface would likely expect the old behavior anyway, but could probably benefit from the additional security if rpcbind, rather than portmap, is running locally.
Unless I've misunderstood the problem, you've done exactly the opposite of what needs to be done:
1. rpcb_{un,}set(3t) need to work whether rpcbind or portmapper are running
2. pmap_{un,}set(3t) should still invoke rpcb_{un,}set(3t) under the covers to ensure they get advanced security when available
This is because you can build legacy apps against libtirpc (which want robust pmap_{un,}set(3t) no matter which portmapper is running), but you can also construct TI-RPC enabled apps that may be running on a system with only portmap (which want robust rpcb_{un,}set(3t)). So I think you have to fix both APIs.
> Olaf
> --
> Neo didn't bring down the Matrix. SOA did. (soafacts.com)
> --------------------------------------------
> Olaf Kirch - Director Server (okir@novell.com)
> SUSE LINUX Products GmbH, Maxfeldstr. 5, D-90409 Nürnberg
> GF: Markus Rex, HRB 16746 (AG Nürnberg)
> -----------------------------
>
> commit 30feadb3ae5a8d001aabc44f8ddad44298ec61a2
> Author: Olaf Kirch <okir@suse.de>
> Date: Mon Aug 23 14:36:13 2010 +0200
>
> pmap_set/unset: allow compat functions to work with old-style portmap
>
> This change fixes a bug when running applications compiled against
> libtirpc on a host with old-style portmap. Without this change, the
> pmap_set/pmap_unset compatibility functions will actually be mapped
> to a RPCB_SET/UNSET call. If the server does not support anything more
> recent than PMAP, the operations will fail completely.
>
> This fix makes pmap_set/unset try the old portmapper functions
> first, and if those fail, try to fall back to the new rpcbind
> interface.
>
> Signed-off-by: Olaf Kirch <okir@suse.de>
>
> diff --git a/src/pmap_clnt.c b/src/pmap_clnt.c
> index 1d5d153..24cf94a 100644
> --- a/src/pmap_clnt.c
> +++ b/src/pmap_clnt.c
> @@ -58,6 +58,10 @@ pmap_set(u_long program, u_long version, int protocol, int
> port)
> struct netconfig *nconf;
> char buf[32];
>
> +#ifdef PORTMAP
> + if (__pmap_set(program, version, protocol, port))
> + return (TRUE);
> +#endif
> if ((protocol != IPPROTO_UDP) && (protocol != IPPROTO_TCP)) {
> return (FALSE);
> }
> @@ -89,6 +93,11 @@ pmap_unset(u_long program, u_long version)
> bool_t udp_rslt = FALSE;
> bool_t tcp_rslt = FALSE;
>
> +#ifdef PORTMAP
> + if (__pmap_set(program, version, IPPROTO_UDP, 0)
> + && __pmap_set(program, version, IPPROTO_TCP, 0))
> + return (TRUE);
> +#endif
> nconf = __rpc_getconfip("udp");
> if (nconf != NULL) {
> udp_rslt = rpcb_unset((rpcprog_t)program, (rpcvers_t)version,
> diff --git a/src/rpc_com.h b/src/rpc_com.h
> index 38c2cfe..0a20a01 100644
> --- a/src/rpc_com.h
> +++ b/src/rpc_com.h
> @@ -86,6 +86,9 @@ bool_t __xdrrec_getrec(XDR *, enum xprt_stat *, bool_t);
> void __xprt_unregister_unlocked(SVCXPRT *);
> void __xprt_set_raddr(SVCXPRT *, const struct sockaddr_storage *);
>
> +#ifdef PORTMAP
> +bool_t __pmap_set(rpcprog_t, rpcvers_t, int protocol, int port);
> +#endif
>
> SVCXPRT **__svc_xports;
> int __svc_maxrec;
> diff --git a/src/rpcb_clnt.c b/src/rpcb_clnt.c
> index 531c619..9562646 100644
> --- a/src/rpcb_clnt.c
> +++ b/src/rpcb_clnt.c
> @@ -476,6 +476,55 @@ getpmaphandle(nconf, hostname, tgtaddr)
> #define IN4_LOCALHOST_STRING "127.0.0.1"
> #define IN6_LOCALHOST_STRING "::1"
>
> +#ifdef PORTMAP
> +/*
> + * Perform a PMAP_SET or PMAP_UNSET call to the
> + * local rpcbind/portmap service.
> + */
> +bool_t
> +__pmap_set(program, version, protocol, port)
> + rpcprog_t program;
> + rpcvers_t version;
> + int protocol;
> + int port;
> +{
> + CLIENT *client;
> + rpcproc_t pmapproc;
> + struct pmap pmapparms;
> + bool_t rslt = FALSE;
> + enum clnt_stat clnt_st;
> +
> + /* Arguments should already have been checked by caller */
> +
> + pmapproc = port? PMAPPROC_SET : PMAPPROC_UNSET;
> + pmapparms.pm_prog = program;
> + pmapparms.pm_vers = version;
> + pmapparms.pm_prot = protocol;
> + pmapparms.pm_port = port;
> +
> + client = getpmaphandle(NULL, IN4_LOCALHOST_STRING, NULL);
> + if (client == NULL)
> + return (FALSE);
> +
> + clnt_st = CLNT_CALL(client, pmapproc,
> + (xdrproc_t) xdr_pmap, (caddr_t)(void *) &pmapparms,
> + (xdrproc_t) xdr_bool, (caddr_t)(void *) &rslt,
> + tottimeout);
> +
> + if (clnt_st == RPC_SUCCESS)
> + return rslt;
> +
> + if (clnt_st != RPC_PROGVERSMISMATCH &&
> + clnt_st != RPC_PROGUNAVAIL) {
> + rpc_createerr.cf_stat = RPC_PMAPFAILURE;
> + clnt_geterr(client, &rpc_createerr.cf_error);
> + return (FALSE);
> + }
> +
> + return (TRUE);
> +}
> +#endif
> +
> /*
> * This routine will return a client handle that is connected to the local
> * rpcbind. Returns NULL on error and free's everything.
--
chuck[dot]lever[at]oracle[dot]com
next prev parent reply other threads:[~2010-09-07 15:39 UTC|newest]
Thread overview: 10+ messages / expand[flat|nested] mbox.gz Atom feed top
2010-08-30 13:03 [PATCH 0/2] Make libtirpc work with old style portmapper Olaf Kirch
2010-08-30 13:04 ` [PATCH 1/2] Introduce new helper function getpmaphandle Olaf Kirch
2010-08-30 13:04 ` [PATCH 2/2] " Olaf Kirch
2010-08-30 15:59 ` [PATCH 0/2] Make libtirpc work with old style portmapper Chuck Lever
2010-08-30 16:19 ` Olaf Kirch
2010-08-30 23:48 ` Chuck Lever
2010-09-07 11:27 ` Olaf Kirch
2010-09-07 15:36 ` Chuck Lever [this message]
2010-09-07 20:58 ` Olaf Kirch
2010-09-07 21:15 ` Chuck Lever
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=F1D592E6-35AD-45DA-B69E-044D59500DFB@oracle.com \
--to=chuck.lever@oracle.com \
--cc=SteveD@redhat.com \
--cc=linux-nfs@vger.kernel.org \
--cc=nfsv4@linux-nfs.org \
--cc=okir@suse.de \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox