#include "k5-int.h"
#include "etypes.h"
#include "dk.h"
#define K5CLENGTH 5
static krb5_error_code
derive_cksum_key(krb5_context context,
struct krb5_enc_provider *enc,
const krb5_keyblock *key,
krb5_keyusage usage,
krb5_keyblock **outkey)
{
krb5_error_code ret = 0;
krb5_keyblock *cached_key = NULL;
krb5_data d1;
unsigned char constantdata[K5CLENGTH];
cached_key = find_derived_key(usage, DK_CKSUM_KEY_BYTE,
(krb5_keyblock *)key);
if (cached_key)
*outkey = cached_key;
else {
*outkey = krb5_create_derived_keyblock(key->length);
if (*outkey == NULL)
return (ENOMEM);
constantdata[0] = (usage>>24)&0xff;
constantdata[1] = (usage>>16)&0xff;
constantdata[2] = (usage>>8)&0xff;
constantdata[3] = usage&0xff;
constantdata[4] = DK_CKSUM_KEY_BYTE;
d1.data = (char *)constantdata;
d1.length = sizeof(constantdata);
ret = krb5_derive_key(context, enc, key,
*outkey, &d1);
if (ret) {
krb5_free_keyblock(context, *outkey);
*outkey = NULL;
return (ret);
}
#ifdef _KERNEL
if (ret == 0 &&
(*outkey)->kef_mt != context->kef_cksum_mt) {
(*outkey)->kef_mt = context->kef_cksum_mt;
if ((*outkey)->key_tmpl != NULL) {
crypto_destroy_ctx_template((*outkey)->key_tmpl);
(*outkey)->key_tmpl = NULL;
}
ret = update_key_template(*outkey);
}
#endif
if (ret == 0)
ret = add_derived_key((krb5_keyblock *)key, usage,
DK_CKSUM_KEY_BYTE,
*outkey);
}
KRB5_LOG0(KRB5_INFO, "derive_cksum_key() end.");
return (ret);
}
krb5_error_code
krb5_dk_make_checksum(context, hash, key, usage, input, output)
krb5_context context;
krb5_const struct krb5_hash_provider *hash;
krb5_const krb5_keyblock *key;
krb5_keyusage usage;
krb5_const krb5_data *input;
krb5_data *output;
{
int i;
krb5_error_code ret;
krb5_keyblock *cksum_key = NULL;
struct krb5_enc_provider *enc = NULL;
KRB5_LOG0(KRB5_INFO, "krb5_dk_make_checksum() start");
for (i=0; i<krb5_enctypes_length; i++) {
if (krb5_enctypes_list[i].etype == key->enctype)
break;
}
if (i == krb5_enctypes_length) {
KRB5_LOG(KRB5_ERR, "krb5_ck_make_checksum bad enctype: %d",
key->enctype);
return(KRB5_BAD_ENCTYPE);
}
enc = (struct krb5_enc_provider *)krb5_enctypes_list[i].enc;
#ifdef _KERNEL
if (key->kef_key.ck_data == NULL &&
(ret = init_key_kef(krb5_enctypes_list[i].kef_cipher_mt,
(krb5_keyblock *)key)))
goto cleanup;
#endif
ret = derive_cksum_key(context, enc, key, usage, &cksum_key);
if (ret != 0)
goto cleanup;
#ifdef _KERNEL
if ((ret = krb5_hmac(context, (krb5_keyblock *)cksum_key,
input, output))) {
KRB5_LOG(KRB5_ERR, "krb5_hmac error: %0x", ret);
(void) memset(output->data, 0, output->length);
}
#else
if ((ret = krb5_hmac(context, hash, cksum_key, 1, input, output)) != 0) {
KRB5_LOG(KRB5_ERR, "krb5_hmac error: %0x", ret);
(void) memset(output->data, 0, output->length);
}
#endif
cleanup:
KRB5_LOG0(KRB5_INFO, "krb5_dk_make_checksum() end");
return(ret);
}