22#include <winpr/config.h>
23#include <winpr/library.h>
32#include <winpr/assert.h>
33#include <winpr/cast.h>
34#include <winpr/asn1.h>
36#include <winpr/interlocked.h>
37#include <winpr/sspi.h>
38#include <winpr/print.h>
39#include <winpr/tchar.h>
40#include <winpr/sysinfo.h>
41#include <winpr/registry.h>
42#include <winpr/endian.h>
43#include <winpr/crypto.h>
44#include <winpr/path.h>
45#include <winpr/wtypes.h>
46#include <winpr/winsock.h>
47#include <winpr/schannel.h>
48#include <winpr/secapi.h>
57#ifdef WITH_KRB5_HEIMDAL
59#include <krb5-protos.h>
71 "Kerberos Security Package"
74static WCHAR KERBEROS_SecPkgInfoW_NameBuffer[32] = WINPR_C_ARRAY_INIT;
75static WCHAR KERBEROS_SecPkgInfoW_CommentBuffer[32] = WINPR_C_ARRAY_INIT;
82 KERBEROS_SecPkgInfoW_NameBuffer,
83 KERBEROS_SecPkgInfoW_CommentBuffer
87#define TAG WINPR_TAG("sspi.Kerberos")
94 KERBEROS_STATE_INITIAL,
95 KERBEROS_STATE_TGT_REQ,
96 KERBEROS_STATE_TGT_REP,
97 KERBEROS_STATE_AP_REQ,
98 KERBEROS_STATE_AP_REP,
102typedef struct KRB_CREDENTIALS_st
104 volatile LONG refCount;
109 krb5_keytab client_keytab;
115 enum KERBEROS_STATE state;
116 KRB_CREDENTIALS* credentials;
117 krb5_auth_context auth_ctx;
127static const WinPrAsn1_OID kerberos_OID = { 9, (
void*)
"\x2a\x86\x48\x86\xf7\x12\x01\x02\x02" };
129 (
void*)
"\x2a\x86\x48\x86\xf7\x12\x01\x02\x02\x03" };
130krb5_error_code kerberos_log_msg(krb5_context ctx, krb5_error_code code,
const char* what,
131 const char* file,
const char* fkt,
size_t line)
140 const DWORD level = WLOG_ERROR;
142 wLog* log = WLog_Get(TAG);
143 if (WLog_IsLevelActive(log, level))
145 const char* msg = krb5_get_error_message(ctx, code);
146 WLog_PrintTextMessage(log, level, line, file, fkt,
"%s (%s [%d])", what, msg, code);
147 krb5_free_error_message(ctx, msg);
155void krb_log_context_encryption(krb5_context ctx, krb5_principal princ)
157#if !defined(WITH_KRB5_HEIMDAL)
158 typedef krb5_error_code KRB5_CALLCONV (*krb5_get_etype_info_fn)(
159 krb5_context context, krb5_principal principal, krb5_get_init_creds_opt* opt,
160 krb5_enctype* enctype_out, krb5_data* salt_out, krb5_data* s2kparams_out);
162 krb5_get_etype_info_fn fn =
163 GetProcAddressAs(
nullptr,
"krb5_get_etype_info", krb5_get_etype_info_fn);
167 krb5_get_init_creds_opt opt = WINPR_C_ARRAY_INIT;
168 krb5_enctype enctype = 0;
169 krb5_data salt = WINPR_C_ARRAY_INIT;
170 krb5_data s2kparam = WINPR_C_ARRAY_INIT;
171 char buffer[128] = WINPR_C_ARRAY_INIT;
172 krb5_error_code rv = krb_log_exec(fn, ctx, princ, &opt, &enctype, &salt, &s2kparam);
173 krb5_enctype_to_string(enctype, buffer,
sizeof(buffer));
174 const char* msg = krb5_get_error_message(ctx, rv);
176 WLog_DBG(TAG,
"[%s] enctype=%s, salt[%u]=%s, s2kparam[%u]=%s", msg, buffer, salt.length,
177 salt.data, s2kparam.length, s2kparam.data);
179 krb5_free_data_contents(ctx, &salt);
180 krb5_free_data_contents(ctx, &s2kparam);
181 krb5_free_error_message(ctx, msg);
187 "kerberos implementation does not support 'krb5_get_etype_info', not displaying "
188 "encryption information");
192static void credentials_unref(KRB_CREDENTIALS* credentials);
194static void kerberos_ContextFree(KRB_CONTEXT* ctx, BOOL allocated)
199 free(ctx->targetHost);
200 ctx->targetHost =
nullptr;
202 if (ctx->credentials)
204 krb5_context krbctx = ctx->credentials->ctx;
208 krb5_auth_con_free(krbctx, ctx->auth_ctx);
210 krb5glue_keys_free(krbctx, &ctx->keyset);
213 credentials_unref(ctx->credentials);
220static KRB_CONTEXT* kerberos_ContextNew(KRB_CREDENTIALS* credentials)
222 KRB_CONTEXT* context =
nullptr;
224 context = (KRB_CONTEXT*)calloc(1,
sizeof(KRB_CONTEXT));
228 context->credentials = credentials;
229 InterlockedIncrement(&credentials->refCount);
233static krb5_error_code krb5_prompter(krb5_context context,
void* data,
234 WINPR_ATTR_UNUSED
const char* name,
235 WINPR_ATTR_UNUSED
const char* banner,
int num_prompts,
236 krb5_prompt prompts[])
238 for (
int i = 0; i < num_prompts; i++)
240 krb5_prompt_type type = krb5glue_get_prompt_type(context, prompts, i);
241 if (type && (type == KRB5_PROMPT_TYPE_PREAUTH || type == KRB5_PROMPT_TYPE_PASSWORD) && data)
243 prompts[i].reply->data = _strdup((
const char*)data);
245 const size_t len = strlen((
const char*)data);
246 if (len > UINT32_MAX)
247 return KRB5KRB_ERR_GENERIC;
248 prompts[i].reply->length = (UINT32)len;
254WINPR_ATTR_NODISCARD
static inline krb5glue_key get_key(
struct krb5glue_keyset* keyset)
256 return keyset->acceptor_key ? keyset->acceptor_key
257 : keyset->initiator_key ? keyset->initiator_key
258 : keyset->session_key;
261static BOOL isValidIPv4(
const char* ipAddress)
263 struct sockaddr_in sa = WINPR_C_ARRAY_INIT;
264 int result = inet_pton(AF_INET, ipAddress, &(sa.sin_addr));
268static BOOL isValidIPv6(
const char* ipAddress)
270 struct sockaddr_in6 sa = WINPR_C_ARRAY_INIT;
271 int result = inet_pton(AF_INET6, ipAddress, &(sa.sin6_addr));
275static BOOL isValidIP(
const char* ipAddress)
277 return isValidIPv4(ipAddress) || isValidIPv6(ipAddress);
280#if defined(WITH_KRB5_MIT)
281WINPR_ATTR_MALLOC(free, 1)
283static
char* get_realm_name(krb5_data realm,
size_t* plen)
287 if ((realm.length <= 0) || (!realm.data))
290 char* name =
nullptr;
291 (void)winpr_asprintf(&name, plen,
"krbtgt/%*s@%*s", realm.length, realm.data, realm.length,
295#elif defined(WITH_KRB5_HEIMDAL)
296WINPR_ATTR_MALLOC(free, 1)
298static
char* get_realm_name(Realm realm,
size_t* plen)
305 char* name =
nullptr;
306 (void)winpr_asprintf(&name, plen,
"krbtgt/%s@%s", realm, realm);
311static int build_krbtgt(krb5_context ctx, krb5_principal principal, krb5_principal* ptarget)
315 krb5_error_code rv = KRB5_CC_NOMEM;
317 char* name = get_realm_name(principal->realm, &len);
318 if (!name || (len == 0))
322 krb5_principal target = WINPR_C_ARRAY_INIT;
323 rv = krb5_parse_name(ctx, name, &target);
333static SECURITY_STATUS SEC_ENTRY kerberos_AcquireCredentialsHandleA(
334 WINPR_ATTR_UNUSED SEC_CHAR* pszPrincipal, WINPR_ATTR_UNUSED SEC_CHAR* pszPackage,
335 WINPR_ATTR_UNUSED ULONG fCredentialUse, WINPR_ATTR_UNUSED
void* pvLogonID,
336 WINPR_ATTR_UNUSED
void* pAuthData, WINPR_ATTR_UNUSED SEC_GET_KEY_FN pGetKeyFn,
337 WINPR_ATTR_UNUSED
void* pvGetKeyArgument, WINPR_ATTR_UNUSED
PCredHandle phCredential,
342 KRB_CREDENTIALS* credentials =
nullptr;
343 krb5_context ctx =
nullptr;
344 krb5_ccache ccache =
nullptr;
345 krb5_keytab keytab =
nullptr;
346 krb5_principal principal =
nullptr;
347 char* domain =
nullptr;
348 char* username =
nullptr;
349 char* password =
nullptr;
350 BOOL own_ccache = FALSE;
351 const char*
const default_ccache_type =
"MEMORY";
355 UINT32 identityFlags = sspi_GetAuthIdentityFlags(pAuthData);
357 if (identityFlags & SEC_WINNT_AUTH_IDENTITY_EXTENDED)
363 WLog_ERR(TAG,
"Failed to copy auth identity fields");
368 pszPrincipal = username;
371 if (krb_log_exec_ptr(krb5_init_context, &ctx))
376 char* udomain = _strdup(domain);
382 krb5_error_code rv = krb_log_exec(krb5_set_default_realm, ctx, udomain);
391 char* cpszPrincipal = _strdup(pszPrincipal);
396 char* p = strchr(cpszPrincipal,
'@');
400 krb5_error_code rv = krb_log_exec(krb5_parse_name, ctx, cpszPrincipal, &principal);
405 WINPR_ASSERT(principal);
408 if (krb_settings && krb_settings->cache)
410 if ((krb_log_exec(krb5_cc_set_default_name, ctx, krb_settings->cache)))
419 if (krb5_cc_cache_match(ctx, principal, &ccache) == KRB5_CC_NOTFOUND)
423 if (krb_log_exec(krb5_cc_new_unique, ctx, default_ccache_type,
nullptr, &ccache))
428 if (krb_log_exec(krb5_cc_resolve, ctx, krb_settings->cache, &ccache))
432 if (krb_log_exec(krb5_cc_initialize, ctx, ccache, principal))
437 if (krb_log_exec(krb5_cc_default, ctx, &ccache))
441 WINPR_ASSERT(ccache);
443 else if (fCredentialUse & SECPKG_CRED_OUTBOUND)
446 if (krb_log_exec(krb5_cc_default, ctx, &ccache))
448 if (krb_log_exec(krb5_cc_get_principal, ctx, ccache, &principal))
450 WINPR_ASSERT(ccache);
457 if (krb_log_exec(krb5_cc_new_unique, ctx, default_ccache_type,
nullptr, &ccache))
462 if (krb_log_exec(krb5_cc_resolve, ctx, krb_settings->cache, &ccache))
465 WINPR_ASSERT(ccache);
468 if (krb_settings && krb_settings->keytab)
470 if (krb_log_exec(krb5_kt_resolve, ctx, krb_settings->keytab, &keytab))
475 if (fCredentialUse & SECPKG_CRED_INBOUND)
476 if (krb_log_exec(krb5_kt_default, ctx, &keytab))
481 if (fCredentialUse & SECPKG_CRED_OUTBOUND)
483 krb5_creds creds = WINPR_C_ARRAY_INIT;
484 krb5_creds matchCreds = WINPR_C_ARRAY_INIT;
485 krb5_flags matchFlags = KRB5_TC_MATCH_TIMES;
487 krb5_timeofday(ctx, &matchCreds.times.endtime);
488 matchCreds.times.endtime += 60;
489 matchCreds.client = principal;
491 WINPR_ASSERT(principal);
493 WINPR_ASSERT(ccache);
494 if (krb_log_exec(build_krbtgt, ctx, principal, &matchCreds.server))
497 int rv = krb5_cc_retrieve_cred(ctx, ccache, matchFlags, &matchCreds, &creds);
498 krb5_free_principal(ctx, matchCreds.server);
499 krb5_free_cred_contents(ctx, &creds);
502 if (krb_log_exec(krb5glue_get_init_creds, ctx, principal, ccache, krb5_prompter,
503 password, krb_settings))
508 credentials = calloc(1,
sizeof(KRB_CREDENTIALS));
511 credentials->refCount = 1;
512 credentials->ctx = ctx;
513 credentials->ccache = ccache;
514 credentials->keytab = keytab;
515 credentials->own_ccache = own_ccache;
524 krb5_free_principal(ctx, principal);
532 krb5_cc_destroy(ctx, ccache);
534 krb5_cc_close(ctx, ccache);
537 krb5_kt_close(ctx, keytab);
539 krb5_free_context(ctx);
546 sspi_SecureHandleSetLowerPointer(phCredential, (
void*)credentials);
547 sspi_SecureHandleSetUpperPointer(phCredential, (
void*)KERBEROS_SSP_NAME);
551 return SEC_E_NO_CREDENTIALS;
553 return SEC_E_UNSUPPORTED_FUNCTION;
557static SECURITY_STATUS SEC_ENTRY kerberos_AcquireCredentialsHandleW(
558 SEC_WCHAR* pszPrincipal, SEC_WCHAR* pszPackage, ULONG fCredentialUse,
void* pvLogonID,
559 void* pAuthData, SEC_GET_KEY_FN pGetKeyFn,
void* pvGetKeyArgument,
PCredHandle phCredential,
562 SECURITY_STATUS status = SEC_E_INSUFFICIENT_MEMORY;
563 char* principal =
nullptr;
564 char*
package = nullptr;
568 principal = ConvertWCharToUtf8Alloc(pszPrincipal,
nullptr);
574 package = ConvertWCharToUtf8Alloc(pszPackage, nullptr);
580 kerberos_AcquireCredentialsHandleA(principal, package, fCredentialUse, pvLogonID, pAuthData,
581 pGetKeyFn, pvGetKeyArgument, phCredential, ptsExpiry);
591static void credentials_unref(KRB_CREDENTIALS* credentials)
593 WINPR_ASSERT(credentials);
595 if (InterlockedDecrement(&credentials->refCount))
598 free(credentials->kdc_url);
600 if (credentials->ccache)
602 if (credentials->own_ccache)
603 krb5_cc_destroy(credentials->ctx, credentials->ccache);
605 krb5_cc_close(credentials->ctx, credentials->ccache);
607 if (credentials->keytab)
608 krb5_kt_close(credentials->ctx, credentials->keytab);
610 krb5_free_context(credentials->ctx);
615static SECURITY_STATUS
616 SEC_ENTRY kerberos_FreeCredentialsHandle(WINPR_ATTR_UNUSED
PCredHandle phCredential)
619 KRB_CREDENTIALS* credentials = sspi_SecureHandleGetLowerPointer(phCredential);
620 sspi_SecureHandleInvalidate(phCredential);
622 return SEC_E_INVALID_HANDLE;
624 credentials_unref(credentials);
628 return SEC_E_UNSUPPORTED_FUNCTION;
632static SECURITY_STATUS SEC_ENTRY kerberos_QueryCredentialsAttributesW(
633 WINPR_ATTR_UNUSED
PCredHandle phCredential, WINPR_ATTR_UNUSED ULONG ulAttribute,
634 WINPR_ATTR_UNUSED
void* pBuffer)
639 case SECPKG_CRED_ATTR_NAMES:
642 WLog_ERR(TAG,
"TODO: QueryCredentialsAttributesW, implement ulAttribute=%08" PRIx32,
644 return SEC_E_UNSUPPORTED_FUNCTION;
648 return SEC_E_UNSUPPORTED_FUNCTION;
652static SECURITY_STATUS SEC_ENTRY kerberos_QueryCredentialsAttributesA(
PCredHandle phCredential,
656 return kerberos_QueryCredentialsAttributesW(phCredential, ulAttribute, pBuffer);
661static BOOL kerberos_mk_tgt_token(
SecBuffer* buf,
int msg_type,
char* sname,
char* host,
662 const krb5_data* ticket)
664 WinPrAsn1Encoder* enc =
nullptr;
673 if (msg_type != KRB_TGT_REQ && msg_type != KRB_TGT_REP)
675 if (msg_type == KRB_TGT_REP && !ticket)
678 enc = WinPrAsn1Encoder_New(WINPR_ASN1_DER);
683 if (!WinPrAsn1EncSeqContainer(enc))
687 if (!WinPrAsn1EncContextualInteger(enc, 0, 5))
691 if (!WinPrAsn1EncContextualInteger(enc, 1, msg_type))
694 if (msg_type == KRB_TGT_REQ && sname)
697 if (!WinPrAsn1EncContextualSeqContainer(enc, 2))
701 if (!WinPrAsn1EncContextualInteger(enc, 0, KRB5_NT_SRV_HST))
705 if (!WinPrAsn1EncContextualSeqContainer(enc, 1))
708 if (!WinPrAsn1EncGeneralString(enc, sname))
711 if (host && !WinPrAsn1EncGeneralString(enc, host))
714 if (!WinPrAsn1EncEndContainer(enc) || !WinPrAsn1EncEndContainer(enc))
717 else if (msg_type == KRB_TGT_REP)
720 data.data = (BYTE*)ticket->data;
721 data.len = ticket->length;
722 if (!WinPrAsn1EncContextualRawContent(enc, 2, &data))
726 if (!WinPrAsn1EncEndContainer(enc))
729 if (!WinPrAsn1EncStreamSize(enc, &len) || len > buf->cbBuffer)
732 Stream_StaticInit(&s, buf->pvBuffer, len);
733 if (!WinPrAsn1EncToStream(enc, &s))
736 token.data = buf->pvBuffer;
737 token.length = (UINT)len;
738 if (sspi_gss_wrap_token(buf, &kerberos_u2u_OID,
739 msg_type == KRB_TGT_REQ ? TOK_ID_TGT_REQ : TOK_ID_TGT_REP, &token))
743 WinPrAsn1Encoder_Free(&enc);
747static BOOL append(
char* dst,
size_t dstSize,
const char* src)
749 const size_t dlen = strnlen(dst, dstSize);
750 const size_t slen = strlen(src);
751 if (dlen + slen >= dstSize)
753 if (!strncat(dst, src, dstSize - dlen))
758static BOOL kerberos_rd_tgt_req_tag2(
WinPrAsn1Decoder* dec,
char* buf,
size_t len)
764 if (!WinPrAsn1DecReadSequence(dec, &seq))
771 WinPrAsn1_INTEGER val = 0;
772 if (!WinPrAsn1DecReadContextualInteger(&seq, 0, &error, &val))
777 if (!WinPrAsn1DecReadContextualSequence(&seq, 1, &error, dec))
782 WinPrAsn1_tag tag = 0;
784 while (WinPrAsn1DecPeekTag(dec, &tag))
786 BOOL success = FALSE;
787 char* lstr =
nullptr;
788 if (!WinPrAsn1DecReadGeneralString(dec, &lstr))
793 if (!append(buf, len,
"/"))
798 if (!append(buf, len, lstr))
814static BOOL kerberos_rd_tgt_req_tag3(
WinPrAsn1Decoder* dec,
char* buf,
size_t len)
818 WinPrAsn1_STRING str =
nullptr;
819 if (!WinPrAsn1DecReadGeneralString(dec, &str))
822 if (!append(buf, len,
"@"))
824 if (!append(buf, len, str))
841 wStream s = WinPrAsn1DecGetStream(dec);
842 const size_t len = Stream_Length(&s);
847 WinPrAsn1_tagId tag = 0;
848 if (WinPrAsn1DecReadContextualTag(dec, &tag, &dec2) == 0)
851 char* buf = calloc(len + 1,
sizeof(
char));
859 BOOL checkForTag3 = TRUE;
862 rc = kerberos_rd_tgt_req_tag2(&dec2, buf, len);
865 const size_t res = WinPrAsn1DecReadContextualTag(dec, &tag, dec);
867 checkForTag3 = FALSE;
874 rc = kerberos_rd_tgt_req_tag3(&dec2, buf, len);
893 WinPrAsn1_tagId tag = 0;
894 if (WinPrAsn1DecReadContextualTag(dec, &tag, &asnTicket) == 0)
900 wStream s = WinPrAsn1DecGetStream(&asnTicket);
901 ticket->data = Stream_BufferAs(&s,
char);
903 const size_t len = Stream_Length(&s);
904 if (len > UINT32_MAX)
906 ticket->length = (UINT32)len;
910static BOOL kerberos_rd_tgt_token(
const sspi_gss_data* token,
char** target, krb5_data* ticket)
913 WinPrAsn1_INTEGER val = 0;
921 WinPrAsn1Decoder_InitMem(&der, WINPR_ASN1_DER, (BYTE*)token->data, token->length);
925 if (!WinPrAsn1DecReadSequence(&der, &seq))
929 if (!WinPrAsn1DecReadContextualInteger(&seq, 0, &error, &val) || val != 5)
933 if (!WinPrAsn1DecReadContextualInteger(&seq, 1, &error, &val))
939 return kerberos_rd_tgt_req(&seq, target);
941 return kerberos_rd_tgt_rep(&seq, ticket);
948static BOOL kerberos_hash_channel_bindings(WINPR_DIGEST_CTX* md5,
SEC_CHANNEL_BINDINGS* bindings)
952 winpr_Data_Write_UINT32(buf, bindings->dwInitiatorAddrType);
953 if (!winpr_Digest_Update(md5, buf, 4))
956 winpr_Data_Write_UINT32(buf, bindings->cbInitiatorLength);
957 if (!winpr_Digest_Update(md5, buf, 4))
960 if (bindings->cbInitiatorLength &&
961 !winpr_Digest_Update(md5, (BYTE*)bindings + bindings->dwInitiatorOffset,
962 bindings->cbInitiatorLength))
965 winpr_Data_Write_UINT32(buf, bindings->dwAcceptorAddrType);
966 if (!winpr_Digest_Update(md5, buf, 4))
969 winpr_Data_Write_UINT32(buf, bindings->cbAcceptorLength);
970 if (!winpr_Digest_Update(md5, buf, 4))
973 if (bindings->cbAcceptorLength &&
974 !winpr_Digest_Update(md5, (BYTE*)bindings + bindings->dwAcceptorOffset,
975 bindings->cbAcceptorLength))
978 winpr_Data_Write_UINT32(buf, bindings->cbApplicationDataLength);
979 if (!winpr_Digest_Update(md5, buf, 4))
982 if (bindings->cbApplicationDataLength &&
983 !winpr_Digest_Update(md5, (BYTE*)bindings + bindings->dwApplicationDataOffset,
984 bindings->cbApplicationDataLength))
992static SECURITY_STATUS SEC_ENTRY kerberos_InitializeSecurityContextA(
994 WINPR_ATTR_UNUSED SEC_CHAR* pszTargetName, WINPR_ATTR_UNUSED ULONG fContextReq,
995 WINPR_ATTR_UNUSED ULONG Reserved1, WINPR_ATTR_UNUSED ULONG TargetDataRep,
996 WINPR_ATTR_UNUSED
PSecBufferDesc pInput, WINPR_ATTR_UNUSED ULONG Reserved2,
998 WINPR_ATTR_UNUSED ULONG* pfContextAttr, WINPR_ATTR_UNUSED
PTimeStamp ptsExpiry)
1004 WINPR_DIGEST_CTX* md5 =
nullptr;
1005 char* target =
nullptr;
1006 char* sname =
nullptr;
1007 char* host =
nullptr;
1008 krb5_data input_token = WINPR_C_ARRAY_INIT;
1009 krb5_data output_token = WINPR_C_ARRAY_INIT;
1010 SECURITY_STATUS status = SEC_E_INTERNAL_ERROR;
1012 uint16_t tok_id = 0;
1013 krb5_ap_rep_enc_part* reply =
nullptr;
1014 krb5_flags ap_flags = AP_OPTS_USE_SUBKEY;
1015 char cksum_contents[24] = WINPR_C_ARRAY_INIT;
1016 krb5_data cksum = WINPR_C_ARRAY_INIT;
1017 krb5_creds in_creds = WINPR_C_ARRAY_INIT;
1018 krb5_creds* creds =
nullptr;
1019 BOOL isNewContext = FALSE;
1020 KRB_CONTEXT* context =
nullptr;
1021 KRB_CREDENTIALS* credentials = sspi_SecureHandleGetLowerPointer(phCredential);
1024 if (phContext && !phContext->dwLower && !phContext->dwUpper)
1025 return SEC_E_INVALID_HANDLE;
1027 context = sspi_SecureHandleGetLowerPointer(phContext);
1030 return SEC_E_NO_CREDENTIALS;
1034 input_buffer = sspi_FindSecBuffer(pInput, SECBUFFER_TOKEN);
1035 bindings_buffer = sspi_FindSecBuffer(pInput, SECBUFFER_CHANNEL_BINDINGS);
1038 output_buffer = sspi_FindSecBuffer(pOutput, SECBUFFER_TOKEN);
1040 if (fContextReq & ISC_REQ_MUTUAL_AUTH)
1041 ap_flags |= AP_OPTS_MUTUAL_REQUIRED;
1043 if (fContextReq & ISC_REQ_USE_SESSION_KEY)
1044 ap_flags |= AP_OPTS_USE_SESSION_KEY;
1049 target = _strdup(pszTargetName);
1052 status = SEC_E_INSUFFICIENT_MEMORY;
1055 host = strchr(target,
'/');
1063 if (isValidIP(host))
1065 status = SEC_E_NO_CREDENTIALS;
1072 context = kerberos_ContextNew(credentials);
1075 status = SEC_E_INSUFFICIENT_MEMORY;
1079 isNewContext = TRUE;
1082 context->targetHost = _strdup(host);
1083 if (!context->targetHost)
1085 status = SEC_E_INSUFFICIENT_MEMORY;
1089 if (fContextReq & ISC_REQ_USE_SESSION_KEY)
1091 context->state = KERBEROS_STATE_TGT_REQ;
1092 context->u2u = TRUE;
1095 context->state = KERBEROS_STATE_AP_REQ;
1099 if (!input_buffer || !sspi_gss_unwrap_token(input_buffer, &oid, &tok_id, &input_token))
1101 if ((context->u2u && !sspi_gss_oid_compare(&oid, &kerberos_u2u_OID)) ||
1102 (!context->u2u && !sspi_gss_oid_compare(&oid, &kerberos_OID)))
1107 context->flags |= (fContextReq & 0x1F);
1108 if ((fContextReq & ISC_REQ_INTEGRITY) && !(fContextReq & ISC_REQ_NO_INTEGRITY))
1109 context->flags |= SSPI_GSS_C_INTEG_FLAG;
1111 switch (context->state)
1113 case KERBEROS_STATE_TGT_REQ:
1115 if (!kerberos_mk_tgt_token(output_buffer, KRB_TGT_REQ, sname, host,
nullptr))
1118 context->state = KERBEROS_STATE_TGT_REP;
1119 status = SEC_I_CONTINUE_NEEDED;
1122 case KERBEROS_STATE_TGT_REP:
1124 if (tok_id != TOK_ID_TGT_REP)
1127 if (!kerberos_rd_tgt_token(&input_token,
nullptr, &in_creds.second_ticket))
1134 case KERBEROS_STATE_AP_REQ:
1137 if (krb_log_exec(krb5_auth_con_init, credentials->ctx, &context->auth_ctx))
1139 if (krb_log_exec(krb5_auth_con_setflags, credentials->ctx, context->auth_ctx,
1140 KRB5_AUTH_CONTEXT_DO_SEQUENCE | KRB5_AUTH_CONTEXT_USE_SUBKEY))
1142 if (krb_log_exec(krb5glue_auth_con_set_cksumtype, credentials->ctx, context->auth_ctx,
1147 if (krb_log_exec(krb5_sname_to_principal, credentials->ctx, host, sname,
1148 KRB5_NT_SRV_HST, &in_creds.server))
1151 if (krb_log_exec(krb5_cc_get_principal, credentials->ctx, credentials->ccache,
1154 status = SEC_E_WRONG_PRINCIPAL;
1158 if (krb_log_exec(krb5_get_credentials, credentials->ctx,
1159 context->u2u ? KRB5_GC_USER_USER : 0, credentials->ccache, &in_creds,
1162 status = SEC_E_NO_CREDENTIALS;
1167 cksum.data = cksum_contents;
1168 cksum.length =
sizeof(cksum_contents);
1169 winpr_Data_Write_UINT32(cksum_contents, 16);
1170 winpr_Data_Write_UINT32((cksum_contents + 20), context->flags);
1172 if (bindings_buffer)
1178 (bindings->cbInitiatorLength + bindings->dwInitiatorOffset) >
1179 bindings_buffer->cbBuffer ||
1180 (bindings->cbAcceptorLength + bindings->dwAcceptorOffset) >
1181 bindings_buffer->cbBuffer ||
1182 (bindings->cbApplicationDataLength + bindings->dwApplicationDataOffset) >
1183 bindings_buffer->cbBuffer)
1185 status = SEC_E_BAD_BINDINGS;
1189 md5 = winpr_Digest_New();
1193 if (!winpr_Digest_Init(md5, WINPR_MD_MD5))
1196 if (!kerberos_hash_channel_bindings(md5, bindings))
1199 if (!winpr_Digest_Final(md5, (BYTE*)cksum_contents + 4, 16))
1204 if (krb_log_exec(krb5_mk_req_extended, credentials->ctx, &context->auth_ctx, ap_flags,
1205 &cksum, creds, &output_token))
1208 if (!sspi_gss_wrap_token(output_buffer,
1209 context->u2u ? &kerberos_u2u_OID : &kerberos_OID,
1210 TOK_ID_AP_REQ, &output_token))
1213 if (context->flags & SSPI_GSS_C_SEQUENCE_FLAG)
1215 if (krb_log_exec(krb5_auth_con_getlocalseqnumber, credentials->ctx,
1216 context->auth_ctx, (INT32*)&context->local_seq))
1218 context->remote_seq ^= context->local_seq;
1221 if (krb_log_exec(krb5glue_update_keyset, credentials->ctx, context->auth_ctx, FALSE,
1225 context->state = KERBEROS_STATE_AP_REP;
1227 if (context->flags & SSPI_GSS_C_MUTUAL_FLAG)
1228 status = SEC_I_CONTINUE_NEEDED;
1233 case KERBEROS_STATE_AP_REP:
1235 if (tok_id == TOK_ID_AP_REP)
1237 if (krb_log_exec(krb5_rd_rep, credentials->ctx, context->auth_ctx, &input_token,
1240 krb5_free_ap_rep_enc_part(credentials->ctx, reply);
1242 else if (tok_id == TOK_ID_ERROR)
1244 krb5glue_log_error(credentials->ctx, &input_token, TAG);
1250 if (context->flags & SSPI_GSS_C_SEQUENCE_FLAG)
1252 if (krb_log_exec(krb5_auth_con_getremoteseqnumber, credentials->ctx,
1253 context->auth_ctx, (INT32*)&context->remote_seq))
1257 if (krb_log_exec(krb5glue_update_keyset, credentials->ctx, context->auth_ctx, FALSE,
1261 context->state = KERBEROS_STATE_FINAL;
1264 output_buffer->cbBuffer = 0;
1268 case KERBEROS_STATE_FINAL:
1270 WLog_ERR(TAG,
"Kerberos in invalid state!");
1277 krb5_data edata = WINPR_C_ARRAY_INIT;
1278 in_creds.second_ticket = edata;
1279 krb5_free_cred_contents(credentials->ctx, &in_creds);
1282 krb5_free_creds(credentials->ctx, creds);
1283 if (output_token.data)
1284 krb5glue_free_data_contents(credentials->ctx, &output_token);
1286 winpr_Digest_Free(md5);
1295 case SEC_I_CONTINUE_NEEDED:
1296 sspi_SecureHandleSetLowerPointer(phNewContext, context);
1297 sspi_SecureHandleSetUpperPointer(phNewContext, KERBEROS_SSP_NAME);
1300 kerberos_ContextFree(context, TRUE);
1301 sspi_SecureHandleInvalidate(phNewContext);
1309 status = SEC_E_INVALID_TOKEN;
1312 return SEC_E_UNSUPPORTED_FUNCTION;
1316static SECURITY_STATUS SEC_ENTRY kerberos_InitializeSecurityContextW(
1318 ULONG Reserved1, ULONG TargetDataRep,
PSecBufferDesc pInput, ULONG Reserved2,
1321 SECURITY_STATUS status = 0;
1322 char* target_name =
nullptr;
1326 target_name = ConvertWCharToUtf8Alloc(pszTargetName,
nullptr);
1328 return SEC_E_INSUFFICIENT_MEMORY;
1331 status = kerberos_InitializeSecurityContextA(phCredential, phContext, target_name, fContextReq,
1332 Reserved1, TargetDataRep, pInput, Reserved2,
1333 phNewContext, pOutput, pfContextAttr, ptsExpiry);
1342static BOOL retrieveTgtForPrincipal(KRB_CREDENTIALS* credentials, krb5_principal principal,
1346 krb5_kt_cursor cur = WINPR_C_ARRAY_INIT;
1347 krb5_keytab_entry entry = WINPR_C_ARRAY_INIT;
1348 if (krb_log_exec(krb5_kt_start_seq_get, credentials->ctx, credentials->keytab, &cur))
1353 krb5_error_code rv =
1354 krb_log_exec(krb5_kt_next_entry, credentials->ctx, credentials->keytab, &entry, &cur);
1355 if (rv == KRB5_KT_END)
1360 if (krb5_principal_compare(credentials->ctx, principal, entry.principal))
1362 rv = krb_log_exec(krb5glue_free_keytab_entry_contents, credentials->ctx, &entry);
1363 memset(&entry, 0,
sizeof(entry));
1368 if (krb_log_exec(krb5_kt_end_seq_get, credentials->ctx, credentials->keytab, &cur))
1371 if (!entry.principal)
1375 if (krb_log_exec(krb5_get_init_creds_keytab, credentials->ctx, creds, entry.principal,
1376 credentials->keytab, 0,
nullptr,
nullptr))
1385static BOOL retrieveSomeTgt(KRB_CREDENTIALS* credentials,
const char* target, krb5_creds* creds)
1388 krb5_principal target_princ = WINPR_C_ARRAY_INIT;
1389 char* default_realm =
nullptr;
1391 krb5_error_code rv =
1392 krb_log_exec(krb5_parse_name_flags, credentials->ctx, target, 0, &target_princ);
1396#if defined(WITH_KRB5_HEIMDAL)
1397 if (!target_princ->realm)
1399 rv = krb_log_exec(krb5_get_default_realm, credentials->ctx, &default_realm);
1403 target_princ->realm = default_realm;
1406 if (!target_princ->realm.length)
1408 rv = krb_log_exec(krb5_get_default_realm, credentials->ctx, &default_realm);
1412 target_princ->realm.data = default_realm;
1413 target_princ->realm.length = (
unsigned int)strlen(default_realm);
1423 if (retrieveTgtForPrincipal(credentials, target_princ, creds))
1428#if defined(WITH_KRB5_MIT)
1433 char hostDollar[300] = WINPR_C_ARRAY_INIT;
1434 if (target_princ->length < 2)
1437 (void)snprintf(hostDollar,
sizeof(hostDollar) - 1,
"%s$@%s", target_princ->data[1].data,
1438 target_princ->realm.data);
1439 krb5_free_principal(credentials->ctx, target_princ);
1441 rv = krb_log_exec(krb5_parse_name_flags, credentials->ctx, hostDollar, 0, &target_princ);
1445 ret = retrieveTgtForPrincipal(credentials, target_princ, creds);
1450 krb5_free_default_realm(credentials->ctx, default_realm);
1452 krb5_free_principal(credentials->ctx, target_princ);
1457static SECURITY_STATUS SEC_ENTRY kerberos_AcceptSecurityContext(
1459 WINPR_ATTR_UNUSED
PSecBufferDesc pInput, WINPR_ATTR_UNUSED ULONG fContextReq,
1460 WINPR_ATTR_UNUSED ULONG TargetDataRep, WINPR_ATTR_UNUSED
PCtxtHandle phNewContext,
1461 WINPR_ATTR_UNUSED
PSecBufferDesc pOutput, WINPR_ATTR_UNUSED ULONG* pfContextAttr,
1465 BOOL isNewContext = FALSE;
1469 uint16_t tok_id = 0;
1470 krb5_data input_token = WINPR_C_ARRAY_INIT;
1471 krb5_data output_token = WINPR_C_ARRAY_INIT;
1472 SECURITY_STATUS status = SEC_E_INTERNAL_ERROR;
1473 krb5_flags ap_flags = 0;
1474 krb5glue_authenticator authenticator =
nullptr;
1475 char* target =
nullptr;
1476 krb5_keytab_entry entry = WINPR_C_ARRAY_INIT;
1477 krb5_creds creds = WINPR_C_ARRAY_INIT;
1480 if (phContext && !phContext->dwLower && !phContext->dwUpper)
1481 return SEC_E_INVALID_HANDLE;
1483 KRB_CONTEXT* context = sspi_SecureHandleGetLowerPointer(phContext);
1484 KRB_CREDENTIALS* credentials = sspi_SecureHandleGetLowerPointer(phCredential);
1487 input_buffer = sspi_FindSecBuffer(pInput, SECBUFFER_TOKEN);
1489 output_buffer = sspi_FindSecBuffer(pOutput, SECBUFFER_TOKEN);
1492 return SEC_E_INVALID_TOKEN;
1494 if (!sspi_gss_unwrap_token(input_buffer, &oid, &tok_id, &input_token))
1495 return SEC_E_INVALID_TOKEN;
1499 isNewContext = TRUE;
1500 context = kerberos_ContextNew(credentials);
1501 context->acceptor = TRUE;
1503 if (sspi_gss_oid_compare(&oid, &kerberos_u2u_OID))
1505 context->u2u = TRUE;
1506 context->state = KERBEROS_STATE_TGT_REQ;
1508 else if (sspi_gss_oid_compare(&oid, &kerberos_OID))
1509 context->state = KERBEROS_STATE_AP_REQ;
1515 if ((context->u2u && !sspi_gss_oid_compare(&oid, &kerberos_u2u_OID)) ||
1516 (!context->u2u && !sspi_gss_oid_compare(&oid, &kerberos_OID)))
1520 if (context->state == KERBEROS_STATE_TGT_REQ && tok_id == TOK_ID_TGT_REQ)
1522 if (!kerberos_rd_tgt_token(&input_token, &target,
nullptr))
1525 if (!retrieveSomeTgt(credentials, target, &creds))
1528 if (!kerberos_mk_tgt_token(output_buffer, KRB_TGT_REP,
nullptr,
nullptr, &creds.ticket))
1531 if (krb_log_exec(krb5_auth_con_init, credentials->ctx, &context->auth_ctx))
1534 if (krb_log_exec(krb5glue_auth_con_setuseruserkey, credentials->ctx, context->auth_ctx,
1535 &krb5glue_creds_getkey(creds)))
1538 context->state = KERBEROS_STATE_AP_REQ;
1540 else if (context->state == KERBEROS_STATE_AP_REQ && tok_id == TOK_ID_AP_REQ)
1542 if (krb_log_exec(krb5_rd_req, credentials->ctx, &context->auth_ctx, &input_token,
nullptr,
1543 credentials->keytab, &ap_flags,
nullptr))
1546 if (krb_log_exec(krb5_auth_con_setflags, credentials->ctx, context->auth_ctx,
1547 KRB5_AUTH_CONTEXT_DO_SEQUENCE | KRB5_AUTH_CONTEXT_USE_SUBKEY))
1551 if (krb_log_exec(krb5_auth_con_getauthenticator, credentials->ctx, context->auth_ctx,
1554 if (!krb5glue_authenticator_validate_chksum(authenticator, GSS_CHECKSUM_TYPE,
1558 if ((ap_flags & AP_OPTS_MUTUAL_REQUIRED) && (context->flags & SSPI_GSS_C_MUTUAL_FLAG))
1562 if (krb_log_exec(krb5_mk_rep, credentials->ctx, context->auth_ctx, &output_token))
1564 if (!sspi_gss_wrap_token(output_buffer,
1565 context->u2u ? &kerberos_u2u_OID : &kerberos_OID,
1566 TOK_ID_AP_REP, &output_token))
1572 output_buffer->cbBuffer = 0;
1575 *pfContextAttr = (context->flags & 0x1F);
1576 if (context->flags & SSPI_GSS_C_INTEG_FLAG)
1577 *pfContextAttr |= ASC_RET_INTEGRITY;
1579 if (context->flags & SSPI_GSS_C_SEQUENCE_FLAG)
1581 if (krb_log_exec(krb5_auth_con_getlocalseqnumber, credentials->ctx, context->auth_ctx,
1582 (INT32*)&context->local_seq))
1584 if (krb_log_exec(krb5_auth_con_getremoteseqnumber, credentials->ctx, context->auth_ctx,
1585 (INT32*)&context->remote_seq))
1589 if (krb_log_exec(krb5glue_update_keyset, credentials->ctx, context->auth_ctx, TRUE,
1593 context->state = KERBEROS_STATE_FINAL;
1599 if (context->state == KERBEROS_STATE_FINAL)
1602 status = SEC_I_CONTINUE_NEEDED;
1606 if (output_token.data)
1607 krb5glue_free_data_contents(credentials->ctx, &output_token);
1608 if (entry.principal)
1609 krb5glue_free_keytab_entry_contents(credentials->ctx, &entry);
1616 case SEC_I_CONTINUE_NEEDED:
1617 sspi_SecureHandleSetLowerPointer(phNewContext, context);
1618 sspi_SecureHandleSetUpperPointer(phNewContext, KERBEROS_SSP_NAME);
1621 kerberos_ContextFree(context, TRUE);
1622 sspi_SecureHandleInvalidate(phNewContext);
1630 status = SEC_E_INVALID_TOKEN;
1633 return SEC_E_UNSUPPORTED_FUNCTION;
1638static KRB_CONTEXT* get_context(
PCtxtHandle phContext)
1643 TCHAR* name = sspi_SecureHandleGetUpperPointer(phContext);
1647 if (_tcsncmp(KERBEROS_SSP_NAME, name, ARRAYSIZE(KERBEROS_SSP_NAME)) != 0)
1649 return sspi_SecureHandleGetLowerPointer(phContext);
1652static BOOL copy_krb5_data(krb5_data* data, PUCHAR* ptr, ULONG* psize)
1656 WINPR_ASSERT(psize);
1658 *ptr = (PUCHAR)malloc(data->length);
1662 *psize = data->length;
1663 memcpy(*ptr, data->data, data->length);
1668static SECURITY_STATUS
1669 SEC_ENTRY kerberos_DeleteSecurityContext(WINPR_ATTR_UNUSED
PCtxtHandle phContext)
1672 KRB_CONTEXT* context = get_context(phContext);
1673 sspi_SecureHandleInvalidate(phContext);
1675 return SEC_E_INVALID_HANDLE;
1677 kerberos_ContextFree(context, TRUE);
1681 return SEC_E_UNSUPPORTED_FUNCTION;
1687static SECURITY_STATUS krb5_error_to_SECURITY_STATUS(krb5_error_code code)
1694 return SEC_E_INTERNAL_ERROR;
1698static SECURITY_STATUS kerberos_ATTR_SIZES(KRB_CONTEXT* context, KRB_CREDENTIALS* credentials,
1704 krb5glue_key key =
nullptr;
1706 WINPR_ASSERT(context);
1707 WINPR_ASSERT(context->auth_ctx);
1713 ContextSizes->cbMaxToken = KERBEROS_SecPkgInfoA.cbMaxToken;
1714 ContextSizes->cbMaxSignature = 0;
1715 ContextSizes->cbBlockSize = 1;
1716 ContextSizes->cbSecurityTrailer = 0;
1718 key = get_key(&context->keyset);
1720 if (context->flags & SSPI_GSS_C_CONF_FLAG)
1722 krb5_error_code rv = krb_log_exec(krb5glue_crypto_length, credentials->ctx, key,
1723 KRB5_CRYPTO_TYPE_HEADER, &header);
1725 return krb5_error_to_SECURITY_STATUS(rv);
1727 rv = krb_log_exec(krb5glue_crypto_length, credentials->ctx, key, KRB5_CRYPTO_TYPE_PADDING,
1730 return krb5_error_to_SECURITY_STATUS(rv);
1732 rv = krb_log_exec(krb5glue_crypto_length, credentials->ctx, key, KRB5_CRYPTO_TYPE_TRAILER,
1735 return krb5_error_to_SECURITY_STATUS(rv);
1738 ContextSizes->cbSecurityTrailer = header + pad + trailer + 32;
1741 if (context->flags & SSPI_GSS_C_INTEG_FLAG)
1743 krb5_error_code rv = krb_log_exec(krb5glue_crypto_length, credentials->ctx, key,
1744 KRB5_CRYPTO_TYPE_CHECKSUM, &ContextSizes->cbMaxSignature);
1746 return krb5_error_to_SECURITY_STATUS(rv);
1748 ContextSizes->cbMaxSignature += 16;
1754static SECURITY_STATUS kerberos_ATTR_AUTH_IDENTITY(KRB_CONTEXT* context,
1755 KRB_CREDENTIALS* credentials,
1760 WINPR_ASSERT(context);
1761 WINPR_ASSERT(context->auth_ctx);
1762 WINPR_ASSERT(credentials);
1764 WINPR_ASSERT(AuthIdentity);
1765 *AuthIdentity = empty;
1767 krb5glue_authenticator authenticator =
nullptr;
1768 krb5_error_code rv = krb_log_exec(krb5_auth_con_getauthenticator, credentials->ctx,
1769 context->auth_ctx, &authenticator);
1776#if defined(WITH_KRB5_HEIMDAL)
1777 const Realm data = authenticator->crealm;
1780 const size_t data_len = length_Realm(&data);
1782 krb5_data* realm_data = krb5_princ_realm(credentials->ctx, authenticator->client);
1785 const char* data = realm_data->data;
1788 const size_t data_len = realm_data->length;
1791 if (data_len > (
sizeof(AuthIdentity->Domain) - 1))
1793 strncpy(AuthIdentity->Domain, data, data_len);
1797#if defined(WITH_KRB5_HEIMDAL)
1798 const PrincipalName* principal = &authenticator->cname;
1799 const size_t name_length = length_PrincipalName(principal);
1800 if (!principal->name_string.val)
1802 const char* name = *principal->name_string.val;
1804 char* name =
nullptr;
1805 rv = krb_log_exec(krb5_unparse_name_flags, credentials->ctx, authenticator->client,
1806 KRB5_PRINCIPAL_UNPARSE_NO_REALM, &name);
1810 const size_t name_length = strlen(name);
1813 const bool ok = (name_length <= (
sizeof(AuthIdentity->User) - 1));
1815 strncpy(AuthIdentity->User, name, name_length);
1819#if !defined(WITH_KRB5_HEIMDAL)
1820 krb5_free_unparsed_name(credentials->ctx, name);
1825 krb5glue_free_authenticator(credentials->ctx, authenticator);
1826 return krb5_error_to_SECURITY_STATUS(rv);
1829static SECURITY_STATUS kerberos_ATTR_PACKAGE_INFO(WINPR_ATTR_UNUSED KRB_CONTEXT* context,
1830 WINPR_ATTR_UNUSED KRB_CREDENTIALS* credentials,
1835 (
SecPkgInfoA*)sspi_ContextBufferAlloc(QuerySecurityPackageInfoIndex, size);
1838 return SEC_E_INSUFFICIENT_MEMORY;
1840 pPackageInfo->fCapabilities = KERBEROS_SecPkgInfoA.fCapabilities;
1841 pPackageInfo->wVersion = KERBEROS_SecPkgInfoA.wVersion;
1842 pPackageInfo->wRPCID = KERBEROS_SecPkgInfoA.wRPCID;
1843 pPackageInfo->cbMaxToken = KERBEROS_SecPkgInfoA.cbMaxToken;
1844 pPackageInfo->Name = _strdup(KERBEROS_SecPkgInfoA.Name);
1845 pPackageInfo->Comment = _strdup(KERBEROS_SecPkgInfoA.Comment);
1847 if (!pPackageInfo->Name || !pPackageInfo->Comment)
1849 sspi_ContextBufferFree(pPackageInfo);
1850 return SEC_E_INSUFFICIENT_MEMORY;
1852 PackageInfo->PackageInfo = pPackageInfo;
1856static SECURITY_STATUS kerberos_ATTR_TICKET_LOGON(KRB_CONTEXT* context,
1857 KRB_CREDENTIALS* credentials,
1860 krb5_creds matchCred = WINPR_C_ARRAY_INIT;
1861 krb5_auth_context authContext =
nullptr;
1862 krb5_flags getCredsFlags = KRB5_GC_CACHED;
1863 BOOL firstRun = TRUE;
1864 krb5_creds* hostCred =
nullptr;
1865 SECURITY_STATUS ret = SEC_E_INSUFFICIENT_MEMORY;
1866 int rv = krb_log_exec(krb5_sname_to_principal, credentials->ctx, context->targetHost,
"HOST",
1867 KRB5_NT_SRV_HST, &matchCred.server);
1871 rv = krb_log_exec(krb5_cc_get_principal, credentials->ctx, credentials->ccache,
1878 rv = krb_log_exec(krb5_get_credentials, credentials->ctx, getCredsFlags, credentials->ccache,
1879 &matchCred, &hostCred);
1884 case KRB5_CC_NOTFOUND:
1893 WLog_ERR(TAG,
"krb5_get_credentials(hostCreds), rv=%d", rv);
1897 if (krb_log_exec(krb5_auth_con_init, credentials->ctx, &authContext))
1901 krb5_data derOut = WINPR_C_ARRAY_INIT;
1902 if (krb_log_exec(krb5_fwd_tgt_creds, credentials->ctx, authContext, context->targetHost,
1903 matchCred.client, matchCred.server, credentials->ccache, 1, &derOut))
1905 ret = SEC_E_LOGON_DENIED;
1909 ticketLogon->MessageType = KerbTicketLogon;
1910 ticketLogon->Flags = KERB_LOGON_FLAG_REDIRECTED;
1912 if (!copy_krb5_data(&hostCred->ticket, &ticketLogon->ServiceTicket,
1913 &ticketLogon->ServiceTicketLength))
1915 krb5_free_data(credentials->ctx, &derOut);
1919 ticketLogon->TicketGrantingTicketLength = derOut.length;
1920 ticketLogon->TicketGrantingTicket = (PUCHAR)derOut.data;
1925 krb5_auth_con_free(credentials->ctx, authContext);
1926 krb5_free_creds(credentials->ctx, hostCred);
1927 krb5_free_cred_contents(credentials->ctx, &matchCred);
1933static SECURITY_STATUS
1934 SEC_ENTRY kerberos_QueryContextAttributesA(
PCtxtHandle phContext,
1935 WINPR_ATTR_UNUSED ULONG ulAttribute,
void* pBuffer)
1938 return SEC_E_INVALID_HANDLE;
1941 return SEC_E_INVALID_PARAMETER;
1944 KRB_CONTEXT* context = get_context(phContext);
1946 return SEC_E_INVALID_PARAMETER;
1948 KRB_CREDENTIALS* credentials = context->credentials;
1950 switch (ulAttribute)
1952 case SECPKG_ATTR_SIZES:
1955 case SECPKG_ATTR_AUTH_IDENTITY:
1956 return kerberos_ATTR_AUTH_IDENTITY(context, credentials,
1959 case SECPKG_ATTR_PACKAGE_INFO:
1960 return kerberos_ATTR_PACKAGE_INFO(context, credentials,
1963 case SECPKG_CRED_ATTR_TICKET_LOGON:
1964 return kerberos_ATTR_TICKET_LOGON(context, credentials, (
KERB_TICKET_LOGON*)pBuffer);
1967 WLog_ERR(TAG,
"TODO: QueryContextAttributes implement ulAttribute=0x%08" PRIx32,
1969 return SEC_E_UNSUPPORTED_FUNCTION;
1972 return SEC_E_UNSUPPORTED_FUNCTION;
1976static SECURITY_STATUS SEC_ENTRY kerberos_QueryContextAttributesW(
PCtxtHandle phContext,
1977 ULONG ulAttribute,
void* pBuffer)
1979 return kerberos_QueryContextAttributesA(phContext, ulAttribute, pBuffer);
1982static SECURITY_STATUS SEC_ENTRY kerberos_SetContextAttributesW(
1983 WINPR_ATTR_UNUSED
PCtxtHandle phContext, WINPR_ATTR_UNUSED ULONG ulAttribute,
1984 WINPR_ATTR_UNUSED
void* pBuffer, WINPR_ATTR_UNUSED ULONG cbBuffer)
1986 return SEC_E_UNSUPPORTED_FUNCTION;
1989static SECURITY_STATUS SEC_ENTRY kerberos_SetContextAttributesA(
1990 WINPR_ATTR_UNUSED
PCtxtHandle phContext, WINPR_ATTR_UNUSED ULONG ulAttribute,
1991 WINPR_ATTR_UNUSED
void* pBuffer, WINPR_ATTR_UNUSED ULONG cbBuffer)
1993 return SEC_E_UNSUPPORTED_FUNCTION;
1996static SECURITY_STATUS SEC_ENTRY kerberos_SetCredentialsAttributesX(
1997 WINPR_ATTR_UNUSED
PCredHandle phCredential, WINPR_ATTR_UNUSED ULONG ulAttribute,
1998 WINPR_ATTR_UNUSED
void* pBuffer, WINPR_ATTR_UNUSED ULONG cbBuffer,
1999 WINPR_ATTR_UNUSED BOOL unicode)
2002 KRB_CREDENTIALS* credentials =
nullptr;
2005 return SEC_E_INVALID_HANDLE;
2007 credentials = sspi_SecureHandleGetLowerPointer(phCredential);
2010 return SEC_E_INVALID_HANDLE;
2013 return SEC_E_INSUFFICIENT_MEMORY;
2015 switch (ulAttribute)
2017 case SECPKG_CRED_ATTR_KDC_PROXY_SETTINGS:
2023 kdc_settings->Version != KDC_PROXY_SETTINGS_V1 ||
2026 kdc_settings->ProxyServerOffset + kdc_settings->ProxyServerLength)
2027 return SEC_E_INVALID_TOKEN;
2029 if (credentials->kdc_url)
2031 free(credentials->kdc_url);
2032 credentials->kdc_url =
nullptr;
2035 if (kdc_settings->ProxyServerLength > 0)
2037 WCHAR* proxy = (WCHAR*)((BYTE*)pBuffer + kdc_settings->ProxyServerOffset);
2039 credentials->kdc_url = ConvertWCharNToUtf8Alloc(
2040 proxy, kdc_settings->ProxyServerLength /
sizeof(WCHAR),
nullptr);
2041 if (!credentials->kdc_url)
2042 return SEC_E_INSUFFICIENT_MEMORY;
2047 case SECPKG_CRED_ATTR_NAMES:
2048 case SECPKG_ATTR_SUPPORTED_ALGS:
2050 WLog_ERR(TAG,
"TODO: SetCredentialsAttributesX implement ulAttribute=0x%08" PRIx32,
2052 return SEC_E_UNSUPPORTED_FUNCTION;
2056 return SEC_E_UNSUPPORTED_FUNCTION;
2060static SECURITY_STATUS SEC_ENTRY kerberos_SetCredentialsAttributesW(
PCredHandle phCredential,
2062 void* pBuffer, ULONG cbBuffer)
2064 return kerberos_SetCredentialsAttributesX(phCredential, ulAttribute, pBuffer, cbBuffer, TRUE);
2067static SECURITY_STATUS SEC_ENTRY kerberos_SetCredentialsAttributesA(
PCredHandle phCredential,
2069 void* pBuffer, ULONG cbBuffer)
2071 return kerberos_SetCredentialsAttributesX(phCredential, ulAttribute, pBuffer, cbBuffer, FALSE);
2074static SECURITY_STATUS SEC_ENTRY kerberos_EncryptMessage(WINPR_ATTR_UNUSED
PCtxtHandle phContext,
2075 WINPR_ATTR_UNUSED ULONG fQOP,
2077 WINPR_ATTR_UNUSED ULONG MessageSeqNo)
2080 KRB_CONTEXT* context = get_context(phContext);
2083 char* header =
nullptr;
2085 krb5glue_key key =
nullptr;
2086 krb5_keyusage usage = 0;
2087 krb5_crypto_iov encrypt_iov[] = { { KRB5_CRYPTO_TYPE_HEADER, { 0 } },
2088 { KRB5_CRYPTO_TYPE_DATA, { 0 } },
2089 { KRB5_CRYPTO_TYPE_DATA, { 0 } },
2090 { KRB5_CRYPTO_TYPE_PADDING, { 0 } },
2091 { KRB5_CRYPTO_TYPE_TRAILER, { 0 } } };
2094 return SEC_E_INVALID_HANDLE;
2096 if (!(context->flags & SSPI_GSS_C_CONF_FLAG))
2097 return SEC_E_UNSUPPORTED_FUNCTION;
2099 KRB_CREDENTIALS* creds = context->credentials;
2101 sig_buffer = sspi_FindSecBuffer(pMessage, SECBUFFER_TOKEN);
2102 data_buffer = sspi_FindSecBuffer(pMessage, SECBUFFER_DATA);
2104 if (!sig_buffer || !data_buffer)
2105 return SEC_E_INVALID_TOKEN;
2108 return SEC_E_QOP_NOT_SUPPORTED;
2110 flags |= context->acceptor ? FLAG_SENDER_IS_ACCEPTOR : 0;
2111 flags |= FLAG_WRAP_CONFIDENTIAL;
2113 key = get_key(&context->keyset);
2115 return SEC_E_INTERNAL_ERROR;
2117 flags |= context->keyset.acceptor_key == key ? FLAG_ACCEPTOR_SUBKEY : 0;
2119 usage = context->acceptor ? KG_USAGE_ACCEPTOR_SEAL : KG_USAGE_INITIATOR_SEAL;
2122 encrypt_iov[1].data.length = data_buffer->cbBuffer;
2123 encrypt_iov[2].data.length = 16;
2126 if (krb_log_exec(krb5glue_crypto_length_iov, creds->ctx, key, encrypt_iov,
2127 ARRAYSIZE(encrypt_iov)))
2128 return SEC_E_INTERNAL_ERROR;
2129 if (sig_buffer->cbBuffer <
2130 encrypt_iov[0].data.length + encrypt_iov[3].data.length + encrypt_iov[4].data.length + 32)
2131 return SEC_E_INSUFFICIENT_MEMORY;
2134 header = sig_buffer->pvBuffer;
2135 encrypt_iov[2].data.data = header + 16;
2136 encrypt_iov[3].data.data = encrypt_iov[2].data.data + encrypt_iov[2].data.length;
2137 encrypt_iov[4].data.data = encrypt_iov[3].data.data + encrypt_iov[3].data.length;
2138 encrypt_iov[0].data.data = encrypt_iov[4].data.data + encrypt_iov[4].data.length;
2139 encrypt_iov[1].data.data = data_buffer->pvBuffer;
2142 winpr_Data_Write_UINT16_BE(header, TOK_ID_WRAP);
2143 header[2] = WINPR_ASSERTING_INT_CAST(
char, flags);
2144 header[3] = (char)0xFF;
2145 winpr_Data_Write_UINT32(header + 4, 0);
2146 winpr_Data_Write_UINT64_BE(header + 8, (context->local_seq + MessageSeqNo));
2149 CopyMemory(encrypt_iov[2].data.data, header, 16);
2152 const size_t len = 16 + encrypt_iov[3].data.length + encrypt_iov[4].data.length;
2153 winpr_Data_Write_UINT16_BE(header + 6, WINPR_ASSERTING_INT_CAST(UINT16, len));
2155 if (krb_log_exec(krb5glue_encrypt_iov, creds->ctx, key, usage, encrypt_iov,
2156 ARRAYSIZE(encrypt_iov)))
2157 return SEC_E_INTERNAL_ERROR;
2161 return SEC_E_UNSUPPORTED_FUNCTION;
2165static SECURITY_STATUS SEC_ENTRY kerberos_DecryptMessage(WINPR_ATTR_UNUSED
PCtxtHandle phContext,
2167 WINPR_ATTR_UNUSED ULONG MessageSeqNo,
2168 WINPR_ATTR_UNUSED ULONG* pfQOP)
2171 KRB_CONTEXT* context = get_context(phContext);
2174 krb5glue_key key =
nullptr;
2175 krb5_keyusage usage = 0;
2176 uint16_t tok_id = 0;
2180 uint64_t seq_no = 0;
2181 krb5_crypto_iov iov[] = { { KRB5_CRYPTO_TYPE_HEADER, { 0 } },
2182 { KRB5_CRYPTO_TYPE_DATA, { 0 } },
2183 { KRB5_CRYPTO_TYPE_DATA, { 0 } },
2184 { KRB5_CRYPTO_TYPE_PADDING, { 0 } },
2185 { KRB5_CRYPTO_TYPE_TRAILER, { 0 } } };
2188 return SEC_E_INVALID_HANDLE;
2190 if (!(context->flags & SSPI_GSS_C_CONF_FLAG))
2191 return SEC_E_UNSUPPORTED_FUNCTION;
2193 KRB_CREDENTIALS* creds = context->credentials;
2195 sig_buffer = sspi_FindSecBuffer(pMessage, SECBUFFER_TOKEN);
2196 data_buffer = sspi_FindSecBuffer(pMessage, SECBUFFER_DATA);
2198 if (!sig_buffer || !data_buffer || sig_buffer->cbBuffer < 16)
2199 return SEC_E_INVALID_TOKEN;
2202 BYTE* header = sig_buffer->pvBuffer;
2203 tok_id = winpr_Data_Get_UINT16_BE(header);
2205 ec = winpr_Data_Get_UINT16_BE(&header[4]);
2206 rrc = winpr_Data_Get_UINT16_BE(&header[6]);
2207 seq_no = winpr_Data_Get_UINT64_BE(&header[8]);
2210 if ((tok_id != TOK_ID_WRAP) || (header[3] != 0xFF))
2211 return SEC_E_INVALID_TOKEN;
2213 if ((flags & FLAG_SENDER_IS_ACCEPTOR) == context->acceptor)
2214 return SEC_E_INVALID_TOKEN;
2216 if ((context->flags & ISC_REQ_SEQUENCE_DETECT) &&
2217 (seq_no != context->remote_seq + MessageSeqNo))
2218 return SEC_E_OUT_OF_SEQUENCE;
2220 if (!(flags & FLAG_WRAP_CONFIDENTIAL))
2221 return SEC_E_INVALID_TOKEN;
2225 return SEC_E_INVALID_TOKEN;
2228 key = get_key(&context->keyset);
2229 if (!key || ((flags & FLAG_ACCEPTOR_SUBKEY) && (context->keyset.acceptor_key != key)))
2230 return SEC_E_INTERNAL_ERROR;
2231 usage = context->acceptor ? KG_USAGE_INITIATOR_SEAL : KG_USAGE_ACCEPTOR_SEAL;
2234 iov[1].data.length = data_buffer->cbBuffer;
2235 iov[2].data.length = 16;
2236 if (krb_log_exec(krb5glue_crypto_length_iov, creds->ctx, key, iov, ARRAYSIZE(iov)))
2237 return SEC_E_INTERNAL_ERROR;
2240 if (rrc != 16 + iov[3].data.length + iov[4].data.length)
2241 return SEC_E_INVALID_TOKEN;
2242 if (sig_buffer->cbBuffer != 16 + rrc + iov[0].data.length)
2243 return SEC_E_INVALID_TOKEN;
2246 iov[0].data.data = (
char*)&header[16 + rrc + ec];
2247 iov[1].data.data = data_buffer->pvBuffer;
2248 iov[2].data.data = (
char*)&header[16 + ec];
2249 char* data2 = iov[2].data.data;
2250 iov[3].data.data = &data2[iov[2].data.length];
2252 char* data3 = iov[3].data.data;
2253 iov[4].data.data = &data3[iov[3].data.length];
2255 if (krb_log_exec(krb5glue_decrypt_iov, creds->ctx, key, usage, iov, ARRAYSIZE(iov)))
2256 return SEC_E_INTERNAL_ERROR;
2259 winpr_Data_Write_UINT16_BE(iov[2].data.data + 4, ec);
2260 winpr_Data_Write_UINT16_BE(iov[2].data.data + 6, rrc);
2261 if (memcmp(iov[2].data.data, header, 16) != 0)
2262 return SEC_E_MESSAGE_ALTERED;
2268 return SEC_E_UNSUPPORTED_FUNCTION;
2272static SECURITY_STATUS SEC_ENTRY kerberos_MakeSignature(WINPR_ATTR_UNUSED
PCtxtHandle phContext,
2273 WINPR_ATTR_UNUSED ULONG fQOP,
2275 WINPR_ATTR_UNUSED ULONG MessageSeqNo)
2278 KRB_CONTEXT* context = get_context(phContext);
2281 krb5glue_key key =
nullptr;
2282 krb5_keyusage usage = 0;
2284 krb5_crypto_iov iov[] = { { KRB5_CRYPTO_TYPE_DATA, { 0 } },
2285 { KRB5_CRYPTO_TYPE_DATA, { 0 } },
2286 { KRB5_CRYPTO_TYPE_CHECKSUM, { 0 } } };
2289 return SEC_E_INVALID_HANDLE;
2291 if (!(context->flags & SSPI_GSS_C_INTEG_FLAG))
2292 return SEC_E_UNSUPPORTED_FUNCTION;
2294 KRB_CREDENTIALS* creds = context->credentials;
2296 sig_buffer = sspi_FindSecBuffer(pMessage, SECBUFFER_TOKEN);
2297 data_buffer = sspi_FindSecBuffer(pMessage, SECBUFFER_DATA);
2299 if (!sig_buffer || !data_buffer)
2300 return SEC_E_INVALID_TOKEN;
2302 flags |= context->acceptor ? FLAG_SENDER_IS_ACCEPTOR : 0;
2304 key = get_key(&context->keyset);
2306 return SEC_E_INTERNAL_ERROR;
2307 usage = context->acceptor ? KG_USAGE_ACCEPTOR_SIGN : KG_USAGE_INITIATOR_SIGN;
2309 flags |= context->keyset.acceptor_key == key ? FLAG_ACCEPTOR_SUBKEY : 0;
2312 iov[0].data.length = data_buffer->cbBuffer;
2313 iov[1].data.length = 16;
2314 if (krb_log_exec(krb5glue_crypto_length_iov, creds->ctx, key, iov, ARRAYSIZE(iov)))
2315 return SEC_E_INTERNAL_ERROR;
2318 if (sig_buffer->cbBuffer < iov[2].data.length + 16)
2319 return SEC_E_INSUFFICIENT_MEMORY;
2322 char* header = sig_buffer->pvBuffer;
2323 winpr_Data_Write_UINT16_BE(header, TOK_ID_MIC);
2324 header[2] = WINPR_ASSERTING_INT_CAST(
char, flags);
2325 memset(header + 3, 0xFF, 5);
2326 winpr_Data_Write_UINT64_BE(header + 8, (context->local_seq + MessageSeqNo));
2329 iov[0].data.data = data_buffer->pvBuffer;
2330 iov[1].data.data = header;
2331 iov[2].data.data = header + 16;
2333 if (krb_log_exec(krb5glue_make_checksum_iov, creds->ctx, key, usage, iov, ARRAYSIZE(iov)))
2334 return SEC_E_INTERNAL_ERROR;
2336 sig_buffer->cbBuffer = iov[2].data.length + 16;
2340 return SEC_E_UNSUPPORTED_FUNCTION;
2344static SECURITY_STATUS SEC_ENTRY kerberos_VerifySignature(WINPR_ATTR_UNUSED
PCtxtHandle phContext,
2346 WINPR_ATTR_UNUSED ULONG MessageSeqNo,
2347 WINPR_ATTR_UNUSED ULONG* pfQOP)
2352 krb5glue_key key =
nullptr;
2353 krb5_keyusage usage = 0;
2355 uint16_t tok_id = 0;
2356 uint64_t seq_no = 0;
2357 krb5_boolean is_valid = 0;
2358 krb5_crypto_iov iov[] = { { KRB5_CRYPTO_TYPE_DATA, { 0 } },
2359 { KRB5_CRYPTO_TYPE_DATA, { 0 } },
2360 { KRB5_CRYPTO_TYPE_CHECKSUM, { 0 } } };
2361 BYTE cmp_filler[] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF };
2363 KRB_CONTEXT* context = get_context(phContext);
2365 return SEC_E_INVALID_HANDLE;
2367 if (!(context->flags & SSPI_GSS_C_INTEG_FLAG))
2368 return SEC_E_UNSUPPORTED_FUNCTION;
2370 sig_buffer = sspi_FindSecBuffer(pMessage, SECBUFFER_TOKEN);
2371 data_buffer = sspi_FindSecBuffer(pMessage, SECBUFFER_DATA);
2373 if (!sig_buffer || !data_buffer || sig_buffer->cbBuffer < 16)
2374 return SEC_E_INVALID_TOKEN;
2377 BYTE* header = sig_buffer->pvBuffer;
2378 tok_id = winpr_Data_Get_UINT16_BE(header);
2380 seq_no = winpr_Data_Get_UINT64_BE((header + 8));
2383 if (tok_id != TOK_ID_MIC)
2384 return SEC_E_INVALID_TOKEN;
2386 if ((flags & FLAG_SENDER_IS_ACCEPTOR) == context->acceptor || flags & FLAG_WRAP_CONFIDENTIAL)
2387 return SEC_E_INVALID_TOKEN;
2389 if (memcmp(header + 3, cmp_filler,
sizeof(cmp_filler)) != 0)
2390 return SEC_E_INVALID_TOKEN;
2392 if (context->flags & ISC_REQ_SEQUENCE_DETECT && seq_no != context->remote_seq + MessageSeqNo)
2393 return SEC_E_OUT_OF_SEQUENCE;
2396 key = get_key(&context->keyset);
2397 if (!key || (flags & FLAG_ACCEPTOR_SUBKEY && context->keyset.acceptor_key != key))
2398 return SEC_E_INTERNAL_ERROR;
2399 usage = context->acceptor ? KG_USAGE_INITIATOR_SIGN : KG_USAGE_ACCEPTOR_SIGN;
2402 KRB_CREDENTIALS* creds = context->credentials;
2403 iov[0].data.length = data_buffer->cbBuffer;
2404 iov[1].data.length = 16;
2405 if (krb_log_exec(krb5glue_crypto_length_iov, creds->ctx, key, iov, ARRAYSIZE(iov)))
2406 return SEC_E_INTERNAL_ERROR;
2408 if (sig_buffer->cbBuffer != iov[2].data.length + 16)
2409 return SEC_E_INTERNAL_ERROR;
2412 iov[0].data.data = data_buffer->pvBuffer;
2413 iov[1].data.data = (
char*)header;
2414 iov[2].data.data = (
char*)&header[16];
2416 if (krb_log_exec(krb5glue_verify_checksum_iov, creds->ctx, key, usage, iov, ARRAYSIZE(iov),
2418 return SEC_E_INTERNAL_ERROR;
2421 return SEC_E_MESSAGE_ALTERED;
2425 return SEC_E_UNSUPPORTED_FUNCTION;
2432 kerberos_QueryCredentialsAttributesA,
2433 kerberos_AcquireCredentialsHandleA,
2434 kerberos_FreeCredentialsHandle,
2436 kerberos_InitializeSecurityContextA,
2437 kerberos_AcceptSecurityContext,
2439 kerberos_DeleteSecurityContext,
2441 kerberos_QueryContextAttributesA,
2444 kerberos_MakeSignature,
2445 kerberos_VerifySignature,
2455 kerberos_EncryptMessage,
2456 kerberos_DecryptMessage,
2457 kerberos_SetContextAttributesA,
2458 kerberos_SetCredentialsAttributesA,
2464 kerberos_QueryCredentialsAttributesW,
2465 kerberos_AcquireCredentialsHandleW,
2466 kerberos_FreeCredentialsHandle,
2468 kerberos_InitializeSecurityContextW,
2469 kerberos_AcceptSecurityContext,
2471 kerberos_DeleteSecurityContext,
2473 kerberos_QueryContextAttributesW,
2476 kerberos_MakeSignature,
2477 kerberos_VerifySignature,
2487 kerberos_EncryptMessage,
2488 kerberos_DecryptMessage,
2489 kerberos_SetContextAttributesW,
2490 kerberos_SetCredentialsAttributesW,
2493BOOL KERBEROS_init(
void)
2495 InitializeConstWCharFromUtf8(KERBEROS_SecPkgInfoA.Name, KERBEROS_SecPkgInfoW_NameBuffer,
2496 ARRAYSIZE(KERBEROS_SecPkgInfoW_NameBuffer));
2497 InitializeConstWCharFromUtf8(KERBEROS_SecPkgInfoA.Comment, KERBEROS_SecPkgInfoW_CommentBuffer,
2498 ARRAYSIZE(KERBEROS_SecPkgInfoW_CommentBuffer));