From: Andy Adamson <andros@xxxxxxxxxx> Store the version used in the context INIT phase with the context. Check the incoming rpcsec gss version against the stored context version. Signed-off-by: Andy Adamson <andros@xxxxxxxxxx> --- include/linux/sunrpc/gss_api.h | 1 + net/sunrpc/auth_gss/svcauth_gss.c | 20 ++++++++++++++------ 2 files changed, 15 insertions(+), 6 deletions(-) diff --git a/include/linux/sunrpc/gss_api.h b/include/linux/sunrpc/gss_api.h index 4e02204..d4cff5a 100644 --- a/include/linux/sunrpc/gss_api.h +++ b/include/linux/sunrpc/gss_api.h @@ -28,6 +28,7 @@ struct gss3_assert { /* The mechanism-independent gss-api context: */ struct gss_ctx { struct gss_api_mech *mech_type; + u32 gss_version; void *internal_ctx_id; }; diff --git a/net/sunrpc/auth_gss/svcauth_gss.c b/net/sunrpc/auth_gss/svcauth_gss.c index aa7cb3b..f7aa8c4 100644 --- a/net/sunrpc/auth_gss/svcauth_gss.c +++ b/net/sunrpc/auth_gss/svcauth_gss.c @@ -976,8 +976,8 @@ struct gss_svc_data { } static inline int -gss_write_init_verf(struct cache_detail *cd, struct svc_rqst *rqstp, - struct xdr_netobj *out_handle, int *major_status) +gss_write_init_verf(struct cache_detail *cd, struct svc_rqst *rqstp, u32 gssv, + struct xdr_netobj *out_handle, int *major_status) { struct rsc *rsci; int rc; @@ -989,6 +989,8 @@ struct gss_svc_data { *major_status = GSS_S_NO_CONTEXT; return gss_write_null_verf(rqstp); } + /* set the RPCSEC_GSS version in the context */ + rsci->mechctx->gss_version = gssv; rc = gss_write_verf(rqstp, rsci->mechctx, GSS_SEQ_WIN); cache_put(&rsci->h, cd); return rc; @@ -1130,7 +1132,7 @@ static int svcauth_gss_legacy_init(struct svc_rqst *rqstp, ret = SVC_CLOSE; /* Got an answer to the upcall; use it: */ - if (gss_write_init_verf(sn->rsc_cache, rqstp, + if (gss_write_init_verf(sn->rsc_cache, rqstp, gc->gc_v, &rsip->out_handle, &rsip->major_status)) goto out; if (gss_write_resv(resv, PAGE_SIZE, @@ -1259,7 +1261,7 @@ static int svcauth_gss_proxy_init(struct svc_rqst *rqstp, } /* Got an answer to the upcall; use it: */ - if (gss_write_init_verf(sn->rsc_cache, rqstp, + if (gss_write_init_verf(sn->rsc_cache, rqstp, gc->gc_v, &cli_handle, &ud.major_status)) goto out; if (gss_write_resv(resv, PAGE_SIZE, @@ -1435,14 +1437,15 @@ static void destroy_use_gss_proxy_proc_entry(struct net *net) {} rpcstart -= 7; /* credential is: - * version(==1), proc(0,1,2,3), seq, service (1,2,3), handle + * version(==1 or 3), proc(0,1,2,3), seq, service (1,2,3), handle * at least 5 u32s, and is preceded by length, so that makes 6. */ if (argv->iov_len < 5 * 4) goto auth_err; crlen = svc_getnl(argv); - if (svc_getnl(argv) != RPC_GSS_VERSION) + gc->gc_v = svc_getnl(argv); + if ((gc->gc_v != RPC_GSS_VERSION) && (gc->gc_v != RPC_GSS3_VERSION)) goto auth_err; gc->gc_proc = svc_getnl(argv); gc->gc_seq = svc_getnl(argv); @@ -1470,6 +1473,11 @@ static void destroy_use_gss_proxy_proc_entry(struct net *net) {} rsci = gss_svc_searchbyctx(sn->rsc_cache, &gc->gc_ctx); if (!rsci) goto auth_err; + if (rsci->mechctx->gss_version != gc->gc_v) { + pr_warn("NFSD: RPCSEC_GSS version mismatch (%u:%u)\n", + rsci->mechctx->gss_version, gc->gc_v); + goto auth_err; + } switch (gss_verify_header(rqstp, rsci, rpcstart, gc, authp)) { case SVC_OK: break; -- 1.8.3.1 -- To unsubscribe from this list: send the line "unsubscribe linux-nfs" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html