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