20#include <winpr/config.h>
22#include <winpr/assert.h>
28#include <winpr/print.h>
29#include <winpr/sysinfo.h>
30#include <winpr/tchar.h>
31#include <winpr/crypto.h>
33#include "ntlm_compute.h"
35#include "ntlm_av_pairs.h"
37#if defined(WITH_DEBUG_NTLM)
39#define TAG WINPR_TAG("sspi.NTLM")
42static BOOL ntlm_av_pair_get_next_offset(
const NTLM_AV_PAIR* pAvPair,
size_t size,
size_t* pOffset);
44static BOOL ntlm_av_pair_check_data(
const NTLM_AV_PAIR* pAvPair,
size_t cbAvPair,
size_t size)
49 if (!ntlm_av_pair_get_next_offset(pAvPair, cbAvPair, &offset))
51 return cbAvPair >= offset;
55static const char* get_av_pair_string(UINT16 pair)
61 case MsvAvNbComputerName:
62 return "MsvAvNbComputerName";
63 case MsvAvNbDomainName:
64 return "MsvAvNbDomainName";
65 case MsvAvDnsComputerName:
66 return "MsvAvDnsComputerName";
67 case MsvAvDnsDomainName:
68 return "MsvAvDnsDomainName";
69 case MsvAvDnsTreeName:
70 return "MsvAvDnsTreeName";
74 return "MsvAvTimestamp";
76 return "MsvAvSingleHost";
78 return "MsvAvTargetName";
79 case MsvAvChannelBindings:
80 return "MsvAvChannelBindings";
87static BOOL ntlm_av_pair_check(
const NTLM_AV_PAIR* pAvPair,
size_t cbAvPair);
90static inline void ntlm_av_pair_set_id(
NTLM_AV_PAIR* pAvPair, UINT16
id)
92 WINPR_ASSERT(pAvPair);
93 winpr_Data_Write_UINT16(&pAvPair->AvId,
id);
96static inline void ntlm_av_pair_set_len(
NTLM_AV_PAIR* pAvPair, UINT16 len)
98 WINPR_ASSERT(pAvPair);
99 winpr_Data_Write_UINT16(&pAvPair->AvLen, len);
102static BOOL ntlm_av_pair_list_init(
NTLM_AV_PAIR* pAvPairList,
size_t cbAvPairList)
109 ntlm_av_pair_set_id(pAvPair, MsvAvEOL);
110 ntlm_av_pair_set_len(pAvPair, 0);
114WINPR_ATTR_NODISCARD
static inline BOOL ntlm_av_pair_get_id(
const NTLM_AV_PAIR* pAvPair,
115 size_t size, UINT16* pair)
117 if (!pAvPair || !pair)
123 const UINT16 AvId = winpr_Data_Get_UINT16(&pAvPair->AvId);
129ULONG ntlm_av_pair_list_length(
NTLM_AV_PAIR* pAvPairList,
size_t cbAvPairList)
134 pAvPair = ntlm_av_pair_get(pAvPairList, cbAvPairList, MsvAvEOL, &cbAvPair);
138 if (pAvPair < pAvPairList)
141 const size_t size = WINPR_ASSERTING_INT_CAST(
size_t, ((PBYTE)pAvPair - (PBYTE)pAvPairList)) +
143 WINPR_ASSERT(size <= UINT32_MAX);
144 WINPR_ASSERT(size >= 0);
148WINPR_ATTR_NODISCARD
static inline BOOL ntlm_av_pair_get_len(
const NTLM_AV_PAIR* pAvPair,
149 size_t size,
size_t* pAvLen)
157 const UINT16 AvLen = winpr_Data_Get_UINT16(&pAvPair->AvLen);
163#ifdef WITH_DEBUG_NTLM
164void ntlm_print_av_pair_list(
NTLM_AV_PAIR* pAvPairList,
size_t cbAvPairList)
167 size_t cbAvPair = cbAvPairList;
170 if (!ntlm_av_pair_check(pAvPair, cbAvPair))
173 WLog_VRB(TAG,
"AV_PAIRs =");
175 while (pAvPair && ntlm_av_pair_get_id(pAvPair, cbAvPair, &pair) && (pair != MsvAvEOL))
178 ntlm_av_pair_get_len(pAvPair, cbAvPair, &cbLen);
180 WLog_VRB(TAG,
"\t%s AvId: %" PRIu16
" AvLen: %" PRIuz
"", get_av_pair_string(pair), pair,
182 winpr_HexDump(TAG, WLOG_TRACE, ntlm_av_pair_get_value_pointer(pAvPair, cbAvPair), cbLen);
184 pAvPair = ntlm_av_pair_next(pAvPair, &cbAvPair);
189static size_t ntlm_av_pair_list_size(
size_t AvPairsCount,
size_t AvPairsValueLength)
192 return ((AvPairsCount + 1) * 4ULL) + AvPairsValueLength;
195PBYTE ntlm_av_pair_get_value_pointer(
NTLM_AV_PAIR* pAvPair,
size_t cbAvPair)
197 WINPR_ASSERT(pAvPair);
203static BOOL ntlm_av_pair_get_next_offset(
const NTLM_AV_PAIR* pAvPair,
size_t size,
size_t* pOffset)
209 if (!ntlm_av_pair_get_len(pAvPair, size, &avLen))
215static BOOL ntlm_av_pair_check(
const NTLM_AV_PAIR* pAvPair,
size_t cbAvPair)
217 return ntlm_av_pair_check_data(pAvPair, cbAvPair, 0);
226 if (!ntlm_av_pair_check(pAvPair, *pcbAvPair))
229 if (!ntlm_av_pair_get_next_offset(pAvPair, *pcbAvPair, &offset))
232 *pcbAvPair -= offset;
234 if (!ntlm_av_pair_check(next, *pcbAvPair))
240 size_t* pcbAvPairListRemaining)
243 size_t cbAvPair = cbAvPairList;
246 if (!ntlm_av_pair_check(pAvPair, cbAvPair))
249 while (pAvPair && ntlm_av_pair_get_id(pAvPair, cbAvPair, &
id))
259 pAvPair = ntlm_av_pair_next(pAvPair, &cbAvPair);
264 if (pcbAvPairListRemaining)
265 *pcbAvPairListRemaining = cbAvPair;
270static BOOL ntlm_av_pair_add(
NTLM_AV_PAIR* pAvPairList,
size_t cbAvPairList, NTLM_AV_ID AvId,
271 PBYTE Value, UINT16 AvLen)
276 pAvPair = ntlm_av_pair_get(pAvPairList, cbAvPairList, MsvAvEOL, &cbAvPair);
279 if (!pAvPair || cbAvPair < 2 *
sizeof(
NTLM_AV_PAIR) + AvLen)
282 ntlm_av_pair_set_id(pAvPair, (UINT16)AvId);
283 ntlm_av_pair_set_len(pAvPair, AvLen);
286 WINPR_ASSERT(Value !=
nullptr);
287 CopyMemory(ntlm_av_pair_get_value_pointer(pAvPair, cbAvPair), Value, AvLen);
290 pAvPair = ntlm_av_pair_next(pAvPair, &cbAvPair);
291 return ntlm_av_pair_list_init(pAvPair, cbAvPair);
294static BOOL ntlm_av_pair_valid(UINT16 pair)
299 case MsvAvNbComputerName:
300 case MsvAvNbDomainName:
301 case MsvAvDnsComputerName:
302 case MsvAvDnsDomainName:
303 case MsvAvDnsTreeName:
306 case MsvAvSingleHost:
307 case MsvAvTargetName:
308 case MsvAvChannelBindings:
315static BOOL ntlm_av_pair_add_copy(
NTLM_AV_PAIR* pAvPairList,
size_t cbAvPairList,
321 if (!ntlm_av_pair_check(pAvPair, cbAvPair))
324 if (!ntlm_av_pair_get_id(pAvPair, cbAvPair, &pair))
327 if (!ntlm_av_pair_get_len(pAvPair, cbAvPair, &avLen))
330 if (!ntlm_av_pair_valid(pair))
333 WINPR_ASSERT(avLen <= UINT16_MAX);
334 return ntlm_av_pair_add(pAvPairList, cbAvPairList, WINPR_ASSERTING_INT_CAST(NTLM_AV_ID, pair),
335 ntlm_av_pair_get_value_pointer(pAvPair, cbAvPair), (UINT16)avLen);
339 WINPR_ATTR_UNUSED COMPUTER_NAME_FORMAT type)
345 char* name = get_computer_name(ComputerNameNetBIOS,
nullptr);
352 pName->Buffer = ConvertUtf8ToWCharAlloc(name, &len);
355 if (!pName->Buffer || (len == 0) || (len > UINT16_MAX /
sizeof(WCHAR)))
358 pName->Buffer =
nullptr;
362 pName->Length = (USHORT)((len) *
sizeof(WCHAR));
363 pName->MaximumLength = pName->Length;
371 if (string->Length > 0)
373 free(string->Buffer);
374 string->Buffer =
nullptr;
376 string->MaximumLength = 0;
409static BOOL ntlm_md5_update_uint32_be(WINPR_DIGEST_CTX* md5, UINT32 num)
412 be32[0] = (num >> 0) & 0xFF;
413 be32[1] = (num >> 8) & 0xFF;
414 be32[2] = (num >> 16) & 0xFF;
415 be32[3] = (num >> 24) & 0xFF;
416 return winpr_Digest_Update(md5, be32, 4);
419static void ntlm_compute_channel_bindings(
NTLM_CONTEXT* context)
421 WINPR_DIGEST_CTX* md5 =
nullptr;
422 BYTE* ChannelBindingToken =
nullptr;
423 UINT32 ChannelBindingTokenLength = 0;
426 WINPR_ASSERT(context);
428 ZeroMemory(context->ChannelBindingsHash, WINPR_MD5_DIGEST_LENGTH);
429 ChannelBindings = context->Bindings.Bindings;
431 if (!ChannelBindings)
434 if (!(md5 = winpr_Digest_New()))
437 if (!winpr_Digest_Init(md5, WINPR_MD_MD5))
441 ChannelBindingToken = &((BYTE*)ChannelBindings)[ChannelBindings->dwApplicationDataOffset];
443 if (!ntlm_md5_update_uint32_be(md5, ChannelBindings->dwInitiatorAddrType))
446 if (!ntlm_md5_update_uint32_be(md5, ChannelBindings->cbInitiatorLength))
449 if (!ntlm_md5_update_uint32_be(md5, ChannelBindings->dwAcceptorAddrType))
452 if (!ntlm_md5_update_uint32_be(md5, ChannelBindings->cbAcceptorLength))
455 if (!ntlm_md5_update_uint32_be(md5, ChannelBindings->cbApplicationDataLength))
458 if (!winpr_Digest_Update(md5, (
void*)ChannelBindingToken, ChannelBindingTokenLength))
461 if (!winpr_Digest_Final(md5, context->ChannelBindingsHash, WINPR_MD5_DIGEST_LENGTH))
465 winpr_Digest_Free(md5);
468static void ntlm_compute_single_host_data(
NTLM_CONTEXT* context)
470 WINPR_ASSERT(context);
479 winpr_Data_Write_UINT32(&context->SingleHostData.Size, 48);
480 winpr_Data_Write_UINT32(&context->SingleHostData.Z4, 0);
481 winpr_Data_Write_UINT32(&context->SingleHostData.DataPresent, 1);
482 winpr_Data_Write_UINT32(&context->SingleHostData.CustomData, SECURITY_MANDATORY_MEDIUM_RID);
483 FillMemory(context->SingleHostData.MachineID, 32, 0xAA);
486BOOL ntlm_construct_challenge_target_info(
NTLM_CONTEXT* context)
489 ULONG AvPairsCount = 0;
490 ULONG AvPairsLength = 0;
492 size_t cbAvPairList = 0;
498 WINPR_ASSERT(context);
500 if (ntlm_get_target_computer_name(&NbDomainName, ComputerNameNetBIOS) < 0)
503 NbComputerName.Buffer =
nullptr;
505 if (ntlm_get_target_computer_name(&NbComputerName, ComputerNameNetBIOS) < 0)
508 DnsDomainName.Buffer =
nullptr;
510 if (ntlm_get_target_computer_name(&DnsDomainName, ComputerNameDnsDomain) < 0)
513 DnsComputerName.Buffer =
nullptr;
515 if (ntlm_get_target_computer_name(&DnsComputerName, ComputerNameDnsHostname) < 0)
519 AvPairsLength = NbDomainName.Length + NbComputerName.Length + DnsDomainName.Length +
520 DnsComputerName.Length + 8;
522 const size_t length = ntlm_av_pair_list_size(AvPairsCount, AvPairsLength);
523 if (!sspi_SecBufferAlloc(&context->ChallengeTargetInfo,
524 WINPR_ASSERTING_INT_CAST(uint32_t, length)))
528 pAvPairList = (
NTLM_AV_PAIR*)context->ChallengeTargetInfo.pvBuffer;
529 cbAvPairList = context->ChallengeTargetInfo.cbBuffer;
531 if (!ntlm_av_pair_list_init(pAvPairList, cbAvPairList))
534 if (!ntlm_av_pair_add(pAvPairList, cbAvPairList, MsvAvNbDomainName, (PBYTE)NbDomainName.Buffer,
535 NbDomainName.Length))
538 if (!ntlm_av_pair_add(pAvPairList, cbAvPairList, MsvAvNbComputerName,
539 (PBYTE)NbComputerName.Buffer, NbComputerName.Length))
542 if (!ntlm_av_pair_add(pAvPairList, cbAvPairList, MsvAvDnsDomainName,
543 (PBYTE)DnsDomainName.Buffer, DnsDomainName.Length))
546 if (!ntlm_av_pair_add(pAvPairList, cbAvPairList, MsvAvDnsComputerName,
547 (PBYTE)DnsComputerName.Buffer, DnsComputerName.Length))
550 if (!ntlm_av_pair_add(pAvPairList, cbAvPairList, MsvAvTimestamp, context->Timestamp,
551 sizeof(context->Timestamp)))
556 ntlm_free_unicode_string(&NbDomainName);
557 ntlm_free_unicode_string(&NbComputerName);
558 ntlm_free_unicode_string(&DnsDomainName);
559 ntlm_free_unicode_string(&DnsComputerName);
563BOOL ntlm_construct_authenticate_target_info(
NTLM_CONTEXT* context)
565 ULONG AvPairsCount = 0;
566 size_t AvPairsValueLength = 0;
575 size_t cbAvTimestamp = 0;
576 size_t cbAvNbDomainName = 0;
577 size_t cbAvNbComputerName = 0;
578 size_t cbAvDnsDomainName = 0;
579 size_t cbAvDnsComputerName = 0;
580 size_t cbAvDnsTreeName = 0;
581 size_t cbChallengeTargetInfo = 0;
582 size_t cbAuthenticateTargetInfo = 0;
584 WINPR_ASSERT(context);
587 ChallengeTargetInfo = (
NTLM_AV_PAIR*)context->ChallengeTargetInfo.pvBuffer;
588 cbChallengeTargetInfo = context->ChallengeTargetInfo.cbBuffer;
589 AvNbDomainName = ntlm_av_pair_get(ChallengeTargetInfo, cbChallengeTargetInfo, MsvAvNbDomainName,
591 AvNbComputerName = ntlm_av_pair_get(ChallengeTargetInfo, cbChallengeTargetInfo,
592 MsvAvNbComputerName, &cbAvNbComputerName);
593 AvDnsDomainName = ntlm_av_pair_get(ChallengeTargetInfo, cbChallengeTargetInfo,
594 MsvAvDnsDomainName, &cbAvDnsDomainName);
595 AvDnsComputerName = ntlm_av_pair_get(ChallengeTargetInfo, cbChallengeTargetInfo,
596 MsvAvDnsComputerName, &cbAvDnsComputerName);
597 AvDnsTreeName = ntlm_av_pair_get(ChallengeTargetInfo, cbChallengeTargetInfo, MsvAvDnsTreeName,
599 AvTimestamp = ntlm_av_pair_get(ChallengeTargetInfo, cbChallengeTargetInfo, MsvAvTimestamp,
605 if (!ntlm_av_pair_get_len(AvNbDomainName, cbAvNbDomainName, &avLen))
608 AvPairsValueLength += avLen;
611 if (AvNbComputerName)
614 if (!ntlm_av_pair_get_len(AvNbComputerName, cbAvNbComputerName, &avLen))
617 AvPairsValueLength += avLen;
623 if (!ntlm_av_pair_get_len(AvDnsDomainName, cbAvDnsDomainName, &avLen))
626 AvPairsValueLength += avLen;
629 if (AvDnsComputerName)
632 if (!ntlm_av_pair_get_len(AvDnsComputerName, cbAvDnsComputerName, &avLen))
635 AvPairsValueLength += avLen;
641 if (!ntlm_av_pair_get_len(AvDnsTreeName, cbAvDnsTreeName, &avLen))
644 AvPairsValueLength += avLen;
648 AvPairsValueLength += 8;
653 AvPairsValueLength += 4;
656 if (context->SendSingleHostData)
659 ntlm_compute_single_host_data(context);
660 AvPairsValueLength += context->SingleHostData.Size;
668 if (!context->SuppressExtendedProtection)
675 AvPairsValueLength += 16;
676 ntlm_compute_channel_bindings(context);
678 if (context->ServicePrincipalName.Length > 0)
681 AvPairsValueLength += context->ServicePrincipalName.Length;
686 size_t size = ntlm_av_pair_list_size(AvPairsCount, AvPairsValueLength);
690 if (!sspi_SecBufferAlloc(&context->AuthenticateTargetInfo,
691 WINPR_ASSERTING_INT_CAST(uint32_t, size)))
695 AuthenticateTargetInfo = (
NTLM_AV_PAIR*)context->AuthenticateTargetInfo.pvBuffer;
696 cbAuthenticateTargetInfo = context->AuthenticateTargetInfo.cbBuffer;
698 if (!ntlm_av_pair_list_init(AuthenticateTargetInfo, cbAuthenticateTargetInfo))
703 if (!ntlm_av_pair_add_copy(AuthenticateTargetInfo, cbAuthenticateTargetInfo, AvNbDomainName,
708 if (AvNbComputerName)
710 if (!ntlm_av_pair_add_copy(AuthenticateTargetInfo, cbAuthenticateTargetInfo,
711 AvNbComputerName, cbAvNbComputerName))
717 if (!ntlm_av_pair_add_copy(AuthenticateTargetInfo, cbAuthenticateTargetInfo,
718 AvDnsDomainName, cbAvDnsDomainName))
722 if (AvDnsComputerName)
724 if (!ntlm_av_pair_add_copy(AuthenticateTargetInfo, cbAuthenticateTargetInfo,
725 AvDnsComputerName, cbAvDnsComputerName))
731 if (!ntlm_av_pair_add_copy(AuthenticateTargetInfo, cbAuthenticateTargetInfo, AvDnsTreeName,
738 if (!ntlm_av_pair_add_copy(AuthenticateTargetInfo, cbAuthenticateTargetInfo, AvTimestamp,
746 winpr_Data_Write_UINT32(&flags, MSV_AV_FLAGS_MESSAGE_INTEGRITY_CHECK);
748 if (!ntlm_av_pair_add(AuthenticateTargetInfo, cbAuthenticateTargetInfo, MsvAvFlags,
753 if (context->SendSingleHostData)
755 WINPR_ASSERT(context->SingleHostData.Size <= UINT16_MAX);
756 if (!ntlm_av_pair_add(AuthenticateTargetInfo, cbAuthenticateTargetInfo, MsvAvSingleHost,
757 (PBYTE)&context->SingleHostData,
758 (UINT16)context->SingleHostData.Size))
762 if (!context->SuppressExtendedProtection)
764 if (!ntlm_av_pair_add(AuthenticateTargetInfo, cbAuthenticateTargetInfo,
765 MsvAvChannelBindings, context->ChannelBindingsHash, 16))
768 if (context->ServicePrincipalName.Length > 0)
770 if (!ntlm_av_pair_add(AuthenticateTargetInfo, cbAuthenticateTargetInfo, MsvAvTargetName,
771 (PBYTE)context->ServicePrincipalName.Buffer,
772 context->ServicePrincipalName.Length))
781 ntlm_av_pair_get(ChallengeTargetInfo, cbChallengeTargetInfo, MsvAvEOL, &cbAvEOL);
783 size_t cbAvEntryLen = 0;
784 if (!ntlm_av_pair_get_len(AvEOL, cbAvEOL, &cbAvEntryLen))
792 sspi_SecBufferFree(&context->AuthenticateTargetInfo);