/* * Copyright 1994 by OpenVision Technologies, Inc. * * Permission to use, copy, modify, distribute, and sell this software * and its documentation for any purpose is hereby granted without fee, * provided that the above copyright notice appears in all copies and * that both that copyright notice and this permission notice appear in * supporting documentation, and that the name of OpenVision not be used * in advertising or publicity pertaining to distribution of the software * without specific, written prior permission. OpenVision makes no * representations about the suitability of this software for any * purpose. It is provided "as is" without express or implied warranty. * * OPENVISION DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO * EVENT SHALL OPENVISION BE LIABLE FOR ANY SPECIAL, INDIRECT OR * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF * USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR * OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR * PERFORMANCE OF THIS SOFTWARE. */ #include #include #include #include #include "krb5.h" #include #include "krb524.h" int krb524_convert_creds_addr(krb5_creds *v5creds, CREDENTIALS *v4creds, struct sockaddr *saddr) { int ret; if (ret = krb524_convert_creds_plain(v5creds, v4creds)) return ret; return krb524_convert_tkt(v5creds->server, &v5creds->ticket, &v4creds->ticket_st, &v4creds->kvno, saddr); } int krb524_convert_creds_kdc(krb5_creds *v5creds, CREDENTIALS *v4creds) { struct sockaddr_in *addrs; int ret, naddrs; if (ret = krb5_locate_kdc(&v5creds->server->realm, &addrs, &naddrs)) return ret; if (naddrs == 0) ret = KRB5_KDC_UNREACH; else { addrs[0].sin_port = 0; /* use krb524 default port */ ret = krb524_convert_creds_addr(v5creds, v4creds, (struct sockaddr *) &addrs[0]); } free(addrs); return ret; } int krb524_convert_creds_plain(krb5_creds *v5creds, CREDENTIALS *v4creds) { unsigned long addr; krb5_data *comp; int ret; memset((char *) v4creds, 0, sizeof(CREDENTIALS)); if (ret = krb524_convert_princs(v5creds->client, v5creds->server, v4creds->pname, v4creds->pinst, v4creds->realm, v4creds->service, v4creds->instance)) return ret; /* Check keytype too */ if (v5creds->keyblock.length != sizeof(C_Block)) { if (krb524_debug) fprintf(stderr, "v5 session keyblock length %d != " "C_Block size %d\n", v5creds->keyblock.length, sizeof(C_Block)); return KRB524_BADKEY; } else memcpy(v4creds->session, (char *) v5creds->keyblock.contents, sizeof(C_Block)); /* V4 has no concept of authtime or renew_till, so ignore them */ /* V4 lifetime is 1 byte, in 5 minute increments */ v4creds->lifetime = 0xff & ((v5creds->times.endtime - v5creds->times.starttime) / 300); v4creds->issue_date = v5creds->times.starttime; /* XXX perhaps we should use the addr of the client host if */ /* v5creds contains more than one addr. Q: Does V4 support */ /* non-INET addresses? */ if (!v5creds->addresses || !v5creds->addresses[0] || v5creds->addresses[0]->addrtype != ADDRTYPE_INET || v5creds->addresses[0]->length != sizeof(addr)) { if (krb524_debug) fprintf(stderr, "Invalid v5creds address information.\n"); return KRB524_BADADDR; } else memcpy((char *) &addr, v5creds->addresses[0]->contents, sizeof(addr)); return 0; }