2042 lines
56 KiB
C
2042 lines
56 KiB
C
/**
|
|
* FreeRDP: A Remote Desktop Protocol Client
|
|
* NT LAN Manager Security Support Provider (NTLMSSP)
|
|
*
|
|
* Copyright 2011 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 <time.h>
|
|
#include <openssl/des.h>
|
|
#include <openssl/md4.h>
|
|
#include <openssl/hmac.h>
|
|
#include <openssl/rand.h>
|
|
#include <openssl/engine.h>
|
|
#include <freerdp/utils/memory.h>
|
|
|
|
#include "ntlmssp.h"
|
|
|
|
#define NTLMSSP_NEGOTIATE_56 0x80000000 /* W (0) */
|
|
#define NTLMSSP_NEGOTIATE_KEY_EXCH 0x40000000 /* V (1) */
|
|
#define NTLMSSP_NEGOTIATE_128 0x20000000 /* U (2) */
|
|
#define NTLMSSP_RESERVED1 0x10000000 /* r1 (3) */
|
|
#define NTLMSSP_RESERVED2 0x08000000 /* r2 (4) */
|
|
#define NTLMSSP_RESERVED3 0x04000000 /* r3 (5) */
|
|
#define NTLMSSP_NEGOTIATE_VERSION 0x02000000 /* T (6) */
|
|
#define NTLMSSP_RESERVED4 0x01000000 /* r4 (7) */
|
|
#define NTLMSSP_NEGOTIATE_TARGET_INFO 0x00800000 /* S (8) */
|
|
#define NTLMSSP_RESERVEDEQUEST_NON_NT_SESSION_KEY 0x00400000 /* R (9) */
|
|
#define NTLMSSP_RESERVED5 0x00200000 /* r5 (10) */
|
|
#define NTLMSSP_NEGOTIATE_IDENTIFY 0x00100000 /* Q (11) */
|
|
#define NTLMSSP_NEGOTIATE_EXTENDED_SESSION_SECURITY 0x00080000 /* P (12) */
|
|
#define NTLMSSP_RESERVED6 0x00040000 /* r6 (13) */
|
|
#define NTLMSSP_TARGET_TYPE_SERVER 0x00020000 /* O (14) */
|
|
#define NTLMSSP_TARGET_TYPE_DOMAIN 0x00010000 /* N (15) */
|
|
#define NTLMSSP_NEGOTIATE_ALWAYS_SIGN 0x00008000 /* M (16) */
|
|
#define NTLMSSP_RESERVED7 0x00004000 /* r7 (17) */
|
|
#define NTLMSSP_NEGOTIATE_WORKSTATION_SUPPLIED 0x00002000 /* L (18) */
|
|
#define NTLMSSP_NEGOTIATE_DOMAIN_SUPPLIED 0x00001000 /* K (19) */
|
|
#define NTLMSSP_NEGOTIATE_ANONYMOUS 0x00000800 /* J (20) */
|
|
#define NTLMSSP_RESERVED8 0x00000400 /* r8 (21) */
|
|
#define NTLMSSP_NEGOTIATE_NTLM 0x00000200 /* H (22) */
|
|
#define NTLMSSP_RESERVED9 0x00000100 /* r9 (23) */
|
|
#define NTLMSSP_NEGOTIATE_LM_KEY 0x00000080 /* G (24) */
|
|
#define NTLMSSP_NEGOTIATE_DATAGRAM 0x00000040 /* F (25) */
|
|
#define NTLMSSP_NEGOTIATE_SEAL 0x00000020 /* E (26) */
|
|
#define NTLMSSP_NEGOTIATE_SIGN 0x00000010 /* D (27) */
|
|
#define NTLMSSP_RESERVED10 0x00000008 /* r10 (28) */
|
|
#define NTLMSSP_REQUEST_TARGET 0x00000004 /* C (29) */
|
|
#define NTLMSSP_NEGOTIATE_OEM 0x00000002 /* B (30) */
|
|
#define NTLMSSP_NEGOTIATE_UNICODE 0x00000001 /* A (31) */
|
|
|
|
#define WINDOWS_MAJOR_VERSION_5 0x05
|
|
#define WINDOWS_MAJOR_VERSION_6 0x06
|
|
#define WINDOWS_MINOR_VERSION_0 0x00
|
|
#define WINDOWS_MINOR_VERSION_1 0x01
|
|
#define WINDOWS_MINOR_VERSION_2 0x02
|
|
#define NTLMSSP_REVISION_W2K3 0x0F
|
|
|
|
#define MESSAGE_TYPE_NEGOTIATE 1
|
|
#define MESSAGE_TYPE_CHALLENGE 2
|
|
#define MESSAGE_TYPE_AUTHENTICATE 3
|
|
|
|
static const char ntlm_signature[] = "NTLMSSP";
|
|
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";
|
|
|
|
static const char* const NTLMSSP_NEGOTIATE_STRINGS[] =
|
|
{
|
|
"NTLMSSP_NEGOTIATE_56",
|
|
"NTLMSSP_NEGOTIATE_KEY_EXCH",
|
|
"NTLMSSP_NEGOTIATE_128",
|
|
"NTLMSSP_RESERVED1",
|
|
"NTLMSSP_RESERVED2",
|
|
"NTLMSSP_RESERVED3",
|
|
"NTLMSSP_NEGOTIATE_VERSION",
|
|
"NTLMSSP_RESERVED4",
|
|
"NTLMSSP_NEGOTIATE_TARGET_INFO",
|
|
"NTLMSSP_REQUEST_NON_NT_SESSION_KEY",
|
|
"NTLMSSP_RESERVED5",
|
|
"NTLMSSP_NEGOTIATE_IDENTIFY",
|
|
"NTLMSSP_NEGOTIATE_EXTENDED_SESSION_SECURITY",
|
|
"NTLMSSP_RESERVED6",
|
|
"NTLMSSP_TARGET_TYPE_SERVER",
|
|
"NTLMSSP_TARGET_TYPE_DOMAIN",
|
|
"NTLMSSP_NEGOTIATE_ALWAYS_SIGN",
|
|
"NTLMSSP_RESERVED7",
|
|
"NTLMSSP_NEGOTIATE_WORKSTATION_SUPPLIED",
|
|
"NTLMSSP_NEGOTIATE_DOMAIN_SUPPLIED",
|
|
"NTLMSSP_NEGOTIATE_ANONYMOUS",
|
|
"NTLMSSP_RESERVED8",
|
|
"NTLMSSP_NEGOTIATE_NTLM",
|
|
"NTLMSSP_RESERVED9",
|
|
"NTLMSSP_NEGOTIATE_LM_KEY",
|
|
"NTLMSSP_NEGOTIATE_DATAGRAM",
|
|
"NTLMSSP_NEGOTIATE_SEAL",
|
|
"NTLMSSP_NEGOTIATE_SIGN",
|
|
"NTLMSSP_RESERVED10",
|
|
"NTLMSSP_REQUEST_TARGET",
|
|
"NTLMSSP_NEGOTIATE_OEM",
|
|
"NTLMSSP_NEGOTIATE_UNICODE"
|
|
};
|
|
|
|
static const char* const AV_PAIRS_STRINGS[] =
|
|
{
|
|
"MsvAvEOL",
|
|
"MsvAvNbComputerName",
|
|
"MsvAvNbDomainName",
|
|
"MsvAvDnsComputerName",
|
|
"MsvAvDnsDomainName",
|
|
"MsvAvDnsTreeName",
|
|
"MsvAvFlags",
|
|
"MsvAvTimestamp",
|
|
"MsvAvRestrictions",
|
|
"MsvAvTargetName",
|
|
"MsvChannelBindings"
|
|
};
|
|
|
|
/**
|
|
* Set NTLMSSP username.
|
|
* @param ntlmssp
|
|
* @param username username
|
|
*/
|
|
|
|
void ntlmssp_set_username(NTLMSSP* ntlmssp, char* username)
|
|
{
|
|
freerdp_blob_free(&ntlmssp->username);
|
|
|
|
if (username != NULL)
|
|
{
|
|
ntlmssp->username.data = freerdp_uniconv_out(ntlmssp->uniconv, username, (size_t*) &(ntlmssp->username.length));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Set NTLMSSP domain name.
|
|
* @param ntlmssp
|
|
* @param domain domain name
|
|
*/
|
|
|
|
void ntlmssp_set_domain(NTLMSSP* ntlmssp, char* domain)
|
|
{
|
|
freerdp_blob_free(&ntlmssp->domain);
|
|
|
|
if (domain != NULL)
|
|
{
|
|
ntlmssp->domain.data = freerdp_uniconv_out(ntlmssp->uniconv, domain, (size_t*) &(ntlmssp->domain.length));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Set NTLMSSP password.
|
|
* @param ntlmssp
|
|
* @param password password
|
|
*/
|
|
|
|
void ntlmssp_set_password(NTLMSSP* ntlmssp, char* password)
|
|
{
|
|
freerdp_blob_free(&ntlmssp->password);
|
|
|
|
if (password != NULL)
|
|
{
|
|
ntlmssp->password.data = freerdp_uniconv_out(ntlmssp->uniconv, password, (size_t*) &(ntlmssp->password.length));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Set NTLMSSP workstation.
|
|
* @param ntlmssp
|
|
* @param workstation workstation
|
|
*/
|
|
|
|
void ntlmssp_set_workstation(NTLMSSP* ntlmssp, char* workstation)
|
|
{
|
|
freerdp_blob_free(&ntlmssp->workstation);
|
|
|
|
if (workstation != NULL)
|
|
{
|
|
ntlmssp->workstation.data = freerdp_uniconv_out(ntlmssp->uniconv, workstation, (size_t*) &(ntlmssp->workstation.length));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Generate client challenge (8-byte nonce).
|
|
* @param ntlmssp
|
|
*/
|
|
|
|
void ntlmssp_generate_client_challenge(NTLMSSP* ntlmssp)
|
|
{
|
|
/* ClientChallenge in computation of LMv2 and NTLMv2 responses */
|
|
crypto_nonce(ntlmssp->client_challenge, 8);
|
|
}
|
|
|
|
/**
|
|
* Generate KeyExchangeKey (the 128-bit SessionBaseKey).\n
|
|
* @msdn{cc236710}
|
|
* @param ntlmssp
|
|
*/
|
|
|
|
void ntlmssp_generate_key_exchange_key(NTLMSSP* ntlmssp)
|
|
{
|
|
/* In NTLMv2, KeyExchangeKey is the 128-bit SessionBaseKey */
|
|
memcpy(ntlmssp->key_exchange_key, ntlmssp->session_base_key, 16);
|
|
}
|
|
|
|
/**
|
|
* Generate RandomSessionKey (16-byte nonce).
|
|
* @param ntlmssp
|
|
*/
|
|
|
|
void ntlmssp_generate_random_session_key(NTLMSSP* ntlmssp)
|
|
{
|
|
crypto_nonce(ntlmssp->random_session_key, 16);
|
|
}
|
|
|
|
/**
|
|
* Generate ExportedSessionKey (the RandomSessionKey, exported)
|
|
* @param ntlmssp
|
|
*/
|
|
|
|
void ntlmssp_generate_exported_session_key(NTLMSSP* ntlmssp)
|
|
{
|
|
memcpy(ntlmssp->exported_session_key, ntlmssp->random_session_key, 16);
|
|
}
|
|
|
|
/**
|
|
* Encrypt RandomSessionKey (RC4-encrypted RandomSessionKey, using KeyExchangeKey as the key).
|
|
* @param ntlmssp
|
|
*/
|
|
|
|
void ntlmssp_encrypt_random_session_key(NTLMSSP* ntlmssp)
|
|
{
|
|
/* In NTLMv2, EncryptedRandomSessionKey is the ExportedSessionKey RC4-encrypted with the KeyExchangeKey */
|
|
credssp_rc4k(ntlmssp->key_exchange_key, 16, ntlmssp->random_session_key, ntlmssp->encrypted_random_session_key);
|
|
}
|
|
|
|
/**
|
|
* Generate timestamp for AUTHENTICATE_MESSAGE.
|
|
* @param ntlmssp
|
|
*/
|
|
|
|
void ntlmssp_generate_timestamp(NTLMSSP* ntlmssp)
|
|
{
|
|
credssp_current_time(ntlmssp->timestamp);
|
|
|
|
if (ntlmssp->ntlm_v2)
|
|
{
|
|
if (ntlmssp->av_pairs->Timestamp.length == 8)
|
|
{
|
|
memcpy(ntlmssp->av_pairs->Timestamp.value, ntlmssp->timestamp, 8);
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (ntlmssp->av_pairs->Timestamp.length != 8)
|
|
{
|
|
ntlmssp->av_pairs->Timestamp.length = 8;
|
|
ntlmssp->av_pairs->Timestamp.value = xmalloc(ntlmssp->av_pairs->Timestamp.length);
|
|
}
|
|
memcpy(ntlmssp->av_pairs->Timestamp.value, ntlmssp->timestamp, 8);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Generate signing key.\n
|
|
* @msdn{cc236711}
|
|
* @param exported_session_key ExportedSessionKey
|
|
* @param sign_magic Sign magic string
|
|
* @param signing_key Destination signing key
|
|
*/
|
|
|
|
void ntlmssp_generate_signing_key(uint8* exported_session_key, rdpBlob* sign_magic, uint8* signing_key)
|
|
{
|
|
int length;
|
|
uint8* value;
|
|
CryptoMd5 md5;
|
|
|
|
length = 16 + sign_magic->length;
|
|
value = (uint8*) xmalloc(length);
|
|
|
|
/* Concatenate ExportedSessionKey with sign magic */
|
|
memcpy(value, exported_session_key, 16);
|
|
memcpy(&value[16], sign_magic->data, sign_magic->length);
|
|
|
|
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 ntlmssp
|
|
*/
|
|
|
|
void ntlmssp_generate_client_signing_key(NTLMSSP* ntlmssp)
|
|
{
|
|
rdpBlob sign_magic;
|
|
sign_magic.data = (void*) client_sign_magic;
|
|
sign_magic.length = sizeof(client_sign_magic);
|
|
ntlmssp_generate_signing_key(ntlmssp->exported_session_key, &sign_magic, ntlmssp->client_signing_key);
|
|
}
|
|
|
|
/**
|
|
* Generate server signing key (ServerSigningKey).\n
|
|
* @msdn{cc236711}
|
|
* @param ntlmssp
|
|
*/
|
|
|
|
void ntlmssp_generate_server_signing_key(NTLMSSP* ntlmssp)
|
|
{
|
|
rdpBlob sign_magic;
|
|
sign_magic.data = (void*) server_sign_magic;
|
|
sign_magic.length = sizeof(server_sign_magic);
|
|
ntlmssp_generate_signing_key(ntlmssp->exported_session_key, &sign_magic, ntlmssp->server_signing_key);
|
|
}
|
|
|
|
/**
|
|
* Generate sealing key.\n
|
|
* @msdn{cc236712}
|
|
* @param exported_session_key ExportedSessionKey
|
|
* @param seal_magic Seal magic string
|
|
* @param sealing_key Destination sealing key
|
|
*/
|
|
|
|
void ntlmssp_generate_sealing_key(uint8* exported_session_key, rdpBlob* seal_magic, uint8* sealing_key)
|
|
{
|
|
uint8* p;
|
|
CryptoMd5 md5;
|
|
rdpBlob blob;
|
|
|
|
freerdp_blob_alloc(&blob, 16 + seal_magic->length);
|
|
p = (uint8*) blob.data;
|
|
|
|
/* Concatenate ExportedSessionKey with seal magic */
|
|
memcpy(p, exported_session_key, 16);
|
|
memcpy(&p[16], seal_magic->data, seal_magic->length);
|
|
|
|
md5 = crypto_md5_init();
|
|
crypto_md5_update(md5, blob.data, blob.length);
|
|
crypto_md5_final(md5, sealing_key);
|
|
|
|
freerdp_blob_free(&blob);
|
|
}
|
|
|
|
/**
|
|
* Generate client sealing key (ClientSealingKey).\n
|
|
* @msdn{cc236712}
|
|
* @param ntlmssp
|
|
*/
|
|
|
|
void ntlmssp_generate_client_sealing_key(NTLMSSP* ntlmssp)
|
|
{
|
|
rdpBlob seal_magic;
|
|
seal_magic.data = (void*) client_seal_magic;
|
|
seal_magic.length = sizeof(client_seal_magic);
|
|
ntlmssp_generate_signing_key(ntlmssp->exported_session_key, &seal_magic, ntlmssp->client_sealing_key);
|
|
}
|
|
|
|
/**
|
|
* Generate server sealing key (ServerSealingKey).\n
|
|
* @msdn{cc236712}
|
|
* @param ntlmssp
|
|
*/
|
|
|
|
void ntlmssp_generate_server_sealing_key(NTLMSSP* ntlmssp)
|
|
{
|
|
rdpBlob seal_magic;
|
|
seal_magic.data = (void*) server_seal_magic;
|
|
seal_magic.length = sizeof(server_seal_magic);
|
|
ntlmssp_generate_signing_key(ntlmssp->exported_session_key, &seal_magic, ntlmssp->server_sealing_key);
|
|
}
|
|
|
|
/**
|
|
* Initialize RC4 stream cipher states for sealing.
|
|
* @param ntlmssp
|
|
*/
|
|
|
|
void ntlmssp_init_rc4_seal_states(NTLMSSP* ntlmssp)
|
|
{
|
|
ntlmssp->send_rc4_seal = crypto_rc4_init(ntlmssp->client_sealing_key, 16);
|
|
ntlmssp->recv_rc4_seal = crypto_rc4_init(ntlmssp->server_sealing_key, 16);
|
|
}
|
|
|
|
/**
|
|
* Get bit from a byte buffer using a bit offset.
|
|
* @param buffer byte buffer
|
|
* @param bit bit offset
|
|
* @return bit value
|
|
*/
|
|
|
|
static int get_bit(char* buffer, int bit)
|
|
{
|
|
return (buffer[(bit - (bit % 8)) / 8] >> (7 - bit % 8) & 1);
|
|
}
|
|
|
|
/**
|
|
* Set bit in a byte buffer using a bit offset.
|
|
* @param buffer byte buffer
|
|
* @param bit bit offset
|
|
* @param value bit value
|
|
*/
|
|
|
|
static void set_bit(char* buffer, int bit, int value)
|
|
{
|
|
buffer[(bit - (bit % 8)) / 8] |= value << (7 - bit % 8);
|
|
}
|
|
|
|
static void ntlmssp_compute_des_key(char* text, char* des_key)
|
|
{
|
|
int i, j;
|
|
int bit;
|
|
int nbits;
|
|
|
|
/* Convert the 7 bytes into a bit stream, and insert a parity-bit (odd parity) after every seven bits. */
|
|
|
|
memset(des_key, '\0', 8);
|
|
|
|
for (i = 0; i < 8; i++)
|
|
{
|
|
nbits = 0;
|
|
|
|
for (j = 0; j < 7; j++)
|
|
{
|
|
/* copy 7 bits, and count the number of bits that are set */
|
|
|
|
bit = get_bit(text, i*7 + j);
|
|
set_bit(des_key, i*7 + i + j, bit);
|
|
nbits += bit;
|
|
}
|
|
|
|
/* insert parity bit (odd parity) */
|
|
|
|
if (nbits % 2 == 0)
|
|
set_bit(des_key, i*7 + i + j, 1);
|
|
}
|
|
}
|
|
|
|
void ntlmssp_compute_lm_hash(char* password, char* hash)
|
|
{
|
|
int i;
|
|
int maxlen;
|
|
char text[14];
|
|
char des_key1[8];
|
|
char des_key2[8];
|
|
des_key_schedule ks;
|
|
|
|
/* LM("password") = E52CAC67419A9A224A3B108F3FA6CB6D */
|
|
|
|
maxlen = (strlen(password) < 14) ? strlen(password) : 14;
|
|
|
|
/* convert to uppercase */
|
|
for (i = 0; i < maxlen; i++)
|
|
{
|
|
if ((password[i] >= 'a') && (password[i] <= 'z'))
|
|
text[i] = password[i] - 32;
|
|
else
|
|
text[i] = password[i];
|
|
}
|
|
|
|
/* pad with nulls up to 14 bytes */
|
|
for (i = maxlen; i < 14; i++)
|
|
text[i] = '\0';
|
|
|
|
ntlmssp_compute_des_key(text, des_key1);
|
|
ntlmssp_compute_des_key(&text[7], des_key2);
|
|
|
|
DES_set_key((const_DES_cblock*)des_key1, &ks);
|
|
DES_ecb_encrypt((const_DES_cblock*) lm_magic, (DES_cblock*)hash, &ks, DES_ENCRYPT);
|
|
|
|
DES_set_key((const_DES_cblock*)des_key2, &ks);
|
|
DES_ecb_encrypt((const_DES_cblock*) lm_magic, (DES_cblock*)&hash[8], &ks, DES_ENCRYPT);
|
|
}
|
|
|
|
void ntlmssp_compute_ntlm_hash(rdpBlob* password, 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->data, password->length);
|
|
MD4_Final((void*) hash, &md4_ctx);
|
|
}
|
|
|
|
void ntlmssp_compute_ntlm_v2_hash(NTLMSSP* ntlmssp, char* hash)
|
|
{
|
|
char* p;
|
|
rdpBlob blob;
|
|
char ntlm_hash[16];
|
|
|
|
freerdp_blob_alloc(&blob, ntlmssp->username.length + ntlmssp->domain.length);
|
|
p = (char*) blob.data;
|
|
|
|
/* First, compute the NTLMv1 hash of the password */
|
|
ntlmssp_compute_ntlm_hash(&ntlmssp->password, ntlm_hash);
|
|
|
|
/* Concatenate(Uppercase(username),domain)*/
|
|
memcpy(p, ntlmssp->username.data, ntlmssp->username.length);
|
|
freerdp_uniconv_uppercase(ntlmssp->uniconv, p, ntlmssp->username.length / 2);
|
|
|
|
memcpy(&p[ntlmssp->username.length], ntlmssp->domain.data, ntlmssp->domain.length);
|
|
|
|
/* 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, blob.data, blob.length, (void*) hash, NULL);
|
|
|
|
freerdp_blob_free(&blob);
|
|
}
|
|
|
|
void ntlmssp_compute_lm_response(char* password, char* challenge, char* response)
|
|
{
|
|
char hash[21];
|
|
char des_key1[8];
|
|
char des_key2[8];
|
|
char des_key3[8];
|
|
des_key_schedule ks;
|
|
|
|
/* A LM hash is 16-bytes long, but the LM response uses a LM hash null-padded to 21 bytes */
|
|
memset(hash, '\0', 21);
|
|
ntlmssp_compute_lm_hash(password, hash);
|
|
|
|
/* Each 7-byte third of the 21-byte null-padded LM hash is used to create a DES key */
|
|
ntlmssp_compute_des_key(hash, des_key1);
|
|
ntlmssp_compute_des_key(&hash[7], des_key2);
|
|
ntlmssp_compute_des_key(&hash[14], des_key3);
|
|
|
|
/* Encrypt the LM challenge with each key, and concatenate the result. This is the LM response (24 bytes) */
|
|
DES_set_key((const_DES_cblock*)des_key1, &ks);
|
|
DES_ecb_encrypt((const_DES_cblock*)challenge, (DES_cblock*)response, &ks, DES_ENCRYPT);
|
|
|
|
DES_set_key((const_DES_cblock*)des_key2, &ks);
|
|
DES_ecb_encrypt((const_DES_cblock*)challenge, (DES_cblock*)&response[8], &ks, DES_ENCRYPT);
|
|
|
|
DES_set_key((const_DES_cblock*)des_key3, &ks);
|
|
DES_ecb_encrypt((const_DES_cblock*)challenge, (DES_cblock*)&response[16], &ks, DES_ENCRYPT);
|
|
}
|
|
|
|
void ntlmssp_compute_lm_v2_response(NTLMSSP* ntlmssp)
|
|
{
|
|
char *response;
|
|
char value[16];
|
|
char ntlm_v2_hash[16];
|
|
|
|
/* Compute the NTLMv2 hash */
|
|
ntlmssp_compute_ntlm_v2_hash(ntlmssp, ntlm_v2_hash);
|
|
|
|
/* Concatenate the server and client challenges */
|
|
memcpy(value, ntlmssp->server_challenge, 8);
|
|
memcpy(&value[8], ntlmssp->client_challenge, 8);
|
|
|
|
freerdp_blob_alloc(&ntlmssp->lm_challenge_response, 24);
|
|
response = (char*) ntlmssp->lm_challenge_response.data;
|
|
|
|
/* 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], ntlmssp->client_challenge, 8);
|
|
}
|
|
|
|
/**
|
|
* Compute NTLMv2 Response.\n
|
|
* NTLMv2_RESPONSE @msdn{cc236653}\n
|
|
* NTLMv2 Authentication @msdn{cc236700}
|
|
* @param ntlmssp
|
|
*/
|
|
|
|
void ntlmssp_compute_ntlm_v2_response(NTLMSSP* ntlmssp)
|
|
{
|
|
uint8* blob;
|
|
uint8 ntlm_v2_hash[16];
|
|
uint8 nt_proof_str[16];
|
|
rdpBlob ntlm_v2_temp;
|
|
rdpBlob ntlm_v2_temp_chal;
|
|
|
|
freerdp_blob_alloc(&ntlm_v2_temp, ntlmssp->target_info.length + 28);
|
|
|
|
memset(ntlm_v2_temp.data, '\0', ntlm_v2_temp.length);
|
|
blob = (uint8*) ntlm_v2_temp.data;
|
|
|
|
/* Compute the NTLMv2 hash */
|
|
ntlmssp_compute_ntlm_v2_hash(ntlmssp, (char*) ntlm_v2_hash);
|
|
|
|
#ifdef WITH_DEBUG_NLA
|
|
printf("Password (length = %d)\n", ntlmssp->password.length);
|
|
freerdp_hexdump(ntlmssp->password.data, ntlmssp->password.length);
|
|
printf("\n");
|
|
|
|
printf("Username (length = %d)\n", ntlmssp->username.length);
|
|
freerdp_hexdump(ntlmssp->username.data, ntlmssp->username.length);
|
|
printf("\n");
|
|
|
|
printf("Domain (length = %d)\n", ntlmssp->domain.length);
|
|
freerdp_hexdump(ntlmssp->domain.data, ntlmssp->domain.length);
|
|
printf("\n");
|
|
|
|
printf("Workstation (length = %d)\n", ntlmssp->workstation.length);
|
|
freerdp_hexdump(ntlmssp->workstation.data, ntlmssp->workstation.length);
|
|
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], ntlmssp->av_pairs->Timestamp.value, 8); /* Timestamp (8 bytes) */
|
|
memcpy(&blob[16], ntlmssp->client_challenge, 8); /* ClientChallenge (8 bytes) */
|
|
/* Reserved3 (4 bytes) */
|
|
memcpy(&blob[28], ntlmssp->target_info.data, ntlmssp->target_info.length);
|
|
|
|
#ifdef WITH_DEBUG_NLA
|
|
printf("NTLMv2 Response Temp Blob\n");
|
|
freerdp_hexdump(ntlm_v2_temp.data, ntlm_v2_temp.length);
|
|
printf("\n");
|
|
#endif
|
|
|
|
/* Concatenate server challenge with temp */
|
|
freerdp_blob_alloc(&ntlm_v2_temp_chal, ntlm_v2_temp.length + 8);
|
|
blob = (uint8*) ntlm_v2_temp_chal.data;
|
|
memcpy(blob, ntlmssp->server_challenge, 8);
|
|
memcpy(&blob[8], ntlm_v2_temp.data, ntlm_v2_temp.length);
|
|
|
|
HMAC(EVP_md5(), (void*) ntlm_v2_hash, 16, (void*) ntlm_v2_temp_chal.data,
|
|
ntlm_v2_temp_chal.length, (void*) nt_proof_str, NULL);
|
|
|
|
/* NtChallengeResponse, Concatenate NTProofStr with temp */
|
|
freerdp_blob_alloc(&ntlmssp->nt_challenge_response, ntlm_v2_temp.length + 16);
|
|
blob = (uint8*) ntlmssp->nt_challenge_response.data;
|
|
memcpy(blob, nt_proof_str, 16);
|
|
memcpy(&blob[16], ntlm_v2_temp.data, ntlm_v2_temp.length);
|
|
|
|
/* 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*) ntlmssp->session_base_key, NULL);
|
|
|
|
freerdp_blob_free(&ntlm_v2_temp);
|
|
freerdp_blob_free(&ntlm_v2_temp_chal);
|
|
}
|
|
|
|
/**
|
|
* Input NegotiateFlags, a 4-byte bit map.
|
|
* @param s
|
|
* @param flags
|
|
*/
|
|
|
|
void ntlmssp_input_negotiate_flags(STREAM* s, uint32* flags)
|
|
{
|
|
*flags = 0;
|
|
stream_read_uint32(s, *flags);
|
|
}
|
|
|
|
/**
|
|
* Output NegotiateFlags, a 4-byte bit map.
|
|
* @param s
|
|
* @param flags
|
|
*/
|
|
|
|
void ntlmssp_output_negotiate_flags(STREAM* s, uint32 flags)
|
|
{
|
|
stream_write_uint32(s, flags);
|
|
}
|
|
|
|
void ntlmssp_print_negotiate_flags(uint32 flags)
|
|
{
|
|
int i;
|
|
const char* str;
|
|
|
|
printf("negotiateFlags \"0x%08X\"{\n", flags);
|
|
|
|
for (i = 31; i >= 0; i--)
|
|
{
|
|
if ((flags >> i) & 1)
|
|
{
|
|
str = NTLMSSP_NEGOTIATE_STRINGS[(31 - i)];
|
|
printf("\t%s (%d),\n", str, (31 - i));
|
|
}
|
|
}
|
|
|
|
printf("}\n");
|
|
}
|
|
|
|
/**
|
|
* Output Restriction_Encoding.\n
|
|
* Restriction_Encoding @msdn{cc236647}
|
|
* @param ntlmssp
|
|
*/
|
|
|
|
static void ntlmssp_output_restriction_encoding(NTLMSSP* ntlmssp)
|
|
{
|
|
AV_PAIR *restrictions = &ntlmssp->av_pairs->Restrictions;
|
|
STREAM* s = stream_new(0);
|
|
|
|
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->data = restrictions->value;
|
|
s->size = restrictions->length;
|
|
s->p = s->data;
|
|
|
|
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 ntlmssp
|
|
*/
|
|
|
|
void ntlmssp_output_target_name(NTLMSSP* ntlmssp)
|
|
{
|
|
STREAM* s = stream_new(0);
|
|
AV_PAIR* target_name = &ntlmssp->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";
|
|
|
|
target_name->length = 42;
|
|
target_name->value = (uint8*) xmalloc(target_name->length);
|
|
|
|
s->data = target_name->value;
|
|
s->size = target_name->length;
|
|
s->p = s->data;
|
|
|
|
stream_write(s, name, target_name->length);
|
|
|
|
xfree(s);
|
|
}
|
|
|
|
/**
|
|
* Output ChannelBindings.\n
|
|
* @param ntlmssp
|
|
*/
|
|
|
|
void ntlmssp_output_channel_bindings(NTLMSSP* ntlmssp)
|
|
{
|
|
STREAM* s = stream_new(0);
|
|
AV_PAIR* channel_bindings = &ntlmssp->av_pairs->ChannelBindings;
|
|
|
|
channel_bindings->value = (uint8*) xmalloc(48);
|
|
channel_bindings->length = 16;
|
|
|
|
s->data = channel_bindings->value;
|
|
s->size = channel_bindings->length;
|
|
s->p = s->data;
|
|
|
|
stream_write_zero(s, 16); /* an all-zero value of the hash is used to indicate absence of channel bindings */
|
|
|
|
xfree(s);
|
|
}
|
|
|
|
/**
|
|
* Populate array of AV_PAIRs.\n
|
|
* AV_PAIR @msdn{cc236646}
|
|
* @param ntlmssp
|
|
*/
|
|
|
|
void ntlmssp_populate_av_pairs(NTLMSSP* ntlmssp)
|
|
{
|
|
STREAM* s;
|
|
rdpBlob target_info;
|
|
AV_PAIRS *av_pairs = ntlmssp->av_pairs;
|
|
|
|
/* MsvAvFlags */
|
|
av_pairs->Flags = 0x00000002; /* Indicates the present of a Message Integrity Check (MIC) */
|
|
|
|
/* Restriction_Encoding */
|
|
ntlmssp_output_restriction_encoding(ntlmssp);
|
|
|
|
/* TargetName */
|
|
ntlmssp_output_target_name(ntlmssp);
|
|
|
|
/* ChannelBindings */
|
|
ntlmssp_output_channel_bindings(ntlmssp);
|
|
|
|
s = stream_new(0);
|
|
s->data = xmalloc(ntlmssp->target_info.length + 512);
|
|
s->p = s->data;
|
|
|
|
ntlmssp_output_av_pairs(ntlmssp, s);
|
|
freerdp_blob_alloc(&target_info, s->p - s->data);
|
|
memcpy(target_info.data, s->data, target_info.length);
|
|
|
|
ntlmssp->target_info.data = target_info.data;
|
|
ntlmssp->target_info.length = target_info.length;
|
|
}
|
|
|
|
/**
|
|
* Input array of AV_PAIRs.\n
|
|
* AV_PAIR @msdn{cc236646}
|
|
* @param ntlmssp
|
|
* @param s
|
|
*/
|
|
|
|
void ntlmssp_input_av_pairs(NTLMSSP* ntlmssp, STREAM* s)
|
|
{
|
|
AV_ID AvId;
|
|
uint16 AvLen;
|
|
uint8* value;
|
|
AV_PAIRS* av_pairs = ntlmssp->av_pairs;
|
|
|
|
#ifdef WITH_DEBUG_NLA
|
|
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_NLA
|
|
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_NLA
|
|
printf("}\n");
|
|
#endif
|
|
}
|
|
|
|
/**
|
|
* Output array of AV_PAIRs.\n
|
|
* AV_PAIR @msdn{cc236646}
|
|
* @param ntlmssp
|
|
* @param s
|
|
*/
|
|
|
|
void ntlmssp_output_av_pairs(NTLMSSP* ntlmssp, STREAM* s)
|
|
{
|
|
AV_PAIRS* av_pairs = ntlmssp->av_pairs;
|
|
|
|
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 (ntlmssp->ntlm_v2)
|
|
{
|
|
stream_write_zero(s, 8);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Print array of AV_PAIRs.\n
|
|
* AV_PAIR @msdn{cc236646}
|
|
* @param ntlmssp
|
|
* @param s
|
|
*/
|
|
|
|
void ntlmssp_print_av_pairs(NTLMSSP* ntlmssp)
|
|
{
|
|
AV_PAIRS* av_pairs = ntlmssp->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 ntlmssp
|
|
*/
|
|
|
|
void ntlmssp_free_av_pairs(NTLMSSP* ntlmssp)
|
|
{
|
|
AV_PAIRS *av_pairs = ntlmssp->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);
|
|
}
|
|
|
|
ntlmssp->av_pairs = NULL;
|
|
}
|
|
|
|
/**
|
|
* Output VERSION structure.\n
|
|
* VERSION @msdn{cc236654}
|
|
* @param s
|
|
*/
|
|
|
|
static void ntlmssp_output_version(STREAM* s)
|
|
{
|
|
/* The following version information was observed with Windows 7 */
|
|
|
|
stream_write_uint8(s, WINDOWS_MAJOR_VERSION_6); /* ProductMajorVersion (1 byte) */
|
|
stream_write_uint8(s, WINDOWS_MINOR_VERSION_1); /* ProductMinorVersion (1 byte) */
|
|
stream_write_uint16(s, 7600); /* ProductBuild (2 bytes) */
|
|
stream_write_zero(s, 3); /* Reserved (3 bytes) */
|
|
stream_write_uint8(s, NTLMSSP_REVISION_W2K3); /* NTLMRevisionCurrent (1 byte) */
|
|
}
|
|
|
|
void ntlmssp_compute_message_integrity_check(NTLMSSP* ntlmssp)
|
|
{
|
|
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, ntlmssp->exported_session_key, 16, EVP_md5(), NULL);
|
|
HMAC_Update(&hmac_ctx, ntlmssp->negotiate_message.data, ntlmssp->negotiate_message.length);
|
|
HMAC_Update(&hmac_ctx, ntlmssp->challenge_message.data, ntlmssp->challenge_message.length);
|
|
HMAC_Update(&hmac_ctx, ntlmssp->authenticate_message.data, ntlmssp->authenticate_message.length);
|
|
HMAC_Final(&hmac_ctx, ntlmssp->message_integrity_check, NULL);
|
|
}
|
|
|
|
/**
|
|
* Encrypt and sign message using NTLMSSP.\n
|
|
* GSS_WrapEx() @msdn{cc236718}\n
|
|
* EncryptMessage() @msdn{aa375378}
|
|
* @param ntlmssp
|
|
* @param[in] msg message to encrypt
|
|
* @param[out] encrypted_msg encrypted message
|
|
* @param[out] signature destination signature
|
|
*/
|
|
|
|
void ntlmssp_encrypt_message(NTLMSSP* ntlmssp, rdpBlob* msg, rdpBlob* encrypted_msg, uint8* signature)
|
|
{
|
|
HMAC_CTX hmac_ctx;
|
|
uint8 digest[16];
|
|
uint8 checksum[8];
|
|
uint32 version = 1;
|
|
|
|
/* Compute the HMAC-MD5 hash of ConcatenationOf(seq_num,msg) using the client signing key */
|
|
HMAC_CTX_init(&hmac_ctx);
|
|
HMAC_Init_ex(&hmac_ctx, ntlmssp->client_signing_key, 16, EVP_md5(), NULL);
|
|
HMAC_Update(&hmac_ctx, (void*) &ntlmssp->send_seq_num, 4);
|
|
HMAC_Update(&hmac_ctx, msg->data, msg->length);
|
|
HMAC_Final(&hmac_ctx, digest, NULL);
|
|
|
|
/* Allocate space for encrypted message */
|
|
freerdp_blob_alloc(encrypted_msg, msg->length);
|
|
|
|
/* Encrypt message using with RC4 */
|
|
crypto_rc4(ntlmssp->send_rc4_seal, msg->length, msg->data, encrypted_msg->data);
|
|
|
|
/* RC4-encrypt first 8 bytes of digest */
|
|
crypto_rc4(ntlmssp->send_rc4_seal, 8, digest, checksum);
|
|
|
|
/* Concatenate version, ciphertext and sequence number to build signature */
|
|
memcpy(signature, (void*) &version, 4);
|
|
memcpy(&signature[4], (void*) checksum, 8);
|
|
memcpy(&signature[12], (void*) &(ntlmssp->send_seq_num), 4);
|
|
|
|
HMAC_CTX_cleanup(&hmac_ctx);
|
|
|
|
ntlmssp->send_seq_num++;
|
|
}
|
|
|
|
/**
|
|
* Decrypt message and verify signature using NTLMSSP.\n
|
|
* GSS_UnwrapEx() @msdn{cc236703}\n
|
|
* DecryptMessage() @msdn{aa375211}
|
|
* @param ntlmssp
|
|
* @param[in] encrypted_msg encrypted message
|
|
* @param[out] msg decrypted message
|
|
* @param[in] signature signature
|
|
* @return
|
|
*/
|
|
|
|
int ntlmssp_decrypt_message(NTLMSSP* ntlmssp, rdpBlob* encrypted_msg, rdpBlob* msg, uint8* signature)
|
|
{
|
|
HMAC_CTX hmac_ctx;
|
|
uint8 digest[16];
|
|
uint8 checksum[8];
|
|
uint32 version = 1;
|
|
uint8 expected_signature[16];
|
|
|
|
/* Allocate space for encrypted message */
|
|
freerdp_blob_alloc(msg, encrypted_msg->length);
|
|
|
|
/* Encrypt message using with RC4 */
|
|
crypto_rc4(ntlmssp->recv_rc4_seal, encrypted_msg->length, encrypted_msg->data, msg->data);
|
|
|
|
/* Compute the HMAC-MD5 hash of ConcatenationOf(seq_num,msg) using the client signing key */
|
|
HMAC_CTX_init(&hmac_ctx);
|
|
HMAC_Init_ex(&hmac_ctx, ntlmssp->server_signing_key, 16, EVP_md5(), NULL);
|
|
HMAC_Update(&hmac_ctx, (void*) &ntlmssp->recv_seq_num, 4);
|
|
HMAC_Update(&hmac_ctx, msg->data, msg->length);
|
|
HMAC_Final(&hmac_ctx, digest, NULL);
|
|
|
|
/* RC4-encrypt first 8 bytes of digest */
|
|
crypto_rc4(ntlmssp->recv_rc4_seal, 8, digest, checksum);
|
|
|
|
/* Concatenate version, ciphertext and sequence number to build signature */
|
|
memcpy(expected_signature, (void*) &version, 4);
|
|
memcpy(&expected_signature[4], (void*) checksum, 8);
|
|
memcpy(&expected_signature[12], (void*) &(ntlmssp->recv_seq_num), 4);
|
|
|
|
if (memcmp(signature, expected_signature, 16) != 0)
|
|
{
|
|
/* signature verification failed! */
|
|
printf("signature verification failed, something nasty is going on!\n");
|
|
return 0;
|
|
}
|
|
|
|
HMAC_CTX_cleanup(&hmac_ctx);
|
|
|
|
ntlmssp->recv_seq_num++;
|
|
return 1;
|
|
}
|
|
|
|
/**
|
|
* Send NTLMSSP NEGOTIATE_MESSAGE.\n
|
|
* NEGOTIATE_MESSAGE @msdn{cc236641}
|
|
* @param ntlmssp
|
|
* @param s
|
|
*/
|
|
|
|
void ntlmssp_send_negotiate_message(NTLMSSP* ntlmssp, STREAM* s)
|
|
{
|
|
int length;
|
|
uint32 negotiateFlags = 0;
|
|
|
|
stream_write(s, ntlm_signature, 8); /* Signature (8 bytes) */
|
|
stream_write_uint32(s, MESSAGE_TYPE_NEGOTIATE); /* MessageType */
|
|
|
|
if (ntlmssp->ntlm_v2)
|
|
{
|
|
DEBUG_NLA("Negotiating NTLMv2");
|
|
/* observed: B7 82 08 E2 (0xE20882B7) (Dmitrij Jasnov) */
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_56;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_KEY_EXCH;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_128;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_VERSION;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_EXTENDED_SESSION_SECURITY;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_ALWAYS_SIGN;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_NTLM;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_LM_KEY;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_SEAL;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_SIGN;
|
|
negotiateFlags |= NTLMSSP_REQUEST_TARGET;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_OEM;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_UNICODE;
|
|
}
|
|
else
|
|
{
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_KEY_EXCH;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_128;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_EXTENDED_SESSION_SECURITY;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_ALWAYS_SIGN;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_NTLM;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_SEAL;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_SIGN;
|
|
negotiateFlags |= NTLMSSP_REQUEST_TARGET;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_UNICODE;
|
|
}
|
|
|
|
ntlmssp_output_negotiate_flags(s, negotiateFlags); /* NegotiateFlags (4 bytes) */
|
|
|
|
#ifdef WITH_DEBUG_NLA
|
|
ntlmssp_print_negotiate_flags(negotiateFlags);
|
|
#endif
|
|
|
|
/* only set if NTLMSSP_NEGOTIATE_DOMAIN_SUPPLIED is set */
|
|
|
|
/* DomainNameFields (8 bytes) */
|
|
stream_write_uint16(s, 0); /* DomainNameLen */
|
|
stream_write_uint16(s, 0); /* DomainNameMaxLen */
|
|
stream_write_uint32(s, 0); /* DomainNameBufferOffset */
|
|
|
|
/* only set if NTLMSSP_NEGOTIATE_WORKSTATION_SUPPLIED is set */
|
|
|
|
/* WorkstationFields (8 bytes) */
|
|
stream_write_uint16(s, 0); /* WorkstationLen */
|
|
stream_write_uint16(s, 0); /* WorkstationMaxLen */
|
|
stream_write_uint32(s, 0); /* WorkstationBufferOffset */
|
|
|
|
if (negotiateFlags & NTLMSSP_NEGOTIATE_VERSION)
|
|
{
|
|
/* Only present if NTLMSSP_NEGOTIATE_VERSION is set */
|
|
ntlmssp_output_version(s);
|
|
|
|
#ifdef WITH_DEBUG_NLA
|
|
printf("Version (length = 8)\n");
|
|
freerdp_hexdump((s->p - 8), 8);
|
|
printf("\n");
|
|
#endif
|
|
}
|
|
|
|
length = s->p - s->data;
|
|
freerdp_blob_alloc(&ntlmssp->negotiate_message, length);
|
|
memcpy(ntlmssp->negotiate_message.data, s->data, length);
|
|
|
|
#ifdef WITH_DEBUG_NLA
|
|
printf("NEGOTIATE_MESSAGE (length = %d)\n", length);
|
|
freerdp_hexdump(s->data, length);
|
|
printf("\n");
|
|
#endif
|
|
|
|
ntlmssp->state = NTLMSSP_STATE_CHALLENGE;
|
|
}
|
|
|
|
/**
|
|
* Receive NTLMSSP NEGOTIATE_MESSAGE.\n
|
|
* NEGOTIATE_MESSAGE @msdn{cc236641}
|
|
* @param ntlmssp
|
|
* @param s
|
|
*/
|
|
|
|
void ntlmssp_recv_negotiate_message(NTLMSSP* ntlmssp, STREAM* s)
|
|
{
|
|
ntlmssp->state = NTLMSSP_STATE_CHALLENGE;
|
|
}
|
|
|
|
/**
|
|
* Send NTLMSSP CHALLENGE_MESSAGE.\n
|
|
* CHALLENGE_MESSAGE @msdn{cc236642}
|
|
* @param ntlmssp
|
|
* @param s
|
|
*/
|
|
|
|
void ntlmssp_send_challenge_message(NTLMSSP* ntlmssp, STREAM* s)
|
|
{
|
|
ntlmssp->state = NTLMSSP_STATE_AUTHENTICATE;
|
|
}
|
|
|
|
/**
|
|
* Receive NTLMSSP CHALLENGE_MESSAGE.\n
|
|
* CHALLENGE_MESSAGE @msdn{cc236642}
|
|
* @param ntlmssp
|
|
* @param s
|
|
*/
|
|
|
|
void ntlmssp_recv_challenge_message(NTLMSSP* ntlmssp, STREAM* s)
|
|
{
|
|
uint8* p;
|
|
int length;
|
|
uint8* start_offset;
|
|
uint8* payload_offset;
|
|
uint16 targetNameLen;
|
|
uint16 targetNameMaxLen;
|
|
uint32 targetNameBufferOffset;
|
|
uint16 targetInfoLen;
|
|
uint16 targetInfoMaxLen;
|
|
uint32 targetInfoBufferOffset;
|
|
|
|
start_offset = s->p - 12;
|
|
|
|
/* TargetNameFields (8 bytes) */
|
|
stream_read_uint16(s, targetNameLen); /* TargetNameLen (2 bytes) */
|
|
stream_read_uint16(s, targetNameMaxLen); /* TargetNameMaxLen (2 bytes) */
|
|
stream_read_uint32(s, targetNameBufferOffset); /* TargetNameBufferOffset (4 bytes) */
|
|
|
|
ntlmssp_input_negotiate_flags(s, &(ntlmssp->negotiate_flags)); /* NegotiateFlags (4 bytes) */
|
|
|
|
#ifdef WITH_DEBUG_NLA
|
|
ntlmssp_print_negotiate_flags(ntlmssp->negotiate_flags);
|
|
#endif
|
|
|
|
stream_read(s, ntlmssp->server_challenge, 8); /* ServerChallenge (8 bytes) */
|
|
stream_seek(s, 8); /* Reserved (8 bytes), should be ignored */
|
|
|
|
/* TargetInfoFields (8 bytes) */
|
|
stream_read_uint16(s, targetInfoLen); /* TargetInfoLen (2 bytes) */
|
|
stream_read_uint16(s, targetInfoMaxLen); /* TargetInfoMaxLen (2 bytes) */
|
|
stream_read_uint32(s, targetInfoBufferOffset); /* TargetInfoBufferOffset (4 bytes) */
|
|
|
|
/* only present if NTLMSSP_NEGOTIATE_VERSION is set */
|
|
|
|
if (ntlmssp->negotiate_flags & NTLMSSP_NEGOTIATE_VERSION)
|
|
{
|
|
stream_seek(s, 8); /* Version (8 bytes), can be ignored */
|
|
}
|
|
|
|
/* Payload (variable) */
|
|
payload_offset = s->p;
|
|
|
|
if (targetNameLen > 0)
|
|
{
|
|
p = start_offset + targetNameBufferOffset;
|
|
freerdp_blob_alloc(&ntlmssp->target_name, targetNameLen);
|
|
memcpy(ntlmssp->target_name.data, p, targetNameLen);
|
|
|
|
#ifdef WITH_DEBUG_NLA
|
|
printf("targetName (length = %d, offset = %d)\n", targetNameLen, targetNameBufferOffset);
|
|
freerdp_hexdump(ntlmssp->target_name.data, ntlmssp->target_name.length);
|
|
printf("\n");
|
|
#endif
|
|
}
|
|
|
|
if (targetInfoLen > 0)
|
|
{
|
|
p = start_offset + targetInfoBufferOffset;
|
|
freerdp_blob_alloc(&ntlmssp->target_info, targetInfoLen);
|
|
memcpy(ntlmssp->target_info.data, p, targetInfoLen);
|
|
|
|
#ifdef WITH_DEBUG_NLA
|
|
printf("targetInfo (length = %d, offset = %d)\n", targetInfoLen, targetInfoBufferOffset);
|
|
freerdp_hexdump(ntlmssp->target_info.data, ntlmssp->target_info.length);
|
|
printf("\n");
|
|
#endif
|
|
|
|
if (ntlmssp->ntlm_v2)
|
|
{
|
|
s->p = p;
|
|
ntlmssp_input_av_pairs(ntlmssp, s);
|
|
}
|
|
}
|
|
|
|
length = (payload_offset - start_offset) + targetNameLen + targetInfoLen;
|
|
|
|
freerdp_blob_alloc(&ntlmssp->challenge_message, length);
|
|
memcpy(ntlmssp->challenge_message.data, start_offset, length);
|
|
|
|
#ifdef WITH_DEBUG_NLA
|
|
printf("CHALLENGE_MESSAGE (length = %d)\n", length);
|
|
freerdp_hexdump(start_offset, length);
|
|
printf("\n");
|
|
#endif
|
|
|
|
/* AV_PAIRs */
|
|
if (ntlmssp->ntlm_v2)
|
|
ntlmssp_populate_av_pairs(ntlmssp);
|
|
|
|
/* Timestamp */
|
|
ntlmssp_generate_timestamp(ntlmssp);
|
|
|
|
/* LmChallengeResponse */
|
|
ntlmssp_compute_lm_v2_response(ntlmssp);
|
|
|
|
if (ntlmssp->ntlm_v2)
|
|
memset(ntlmssp->lm_challenge_response.data, '\0', 24);
|
|
|
|
/* NtChallengeResponse */
|
|
ntlmssp_compute_ntlm_v2_response(ntlmssp);
|
|
|
|
/* KeyExchangeKey */
|
|
ntlmssp_generate_key_exchange_key(ntlmssp);
|
|
|
|
/* EncryptedRandomSessionKey */
|
|
ntlmssp_encrypt_random_session_key(ntlmssp);
|
|
|
|
/* Generate signing keys */
|
|
ntlmssp_generate_client_signing_key(ntlmssp);
|
|
ntlmssp_generate_server_signing_key(ntlmssp);
|
|
|
|
/* Generate sealing keys */
|
|
ntlmssp_generate_client_sealing_key(ntlmssp);
|
|
ntlmssp_generate_server_sealing_key(ntlmssp);
|
|
|
|
/* Initialize RC4 seal state using client sealing key */
|
|
ntlmssp_init_rc4_seal_states(ntlmssp);
|
|
|
|
#ifdef WITH_DEBUG_NLA
|
|
printf("ClientChallenge\n");
|
|
freerdp_hexdump(ntlmssp->client_challenge, 8);
|
|
printf("\n");
|
|
|
|
printf("ServerChallenge\n");
|
|
freerdp_hexdump(ntlmssp->server_challenge, 8);
|
|
printf("\n");
|
|
|
|
printf("SessionBaseKey\n");
|
|
freerdp_hexdump(ntlmssp->session_base_key, 16);
|
|
printf("\n");
|
|
|
|
printf("KeyExchangeKey\n");
|
|
freerdp_hexdump(ntlmssp->key_exchange_key, 16);
|
|
printf("\n");
|
|
|
|
printf("ExportedSessionKey\n");
|
|
freerdp_hexdump(ntlmssp->exported_session_key, 16);
|
|
printf("\n");
|
|
|
|
printf("RandomSessionKey\n");
|
|
freerdp_hexdump(ntlmssp->random_session_key, 16);
|
|
printf("\n");
|
|
|
|
printf("ClientSignKey\n");
|
|
freerdp_hexdump(ntlmssp->client_signing_key, 16);
|
|
printf("\n");
|
|
|
|
printf("ClientSealingKey\n");
|
|
freerdp_hexdump(ntlmssp->client_sealing_key, 16);
|
|
printf("\n");
|
|
|
|
printf("Timestamp\n");
|
|
freerdp_hexdump(ntlmssp->timestamp, 8);
|
|
printf("\n");
|
|
#endif
|
|
|
|
ntlmssp->state = NTLMSSP_STATE_AUTHENTICATE;
|
|
}
|
|
|
|
/**
|
|
* Send NTLMSSP AUTHENTICATE_MESSAGE.\n
|
|
* AUTHENTICATE_MESSAGE @msdn{cc236643}
|
|
* @param ntlmssp
|
|
* @param s
|
|
*/
|
|
|
|
void ntlmssp_send_authenticate_message(NTLMSSP* ntlmssp, STREAM* s)
|
|
{
|
|
int length;
|
|
uint32 negotiateFlags = 0;
|
|
uint8* mic_offset = NULL;
|
|
|
|
uint16 DomainNameLen;
|
|
uint16 UserNameLen;
|
|
uint16 WorkstationLen;
|
|
uint16 LmChallengeResponseLen;
|
|
uint16 NtChallengeResponseLen;
|
|
uint16 EncryptedRandomSessionKeyLen;
|
|
|
|
uint32 PayloadBufferOffset;
|
|
uint32 DomainNameBufferOffset;
|
|
uint32 UserNameBufferOffset;
|
|
uint32 WorkstationBufferOffset;
|
|
uint32 LmChallengeResponseBufferOffset;
|
|
uint32 NtChallengeResponseBufferOffset;
|
|
uint32 EncryptedRandomSessionKeyBufferOffset;
|
|
|
|
uint8* UserNameBuffer;
|
|
uint8* DomainNameBuffer;
|
|
uint8* WorkstationBuffer;
|
|
uint8* EncryptedRandomSessionKeyBuffer;
|
|
|
|
WorkstationLen = ntlmssp->workstation.length;
|
|
WorkstationBuffer = ntlmssp->workstation.data;
|
|
|
|
if (ntlmssp->ntlm_v2 < 1)
|
|
WorkstationLen = 0;
|
|
|
|
DomainNameLen = ntlmssp->domain.length;
|
|
DomainNameBuffer = ntlmssp->domain.data;
|
|
|
|
UserNameLen = ntlmssp->username.length;
|
|
UserNameBuffer = ntlmssp->username.data;
|
|
|
|
LmChallengeResponseLen = ntlmssp->lm_challenge_response.length;
|
|
NtChallengeResponseLen = ntlmssp->nt_challenge_response.length;
|
|
|
|
EncryptedRandomSessionKeyLen = 16;
|
|
EncryptedRandomSessionKeyBuffer = ntlmssp->encrypted_random_session_key;
|
|
|
|
if (ntlmssp->ntlm_v2)
|
|
{
|
|
/* observed: 35 82 88 e2 (0xE2888235) */
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_56;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_KEY_EXCH;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_128;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_VERSION;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_TARGET_INFO;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_EXTENDED_SESSION_SECURITY;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_ALWAYS_SIGN;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_NTLM;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_SEAL;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_SIGN;
|
|
negotiateFlags |= NTLMSSP_REQUEST_TARGET;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_UNICODE;
|
|
}
|
|
else
|
|
{
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_KEY_EXCH;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_128;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_EXTENDED_SESSION_SECURITY;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_ALWAYS_SIGN;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_NTLM;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_SEAL;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_SIGN;
|
|
negotiateFlags |= NTLMSSP_REQUEST_TARGET;
|
|
negotiateFlags |= NTLMSSP_NEGOTIATE_UNICODE;
|
|
}
|
|
|
|
if (ntlmssp->ntlm_v2)
|
|
PayloadBufferOffset = 80; /* starting buffer offset */
|
|
else
|
|
PayloadBufferOffset = 64; /* starting buffer offset */
|
|
|
|
if (negotiateFlags & NTLMSSP_NEGOTIATE_VERSION)
|
|
PayloadBufferOffset += 8;
|
|
|
|
DomainNameBufferOffset = PayloadBufferOffset;
|
|
UserNameBufferOffset = DomainNameBufferOffset + DomainNameLen;
|
|
WorkstationBufferOffset = UserNameBufferOffset + UserNameLen;
|
|
LmChallengeResponseBufferOffset = WorkstationBufferOffset + WorkstationLen;
|
|
NtChallengeResponseBufferOffset = LmChallengeResponseBufferOffset + LmChallengeResponseLen;
|
|
EncryptedRandomSessionKeyBufferOffset = NtChallengeResponseBufferOffset + NtChallengeResponseLen;
|
|
|
|
stream_write(s, ntlm_signature, 8); /* Signature (8 bytes) */
|
|
stream_write_uint32(s, MESSAGE_TYPE_AUTHENTICATE); /* MessageType */
|
|
|
|
/* LmChallengeResponseFields (8 bytes) */
|
|
stream_write_uint16(s, LmChallengeResponseLen); /* LmChallengeResponseLen */
|
|
stream_write_uint16(s, LmChallengeResponseLen); /* LmChallengeResponseMaxLen */
|
|
stream_write_uint32(s, LmChallengeResponseBufferOffset); /* LmChallengeResponseBufferOffset */
|
|
|
|
/* NtChallengeResponseFields (8 bytes) */
|
|
stream_write_uint16(s, NtChallengeResponseLen); /* NtChallengeResponseLen */
|
|
stream_write_uint16(s, NtChallengeResponseLen); /* NtChallengeResponseMaxLen */
|
|
stream_write_uint32(s, NtChallengeResponseBufferOffset); /* NtChallengeResponseBufferOffset */
|
|
|
|
/* only set if NTLMSSP_NEGOTIATE_DOMAIN_SUPPLIED is set */
|
|
|
|
/* DomainNameFields (8 bytes) */
|
|
stream_write_uint16(s, DomainNameLen); /* DomainNameLen */
|
|
stream_write_uint16(s, DomainNameLen); /* DomainNameMaxLen */
|
|
stream_write_uint32(s, DomainNameBufferOffset); /* DomainNameBufferOffset */
|
|
|
|
/* UserNameFields (8 bytes) */
|
|
stream_write_uint16(s, UserNameLen); /* UserNameLen */
|
|
stream_write_uint16(s, UserNameLen); /* UserNameMaxLen */
|
|
stream_write_uint32(s, UserNameBufferOffset); /* UserNameBufferOffset */
|
|
|
|
/* only set if NTLMSSP_NEGOTIATE_WORKSTATION_SUPPLIED is set */
|
|
|
|
/* WorkstationFields (8 bytes) */
|
|
stream_write_uint16(s, WorkstationLen); /* WorkstationLen */
|
|
stream_write_uint16(s, WorkstationLen); /* WorkstationMaxLen */
|
|
stream_write_uint32(s, WorkstationBufferOffset); /* WorkstationBufferOffset */
|
|
|
|
/* EncryptedRandomSessionKeyFields (8 bytes) */
|
|
stream_write_uint16(s, EncryptedRandomSessionKeyLen); /* EncryptedRandomSessionKeyLen */
|
|
stream_write_uint16(s, EncryptedRandomSessionKeyLen); /* EncryptedRandomSessionKeyMaxLen */
|
|
stream_write_uint32(s, EncryptedRandomSessionKeyBufferOffset); /* EncryptedRandomSessionKeyBufferOffset */
|
|
|
|
ntlmssp_output_negotiate_flags(s, negotiateFlags); /* NegotiateFlags (4 bytes) */
|
|
|
|
#ifdef WITH_DEBUG_NLA
|
|
ntlmssp_print_negotiate_flags(negotiateFlags);
|
|
#endif
|
|
|
|
if (negotiateFlags & NTLMSSP_NEGOTIATE_VERSION)
|
|
{
|
|
/* Only present if NTLMSSP_NEGOTIATE_VERSION is set */
|
|
ntlmssp_output_version(s);
|
|
|
|
#ifdef WITH_DEBUG_NLA
|
|
printf("Version (length = 8)\n");
|
|
freerdp_hexdump((s->p - 8), 8);
|
|
printf("\n");
|
|
#endif
|
|
}
|
|
|
|
if (ntlmssp->ntlm_v2)
|
|
{
|
|
/* Message Integrity Check */
|
|
mic_offset = s->p;
|
|
stream_write_zero(s, 16);
|
|
}
|
|
|
|
/* DomainName */
|
|
if (DomainNameLen > 0)
|
|
{
|
|
stream_write(s, DomainNameBuffer, DomainNameLen);
|
|
#ifdef WITH_DEBUG_NLA
|
|
printf("DomainName (length = %d, offset = %d)\n", DomainNameLen, DomainNameBufferOffset);
|
|
freerdp_hexdump(DomainNameBuffer, DomainNameLen);
|
|
printf("\n");
|
|
#endif
|
|
}
|
|
|
|
/* UserName */
|
|
stream_write(s, UserNameBuffer, UserNameLen);
|
|
|
|
#ifdef WITH_DEBUG_NLA
|
|
printf("UserName (length = %d, offset = %d)\n", UserNameLen, UserNameBufferOffset);
|
|
freerdp_hexdump(UserNameBuffer, UserNameLen);
|
|
printf("\n");
|
|
#endif
|
|
|
|
/* Workstation */
|
|
if (WorkstationLen > 0)
|
|
{
|
|
stream_write(s, WorkstationBuffer, WorkstationLen);
|
|
#ifdef WITH_DEBUG_NLA
|
|
printf("Workstation (length = %d, offset = %d)\n", WorkstationLen, WorkstationBufferOffset);
|
|
freerdp_hexdump(WorkstationBuffer, WorkstationLen);
|
|
printf("\n");
|
|
#endif
|
|
}
|
|
|
|
/* LmChallengeResponse */
|
|
stream_write(s, ntlmssp->lm_challenge_response.data, LmChallengeResponseLen);
|
|
|
|
#ifdef WITH_DEBUG_NLA
|
|
printf("LmChallengeResponse (length = %d, offset = %d)\n", LmChallengeResponseLen, LmChallengeResponseBufferOffset);
|
|
freerdp_hexdump(ntlmssp->lm_challenge_response.data, LmChallengeResponseLen);
|
|
printf("\n");
|
|
#endif
|
|
|
|
/* NtChallengeResponse */
|
|
stream_write(s, ntlmssp->nt_challenge_response.data, NtChallengeResponseLen);
|
|
|
|
#ifdef WITH_DEBUG_NLA
|
|
if (ntlmssp->ntlm_v2)
|
|
{
|
|
ntlmssp_print_av_pairs(ntlmssp);
|
|
|
|
printf("targetInfo (length = %d)\n", ntlmssp->target_info.length);
|
|
freerdp_hexdump(ntlmssp->target_info.data, ntlmssp->target_info.length);
|
|
printf("\n");
|
|
}
|
|
#endif
|
|
|
|
#ifdef WITH_DEBUG_NLA
|
|
printf("NtChallengeResponse (length = %d, offset = %d)\n", NtChallengeResponseLen, NtChallengeResponseBufferOffset);
|
|
freerdp_hexdump(ntlmssp->nt_challenge_response.data, NtChallengeResponseLen);
|
|
printf("\n");
|
|
#endif
|
|
|
|
/* EncryptedRandomSessionKey */
|
|
stream_write(s, EncryptedRandomSessionKeyBuffer, EncryptedRandomSessionKeyLen);
|
|
|
|
#ifdef WITH_DEBUG_NLA
|
|
printf("EncryptedRandomSessionKey (length = %d, offset = %d)\n", EncryptedRandomSessionKeyLen, EncryptedRandomSessionKeyBufferOffset);
|
|
freerdp_hexdump(EncryptedRandomSessionKeyBuffer, EncryptedRandomSessionKeyLen);
|
|
printf("\n");
|
|
#endif
|
|
|
|
length = s->p - s->data;
|
|
freerdp_blob_alloc(&ntlmssp->authenticate_message, length);
|
|
memcpy(ntlmssp->authenticate_message.data, s->data, length);
|
|
|
|
if (ntlmssp->ntlm_v2)
|
|
{
|
|
/* Message Integrity Check */
|
|
ntlmssp_compute_message_integrity_check(ntlmssp);
|
|
|
|
s->p = mic_offset;
|
|
stream_write(s, ntlmssp->message_integrity_check, 16);
|
|
s->p = s->data + length;
|
|
|
|
#ifdef WITH_DEBUG_NLA
|
|
printf("MessageIntegrityCheck (length = 16)\n");
|
|
freerdp_hexdump(mic_offset, 16);
|
|
printf("\n");
|
|
#endif
|
|
}
|
|
|
|
#ifdef WITH_DEBUG_NLA
|
|
printf("AUTHENTICATE_MESSAGE (length = %d)\n", length);
|
|
freerdp_hexdump(s->data, length);
|
|
printf("\n");
|
|
#endif
|
|
|
|
ntlmssp->state = NTLMSSP_STATE_FINAL;
|
|
}
|
|
|
|
/**
|
|
* Receive NTLMSSP AUTHENTICATE_MESSAGE.\n
|
|
* AUTHENTICATE_MESSAGE @msdn{cc236643}
|
|
* @param ntlmssp
|
|
* @param s
|
|
*/
|
|
|
|
void ntlmssp_recv_authenticate_message(NTLMSSP* ntlmssp, STREAM* s)
|
|
{
|
|
ntlmssp->state = NTLMSSP_STATE_FINAL;
|
|
}
|
|
|
|
/**
|
|
* Send NTLMSSP message (client).
|
|
* @param ntlmssp
|
|
* @param s
|
|
* @return
|
|
*/
|
|
|
|
int ntlmssp_client_send(NTLMSSP* ntlmssp, STREAM* s)
|
|
{
|
|
if (ntlmssp->state == NTLMSSP_STATE_INITIAL)
|
|
ntlmssp->state = NTLMSSP_STATE_NEGOTIATE;
|
|
|
|
if (ntlmssp->state == NTLMSSP_STATE_NEGOTIATE)
|
|
ntlmssp_send_negotiate_message(ntlmssp, s);
|
|
else if (ntlmssp->state == NTLMSSP_STATE_AUTHENTICATE)
|
|
ntlmssp_send_authenticate_message(ntlmssp, s);
|
|
|
|
return (ntlmssp->state == NTLMSSP_STATE_FINAL) ? 0 : 1;
|
|
}
|
|
|
|
/**
|
|
* Send NTLMSSP message (server).
|
|
* @param ntlmssp
|
|
* @param s
|
|
* @return
|
|
*/
|
|
|
|
int ntlmssp_server_send(NTLMSSP* ntlmssp, STREAM* s)
|
|
{
|
|
if (ntlmssp->state == NTLMSSP_STATE_CHALLENGE)
|
|
ntlmssp_send_challenge_message(ntlmssp, s);
|
|
|
|
return (ntlmssp->state == NTLMSSP_STATE_FINAL) ? 0 : 1;
|
|
}
|
|
|
|
/**
|
|
* Send NTLMSSP message.
|
|
* @param ntlmssp
|
|
* @param s
|
|
* @return
|
|
*/
|
|
|
|
int ntlmssp_send(NTLMSSP* ntlmssp, STREAM* s)
|
|
{
|
|
if (ntlmssp->server)
|
|
return ntlmssp_server_send(ntlmssp, s);
|
|
else
|
|
return ntlmssp_client_send(ntlmssp, s);
|
|
}
|
|
|
|
/**
|
|
* Receive NTLMSSP message (client).
|
|
* @param ntlmssp
|
|
* @param s
|
|
* @return
|
|
*/
|
|
|
|
int ntlmssp_client_recv(NTLMSSP* ntlmssp, STREAM* s)
|
|
{
|
|
char signature[8]; /* Signature, "NTLMSSP" */
|
|
uint32 messageType; /* MessageType */
|
|
|
|
stream_read(s, signature, 8);
|
|
stream_read_uint32(s, messageType);
|
|
|
|
if (memcmp(signature, ntlm_signature, 8) != 0)
|
|
{
|
|
printf("Unexpected NTLM signature: %s, expected:%s\n", signature, ntlm_signature);
|
|
return -1;
|
|
}
|
|
|
|
if (messageType == MESSAGE_TYPE_CHALLENGE && ntlmssp->state == NTLMSSP_STATE_CHALLENGE)
|
|
ntlmssp_recv_challenge_message(ntlmssp, s);
|
|
|
|
return 1;
|
|
}
|
|
|
|
/**
|
|
* Receive NTLMSSP message (server).
|
|
* @param ntlmssp
|
|
* @param s
|
|
* @return
|
|
*/
|
|
|
|
int ntlmssp_server_recv(NTLMSSP* ntlmssp, STREAM* s)
|
|
{
|
|
char signature[8]; /* Signature, "NTLMSSP" */
|
|
uint32 messageType; /* MessageType */
|
|
|
|
stream_read(s, signature, 8);
|
|
stream_read_uint32(s, messageType);
|
|
|
|
if (memcmp(signature, ntlm_signature, 8) != 0)
|
|
{
|
|
printf("Unexpected NTLM signature: %s, expected:%s\n", signature, ntlm_signature);
|
|
return -1;
|
|
}
|
|
|
|
if (messageType == MESSAGE_TYPE_NEGOTIATE && ntlmssp->state == NTLMSSP_STATE_INITIAL)
|
|
ntlmssp_recv_negotiate_message(ntlmssp, s);
|
|
else if (messageType == MESSAGE_TYPE_AUTHENTICATE && ntlmssp->state == NTLMSSP_STATE_AUTHENTICATE)
|
|
ntlmssp_recv_authenticate_message(ntlmssp, s);
|
|
|
|
return 1;
|
|
}
|
|
|
|
/**
|
|
* Receive NTLMSSP message.
|
|
* @param ntlmssp
|
|
* @param s
|
|
* @return
|
|
*/
|
|
|
|
int ntlmssp_recv(NTLMSSP* ntlmssp, STREAM* s)
|
|
{
|
|
if (ntlmssp->server)
|
|
return ntlmssp_server_recv(ntlmssp, s);
|
|
else
|
|
return ntlmssp_client_recv(ntlmssp, s);
|
|
}
|
|
|
|
/**
|
|
* Create new NTLMSSP state machine instance.
|
|
* @return
|
|
*/
|
|
|
|
NTLMSSP* ntlmssp_new()
|
|
{
|
|
NTLMSSP* ntlmssp = (NTLMSSP*) xmalloc(sizeof(NTLMSSP));
|
|
|
|
if (ntlmssp != NULL)
|
|
{
|
|
memset(ntlmssp, '\0', sizeof(NTLMSSP));
|
|
ntlmssp->av_pairs = (AV_PAIRS*) xmalloc(sizeof(AV_PAIRS));
|
|
memset(ntlmssp->av_pairs, 0, sizeof(AV_PAIRS));
|
|
ntlmssp_init(ntlmssp);
|
|
}
|
|
|
|
return ntlmssp;
|
|
}
|
|
|
|
/**
|
|
* Create new NTLMSSP client state machine instance.
|
|
* @return
|
|
*/
|
|
|
|
NTLMSSP* ntlmssp_client_new()
|
|
{
|
|
NTLMSSP* ntlmssp = ntlmssp_new();
|
|
ntlmssp->server = false;
|
|
return ntlmssp;
|
|
}
|
|
|
|
/**
|
|
* Create new NTLMSSP client state machine instance.
|
|
* @return
|
|
*/
|
|
|
|
NTLMSSP* ntlmssp_server_new()
|
|
{
|
|
NTLMSSP* ntlmssp = ntlmssp_new();
|
|
ntlmssp->server = true;
|
|
return ntlmssp;
|
|
}
|
|
|
|
/**
|
|
* Initialize NTLMSSP state machine.
|
|
* @param ntlmssp
|
|
*/
|
|
|
|
void ntlmssp_init(NTLMSSP* ntlmssp)
|
|
{
|
|
ntlmssp->state = NTLMSSP_STATE_INITIAL;
|
|
ntlmssp->uniconv = freerdp_uniconv_new();
|
|
}
|
|
|
|
/**
|
|
* Finalize NTLMSSP state machine.
|
|
* @param ntlmssp
|
|
*/
|
|
|
|
void ntlmssp_uninit(NTLMSSP* ntlmssp)
|
|
{
|
|
freerdp_blob_free(&ntlmssp->username);
|
|
freerdp_blob_free(&ntlmssp->password);
|
|
freerdp_blob_free(&ntlmssp->domain);
|
|
|
|
freerdp_blob_free(&ntlmssp->spn);
|
|
freerdp_blob_free(&ntlmssp->workstation);
|
|
freerdp_blob_free(&ntlmssp->target_info);
|
|
freerdp_blob_free(&ntlmssp->target_name);
|
|
|
|
freerdp_blob_free(&ntlmssp->negotiate_message);
|
|
freerdp_blob_free(&ntlmssp->challenge_message);
|
|
freerdp_blob_free(&ntlmssp->authenticate_message);
|
|
|
|
freerdp_blob_free(&ntlmssp->lm_challenge_response);
|
|
freerdp_blob_free(&ntlmssp->nt_challenge_response);
|
|
|
|
ntlmssp_free_av_pairs(ntlmssp);
|
|
freerdp_uniconv_free(ntlmssp->uniconv);
|
|
|
|
ntlmssp->state = NTLMSSP_STATE_FINAL;
|
|
}
|
|
|
|
/**
|
|
* Free NTLMSSP state machine.
|
|
* @param ntlmssp
|
|
*/
|
|
|
|
void ntlmssp_free(NTLMSSP* ntlmssp)
|
|
{
|
|
ntlmssp_uninit(ntlmssp);
|
|
|
|
if (ntlmssp->send_rc4_seal)
|
|
crypto_rc4_free(ntlmssp->send_rc4_seal);
|
|
if (ntlmssp->recv_rc4_seal)
|
|
crypto_rc4_free(ntlmssp->recv_rc4_seal);
|
|
xfree(ntlmssp);
|
|
}
|