1 /* $OpenBSD: gss-genr.c,v 1.3 2003/11/21 11:57:03 djm Exp $ */
4 * Copyright (c) 2001-2003 Simon Wilkinson. All rights reserved.
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions
9 * 1. Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * 2. Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
15 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR `AS IS'' AND ANY EXPRESS OR
16 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
17 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
18 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
19 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
20 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
21 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
22 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
23 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
24 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
35 #include "monitor_wrap.h"
40 extern u_char
*session_id2
;
41 extern u_int session_id2_len
;
43 /* Check that the OID in a data stream matches that in the context */
45 ssh_gssapi_check_oid(Gssctxt
*ctx
, void *data
, size_t len
)
47 return (ctx
!= NULL
&& ctx
->oid
!= GSS_C_NO_OID
&&
48 ctx
->oid
->length
== len
&&
49 memcmp(ctx
->oid
->elements
, data
, len
) == 0);
52 /* Set the contexts OID from a data stream */
54 ssh_gssapi_set_oid_data(Gssctxt
*ctx
, void *data
, size_t len
)
56 if (ctx
->oid
!= GSS_C_NO_OID
) {
57 xfree(ctx
->oid
->elements
);
60 ctx
->oid
= xmalloc(sizeof(gss_OID_desc
));
61 ctx
->oid
->length
= len
;
62 ctx
->oid
->elements
= xmalloc(len
);
63 memcpy(ctx
->oid
->elements
, data
, len
);
66 /* Set the contexts OID */
68 ssh_gssapi_set_oid(Gssctxt
*ctx
, gss_OID oid
)
70 ssh_gssapi_set_oid_data(ctx
, oid
->elements
, oid
->length
);
73 /* All this effort to report an error ... */
75 ssh_gssapi_error(Gssctxt
*ctxt
)
77 debug("%s", ssh_gssapi_last_error(ctxt
, NULL
, NULL
));
81 ssh_gssapi_last_error(Gssctxt
*ctxt
,
82 OM_uint32
*major_status
, OM_uint32
*minor_status
)
85 gss_buffer_desc msg
= GSS_C_EMPTY_BUFFER
;
92 if (major_status
!= NULL
)
93 *major_status
= ctxt
->major
;
94 if (minor_status
!= NULL
)
95 *minor_status
= ctxt
->minor
;
98 /* The GSSAPI error */
100 gss_display_status(&lmin
, ctxt
->major
,
101 GSS_C_GSS_CODE
, GSS_C_NULL_OID
, &ctx
, &msg
);
103 buffer_append(&b
, msg
.value
, msg
.length
);
104 buffer_put_char(&b
, '\n');
106 gss_release_buffer(&lmin
, &msg
);
109 /* The mechanism specific error */
111 gss_display_status(&lmin
, ctxt
->minor
,
112 GSS_C_MECH_CODE
, GSS_C_NULL_OID
, &ctx
, &msg
);
114 buffer_append(&b
, msg
.value
, msg
.length
);
115 buffer_put_char(&b
, '\n');
117 gss_release_buffer(&lmin
, &msg
);
120 buffer_put_char(&b
, '\0');
121 ret
= xmalloc(buffer_len(&b
));
122 buffer_get(&b
, ret
, buffer_len(&b
));
128 * Initialise our GSSAPI context. We use this opaque structure to contain all
129 * of the data which both the client and server need to persist across
130 * {accept,init}_sec_context calls, so that when we do it from the userauth
131 * stuff life is a little easier
134 ssh_gssapi_build_ctx(Gssctxt
**ctx
)
136 *ctx
= xmalloc(sizeof (Gssctxt
));
139 (*ctx
)->context
= GSS_C_NO_CONTEXT
;
140 (*ctx
)->name
= GSS_C_NO_NAME
;
141 (*ctx
)->oid
= GSS_C_NO_OID
;
142 (*ctx
)->creds
= GSS_C_NO_CREDENTIAL
;
143 (*ctx
)->client
= GSS_C_NO_NAME
;
144 (*ctx
)->client_creds
= GSS_C_NO_CREDENTIAL
;
147 /* Delete our context, providing it has been built correctly */
149 ssh_gssapi_delete_ctx(Gssctxt
**ctx
)
155 if ((*ctx
)->context
!= GSS_C_NO_CONTEXT
)
156 gss_delete_sec_context(&ms
, &(*ctx
)->context
, GSS_C_NO_BUFFER
);
157 if ((*ctx
)->name
!= GSS_C_NO_NAME
)
158 gss_release_name(&ms
, &(*ctx
)->name
);
159 if ((*ctx
)->oid
!= GSS_C_NO_OID
) {
160 xfree((*ctx
)->oid
->elements
);
162 (*ctx
)->oid
= GSS_C_NO_OID
;
164 if ((*ctx
)->creds
!= GSS_C_NO_CREDENTIAL
)
165 gss_release_cred(&ms
, &(*ctx
)->creds
);
166 if ((*ctx
)->client
!= GSS_C_NO_NAME
)
167 gss_release_name(&ms
, &(*ctx
)->client
);
168 if ((*ctx
)->client_creds
!= GSS_C_NO_CREDENTIAL
)
169 gss_release_cred(&ms
, &(*ctx
)->client_creds
);
176 * Wrapper to init_sec_context
177 * Requires that the context contains:
179 * server name (from ssh_gssapi_import_name)
182 ssh_gssapi_init_ctx(Gssctxt
*ctx
, int deleg_creds
, gss_buffer_desc
*recv_tok
,
183 gss_buffer_desc
* send_tok
, OM_uint32
*flags
)
188 deleg_flag
= GSS_C_DELEG_FLAG
;
189 debug("Delegating credentials");
192 ctx
->major
= gss_init_sec_context(&ctx
->minor
,
193 GSS_C_NO_CREDENTIAL
, &ctx
->context
, ctx
->name
, ctx
->oid
,
194 GSS_C_MUTUAL_FLAG
| GSS_C_INTEG_FLAG
| deleg_flag
,
195 0, NULL
, recv_tok
, NULL
, send_tok
, flags
, NULL
);
197 if (GSS_ERROR(ctx
->major
))
198 ssh_gssapi_error(ctx
);
203 /* Create a service name for the given host */
205 ssh_gssapi_import_name(Gssctxt
*ctx
, const char *host
)
207 gss_buffer_desc gssbuf
;
209 gssbuf
.length
= sizeof("host@") + strlen(host
);
210 gssbuf
.value
= xmalloc(gssbuf
.length
);
211 snprintf(gssbuf
.value
, gssbuf
.length
, "host@%s", host
);
213 if ((ctx
->major
= gss_import_name(&ctx
->minor
,
214 &gssbuf
, GSS_C_NT_HOSTBASED_SERVICE
, &ctx
->name
)))
215 ssh_gssapi_error(ctx
);
221 /* Acquire credentials for a server running on the current host.
222 * Requires that the context structure contains a valid OID
225 /* Returns a GSSAPI error code */
227 ssh_gssapi_acquire_cred(Gssctxt
*ctx
)
230 char lname
[MAXHOSTNAMELEN
];
233 gss_create_empty_oid_set(&status
, &oidset
);
234 gss_add_oid_set_member(&status
, ctx
->oid
, &oidset
);
236 if (gethostname(lname
, MAXHOSTNAMELEN
))
239 if (GSS_ERROR(ssh_gssapi_import_name(ctx
, lname
)))
242 if ((ctx
->major
= gss_acquire_cred(&ctx
->minor
,
243 ctx
->name
, 0, oidset
, GSS_C_ACCEPT
, &ctx
->creds
, NULL
, NULL
)))
244 ssh_gssapi_error(ctx
);
246 gss_release_oid_set(&status
, &oidset
);
251 ssh_gssapi_sign(Gssctxt
*ctx
, gss_buffer_t buffer
, gss_buffer_t hash
)
253 if ((ctx
->major
= gss_get_mic(&ctx
->minor
, ctx
->context
,
254 GSS_C_QOP_DEFAULT
, buffer
, hash
)))
255 ssh_gssapi_error(ctx
);
261 ssh_gssapi_buildmic(Buffer
*b
, const char *user
, const char *service
,
265 buffer_put_string(b
, session_id2
, session_id2_len
);
266 buffer_put_char(b
, SSH2_MSG_USERAUTH_REQUEST
);
267 buffer_put_cstring(b
, user
);
268 buffer_put_cstring(b
, service
);
269 buffer_put_cstring(b
, context
);
273 ssh_gssapi_server_ctx(Gssctxt
**ctx
, gss_OID oid
) {
275 ssh_gssapi_delete_ctx(ctx
);
276 ssh_gssapi_build_ctx(ctx
);
277 ssh_gssapi_set_oid(*ctx
, oid
);
278 return (ssh_gssapi_acquire_cred(*ctx
));