xref: /freebsd-src/crypto/heimdal/lib/gssapi/mech/gss_inquire_cred.c (revision 6a068746777241722b2b32c5d0bc443a2a64d80b)
1*ae771770SStanislav Sedov /*-
2*ae771770SStanislav Sedov  * Copyright (c) 2005 Doug Rabson
3*ae771770SStanislav Sedov  * All rights reserved.
4*ae771770SStanislav Sedov  *
5*ae771770SStanislav Sedov  * Redistribution and use in source and binary forms, with or without
6*ae771770SStanislav Sedov  * modification, are permitted provided that the following conditions
7*ae771770SStanislav Sedov  * are met:
8*ae771770SStanislav Sedov  * 1. Redistributions of source code must retain the above copyright
9*ae771770SStanislav Sedov  *    notice, this list of conditions and the following disclaimer.
10*ae771770SStanislav Sedov  * 2. Redistributions in binary form must reproduce the above copyright
11*ae771770SStanislav Sedov  *    notice, this list of conditions and the following disclaimer in the
12*ae771770SStanislav Sedov  *    documentation and/or other materials provided with the distribution.
13*ae771770SStanislav Sedov  *
14*ae771770SStanislav Sedov  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15*ae771770SStanislav Sedov  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16*ae771770SStanislav Sedov  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17*ae771770SStanislav Sedov  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18*ae771770SStanislav Sedov  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19*ae771770SStanislav Sedov  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20*ae771770SStanislav Sedov  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21*ae771770SStanislav Sedov  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22*ae771770SStanislav Sedov  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23*ae771770SStanislav Sedov  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24*ae771770SStanislav Sedov  * SUCH DAMAGE.
25*ae771770SStanislav Sedov  *
26*ae771770SStanislav Sedov  *	$FreeBSD: src/lib/libgssapi/gss_inquire_cred.c,v 1.1 2005/12/29 14:40:20 dfr Exp $
27*ae771770SStanislav Sedov  */
28*ae771770SStanislav Sedov 
29*ae771770SStanislav Sedov #include "mech_locl.h"
30*ae771770SStanislav Sedov 
31*ae771770SStanislav Sedov #define AUSAGE 1
32*ae771770SStanislav Sedov #define IUSAGE 2
33*ae771770SStanislav Sedov 
34*ae771770SStanislav Sedov static void
updateusage(gss_cred_usage_t usage,int * usagemask)35*ae771770SStanislav Sedov updateusage(gss_cred_usage_t usage, int *usagemask)
36*ae771770SStanislav Sedov {
37*ae771770SStanislav Sedov     if (usage == GSS_C_BOTH)
38*ae771770SStanislav Sedov 	*usagemask |= AUSAGE | IUSAGE;
39*ae771770SStanislav Sedov     else if (usage == GSS_C_ACCEPT)
40*ae771770SStanislav Sedov 	*usagemask |= AUSAGE;
41*ae771770SStanislav Sedov     else if (usage == GSS_C_INITIATE)
42*ae771770SStanislav Sedov 	*usagemask |= IUSAGE;
43*ae771770SStanislav Sedov }
44*ae771770SStanislav Sedov 
45*ae771770SStanislav Sedov GSSAPI_LIB_FUNCTION OM_uint32 GSSAPI_LIB_CALL
gss_inquire_cred(OM_uint32 * minor_status,const gss_cred_id_t cred_handle,gss_name_t * name_ret,OM_uint32 * lifetime,gss_cred_usage_t * cred_usage,gss_OID_set * mechanisms)46*ae771770SStanislav Sedov gss_inquire_cred(OM_uint32 *minor_status,
47*ae771770SStanislav Sedov     const gss_cred_id_t cred_handle,
48*ae771770SStanislav Sedov     gss_name_t *name_ret,
49*ae771770SStanislav Sedov     OM_uint32 *lifetime,
50*ae771770SStanislav Sedov     gss_cred_usage_t *cred_usage,
51*ae771770SStanislav Sedov     gss_OID_set *mechanisms)
52*ae771770SStanislav Sedov {
53*ae771770SStanislav Sedov 	OM_uint32 major_status;
54*ae771770SStanislav Sedov 	struct _gss_mech_switch *m;
55*ae771770SStanislav Sedov 	struct _gss_cred *cred = (struct _gss_cred *) cred_handle;
56*ae771770SStanislav Sedov 	struct _gss_name *name;
57*ae771770SStanislav Sedov 	struct _gss_mechanism_name *mn;
58*ae771770SStanislav Sedov 	OM_uint32 min_lifetime;
59*ae771770SStanislav Sedov 	int found = 0;
60*ae771770SStanislav Sedov 	int usagemask = 0;
61*ae771770SStanislav Sedov 	gss_cred_usage_t usage;
62*ae771770SStanislav Sedov 
63*ae771770SStanislav Sedov 	_gss_load_mech();
64*ae771770SStanislav Sedov 
65*ae771770SStanislav Sedov 	*minor_status = 0;
66*ae771770SStanislav Sedov 	if (name_ret)
67*ae771770SStanislav Sedov 		*name_ret = GSS_C_NO_NAME;
68*ae771770SStanislav Sedov 	if (lifetime)
69*ae771770SStanislav Sedov 		*lifetime = 0;
70*ae771770SStanislav Sedov 	if (cred_usage)
71*ae771770SStanislav Sedov 		*cred_usage = 0;
72*ae771770SStanislav Sedov 	if (mechanisms)
73*ae771770SStanislav Sedov 		*mechanisms = GSS_C_NO_OID_SET;
74*ae771770SStanislav Sedov 
75*ae771770SStanislav Sedov 	if (name_ret) {
76*ae771770SStanislav Sedov 		name = calloc(1, sizeof(*name));
77*ae771770SStanislav Sedov 		if (name == NULL) {
78*ae771770SStanislav Sedov 			*minor_status = ENOMEM;
79*ae771770SStanislav Sedov 			return (GSS_S_FAILURE);
80*ae771770SStanislav Sedov 		}
81*ae771770SStanislav Sedov 		HEIM_SLIST_INIT(&name->gn_mn);
82*ae771770SStanislav Sedov 	} else {
83*ae771770SStanislav Sedov 		name = NULL;
84*ae771770SStanislav Sedov 	}
85*ae771770SStanislav Sedov 
86*ae771770SStanislav Sedov 	if (mechanisms) {
87*ae771770SStanislav Sedov 		major_status = gss_create_empty_oid_set(minor_status,
88*ae771770SStanislav Sedov 		    mechanisms);
89*ae771770SStanislav Sedov 		if (major_status) {
90*ae771770SStanislav Sedov 			if (name) free(name);
91*ae771770SStanislav Sedov 			return (major_status);
92*ae771770SStanislav Sedov 		}
93*ae771770SStanislav Sedov 	}
94*ae771770SStanislav Sedov 
95*ae771770SStanislav Sedov 	min_lifetime = GSS_C_INDEFINITE;
96*ae771770SStanislav Sedov 	if (cred) {
97*ae771770SStanislav Sedov 		struct _gss_mechanism_cred *mc;
98*ae771770SStanislav Sedov 
99*ae771770SStanislav Sedov 		HEIM_SLIST_FOREACH(mc, &cred->gc_mc, gmc_link) {
100*ae771770SStanislav Sedov 			gss_name_t mc_name;
101*ae771770SStanislav Sedov 			OM_uint32 mc_lifetime;
102*ae771770SStanislav Sedov 
103*ae771770SStanislav Sedov 			major_status = mc->gmc_mech->gm_inquire_cred(minor_status,
104*ae771770SStanislav Sedov 			    mc->gmc_cred, &mc_name, &mc_lifetime, &usage, NULL);
105*ae771770SStanislav Sedov 			if (major_status)
106*ae771770SStanislav Sedov 				continue;
107*ae771770SStanislav Sedov 
108*ae771770SStanislav Sedov 			updateusage(usage, &usagemask);
109*ae771770SStanislav Sedov 			if (name) {
110*ae771770SStanislav Sedov 				mn = malloc(sizeof(struct _gss_mechanism_name));
111*ae771770SStanislav Sedov 				if (!mn) {
112*ae771770SStanislav Sedov 					mc->gmc_mech->gm_release_name(minor_status,
113*ae771770SStanislav Sedov 					    &mc_name);
114*ae771770SStanislav Sedov 					continue;
115*ae771770SStanislav Sedov 				}
116*ae771770SStanislav Sedov 				mn->gmn_mech = mc->gmc_mech;
117*ae771770SStanislav Sedov 				mn->gmn_mech_oid = mc->gmc_mech_oid;
118*ae771770SStanislav Sedov 				mn->gmn_name = mc_name;
119*ae771770SStanislav Sedov 				HEIM_SLIST_INSERT_HEAD(&name->gn_mn, mn, gmn_link);
120*ae771770SStanislav Sedov 			} else {
121*ae771770SStanislav Sedov 				mc->gmc_mech->gm_release_name(minor_status,
122*ae771770SStanislav Sedov 				    &mc_name);
123*ae771770SStanislav Sedov 			}
124*ae771770SStanislav Sedov 
125*ae771770SStanislav Sedov 			if (mc_lifetime < min_lifetime)
126*ae771770SStanislav Sedov 				min_lifetime = mc_lifetime;
127*ae771770SStanislav Sedov 
128*ae771770SStanislav Sedov 			if (mechanisms)
129*ae771770SStanislav Sedov 				gss_add_oid_set_member(minor_status,
130*ae771770SStanislav Sedov 				    mc->gmc_mech_oid, mechanisms);
131*ae771770SStanislav Sedov 			found++;
132*ae771770SStanislav Sedov 		}
133*ae771770SStanislav Sedov 	} else {
134*ae771770SStanislav Sedov 		HEIM_SLIST_FOREACH(m, &_gss_mechs, gm_link) {
135*ae771770SStanislav Sedov 			gss_name_t mc_name;
136*ae771770SStanislav Sedov 			OM_uint32 mc_lifetime;
137*ae771770SStanislav Sedov 
138*ae771770SStanislav Sedov 			major_status = m->gm_mech.gm_inquire_cred(minor_status,
139*ae771770SStanislav Sedov 			    GSS_C_NO_CREDENTIAL, &mc_name, &mc_lifetime,
140*ae771770SStanislav Sedov 			    &usage, NULL);
141*ae771770SStanislav Sedov 			if (major_status)
142*ae771770SStanislav Sedov 				continue;
143*ae771770SStanislav Sedov 
144*ae771770SStanislav Sedov 			updateusage(usage, &usagemask);
145*ae771770SStanislav Sedov 			if (name && mc_name) {
146*ae771770SStanislav Sedov 				mn = malloc(
147*ae771770SStanislav Sedov 					sizeof(struct _gss_mechanism_name));
148*ae771770SStanislav Sedov 				if (!mn) {
149*ae771770SStanislav Sedov 					m->gm_mech.gm_release_name(
150*ae771770SStanislav Sedov 						minor_status, &mc_name);
151*ae771770SStanislav Sedov 					continue;
152*ae771770SStanislav Sedov 				}
153*ae771770SStanislav Sedov 				mn->gmn_mech = &m->gm_mech;
154*ae771770SStanislav Sedov 				mn->gmn_mech_oid = &m->gm_mech_oid;
155*ae771770SStanislav Sedov 				mn->gmn_name = mc_name;
156*ae771770SStanislav Sedov 				HEIM_SLIST_INSERT_HEAD(&name->gn_mn, mn, gmn_link);
157*ae771770SStanislav Sedov 			} else if (mc_name) {
158*ae771770SStanislav Sedov 				m->gm_mech.gm_release_name(minor_status,
159*ae771770SStanislav Sedov 				    &mc_name);
160*ae771770SStanislav Sedov 			}
161*ae771770SStanislav Sedov 
162*ae771770SStanislav Sedov 			if (mc_lifetime < min_lifetime)
163*ae771770SStanislav Sedov 				min_lifetime = mc_lifetime;
164*ae771770SStanislav Sedov 
165*ae771770SStanislav Sedov 			if (mechanisms)
166*ae771770SStanislav Sedov 				gss_add_oid_set_member(minor_status,
167*ae771770SStanislav Sedov 				    &m->gm_mech_oid, mechanisms);
168*ae771770SStanislav Sedov 			found++;
169*ae771770SStanislav Sedov 		}
170*ae771770SStanislav Sedov 	}
171*ae771770SStanislav Sedov 
172*ae771770SStanislav Sedov 	if (found == 0) {
173*ae771770SStanislav Sedov 		gss_name_t n = (gss_name_t)name;
174*ae771770SStanislav Sedov 		if (n)
175*ae771770SStanislav Sedov 			gss_release_name(minor_status, &n);
176*ae771770SStanislav Sedov 		gss_release_oid_set(minor_status, mechanisms);
177*ae771770SStanislav Sedov 		*minor_status = 0;
178*ae771770SStanislav Sedov 		return (GSS_S_NO_CRED);
179*ae771770SStanislav Sedov 	}
180*ae771770SStanislav Sedov 
181*ae771770SStanislav Sedov 	*minor_status = 0;
182*ae771770SStanislav Sedov 	if (name_ret)
183*ae771770SStanislav Sedov 		*name_ret = (gss_name_t) name;
184*ae771770SStanislav Sedov 	if (lifetime)
185*ae771770SStanislav Sedov 		*lifetime = min_lifetime;
186*ae771770SStanislav Sedov 	if (cred_usage) {
187*ae771770SStanislav Sedov 		if ((usagemask & (AUSAGE|IUSAGE)) == (AUSAGE|IUSAGE))
188*ae771770SStanislav Sedov 			*cred_usage = GSS_C_BOTH;
189*ae771770SStanislav Sedov 		else if (usagemask & IUSAGE)
190*ae771770SStanislav Sedov 			*cred_usage = GSS_C_INITIATE;
191*ae771770SStanislav Sedov 		else if (usagemask & AUSAGE)
192*ae771770SStanislav Sedov 			*cred_usage = GSS_C_ACCEPT;
193*ae771770SStanislav Sedov 	}
194*ae771770SStanislav Sedov 	return (GSS_S_COMPLETE);
195*ae771770SStanislav Sedov }
196