mirror of
https://github.com/samba-team/samba.git
synced 2025-01-13 13:18:06 +03:00
feb36dbebf
Since memcmp_const_time() doesn't act as an exact replacement for memcmp(), and its return value is only ever compared with zero, simplify it and emphasize the intention of checking equality by returning a bool instead. Signed-off-by: Joseph Sutton <josephsutton@catalyst.net.nz> Reviewed-by: Andrew Bartlett <abartlet@samba.org>
610 lines
15 KiB
C
610 lines
15 KiB
C
/*
|
|
* GSSAPI Security Extensions
|
|
* Krb5 helpers
|
|
* Copyright (C) Simo Sorce 2010.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "includes.h"
|
|
#include "smb_krb5.h"
|
|
#include "secrets.h"
|
|
#include "librpc/gen_ndr/secrets.h"
|
|
#include "gse_krb5.h"
|
|
#include "lib/param/loadparm.h"
|
|
#include "libads/kerberos_proto.h"
|
|
#include "lib/util/string_wrappers.h"
|
|
|
|
#ifdef HAVE_KRB5
|
|
|
|
static krb5_error_code flush_keytab(krb5_context krbctx, krb5_keytab keytab)
|
|
{
|
|
krb5_error_code ret;
|
|
krb5_kt_cursor kt_cursor;
|
|
krb5_keytab_entry kt_entry;
|
|
|
|
ZERO_STRUCT(kt_entry);
|
|
|
|
ret = krb5_kt_start_seq_get(krbctx, keytab, &kt_cursor);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
|
|
ret = krb5_kt_next_entry(krbctx, keytab, &kt_entry, &kt_cursor);
|
|
while (ret == 0) {
|
|
|
|
/* we need to close and reopen enumeration because we modify
|
|
* the keytab */
|
|
ret = krb5_kt_end_seq_get(krbctx, keytab, &kt_cursor);
|
|
if (ret != 0) {
|
|
DEBUG(1, (__location__ ": krb5_kt_end_seq_get() "
|
|
"failed (%s)\n", error_message(ret)));
|
|
goto out;
|
|
}
|
|
|
|
/* remove the entry */
|
|
ret = krb5_kt_remove_entry(krbctx, keytab, &kt_entry);
|
|
if (ret != 0) {
|
|
DEBUG(1, (__location__ ": krb5_kt_remove_entry() "
|
|
"failed (%s)\n", error_message(ret)));
|
|
goto out;
|
|
}
|
|
smb_krb5_kt_free_entry(krbctx, &kt_entry);
|
|
ZERO_STRUCT(kt_entry);
|
|
|
|
/* now reopen */
|
|
ret = krb5_kt_start_seq_get(krbctx, keytab, &kt_cursor);
|
|
if (ret != 0) {
|
|
DEBUG(1, (__location__ ": krb5_kt_start_seq() failed "
|
|
"(%s)\n", error_message(ret)));
|
|
goto out;
|
|
}
|
|
|
|
ret = krb5_kt_next_entry(krbctx, keytab,
|
|
&kt_entry, &kt_cursor);
|
|
}
|
|
|
|
if (ret != KRB5_KT_END && ret != ENOENT) {
|
|
DEBUG(1, (__location__ ": flushing keytab we got [%s]!\n",
|
|
error_message(ret)));
|
|
}
|
|
|
|
ret = krb5_kt_end_seq_get(krbctx, keytab, &kt_cursor);
|
|
if (ret != 0) {
|
|
DEBUG(1, (__location__ ": krb5_kt_end_seq_get() "
|
|
"failed (%s)\n", error_message(ret)));
|
|
goto out;
|
|
}
|
|
ret = 0;
|
|
|
|
out:
|
|
return ret;
|
|
}
|
|
|
|
static krb5_error_code fill_keytab_from_password(krb5_context krbctx,
|
|
krb5_keytab keytab,
|
|
krb5_principal princ,
|
|
krb5_kvno vno,
|
|
struct secrets_domain_info1_password *pw)
|
|
{
|
|
krb5_error_code ret;
|
|
krb5_enctype *enctypes;
|
|
uint16_t i;
|
|
|
|
ret = smb_krb5_get_allowed_etypes(krbctx, &enctypes);
|
|
if (ret) {
|
|
DEBUG(1, (__location__
|
|
": Can't determine permitted enctypes!\n"));
|
|
return ret;
|
|
}
|
|
|
|
for (i = 0; i < pw->num_keys; i++) {
|
|
krb5_keytab_entry kt_entry;
|
|
krb5_keyblock *key = NULL;
|
|
unsigned int ei;
|
|
bool found_etype = false;
|
|
|
|
for (ei=0; enctypes[ei] != 0; ei++) {
|
|
if ((uint32_t)enctypes[ei] != pw->keys[i].keytype) {
|
|
continue;
|
|
}
|
|
|
|
found_etype = true;
|
|
break;
|
|
}
|
|
|
|
if (!found_etype) {
|
|
continue;
|
|
}
|
|
|
|
ZERO_STRUCT(kt_entry);
|
|
kt_entry.principal = princ;
|
|
kt_entry.vno = vno;
|
|
|
|
key = KRB5_KT_KEY(&kt_entry);
|
|
KRB5_KEY_TYPE(key) = pw->keys[i].keytype;
|
|
KRB5_KEY_DATA(key) = pw->keys[i].value.data;
|
|
KRB5_KEY_LENGTH(key) = pw->keys[i].value.length;
|
|
|
|
ret = krb5_kt_add_entry(krbctx, keytab, &kt_entry);
|
|
if (ret) {
|
|
DEBUG(1, (__location__ ": Failed to add entry to "
|
|
"keytab for enctype %d (error: %s)\n",
|
|
(unsigned)pw->keys[i].keytype,
|
|
error_message(ret)));
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
ret = 0;
|
|
|
|
out:
|
|
SAFE_FREE(enctypes);
|
|
return ret;
|
|
}
|
|
|
|
#define SRV_MEM_KEYTAB_NAME "MEMORY:cifs_srv_keytab"
|
|
#define CLEARTEXT_PRIV_ENCTYPE -99
|
|
|
|
static krb5_error_code fill_mem_keytab_from_secrets(krb5_context krbctx,
|
|
krb5_keytab *keytab)
|
|
{
|
|
TALLOC_CTX *frame = talloc_stackframe();
|
|
krb5_error_code ret, ret2;
|
|
const char *domain = lp_workgroup();
|
|
struct secrets_domain_info1 *info = NULL;
|
|
const char *realm = NULL;
|
|
const DATA_BLOB *ct = NULL;
|
|
krb5_kt_cursor kt_cursor;
|
|
krb5_keytab_entry kt_entry;
|
|
krb5_principal princ = NULL;
|
|
krb5_kvno kvno = 0; /* FIXME: fetch current vno from KDC ? */
|
|
NTSTATUS status;
|
|
|
|
if (!secrets_init()) {
|
|
DEBUG(1, (__location__ ": secrets_init failed\n"));
|
|
TALLOC_FREE(frame);
|
|
return KRB5_CONFIG_CANTOPEN;
|
|
}
|
|
|
|
status = secrets_fetch_or_upgrade_domain_info(domain,
|
|
frame,
|
|
&info);
|
|
if (!NT_STATUS_IS_OK(status)) {
|
|
DBG_WARNING("secrets_fetch_or_upgrade_domain_info(%s) - %s\n",
|
|
domain, nt_errstr(status));
|
|
TALLOC_FREE(frame);
|
|
return KRB5_LIBOS_CANTREADPWD;
|
|
}
|
|
ct = &info->password->cleartext_blob;
|
|
|
|
if (info->domain_info.dns_domain.string != NULL) {
|
|
realm = strupper_talloc(frame,
|
|
info->domain_info.dns_domain.string);
|
|
if (realm == NULL) {
|
|
TALLOC_FREE(frame);
|
|
return ENOMEM;
|
|
}
|
|
}
|
|
|
|
ZERO_STRUCT(kt_entry);
|
|
ZERO_STRUCT(kt_cursor);
|
|
|
|
/* check if the keytab already has any entry */
|
|
ret = krb5_kt_start_seq_get(krbctx, *keytab, &kt_cursor);
|
|
if (ret != 0) {
|
|
goto out;
|
|
}
|
|
|
|
/* check if we have our special enctype used to hold
|
|
* the clear text password. If so, check it out so that
|
|
* we can verify if the keytab needs to be upgraded */
|
|
while ((ret = krb5_kt_next_entry(krbctx, *keytab,
|
|
&kt_entry, &kt_cursor)) == 0) {
|
|
if (smb_krb5_kt_get_enctype_from_entry(&kt_entry) ==
|
|
CLEARTEXT_PRIV_ENCTYPE) {
|
|
break;
|
|
}
|
|
smb_krb5_kt_free_entry(krbctx, &kt_entry);
|
|
ZERO_STRUCT(kt_entry);
|
|
}
|
|
|
|
ret2 = krb5_kt_end_seq_get(krbctx, *keytab, &kt_cursor);
|
|
if (ret2 != 0) {
|
|
ret = ret2;
|
|
DEBUG(1, (__location__ ": krb5_kt_end_seq_get() "
|
|
"failed (%s)\n", error_message(ret)));
|
|
goto out;
|
|
}
|
|
|
|
if (ret != 0 && ret != KRB5_KT_END && ret != ENOENT ) {
|
|
/* Error parsing keytab */
|
|
DEBUG(1, (__location__ ": Failed to parse memory "
|
|
"keytab!\n"));
|
|
goto out;
|
|
}
|
|
|
|
if (ret == 0) {
|
|
/* found private entry,
|
|
* check if keytab is up to date */
|
|
|
|
if ((ct->length == KRB5_KEY_LENGTH(KRB5_KT_KEY(&kt_entry))) &&
|
|
(mem_equal_const_time(KRB5_KEY_DATA(KRB5_KT_KEY(&kt_entry)),
|
|
ct->data, ct->length))) {
|
|
/* keytab is already up to date, return */
|
|
smb_krb5_kt_free_entry(krbctx, &kt_entry);
|
|
goto out;
|
|
}
|
|
|
|
smb_krb5_kt_free_entry(krbctx, &kt_entry);
|
|
ZERO_STRUCT(kt_entry);
|
|
|
|
|
|
/* flush keytab, we need to regen it */
|
|
ret = flush_keytab(krbctx, *keytab);
|
|
if (ret) {
|
|
DEBUG(1, (__location__ ": Failed to flush "
|
|
"memory keytab!\n"));
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
/* keytab is not up to date, fill it up */
|
|
|
|
ret = smb_krb5_make_principal(krbctx, &princ, realm,
|
|
info->account_name, NULL);
|
|
if (ret) {
|
|
DEBUG(1, (__location__ ": Failed to get host principal!\n"));
|
|
goto out;
|
|
}
|
|
|
|
ret = fill_keytab_from_password(krbctx, *keytab,
|
|
princ, kvno,
|
|
info->password);
|
|
if (ret) {
|
|
DBG_WARNING("fill_keytab_from_password() failed for "
|
|
"info->password.\n.");
|
|
goto out;
|
|
}
|
|
|
|
if (info->old_password != NULL) {
|
|
ret = fill_keytab_from_password(krbctx, *keytab,
|
|
princ, kvno - 1,
|
|
info->old_password);
|
|
if (ret) {
|
|
DBG_WARNING("fill_keytab_from_password() failed for "
|
|
"info->old_password.\n.");
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
if (info->older_password != NULL) {
|
|
ret = fill_keytab_from_password(krbctx, *keytab,
|
|
princ, kvno - 2,
|
|
info->older_password);
|
|
if (ret) {
|
|
DBG_WARNING("fill_keytab_from_password() failed for "
|
|
"info->older_password.\n.");
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
if (info->next_change != NULL) {
|
|
ret = fill_keytab_from_password(krbctx, *keytab,
|
|
princ, kvno - 3,
|
|
info->next_change->password);
|
|
if (ret) {
|
|
DBG_WARNING("fill_keytab_from_password() failed for "
|
|
"info->next_change->password.\n.");
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
/* add our private enctype + cleartext password so that we can
|
|
* update the keytab if secrets change later on */
|
|
ZERO_STRUCT(kt_entry);
|
|
kt_entry.principal = princ;
|
|
kt_entry.vno = 0;
|
|
|
|
KRB5_KEY_TYPE(KRB5_KT_KEY(&kt_entry)) = CLEARTEXT_PRIV_ENCTYPE;
|
|
KRB5_KEY_LENGTH(KRB5_KT_KEY(&kt_entry)) = ct->length;
|
|
KRB5_KEY_DATA(KRB5_KT_KEY(&kt_entry)) = ct->data;
|
|
|
|
ret = krb5_kt_add_entry(krbctx, *keytab, &kt_entry);
|
|
if (ret) {
|
|
DEBUG(1, (__location__ ": Failed to add entry to "
|
|
"keytab for private enctype (%d) (error: %s)\n",
|
|
CLEARTEXT_PRIV_ENCTYPE, error_message(ret)));
|
|
goto out;
|
|
}
|
|
|
|
ret = 0;
|
|
|
|
out:
|
|
|
|
if (princ) {
|
|
krb5_free_principal(krbctx, princ);
|
|
}
|
|
|
|
TALLOC_FREE(frame);
|
|
return ret;
|
|
}
|
|
|
|
static krb5_error_code fill_mem_keytab_from_system_keytab(krb5_context krbctx,
|
|
krb5_keytab *mkeytab)
|
|
{
|
|
krb5_error_code ret = 0;
|
|
krb5_keytab keytab = NULL;
|
|
krb5_kt_cursor kt_cursor = { 0, };
|
|
krb5_keytab_entry kt_entry = { 0, };
|
|
char *valid_princ_formats[7] = { NULL, NULL, NULL,
|
|
NULL, NULL, NULL, NULL };
|
|
char *entry_princ_s = NULL;
|
|
fstring my_name, my_fqdn;
|
|
unsigned i;
|
|
int err;
|
|
|
|
/* Generate the list of principal names which we expect
|
|
* clients might want to use for authenticating to the file
|
|
* service. We allow name$,{host,cifs}/{name,fqdn,name.REALM}. */
|
|
|
|
fstrcpy(my_name, lp_netbios_name());
|
|
|
|
my_fqdn[0] = '\0';
|
|
name_to_fqdn(my_fqdn, lp_netbios_name());
|
|
|
|
err = asprintf(&valid_princ_formats[0],
|
|
"%s$@%s", my_name, lp_realm());
|
|
if (err == -1) {
|
|
ret = ENOMEM;
|
|
goto out;
|
|
}
|
|
err = asprintf(&valid_princ_formats[1],
|
|
"host/%s@%s", my_name, lp_realm());
|
|
if (err == -1) {
|
|
ret = ENOMEM;
|
|
goto out;
|
|
}
|
|
err = asprintf(&valid_princ_formats[2],
|
|
"host/%s@%s", my_fqdn, lp_realm());
|
|
if (err == -1) {
|
|
ret = ENOMEM;
|
|
goto out;
|
|
}
|
|
err = asprintf(&valid_princ_formats[3],
|
|
"host/%s.%s@%s", my_name, lp_realm(), lp_realm());
|
|
if (err == -1) {
|
|
ret = ENOMEM;
|
|
goto out;
|
|
}
|
|
err = asprintf(&valid_princ_formats[4],
|
|
"cifs/%s@%s", my_name, lp_realm());
|
|
if (err == -1) {
|
|
ret = ENOMEM;
|
|
goto out;
|
|
}
|
|
err = asprintf(&valid_princ_formats[5],
|
|
"cifs/%s@%s", my_fqdn, lp_realm());
|
|
if (err == -1) {
|
|
ret = ENOMEM;
|
|
goto out;
|
|
}
|
|
err = asprintf(&valid_princ_formats[6],
|
|
"cifs/%s.%s@%s", my_name, lp_realm(), lp_realm());
|
|
if (err == -1) {
|
|
ret = ENOMEM;
|
|
goto out;
|
|
}
|
|
|
|
ret = smb_krb5_kt_open_relative(krbctx, NULL, false, &keytab);
|
|
if (ret) {
|
|
DEBUG(1, ("smb_krb5_kt_open failed (%s)\n",
|
|
error_message(ret)));
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* Iterate through the keytab. For each key, if the principal
|
|
* name case-insensitively matches one of the allowed formats,
|
|
* copy it to the memory keytab.
|
|
*/
|
|
|
|
ret = krb5_kt_start_seq_get(krbctx, keytab, &kt_cursor);
|
|
if (ret) {
|
|
DEBUG(1, (__location__ ": krb5_kt_start_seq_get failed (%s)\n",
|
|
error_message(ret)));
|
|
/*
|
|
* krb5_kt_start_seq_get() may leaves bogus data
|
|
* in kt_cursor. And we want to use the all_zero()
|
|
* logic below.
|
|
*
|
|
* See bug #10490
|
|
*/
|
|
ZERO_STRUCT(kt_cursor);
|
|
goto out;
|
|
}
|
|
|
|
while ((krb5_kt_next_entry(krbctx, keytab,
|
|
&kt_entry, &kt_cursor) == 0)) {
|
|
ret = smb_krb5_unparse_name(talloc_tos(), krbctx,
|
|
kt_entry.principal,
|
|
&entry_princ_s);
|
|
if (ret) {
|
|
DEBUG(1, (__location__ ": smb_krb5_unparse_name "
|
|
"failed (%s)\n", error_message(ret)));
|
|
goto out;
|
|
}
|
|
|
|
for (i = 0; i < ARRAY_SIZE(valid_princ_formats); i++) {
|
|
|
|
if (!strequal(entry_princ_s, valid_princ_formats[i])) {
|
|
continue;
|
|
}
|
|
|
|
ret = krb5_kt_add_entry(krbctx, *mkeytab, &kt_entry);
|
|
if (ret) {
|
|
DEBUG(1, (__location__ ": smb_krb5_unparse_name "
|
|
"failed (%s)\n", error_message(ret)));
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
/* Free the name we parsed. */
|
|
TALLOC_FREE(entry_princ_s);
|
|
|
|
/* Free the entry we just read. */
|
|
smb_krb5_kt_free_entry(krbctx, &kt_entry);
|
|
ZERO_STRUCT(kt_entry);
|
|
}
|
|
krb5_kt_end_seq_get(krbctx, keytab, &kt_cursor);
|
|
|
|
ZERO_STRUCT(kt_cursor);
|
|
|
|
out:
|
|
|
|
for (i = 0; i < ARRAY_SIZE(valid_princ_formats); i++) {
|
|
SAFE_FREE(valid_princ_formats[i]);
|
|
}
|
|
|
|
TALLOC_FREE(entry_princ_s);
|
|
|
|
if (!all_zero((uint8_t *)&kt_entry, sizeof(kt_entry))) {
|
|
smb_krb5_kt_free_entry(krbctx, &kt_entry);
|
|
}
|
|
|
|
if (!all_zero((uint8_t *)&kt_cursor, sizeof(kt_cursor)) && keytab) {
|
|
krb5_kt_end_seq_get(krbctx, keytab, &kt_cursor);
|
|
}
|
|
|
|
if (keytab) {
|
|
krb5_kt_close(krbctx, keytab);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static krb5_error_code fill_mem_keytab_from_dedicated_keytab(krb5_context krbctx,
|
|
krb5_keytab *mkeytab)
|
|
{
|
|
krb5_error_code ret = 0;
|
|
krb5_keytab keytab = NULL;
|
|
krb5_kt_cursor kt_cursor;
|
|
krb5_keytab_entry kt_entry;
|
|
|
|
ret = smb_krb5_kt_open(krbctx, lp_dedicated_keytab_file(),
|
|
false, &keytab);
|
|
if (ret) {
|
|
DEBUG(1, ("smb_krb5_kt_open of %s failed (%s)\n",
|
|
lp_dedicated_keytab_file(),
|
|
error_message(ret)));
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Copy the dedicated keyab to our in-memory keytab.
|
|
*/
|
|
|
|
ret = krb5_kt_start_seq_get(krbctx, keytab, &kt_cursor);
|
|
if (ret) {
|
|
DEBUG(1, (__location__ ": krb5_kt_start_seq_get on %s "
|
|
"failed (%s)\n",
|
|
lp_dedicated_keytab_file(),
|
|
error_message(ret)));
|
|
goto out;
|
|
}
|
|
|
|
while ((krb5_kt_next_entry(krbctx, keytab,
|
|
&kt_entry, &kt_cursor) == 0)) {
|
|
|
|
ret = krb5_kt_add_entry(krbctx, *mkeytab, &kt_entry);
|
|
|
|
/* Free the entry we just read. */
|
|
smb_krb5_kt_free_entry(krbctx, &kt_entry);
|
|
|
|
if (ret) {
|
|
DEBUG(1, (__location__ ": smb_krb5_unparse_name "
|
|
"failed (%s)\n", error_message(ret)));
|
|
break;
|
|
}
|
|
}
|
|
krb5_kt_end_seq_get(krbctx, keytab, &kt_cursor);
|
|
|
|
out:
|
|
|
|
krb5_kt_close(krbctx, keytab);
|
|
|
|
return ret;
|
|
}
|
|
|
|
krb5_error_code gse_krb5_get_server_keytab(krb5_context krbctx,
|
|
krb5_keytab *keytab)
|
|
{
|
|
krb5_error_code ret = 0;
|
|
krb5_error_code ret1 = 0;
|
|
krb5_error_code ret2 = 0;
|
|
|
|
*keytab = NULL;
|
|
|
|
/* create memory keytab */
|
|
ret = krb5_kt_resolve(krbctx, SRV_MEM_KEYTAB_NAME, keytab);
|
|
if (ret) {
|
|
DEBUG(1, (__location__ ": Failed to get memory "
|
|
"keytab!\n"));
|
|
return ret;
|
|
}
|
|
|
|
switch (lp_kerberos_method()) {
|
|
default:
|
|
case KERBEROS_VERIFY_SECRETS:
|
|
ret = fill_mem_keytab_from_secrets(krbctx, keytab);
|
|
break;
|
|
case KERBEROS_VERIFY_SYSTEM_KEYTAB:
|
|
ret = fill_mem_keytab_from_system_keytab(krbctx, keytab);
|
|
break;
|
|
case KERBEROS_VERIFY_DEDICATED_KEYTAB:
|
|
/* just use whatever keytab is configured */
|
|
ret = fill_mem_keytab_from_dedicated_keytab(krbctx, keytab);
|
|
break;
|
|
case KERBEROS_VERIFY_SECRETS_AND_KEYTAB:
|
|
ret1 = fill_mem_keytab_from_secrets(krbctx, keytab);
|
|
if (ret1) {
|
|
DEBUG(3, (__location__ ": Warning! Unable to set mem "
|
|
"keytab from secrets!\n"));
|
|
}
|
|
/* Now append system keytab keys too */
|
|
ret2 = fill_mem_keytab_from_system_keytab(krbctx, keytab);
|
|
if (ret2) {
|
|
DEBUG(3, (__location__ ": Warning! Unable to set mem "
|
|
"keytab from system keytab!\n"));
|
|
}
|
|
if (ret1 == 0 || ret2 == 0) {
|
|
ret = 0;
|
|
} else {
|
|
ret = ret1;
|
|
}
|
|
break;
|
|
}
|
|
|
|
if (ret) {
|
|
krb5_kt_close(krbctx, *keytab);
|
|
*keytab = NULL;
|
|
DEBUG(1,("%s: Error! Unable to set mem keytab - %d\n",
|
|
__location__, ret));
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
#endif /* HAVE_KRB5 */
|