952 lines
26 KiB
C
952 lines
26 KiB
C
/**
|
|
* FreeRDP: A Remote Desktop Protocol Implementation
|
|
* NTLM Security Package (Compute)
|
|
*
|
|
* Copyright 2011-2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#include "ntlm.h"
|
|
#include "../sspi.h"
|
|
|
|
#include <time.h>
|
|
#include <openssl/des.h>
|
|
#include <openssl/md4.h>
|
|
#include <openssl/hmac.h>
|
|
#include <openssl/rand.h>
|
|
#include <openssl/engine.h>
|
|
#include <freerdp/crypto/crypto.h>
|
|
|
|
#include <freerdp/utils/stream.h>
|
|
#include <freerdp/utils/memory.h>
|
|
#include <freerdp/utils/hexdump.h>
|
|
|
|
#include "ntlm_compute.h"
|
|
|
|
const char* const AV_PAIRS_STRINGS[] =
|
|
{
|
|
"MsvAvEOL",
|
|
"MsvAvNbComputerName",
|
|
"MsvAvNbDomainName",
|
|
"MsvAvDnsComputerName",
|
|
"MsvAvDnsDomainName",
|
|
"MsvAvDnsTreeName",
|
|
"MsvAvFlags",
|
|
"MsvAvTimestamp",
|
|
"MsvAvRestrictions",
|
|
"MsvAvTargetName",
|
|
"MsvChannelBindings"
|
|
};
|
|
|
|
static const char lm_magic[] = "KGS!@#$%";
|
|
|
|
static const char client_sign_magic[] = "session key to client-to-server signing key magic constant";
|
|
static const char server_sign_magic[] = "session key to server-to-client signing key magic constant";
|
|
static const char client_seal_magic[] = "session key to client-to-server sealing key magic constant";
|
|
static const char server_seal_magic[] = "session key to server-to-client sealing key magic constant";
|
|
|
|
/**
|
|
* Output Restriction_Encoding.\n
|
|
* Restriction_Encoding @msdn{cc236647}
|
|
* @param NTLM context
|
|
*/
|
|
|
|
void ntlm_output_restriction_encoding(NTLM_CONTEXT* context)
|
|
{
|
|
STREAM* s;
|
|
AV_PAIR* restrictions = &context->av_pairs->Restrictions;
|
|
|
|
uint8 machineID[32] =
|
|
"\x3A\x15\x8E\xA6\x75\x82\xD8\xF7\x3E\x06\xFA\x7A\xB4\xDF\xFD\x43"
|
|
"\x84\x6C\x02\x3A\xFD\x5A\x94\xFE\xCF\x97\x0F\x3D\x19\x2C\x38\x20";
|
|
|
|
restrictions->value = xmalloc(48);
|
|
restrictions->length = 48;
|
|
|
|
s = stream_new(0);
|
|
stream_attach(s, restrictions->value, restrictions->length);
|
|
|
|
stream_write_uint32(s, 48); /* Size */
|
|
stream_write_zero(s, 4); /* Z4 (set to zero) */
|
|
|
|
/* IntegrityLevel (bit 31 set to 1) */
|
|
stream_write_uint8(s, 1);
|
|
stream_write_zero(s, 3);
|
|
|
|
stream_write_uint32(s, 0x00002000); /* SubjectIntegrityLevel */
|
|
stream_write(s, machineID, 32); /* MachineID */
|
|
|
|
xfree(s);
|
|
}
|
|
|
|
/**
|
|
* Output TargetName.\n
|
|
* @param NTLM context
|
|
*/
|
|
|
|
void ntlm_output_target_name(NTLM_CONTEXT* context)
|
|
{
|
|
STREAM* s;
|
|
AV_PAIR* TargetName = &context->av_pairs->TargetName;
|
|
|
|
/*
|
|
* TODO: No idea what should be set here (observed MsvAvTargetName = MsvAvDnsComputerName or
|
|
* MsvAvTargetName should be the name of the service be accessed after authentication)
|
|
* here used: "TERMSRV/192.168.0.123" in unicode (Dmitrij Jasnov)
|
|
*/
|
|
uint8 name[42] =
|
|
"\x54\x00\x45\x00\x52\x00\x4d\x00\x53\x00\x52\x00\x56\x00\x2f\x00\x31\x00\x39\x00\x32"
|
|
"\x00\x2e\x00\x31\x00\x36\x00\x38\x00\x2e\x00\x30\x00\x2e\x00\x31\x00\x32\x00\x33\x00";
|
|
|
|
TargetName->length = 42;
|
|
TargetName->value = (uint8*) xmalloc(TargetName->length);
|
|
|
|
s = stream_new(0);
|
|
stream_attach(s, TargetName->value, TargetName->length);
|
|
|
|
stream_write(s, name, TargetName->length);
|
|
|
|
xfree(s);
|
|
}
|
|
|
|
/**
|
|
* Output ChannelBindings.\n
|
|
* @param NTLM context
|
|
*/
|
|
|
|
void ntlm_output_channel_bindings(NTLM_CONTEXT* context)
|
|
{
|
|
STREAM* s;
|
|
AV_PAIR* ChannelBindings = &context->av_pairs->ChannelBindings;
|
|
|
|
ChannelBindings->value = (uint8*) xmalloc(48);
|
|
ChannelBindings->length = 16;
|
|
|
|
s = stream_new(0);
|
|
stream_attach(s, ChannelBindings->value, ChannelBindings->length);
|
|
|
|
stream_write_zero(s, 16); /* an all-zero value of the hash is used to indicate absence of channel bindings */
|
|
|
|
xfree(s);
|
|
}
|
|
|
|
/**
|
|
* Input array of AV_PAIRs.\n
|
|
* AV_PAIR @msdn{cc236646}
|
|
* @param NTLM context
|
|
* @param s
|
|
*/
|
|
|
|
void ntlm_input_av_pairs(NTLM_CONTEXT* context, STREAM* s)
|
|
{
|
|
AV_ID AvId;
|
|
uint16 AvLen;
|
|
uint8* value;
|
|
AV_PAIRS* av_pairs = context->av_pairs;
|
|
|
|
#ifdef WITH_DEBUG_NTLM
|
|
printf("AV_PAIRS = {\n");
|
|
#endif
|
|
|
|
do
|
|
{
|
|
value = NULL;
|
|
stream_read_uint16(s, AvId);
|
|
stream_read_uint16(s, AvLen);
|
|
|
|
if (AvLen > 0)
|
|
{
|
|
if (AvId != MsvAvFlags)
|
|
{
|
|
value = xmalloc(AvLen);
|
|
stream_read(s, value, AvLen);
|
|
}
|
|
else
|
|
{
|
|
stream_read_uint32(s, av_pairs->Flags);
|
|
}
|
|
}
|
|
|
|
switch (AvId)
|
|
{
|
|
case MsvAvNbComputerName:
|
|
av_pairs->NbComputerName.length = AvLen;
|
|
av_pairs->NbComputerName.value = value;
|
|
break;
|
|
|
|
case MsvAvNbDomainName:
|
|
av_pairs->NbDomainName.length = AvLen;
|
|
av_pairs->NbDomainName.value = value;
|
|
break;
|
|
|
|
case MsvAvDnsComputerName:
|
|
av_pairs->DnsComputerName.length = AvLen;
|
|
av_pairs->DnsComputerName.value = value;
|
|
break;
|
|
|
|
case MsvAvDnsDomainName:
|
|
av_pairs->DnsDomainName.length = AvLen;
|
|
av_pairs->DnsDomainName.value = value;
|
|
break;
|
|
|
|
case MsvAvDnsTreeName:
|
|
av_pairs->DnsTreeName.length = AvLen;
|
|
av_pairs->DnsTreeName.value = value;
|
|
break;
|
|
|
|
case MsvAvTimestamp:
|
|
av_pairs->Timestamp.length = AvLen;
|
|
av_pairs->Timestamp.value = value;
|
|
break;
|
|
|
|
case MsvAvRestrictions:
|
|
av_pairs->Restrictions.length = AvLen;
|
|
av_pairs->Restrictions.value = value;
|
|
break;
|
|
|
|
case MsvAvTargetName:
|
|
av_pairs->TargetName.length = AvLen;
|
|
av_pairs->TargetName.value = value;
|
|
break;
|
|
|
|
case MsvChannelBindings:
|
|
av_pairs->ChannelBindings.length = AvLen;
|
|
av_pairs->ChannelBindings.value = value;
|
|
break;
|
|
|
|
default:
|
|
if (value != NULL)
|
|
xfree(value);
|
|
break;
|
|
}
|
|
|
|
#ifdef WITH_DEBUG_NTLM
|
|
if (AvId < 10)
|
|
printf("\tAvId: %s, AvLen: %d\n", AV_PAIRS_STRINGS[AvId], AvLen);
|
|
else
|
|
printf("\tAvId: %s, AvLen: %d\n", "Unknown", AvLen);
|
|
|
|
freerdp_hexdump(value, AvLen);
|
|
#endif
|
|
}
|
|
while (AvId != MsvAvEOL);
|
|
|
|
#ifdef WITH_DEBUG_NTLM
|
|
printf("}\n");
|
|
#endif
|
|
}
|
|
|
|
/**
|
|
* Output array of AV_PAIRs.\n
|
|
* AV_PAIR @msdn{cc236646}
|
|
* @param NTLM context
|
|
* @param s
|
|
*/
|
|
|
|
void ntlm_output_av_pairs(NTLM_CONTEXT* context, SecBuffer* buffer)
|
|
{
|
|
STREAM* s;
|
|
AV_PAIRS* av_pairs = context->av_pairs;
|
|
|
|
s = stream_new(0);
|
|
stream_attach(s, buffer->pvBuffer, buffer->cbBuffer);
|
|
|
|
if (av_pairs->NbDomainName.length > 0)
|
|
{
|
|
stream_write_uint16(s, MsvAvNbDomainName); /* AvId */
|
|
stream_write_uint16(s, av_pairs->NbDomainName.length); /* AvLen */
|
|
stream_write(s, av_pairs->NbDomainName.value, av_pairs->NbDomainName.length); /* Value */
|
|
}
|
|
|
|
if (av_pairs->NbComputerName.length > 0)
|
|
{
|
|
stream_write_uint16(s, MsvAvNbComputerName); /* AvId */
|
|
stream_write_uint16(s, av_pairs->NbComputerName.length); /* AvLen */
|
|
stream_write(s, av_pairs->NbComputerName.value, av_pairs->NbComputerName.length); /* Value */
|
|
}
|
|
|
|
if (av_pairs->DnsDomainName.length > 0)
|
|
{
|
|
stream_write_uint16(s, MsvAvDnsDomainName); /* AvId */
|
|
stream_write_uint16(s, av_pairs->DnsDomainName.length); /* AvLen */
|
|
stream_write(s, av_pairs->DnsDomainName.value, av_pairs->DnsDomainName.length); /* Value */
|
|
}
|
|
|
|
if (av_pairs->DnsComputerName.length > 0)
|
|
{
|
|
stream_write_uint16(s, MsvAvDnsComputerName); /* AvId */
|
|
stream_write_uint16(s, av_pairs->DnsComputerName.length); /* AvLen */
|
|
stream_write(s, av_pairs->DnsComputerName.value, av_pairs->DnsComputerName.length); /* Value */
|
|
}
|
|
|
|
if (av_pairs->DnsTreeName.length > 0)
|
|
{
|
|
stream_write_uint16(s, MsvAvDnsTreeName); /* AvId */
|
|
stream_write_uint16(s, av_pairs->DnsTreeName.length); /* AvLen */
|
|
stream_write(s, av_pairs->DnsTreeName.value, av_pairs->DnsTreeName.length); /* Value */
|
|
}
|
|
|
|
if (av_pairs->Timestamp.length > 0)
|
|
{
|
|
stream_write_uint16(s, MsvAvTimestamp); /* AvId */
|
|
stream_write_uint16(s, av_pairs->Timestamp.length); /* AvLen */
|
|
stream_write(s, av_pairs->Timestamp.value, av_pairs->Timestamp.length); /* Value */
|
|
}
|
|
|
|
if (av_pairs->Flags > 0)
|
|
{
|
|
stream_write_uint16(s, MsvAvFlags); /* AvId */
|
|
stream_write_uint16(s, 4); /* AvLen */
|
|
stream_write_uint32(s, av_pairs->Flags); /* Value */
|
|
}
|
|
|
|
if (av_pairs->Restrictions.length > 0)
|
|
{
|
|
stream_write_uint16(s, MsvAvRestrictions); /* AvId */
|
|
stream_write_uint16(s, av_pairs->Restrictions.length); /* AvLen */
|
|
stream_write(s, av_pairs->Restrictions.value, av_pairs->Restrictions.length); /* Value */
|
|
}
|
|
|
|
if (av_pairs->ChannelBindings.length > 0)
|
|
{
|
|
stream_write_uint16(s, MsvChannelBindings); /* AvId */
|
|
stream_write_uint16(s, av_pairs->ChannelBindings.length); /* AvLen */
|
|
stream_write(s, av_pairs->ChannelBindings.value, av_pairs->ChannelBindings.length); /* Value */
|
|
}
|
|
|
|
if (av_pairs->TargetName.length > 0)
|
|
{
|
|
stream_write_uint16(s, MsvAvTargetName); /* AvId */
|
|
stream_write_uint16(s, av_pairs->TargetName.length); /* AvLen */
|
|
stream_write(s, av_pairs->TargetName.value, av_pairs->TargetName.length); /* Value */
|
|
}
|
|
|
|
/* This indicates the end of the AV_PAIR array */
|
|
stream_write_uint16(s, MsvAvEOL); /* AvId */
|
|
stream_write_uint16(s, 0); /* AvLen */
|
|
|
|
if (context->ntlm_v2)
|
|
{
|
|
stream_write_zero(s, 8);
|
|
}
|
|
|
|
xfree(s);
|
|
}
|
|
|
|
/**
|
|
* Compute AV_PAIRs length.\n
|
|
* AV_PAIR @msdn{cc236646}
|
|
* @param NTLM context
|
|
*/
|
|
|
|
int ntlm_compute_av_pairs_length(NTLM_CONTEXT* context)
|
|
{
|
|
int length = 0;
|
|
AV_PAIRS* av_pairs = context->av_pairs;
|
|
|
|
if (av_pairs->NbDomainName.length > 0)
|
|
length += av_pairs->NbDomainName.length + 4;
|
|
|
|
if (av_pairs->NbComputerName.length > 0)
|
|
length += av_pairs->NbComputerName.length + 4;
|
|
|
|
if (av_pairs->DnsDomainName.length > 0)
|
|
length += av_pairs->DnsDomainName.length + 4;
|
|
|
|
if (av_pairs->DnsComputerName.length > 0)
|
|
length += av_pairs->DnsComputerName.length + 4;
|
|
|
|
if (av_pairs->DnsTreeName.length > 0)
|
|
length += av_pairs->DnsTreeName.length + 4;
|
|
|
|
if (av_pairs->Timestamp.length > 0)
|
|
length += av_pairs->Timestamp.length;
|
|
|
|
if (av_pairs->Flags > 0)
|
|
length += 4 + 4;
|
|
|
|
if (av_pairs->Restrictions.length > 0)
|
|
length += av_pairs->Restrictions.length + 4;
|
|
|
|
if (av_pairs->ChannelBindings.length > 0)
|
|
length += av_pairs->ChannelBindings.length + 4;
|
|
|
|
if (av_pairs->TargetName.length > 0)
|
|
length += av_pairs->TargetName.length + 4;
|
|
|
|
length += 4;
|
|
|
|
if (context->ntlm_v2)
|
|
length += 8;
|
|
|
|
return length;
|
|
}
|
|
|
|
/**
|
|
* Populate array of AV_PAIRs.\n
|
|
* AV_PAIR @msdn{cc236646}
|
|
* @param NTLM context
|
|
*/
|
|
|
|
void ntlm_populate_av_pairs(NTLM_CONTEXT* context)
|
|
{
|
|
int length;
|
|
AV_PAIRS* av_pairs = context->av_pairs;
|
|
|
|
/* MsvAvFlags */
|
|
av_pairs->Flags = 0x00000002; /* Indicates the present of a Message Integrity Check (MIC) */
|
|
|
|
/* Restriction_Encoding */
|
|
ntlm_output_restriction_encoding(context);
|
|
|
|
/* TargetName */
|
|
ntlm_output_target_name(context);
|
|
|
|
/* ChannelBindings */
|
|
ntlm_output_channel_bindings(context);
|
|
|
|
length = ntlm_compute_av_pairs_length(context);
|
|
sspi_SecBufferAlloc(&context->TargetInfo, length);
|
|
ntlm_output_av_pairs(context, &context->TargetInfo);
|
|
}
|
|
|
|
/**
|
|
* Print array of AV_PAIRs.\n
|
|
* AV_PAIR @msdn{cc236646}
|
|
* @param NTLM context
|
|
*/
|
|
|
|
void ntlm_print_av_pairs(NTLM_CONTEXT* context)
|
|
{
|
|
AV_PAIRS* av_pairs = context->av_pairs;
|
|
|
|
printf("AV_PAIRS = {\n");
|
|
|
|
if (av_pairs->NbDomainName.length > 0)
|
|
{
|
|
printf("\tAvId: MsvAvNbDomainName AvLen: %d\n", av_pairs->NbDomainName.length);
|
|
freerdp_hexdump(av_pairs->NbDomainName.value, av_pairs->NbDomainName.length);
|
|
}
|
|
|
|
if (av_pairs->NbComputerName.length > 0)
|
|
{
|
|
printf("\tAvId: MsvAvNbComputerName AvLen: %d\n", av_pairs->NbComputerName.length);
|
|
freerdp_hexdump(av_pairs->NbComputerName.value, av_pairs->NbComputerName.length);
|
|
}
|
|
|
|
if (av_pairs->DnsDomainName.length > 0)
|
|
{
|
|
printf("\tAvId: MsvAvDnsDomainName AvLen: %d\n", av_pairs->DnsDomainName.length);
|
|
freerdp_hexdump(av_pairs->DnsDomainName.value, av_pairs->DnsDomainName.length);
|
|
}
|
|
|
|
if (av_pairs->DnsComputerName.length > 0)
|
|
{
|
|
printf("\tAvId: MsvAvDnsComputerName AvLen: %d\n", av_pairs->DnsComputerName.length);
|
|
freerdp_hexdump(av_pairs->DnsComputerName.value, av_pairs->DnsComputerName.length);
|
|
}
|
|
|
|
if (av_pairs->DnsTreeName.length > 0)
|
|
{
|
|
printf("\tAvId: MsvAvDnsTreeName AvLen: %d\n", av_pairs->DnsTreeName.length);
|
|
freerdp_hexdump(av_pairs->DnsTreeName.value, av_pairs->DnsTreeName.length);
|
|
}
|
|
|
|
if (av_pairs->Timestamp.length > 0)
|
|
{
|
|
printf("\tAvId: MsvAvTimestamp AvLen: %d\n", av_pairs->Timestamp.length);
|
|
freerdp_hexdump(av_pairs->Timestamp.value, av_pairs->Timestamp.length);
|
|
}
|
|
|
|
if (av_pairs->Flags > 0)
|
|
{
|
|
printf("\tAvId: MsvAvFlags AvLen: %d\n", 4);
|
|
printf("0x%08X\n", av_pairs->Flags);
|
|
}
|
|
|
|
if (av_pairs->Restrictions.length > 0)
|
|
{
|
|
printf("\tAvId: MsvAvRestrictions AvLen: %d\n", av_pairs->Restrictions.length);
|
|
freerdp_hexdump(av_pairs->Restrictions.value, av_pairs->Restrictions.length);
|
|
}
|
|
|
|
if (av_pairs->ChannelBindings.length > 0)
|
|
{
|
|
printf("\tAvId: MsvChannelBindings AvLen: %d\n", av_pairs->ChannelBindings.length);
|
|
freerdp_hexdump(av_pairs->ChannelBindings.value, av_pairs->ChannelBindings.length);
|
|
}
|
|
|
|
if (av_pairs->TargetName.length > 0)
|
|
{
|
|
printf("\tAvId: MsvAvTargetName AvLen: %d\n", av_pairs->TargetName.length);
|
|
freerdp_hexdump(av_pairs->TargetName.value, av_pairs->TargetName.length);
|
|
}
|
|
|
|
printf("}\n");
|
|
}
|
|
|
|
/**
|
|
* Free array of AV_PAIRs.\n
|
|
* AV_PAIR @msdn{cc236646}
|
|
* @param NTLM context
|
|
*/
|
|
|
|
void ntlm_free_av_pairs(NTLM_CONTEXT* context)
|
|
{
|
|
AV_PAIRS* av_pairs = context->av_pairs;
|
|
|
|
if (av_pairs != NULL)
|
|
{
|
|
if (av_pairs->NbComputerName.value != NULL)
|
|
xfree(av_pairs->NbComputerName.value);
|
|
if (av_pairs->NbDomainName.value != NULL)
|
|
xfree(av_pairs->NbDomainName.value);
|
|
if (av_pairs->DnsComputerName.value != NULL)
|
|
xfree(av_pairs->DnsComputerName.value);
|
|
if (av_pairs->DnsDomainName.value != NULL)
|
|
xfree(av_pairs->DnsDomainName.value);
|
|
if (av_pairs->DnsTreeName.value != NULL)
|
|
xfree(av_pairs->DnsTreeName.value);
|
|
if (av_pairs->Timestamp.value != NULL)
|
|
xfree(av_pairs->Timestamp.value);
|
|
if (av_pairs->Restrictions.value != NULL)
|
|
xfree(av_pairs->Restrictions.value);
|
|
if (av_pairs->TargetName.value != NULL)
|
|
xfree(av_pairs->TargetName.value);
|
|
if (av_pairs->ChannelBindings.value != NULL)
|
|
xfree(av_pairs->ChannelBindings.value);
|
|
|
|
xfree(av_pairs);
|
|
}
|
|
|
|
context->av_pairs = NULL;
|
|
}
|
|
|
|
/**
|
|
* Get current time, in tenths of microseconds since midnight of January 1, 1601.
|
|
* @param[out] timestamp 64-bit little-endian timestamp
|
|
*/
|
|
|
|
void ntlm_current_time(uint8* timestamp)
|
|
{
|
|
uint64 time64;
|
|
|
|
/* Timestamp (8 bytes), represented as the number of tenths of microseconds since midnight of January 1, 1601 */
|
|
time64 = time(NULL) + 11644473600LL; /* Seconds since January 1, 1601 */
|
|
time64 *= 10000000; /* Convert timestamp to tenths of a microsecond */
|
|
|
|
memcpy(timestamp, &time64, 8); /* Copy into timestamp in little-endian */
|
|
}
|
|
|
|
/**
|
|
* Generate timestamp for AUTHENTICATE_MESSAGE.
|
|
* @param NTLM context
|
|
*/
|
|
|
|
void ntlm_generate_timestamp(NTLM_CONTEXT* context)
|
|
{
|
|
ntlm_current_time(context->Timestamp);
|
|
|
|
if (context->ntlm_v2)
|
|
{
|
|
if (context->av_pairs->Timestamp.length == 8)
|
|
{
|
|
memcpy(context->av_pairs->Timestamp.value, context->Timestamp, 8);
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (context->av_pairs->Timestamp.length != 8)
|
|
{
|
|
context->av_pairs->Timestamp.length = 8;
|
|
context->av_pairs->Timestamp.value = xmalloc(context->av_pairs->Timestamp.length);
|
|
}
|
|
|
|
memcpy(context->av_pairs->Timestamp.value, context->Timestamp, 8);
|
|
}
|
|
}
|
|
|
|
void ntlm_compute_ntlm_hash(uint16* password, uint32 length, char* hash)
|
|
{
|
|
/* NTLMv1("password") = 8846F7EAEE8FB117AD06BDD830B7586C */
|
|
|
|
MD4_CTX md4_ctx;
|
|
|
|
/* Password needs to be in unicode */
|
|
|
|
/* Apply the MD4 digest algorithm on the password in unicode, the result is the NTLM hash */
|
|
|
|
MD4_Init(&md4_ctx);
|
|
MD4_Update(&md4_ctx, password, length);
|
|
MD4_Final((void*) hash, &md4_ctx);
|
|
}
|
|
|
|
void ntlm_compute_ntlm_v2_hash(NTLM_CONTEXT* context, char* hash)
|
|
{
|
|
char* p;
|
|
SecBuffer buffer;
|
|
char ntlm_hash[16];
|
|
|
|
sspi_SecBufferAlloc(&buffer, context->identity.UserLength + context->identity.DomainLength);
|
|
p = (char*) buffer.pvBuffer;
|
|
|
|
/* First, compute the NTLMv1 hash of the password */
|
|
ntlm_compute_ntlm_hash(context->identity.Password, context->identity.PasswordLength, ntlm_hash);
|
|
|
|
/* Concatenate(Uppercase(username),domain)*/
|
|
memcpy(p, context->identity.User, context->identity.UserLength);
|
|
freerdp_uniconv_uppercase(context->uniconv, p, context->identity.UserLength / 2);
|
|
|
|
memcpy(&p[context->identity.UserLength], context->identity.Domain, context->identity.DomainLength);
|
|
|
|
/* Compute the HMAC-MD5 hash of the above value using the NTLMv1 hash as the key, the result is the NTLMv2 hash */
|
|
HMAC(EVP_md5(), (void*) ntlm_hash, 16, buffer.pvBuffer, buffer.cbBuffer, (void*) hash, NULL);
|
|
|
|
sspi_SecBufferFree(&buffer);
|
|
}
|
|
|
|
void ntlm_compute_lm_v2_response(NTLM_CONTEXT* context)
|
|
{
|
|
char* response;
|
|
char value[16];
|
|
char ntlm_v2_hash[16];
|
|
|
|
/* Compute the NTLMv2 hash */
|
|
ntlm_compute_ntlm_v2_hash(context, ntlm_v2_hash);
|
|
|
|
/* Concatenate the server and client challenges */
|
|
memcpy(value, context->ServerChallenge, 8);
|
|
memcpy(&value[8], context->ClientChallenge, 8);
|
|
|
|
sspi_SecBufferAlloc(&context->LmChallengeResponse, 24);
|
|
response = (char*) context->LmChallengeResponse.pvBuffer;
|
|
|
|
/* Compute the HMAC-MD5 hash of the resulting value using the NTLMv2 hash as the key */
|
|
HMAC(EVP_md5(), (void*) ntlm_v2_hash, 16, (void*) value, 16, (void*) response, NULL);
|
|
|
|
/* Concatenate the resulting HMAC-MD5 hash and the client challenge, giving us the LMv2 response (24 bytes) */
|
|
memcpy(&response[16], context->ClientChallenge, 8);
|
|
}
|
|
|
|
/**
|
|
* Compute NTLMv2 Response.\n
|
|
* NTLMv2_RESPONSE @msdn{cc236653}\n
|
|
* NTLMv2 Authentication @msdn{cc236700}
|
|
* @param NTLM context
|
|
*/
|
|
|
|
void ntlm_compute_ntlm_v2_response(NTLM_CONTEXT* context)
|
|
{
|
|
uint8* blob;
|
|
uint8 ntlm_v2_hash[16];
|
|
uint8 nt_proof_str[16];
|
|
SecBuffer ntlm_v2_temp;
|
|
SecBuffer ntlm_v2_temp_chal;
|
|
|
|
sspi_SecBufferAlloc(&ntlm_v2_temp, context->TargetInfo.cbBuffer + 28);
|
|
|
|
memset(ntlm_v2_temp.pvBuffer, '\0', ntlm_v2_temp.cbBuffer);
|
|
blob = (uint8*) ntlm_v2_temp.pvBuffer;
|
|
|
|
/* Compute the NTLMv2 hash */
|
|
ntlm_compute_ntlm_v2_hash(context, (char*) ntlm_v2_hash);
|
|
|
|
#ifdef WITH_DEBUG_NTLM
|
|
printf("Password (length = %d)\n", context->identity.PasswordLength);
|
|
freerdp_hexdump((uint8*) context->identity.Password, context->identity.PasswordLength);
|
|
printf("\n");
|
|
|
|
printf("Username (length = %d)\n", context->identity.UserLength);
|
|
freerdp_hexdump((uint8*) context->identity.User, context->identity.UserLength);
|
|
printf("\n");
|
|
|
|
printf("Domain (length = %d)\n", context->identity.DomainLength);
|
|
freerdp_hexdump((uint8*) context->identity.Domain, context->identity.DomainLength);
|
|
printf("\n");
|
|
|
|
printf("Workstation (length = %d)\n", context->WorkstationLength);
|
|
freerdp_hexdump((uint8*) context->Workstation, context->WorkstationLength);
|
|
printf("\n");
|
|
|
|
printf("NTOWFv2, NTLMv2 Hash\n");
|
|
freerdp_hexdump(ntlm_v2_hash, 16);
|
|
printf("\n");
|
|
#endif
|
|
|
|
/* Construct temp */
|
|
blob[0] = 1; /* RespType (1 byte) */
|
|
blob[1] = 1; /* HighRespType (1 byte) */
|
|
/* Reserved1 (2 bytes) */
|
|
/* Reserved2 (4 bytes) */
|
|
memcpy(&blob[8], context->av_pairs->Timestamp.value, 8); /* Timestamp (8 bytes) */
|
|
memcpy(&blob[16], context->ClientChallenge, 8); /* ClientChallenge (8 bytes) */
|
|
/* Reserved3 (4 bytes) */
|
|
memcpy(&blob[28], context->TargetInfo.pvBuffer, context->TargetInfo.cbBuffer);
|
|
|
|
#ifdef WITH_DEBUG_NTLM
|
|
printf("NTLMv2 Response Temp Blob\n");
|
|
freerdp_hexdump(ntlm_v2_temp.pvBuffer, ntlm_v2_temp.cbBuffer);
|
|
printf("\n");
|
|
#endif
|
|
|
|
/* Concatenate server challenge with temp */
|
|
sspi_SecBufferAlloc(&ntlm_v2_temp_chal, ntlm_v2_temp.cbBuffer + 8);
|
|
blob = (uint8*) ntlm_v2_temp_chal.pvBuffer;
|
|
memcpy(blob, context->ServerChallenge, 8);
|
|
memcpy(&blob[8], ntlm_v2_temp.pvBuffer, ntlm_v2_temp.cbBuffer);
|
|
|
|
HMAC(EVP_md5(), (void*) ntlm_v2_hash, 16, ntlm_v2_temp_chal.pvBuffer,
|
|
ntlm_v2_temp_chal.cbBuffer, (void*) nt_proof_str, NULL);
|
|
|
|
/* NtChallengeResponse, Concatenate NTProofStr with temp */
|
|
sspi_SecBufferAlloc(&context->NtChallengeResponse, ntlm_v2_temp.cbBuffer + 16);
|
|
blob = (uint8*) context->NtChallengeResponse.pvBuffer;
|
|
memcpy(blob, nt_proof_str, 16);
|
|
memcpy(&blob[16], ntlm_v2_temp.pvBuffer, ntlm_v2_temp.cbBuffer);
|
|
|
|
/* Compute SessionBaseKey, the HMAC-MD5 hash of NTProofStr using the NTLMv2 hash as the key */
|
|
HMAC(EVP_md5(), (void*) ntlm_v2_hash, 16, (void*) nt_proof_str, 16, (void*) context->SessionBaseKey, NULL);
|
|
|
|
sspi_SecBufferFree(&ntlm_v2_temp);
|
|
sspi_SecBufferFree(&ntlm_v2_temp_chal);
|
|
}
|
|
|
|
/**
|
|
* Encrypt the given plain text using RC4 and the given key.
|
|
* @param key RC4 key
|
|
* @param length text length
|
|
* @param plaintext plain text
|
|
* @param ciphertext cipher text
|
|
*/
|
|
|
|
void ntlm_rc4k(uint8* key, int length, uint8* plaintext, uint8* ciphertext)
|
|
{
|
|
CryptoRc4 rc4;
|
|
|
|
/* Initialize RC4 cipher with key */
|
|
rc4 = crypto_rc4_init((void*) key, 16);
|
|
|
|
/* Encrypt plaintext with key */
|
|
crypto_rc4(rc4, length, (void*) plaintext, (void*) ciphertext);
|
|
|
|
/* Free RC4 Cipher */
|
|
crypto_rc4_free(rc4);
|
|
}
|
|
|
|
/**
|
|
* Generate client challenge (8-byte nonce).
|
|
* @param NTLM context
|
|
*/
|
|
|
|
void ntlm_generate_client_challenge(NTLM_CONTEXT* context)
|
|
{
|
|
/* ClientChallenge is used in computation of LMv2 and NTLMv2 responses */
|
|
crypto_nonce(context->ClientChallenge, 8);
|
|
}
|
|
|
|
/**
|
|
* Generate server challenge (8-byte nonce).
|
|
* @param NTLM context
|
|
*/
|
|
|
|
void ntlm_generate_server_challenge(NTLM_CONTEXT* context)
|
|
{
|
|
crypto_nonce(context->ServerChallenge, 8);
|
|
}
|
|
|
|
/**
|
|
* Generate KeyExchangeKey (the 128-bit SessionBaseKey).\n
|
|
* @msdn{cc236710}
|
|
* @param NTLM context
|
|
*/
|
|
|
|
void ntlm_generate_key_exchange_key(NTLM_CONTEXT* context)
|
|
{
|
|
/* In NTLMv2, KeyExchangeKey is the 128-bit SessionBaseKey */
|
|
memcpy(context->KeyExchangeKey, context->SessionBaseKey, 16);
|
|
}
|
|
|
|
/**
|
|
* Generate RandomSessionKey (16-byte nonce).
|
|
* @param NTLM context
|
|
*/
|
|
|
|
void ntlm_generate_random_session_key(NTLM_CONTEXT* context)
|
|
{
|
|
crypto_nonce(context->RandomSessionKey, 16);
|
|
}
|
|
|
|
/**
|
|
* Generate ExportedSessionKey (the RandomSessionKey, exported)
|
|
* @param NTLM context
|
|
*/
|
|
|
|
void ntlm_generate_exported_session_key(NTLM_CONTEXT* context)
|
|
{
|
|
memcpy(context->ExportedSessionKey, context->RandomSessionKey, 16);
|
|
}
|
|
|
|
/**
|
|
* Encrypt RandomSessionKey (RC4-encrypted RandomSessionKey, using KeyExchangeKey as the key).
|
|
* @param NTLM context
|
|
*/
|
|
|
|
void ntlm_encrypt_random_session_key(NTLM_CONTEXT* context)
|
|
{
|
|
/* In NTLMv2, EncryptedRandomSessionKey is the ExportedSessionKey RC4-encrypted with the KeyExchangeKey */
|
|
ntlm_rc4k(context->KeyExchangeKey, 16, context->RandomSessionKey, context->EncryptedRandomSessionKey);
|
|
}
|
|
|
|
/**
|
|
* Generate signing key.\n
|
|
* @msdn{cc236711}
|
|
* @param exported_session_key ExportedSessionKey
|
|
* @param sign_magic Sign magic string
|
|
* @param signing_key Destination signing key
|
|
*/
|
|
|
|
void ntlm_generate_signing_key(uint8* exported_session_key, SecBuffer* sign_magic, uint8* signing_key)
|
|
{
|
|
int length;
|
|
uint8* value;
|
|
CryptoMd5 md5;
|
|
|
|
length = 16 + sign_magic->cbBuffer;
|
|
value = (uint8*) xmalloc(length);
|
|
|
|
/* Concatenate ExportedSessionKey with sign magic */
|
|
memcpy(value, exported_session_key, 16);
|
|
memcpy(&value[16], sign_magic->pvBuffer, sign_magic->cbBuffer);
|
|
|
|
md5 = crypto_md5_init();
|
|
crypto_md5_update(md5, value, length);
|
|
crypto_md5_final(md5, signing_key);
|
|
|
|
xfree(value);
|
|
}
|
|
|
|
/**
|
|
* Generate client signing key (ClientSigningKey).\n
|
|
* @msdn{cc236711}
|
|
* @param NTLM context
|
|
*/
|
|
|
|
void ntlm_generate_client_signing_key(NTLM_CONTEXT* context)
|
|
{
|
|
SecBuffer sign_magic;
|
|
sign_magic.pvBuffer = (void*) client_sign_magic;
|
|
sign_magic.cbBuffer = sizeof(client_sign_magic);
|
|
ntlm_generate_signing_key(context->ExportedSessionKey, &sign_magic, context->ClientSigningKey);
|
|
}
|
|
|
|
/**
|
|
* Generate server signing key (ServerSigningKey).\n
|
|
* @msdn{cc236711}
|
|
* @param NTLM context
|
|
*/
|
|
|
|
void ntlm_generate_server_signing_key(NTLM_CONTEXT* context)
|
|
{
|
|
SecBuffer sign_magic;
|
|
sign_magic.pvBuffer = (void*) server_sign_magic;
|
|
sign_magic.cbBuffer = sizeof(server_sign_magic);
|
|
ntlm_generate_signing_key(context->ExportedSessionKey, &sign_magic, context->ServerSigningKey);
|
|
}
|
|
|
|
/**
|
|
* Generate sealing key.\n
|
|
* @msdn{cc236712}
|
|
* @param exported_session_key ExportedSessionKey
|
|
* @param seal_magic Seal magic string
|
|
* @param sealing_key Destination sealing key
|
|
*/
|
|
|
|
void ntlm_generate_sealing_key(uint8* exported_session_key, SecBuffer* seal_magic, uint8* sealing_key)
|
|
{
|
|
uint8* p;
|
|
CryptoMd5 md5;
|
|
SecBuffer buffer;
|
|
|
|
sspi_SecBufferAlloc(&buffer, 16 + seal_magic->cbBuffer);
|
|
p = (uint8*) buffer.pvBuffer;
|
|
|
|
/* Concatenate ExportedSessionKey with seal magic */
|
|
memcpy(p, exported_session_key, 16);
|
|
memcpy(&p[16], seal_magic->pvBuffer, seal_magic->cbBuffer);
|
|
|
|
md5 = crypto_md5_init();
|
|
crypto_md5_update(md5, buffer.pvBuffer, buffer.cbBuffer);
|
|
crypto_md5_final(md5, sealing_key);
|
|
|
|
sspi_SecBufferFree(&buffer);
|
|
}
|
|
|
|
/**
|
|
* Generate client sealing key (ClientSealingKey).\n
|
|
* @msdn{cc236712}
|
|
* @param NTLM context
|
|
*/
|
|
|
|
void ntlm_generate_client_sealing_key(NTLM_CONTEXT* context)
|
|
{
|
|
SecBuffer seal_magic;
|
|
seal_magic.pvBuffer = (void*) client_seal_magic;
|
|
seal_magic.cbBuffer = sizeof(client_seal_magic);
|
|
ntlm_generate_signing_key(context->ExportedSessionKey, &seal_magic, context->ClientSealingKey);
|
|
}
|
|
|
|
/**
|
|
* Generate server sealing key (ServerSealingKey).\n
|
|
* @msdn{cc236712}
|
|
* @param NTLM context
|
|
*/
|
|
|
|
void ntlm_generate_server_sealing_key(NTLM_CONTEXT* context)
|
|
{
|
|
SecBuffer seal_magic;
|
|
seal_magic.pvBuffer = (void*) server_seal_magic;
|
|
seal_magic.cbBuffer = sizeof(server_seal_magic);
|
|
ntlm_generate_signing_key(context->ExportedSessionKey, &seal_magic, context->ServerSealingKey);
|
|
}
|
|
|
|
/**
|
|
* Initialize RC4 stream cipher states for sealing.
|
|
* @param NTLM context
|
|
*/
|
|
|
|
void ntlm_init_rc4_seal_states(NTLM_CONTEXT* context)
|
|
{
|
|
context->send_rc4_seal = crypto_rc4_init(context->ClientSealingKey, 16);
|
|
context->recv_rc4_seal = crypto_rc4_init(context->ServerSealingKey, 16);
|
|
}
|
|
|
|
void ntlm_compute_message_integrity_check(NTLM_CONTEXT* context)
|
|
{
|
|
HMAC_CTX hmac_ctx;
|
|
|
|
/*
|
|
* Compute the HMAC-MD5 hash of ConcatenationOf(NEGOTIATE_MESSAGE,
|
|
* CHALLENGE_MESSAGE, AUTHENTICATE_MESSAGE) using the ExportedSessionKey
|
|
*/
|
|
|
|
HMAC_CTX_init(&hmac_ctx);
|
|
HMAC_Init_ex(&hmac_ctx, context->ExportedSessionKey, 16, EVP_md5(), NULL);
|
|
HMAC_Update(&hmac_ctx, context->NegotiateMessage.pvBuffer, context->NegotiateMessage.cbBuffer);
|
|
HMAC_Update(&hmac_ctx, context->ChallengeMessage.pvBuffer, context->ChallengeMessage.cbBuffer);
|
|
HMAC_Update(&hmac_ctx, context->AuthenticateMessage.pvBuffer, context->AuthenticateMessage.cbBuffer);
|
|
HMAC_Final(&hmac_ctx, context->MessageIntegrityCheck, NULL);
|
|
}
|
|
|