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);
338static char* get_name(COMPUTER_NAME_FORMAT type)
342 if (GetComputerNameExA(type,
nullptr, &nSize))
345 if (GetLastError() != ERROR_MORE_DATA)
348 char* computerName = calloc(1, nSize);
353 if (!GetComputerNameExA(type, computerName, &nSize))
363 WINPR_ATTR_UNUSED COMPUTER_NAME_FORMAT type)
369 char* name = get_name(ComputerNameNetBIOS);
376 pName->Buffer = ConvertUtf8ToWCharAlloc(name, &len);
379 if (!pName->Buffer || (len == 0) || (len > UINT16_MAX /
sizeof(WCHAR)))
382 pName->Buffer =
nullptr;
386 pName->Length = (USHORT)((len) *
sizeof(WCHAR));
387 pName->MaximumLength = pName->Length;
395 if (string->Length > 0)
397 free(string->Buffer);
398 string->Buffer =
nullptr;
400 string->MaximumLength = 0;
433static BOOL ntlm_md5_update_uint32_be(WINPR_DIGEST_CTX* md5, UINT32 num)
436 be32[0] = (num >> 0) & 0xFF;
437 be32[1] = (num >> 8) & 0xFF;
438 be32[2] = (num >> 16) & 0xFF;
439 be32[3] = (num >> 24) & 0xFF;
440 return winpr_Digest_Update(md5, be32, 4);
443static void ntlm_compute_channel_bindings(
NTLM_CONTEXT* context)
445 WINPR_DIGEST_CTX* md5 =
nullptr;
446 BYTE* ChannelBindingToken =
nullptr;
447 UINT32 ChannelBindingTokenLength = 0;
450 WINPR_ASSERT(context);
452 ZeroMemory(context->ChannelBindingsHash, WINPR_MD5_DIGEST_LENGTH);
453 ChannelBindings = context->Bindings.Bindings;
455 if (!ChannelBindings)
458 if (!(md5 = winpr_Digest_New()))
461 if (!winpr_Digest_Init(md5, WINPR_MD_MD5))
465 ChannelBindingToken = &((BYTE*)ChannelBindings)[ChannelBindings->dwApplicationDataOffset];
467 if (!ntlm_md5_update_uint32_be(md5, ChannelBindings->dwInitiatorAddrType))
470 if (!ntlm_md5_update_uint32_be(md5, ChannelBindings->cbInitiatorLength))
473 if (!ntlm_md5_update_uint32_be(md5, ChannelBindings->dwAcceptorAddrType))
476 if (!ntlm_md5_update_uint32_be(md5, ChannelBindings->cbAcceptorLength))
479 if (!ntlm_md5_update_uint32_be(md5, ChannelBindings->cbApplicationDataLength))
482 if (!winpr_Digest_Update(md5, (
void*)ChannelBindingToken, ChannelBindingTokenLength))
485 if (!winpr_Digest_Final(md5, context->ChannelBindingsHash, WINPR_MD5_DIGEST_LENGTH))
489 winpr_Digest_Free(md5);
492static void ntlm_compute_single_host_data(
NTLM_CONTEXT* context)
494 WINPR_ASSERT(context);
503 winpr_Data_Write_UINT32(&context->SingleHostData.Size, 48);
504 winpr_Data_Write_UINT32(&context->SingleHostData.Z4, 0);
505 winpr_Data_Write_UINT32(&context->SingleHostData.DataPresent, 1);
506 winpr_Data_Write_UINT32(&context->SingleHostData.CustomData, SECURITY_MANDATORY_MEDIUM_RID);
507 FillMemory(context->SingleHostData.MachineID, 32, 0xAA);
510BOOL ntlm_construct_challenge_target_info(
NTLM_CONTEXT* context)
513 ULONG AvPairsCount = 0;
514 ULONG AvPairsLength = 0;
516 size_t cbAvPairList = 0;
522 WINPR_ASSERT(context);
524 if (ntlm_get_target_computer_name(&NbDomainName, ComputerNameNetBIOS) < 0)
527 NbComputerName.Buffer =
nullptr;
529 if (ntlm_get_target_computer_name(&NbComputerName, ComputerNameNetBIOS) < 0)
532 DnsDomainName.Buffer =
nullptr;
534 if (ntlm_get_target_computer_name(&DnsDomainName, ComputerNameDnsDomain) < 0)
537 DnsComputerName.Buffer =
nullptr;
539 if (ntlm_get_target_computer_name(&DnsComputerName, ComputerNameDnsHostname) < 0)
543 AvPairsLength = NbDomainName.Length + NbComputerName.Length + DnsDomainName.Length +
544 DnsComputerName.Length + 8;
546 const size_t length = ntlm_av_pair_list_size(AvPairsCount, AvPairsLength);
547 if (!sspi_SecBufferAlloc(&context->ChallengeTargetInfo,
548 WINPR_ASSERTING_INT_CAST(uint32_t, length)))
552 pAvPairList = (
NTLM_AV_PAIR*)context->ChallengeTargetInfo.pvBuffer;
553 cbAvPairList = context->ChallengeTargetInfo.cbBuffer;
555 if (!ntlm_av_pair_list_init(pAvPairList, cbAvPairList))
558 if (!ntlm_av_pair_add(pAvPairList, cbAvPairList, MsvAvNbDomainName, (PBYTE)NbDomainName.Buffer,
559 NbDomainName.Length))
562 if (!ntlm_av_pair_add(pAvPairList, cbAvPairList, MsvAvNbComputerName,
563 (PBYTE)NbComputerName.Buffer, NbComputerName.Length))
566 if (!ntlm_av_pair_add(pAvPairList, cbAvPairList, MsvAvDnsDomainName,
567 (PBYTE)DnsDomainName.Buffer, DnsDomainName.Length))
570 if (!ntlm_av_pair_add(pAvPairList, cbAvPairList, MsvAvDnsComputerName,
571 (PBYTE)DnsComputerName.Buffer, DnsComputerName.Length))
574 if (!ntlm_av_pair_add(pAvPairList, cbAvPairList, MsvAvTimestamp, context->Timestamp,
575 sizeof(context->Timestamp)))
580 ntlm_free_unicode_string(&NbDomainName);
581 ntlm_free_unicode_string(&NbComputerName);
582 ntlm_free_unicode_string(&DnsDomainName);
583 ntlm_free_unicode_string(&DnsComputerName);
587BOOL ntlm_construct_authenticate_target_info(
NTLM_CONTEXT* context)
589 ULONG AvPairsCount = 0;
590 size_t AvPairsValueLength = 0;
599 size_t cbAvTimestamp = 0;
600 size_t cbAvNbDomainName = 0;
601 size_t cbAvNbComputerName = 0;
602 size_t cbAvDnsDomainName = 0;
603 size_t cbAvDnsComputerName = 0;
604 size_t cbAvDnsTreeName = 0;
605 size_t cbChallengeTargetInfo = 0;
606 size_t cbAuthenticateTargetInfo = 0;
608 WINPR_ASSERT(context);
611 ChallengeTargetInfo = (
NTLM_AV_PAIR*)context->ChallengeTargetInfo.pvBuffer;
612 cbChallengeTargetInfo = context->ChallengeTargetInfo.cbBuffer;
613 AvNbDomainName = ntlm_av_pair_get(ChallengeTargetInfo, cbChallengeTargetInfo, MsvAvNbDomainName,
615 AvNbComputerName = ntlm_av_pair_get(ChallengeTargetInfo, cbChallengeTargetInfo,
616 MsvAvNbComputerName, &cbAvNbComputerName);
617 AvDnsDomainName = ntlm_av_pair_get(ChallengeTargetInfo, cbChallengeTargetInfo,
618 MsvAvDnsDomainName, &cbAvDnsDomainName);
619 AvDnsComputerName = ntlm_av_pair_get(ChallengeTargetInfo, cbChallengeTargetInfo,
620 MsvAvDnsComputerName, &cbAvDnsComputerName);
621 AvDnsTreeName = ntlm_av_pair_get(ChallengeTargetInfo, cbChallengeTargetInfo, MsvAvDnsTreeName,
623 AvTimestamp = ntlm_av_pair_get(ChallengeTargetInfo, cbChallengeTargetInfo, MsvAvTimestamp,
629 if (!ntlm_av_pair_get_len(AvNbDomainName, cbAvNbDomainName, &avLen))
632 AvPairsValueLength += avLen;
635 if (AvNbComputerName)
638 if (!ntlm_av_pair_get_len(AvNbComputerName, cbAvNbComputerName, &avLen))
641 AvPairsValueLength += avLen;
647 if (!ntlm_av_pair_get_len(AvDnsDomainName, cbAvDnsDomainName, &avLen))
650 AvPairsValueLength += avLen;
653 if (AvDnsComputerName)
656 if (!ntlm_av_pair_get_len(AvDnsComputerName, cbAvDnsComputerName, &avLen))
659 AvPairsValueLength += avLen;
665 if (!ntlm_av_pair_get_len(AvDnsTreeName, cbAvDnsTreeName, &avLen))
668 AvPairsValueLength += avLen;
672 AvPairsValueLength += 8;
677 AvPairsValueLength += 4;
680 if (context->SendSingleHostData)
683 ntlm_compute_single_host_data(context);
684 AvPairsValueLength += context->SingleHostData.Size;
692 if (!context->SuppressExtendedProtection)
699 AvPairsValueLength += 16;
700 ntlm_compute_channel_bindings(context);
702 if (context->ServicePrincipalName.Length > 0)
705 AvPairsValueLength += context->ServicePrincipalName.Length;
710 size_t size = ntlm_av_pair_list_size(AvPairsCount, AvPairsValueLength);
714 if (!sspi_SecBufferAlloc(&context->AuthenticateTargetInfo,
715 WINPR_ASSERTING_INT_CAST(uint32_t, size)))
719 AuthenticateTargetInfo = (
NTLM_AV_PAIR*)context->AuthenticateTargetInfo.pvBuffer;
720 cbAuthenticateTargetInfo = context->AuthenticateTargetInfo.cbBuffer;
722 if (!ntlm_av_pair_list_init(AuthenticateTargetInfo, cbAuthenticateTargetInfo))
727 if (!ntlm_av_pair_add_copy(AuthenticateTargetInfo, cbAuthenticateTargetInfo, AvNbDomainName,
732 if (AvNbComputerName)
734 if (!ntlm_av_pair_add_copy(AuthenticateTargetInfo, cbAuthenticateTargetInfo,
735 AvNbComputerName, cbAvNbComputerName))
741 if (!ntlm_av_pair_add_copy(AuthenticateTargetInfo, cbAuthenticateTargetInfo,
742 AvDnsDomainName, cbAvDnsDomainName))
746 if (AvDnsComputerName)
748 if (!ntlm_av_pair_add_copy(AuthenticateTargetInfo, cbAuthenticateTargetInfo,
749 AvDnsComputerName, cbAvDnsComputerName))
755 if (!ntlm_av_pair_add_copy(AuthenticateTargetInfo, cbAuthenticateTargetInfo, AvDnsTreeName,
762 if (!ntlm_av_pair_add_copy(AuthenticateTargetInfo, cbAuthenticateTargetInfo, AvTimestamp,
770 winpr_Data_Write_UINT32(&flags, MSV_AV_FLAGS_MESSAGE_INTEGRITY_CHECK);
772 if (!ntlm_av_pair_add(AuthenticateTargetInfo, cbAuthenticateTargetInfo, MsvAvFlags,
777 if (context->SendSingleHostData)
779 WINPR_ASSERT(context->SingleHostData.Size <= UINT16_MAX);
780 if (!ntlm_av_pair_add(AuthenticateTargetInfo, cbAuthenticateTargetInfo, MsvAvSingleHost,
781 (PBYTE)&context->SingleHostData,
782 (UINT16)context->SingleHostData.Size))
786 if (!context->SuppressExtendedProtection)
788 if (!ntlm_av_pair_add(AuthenticateTargetInfo, cbAuthenticateTargetInfo,
789 MsvAvChannelBindings, context->ChannelBindingsHash, 16))
792 if (context->ServicePrincipalName.Length > 0)
794 if (!ntlm_av_pair_add(AuthenticateTargetInfo, cbAuthenticateTargetInfo, MsvAvTargetName,
795 (PBYTE)context->ServicePrincipalName.Buffer,
796 context->ServicePrincipalName.Length))
805 ntlm_av_pair_get(ChallengeTargetInfo, cbChallengeTargetInfo, MsvAvEOL, &cbAvEOL);
807 size_t cbAvEntryLen = 0;
808 if (!ntlm_av_pair_get_len(AvEOL, cbAvEOL, &cbAvEntryLen))
816 sspi_SecBufferFree(&context->AuthenticateTargetInfo);