2012-06-03 02:21:04 +04:00
|
|
|
/**
|
|
|
|
* WinPR: Windows Portable Runtime
|
|
|
|
* NTLM Utils
|
|
|
|
*
|
|
|
|
* Copyright 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.
|
|
|
|
*/
|
|
|
|
|
2022-02-16 12:08:00 +03:00
|
|
|
#include <winpr/config.h>
|
2012-08-15 01:20:53 +04:00
|
|
|
|
2012-06-03 02:21:04 +04:00
|
|
|
#include <winpr/ntlm.h>
|
2021-06-17 12:25:58 +03:00
|
|
|
#include <winpr/assert.h>
|
2012-06-03 02:21:04 +04:00
|
|
|
|
|
|
|
#include <winpr/crt.h>
|
2015-10-09 22:57:41 +03:00
|
|
|
#include <winpr/crypto.h>
|
2012-06-03 02:21:04 +04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Define NTOWFv1(Password, User, Domain) as
|
|
|
|
* MD4(UNICODE(Password))
|
|
|
|
* EndDefine
|
|
|
|
*/
|
|
|
|
|
2018-08-20 11:55:29 +03:00
|
|
|
BOOL NTOWFv1W(LPWSTR Password, UINT32 PasswordLength, BYTE* NtHash)
|
2012-06-03 02:21:04 +04:00
|
|
|
{
|
2018-08-20 11:55:29 +03:00
|
|
|
if (!Password || !NtHash)
|
|
|
|
return FALSE;
|
2016-11-24 16:53:19 +03:00
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
if (!winpr_Digest(WINPR_MD_MD4, (BYTE*)Password, (size_t)PasswordLength, NtHash,
|
2018-08-22 14:16:31 +03:00
|
|
|
WINPR_MD4_DIGEST_LENGTH))
|
2018-08-20 11:55:29 +03:00
|
|
|
return FALSE;
|
2012-06-03 02:21:04 +04:00
|
|
|
|
2018-08-20 11:55:29 +03:00
|
|
|
return TRUE;
|
2012-06-03 02:21:04 +04:00
|
|
|
}
|
|
|
|
|
2018-08-20 11:55:29 +03:00
|
|
|
BOOL NTOWFv1A(LPSTR Password, UINT32 PasswordLength, BYTE* NtHash)
|
2012-06-03 02:21:04 +04:00
|
|
|
{
|
|
|
|
LPWSTR PasswordW = NULL;
|
2018-08-20 11:55:29 +03:00
|
|
|
BOOL result = FALSE;
|
2022-10-28 09:09:27 +03:00
|
|
|
size_t pwdCharLength = 0;
|
2018-08-20 11:55:29 +03:00
|
|
|
|
|
|
|
if (!NtHash)
|
|
|
|
return FALSE;
|
2012-06-03 02:21:04 +04:00
|
|
|
|
2022-10-28 09:09:27 +03:00
|
|
|
PasswordW = ConvertUtf8NToWCharAlloc(Password, PasswordLength, &pwdCharLength);
|
|
|
|
if (!PasswordW)
|
2018-08-20 11:55:29 +03:00
|
|
|
return FALSE;
|
2015-04-03 17:21:01 +03:00
|
|
|
|
2023-07-28 10:42:01 +03:00
|
|
|
if (!NTOWFv1W(PasswordW, (UINT32)pwdCharLength * sizeof(WCHAR), NtHash))
|
2018-08-20 11:55:29 +03:00
|
|
|
goto out_fail;
|
|
|
|
|
|
|
|
result = TRUE;
|
|
|
|
out_fail:
|
2012-06-03 02:21:04 +04:00
|
|
|
free(PasswordW);
|
2018-08-20 11:55:29 +03:00
|
|
|
return result;
|
2012-06-03 02:21:04 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Define NTOWFv2(Password, User, Domain) as
|
|
|
|
* HMAC_MD5(MD4(UNICODE(Password)),
|
|
|
|
* UNICODE(ConcatenationOf(UpperCase(User), Domain)))
|
|
|
|
* EndDefine
|
|
|
|
*/
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
BOOL NTOWFv2W(LPWSTR Password, UINT32 PasswordLength, LPWSTR User, UINT32 UserLength, LPWSTR Domain,
|
|
|
|
UINT32 DomainLength, BYTE* NtHash)
|
2012-06-03 02:21:04 +04:00
|
|
|
{
|
2020-04-24 01:01:58 +03:00
|
|
|
BYTE NtHashV1[WINPR_MD5_DIGEST_LENGTH];
|
2012-06-03 02:21:04 +04:00
|
|
|
|
2018-08-20 11:55:29 +03:00
|
|
|
if ((!User) || (!Password) || (!NtHash))
|
|
|
|
return FALSE;
|
2012-06-03 02:21:04 +04:00
|
|
|
|
2015-04-03 17:21:01 +03:00
|
|
|
if (!NTOWFv1W(Password, PasswordLength, NtHashV1))
|
2018-08-20 11:55:29 +03:00
|
|
|
return FALSE;
|
2012-06-03 02:21:04 +04:00
|
|
|
|
2020-04-23 02:04:33 +03:00
|
|
|
return NTOWFv2FromHashW(NtHashV1, User, UserLength, Domain, DomainLength, NtHash);
|
2012-06-03 02:21:04 +04:00
|
|
|
}
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
BOOL NTOWFv2A(LPSTR Password, UINT32 PasswordLength, LPSTR User, UINT32 UserLength, LPSTR Domain,
|
|
|
|
UINT32 DomainLength, BYTE* NtHash)
|
2012-06-03 02:21:04 +04:00
|
|
|
{
|
|
|
|
LPWSTR UserW = NULL;
|
|
|
|
LPWSTR DomainW = NULL;
|
|
|
|
LPWSTR PasswordW = NULL;
|
2018-08-20 11:55:29 +03:00
|
|
|
BOOL result = FALSE;
|
2022-10-28 09:09:27 +03:00
|
|
|
size_t userCharLength = 0;
|
|
|
|
size_t domainCharLength = 0;
|
|
|
|
size_t pwdCharLength = 0;
|
2018-08-20 11:55:29 +03:00
|
|
|
|
|
|
|
if (!NtHash)
|
|
|
|
return FALSE;
|
|
|
|
|
2022-10-28 09:09:27 +03:00
|
|
|
UserW = ConvertUtf8NToWCharAlloc(User, UserLength, &userCharLength);
|
|
|
|
DomainW = ConvertUtf8NToWCharAlloc(Domain, DomainLength, &domainCharLength);
|
|
|
|
PasswordW = ConvertUtf8NToWCharAlloc(Password, PasswordLength, &pwdCharLength);
|
2012-06-03 02:21:04 +04:00
|
|
|
|
2015-04-03 17:21:01 +03:00
|
|
|
if (!UserW || !DomainW || !PasswordW)
|
|
|
|
goto out_fail;
|
|
|
|
|
2023-07-28 10:42:01 +03:00
|
|
|
if (!NTOWFv2W(PasswordW, (UINT32)pwdCharLength * sizeof(WCHAR), UserW,
|
|
|
|
(UINT32)userCharLength * sizeof(WCHAR), DomainW,
|
|
|
|
(UINT32)domainCharLength * sizeof(WCHAR), NtHash))
|
2018-08-20 11:55:29 +03:00
|
|
|
goto out_fail;
|
|
|
|
|
|
|
|
result = TRUE;
|
2015-04-03 17:21:01 +03:00
|
|
|
out_fail:
|
2012-06-03 02:21:04 +04:00
|
|
|
free(UserW);
|
|
|
|
free(DomainW);
|
|
|
|
free(PasswordW);
|
2018-08-20 11:55:29 +03:00
|
|
|
return result;
|
2012-06-03 02:21:04 +04:00
|
|
|
}
|
|
|
|
|
2018-08-20 11:55:29 +03:00
|
|
|
BOOL NTOWFv2FromHashW(BYTE* NtHashV1, LPWSTR User, UINT32 UserLength, LPWSTR Domain,
|
|
|
|
UINT32 DomainLength, BYTE* NtHash)
|
2012-07-23 07:23:23 +04:00
|
|
|
{
|
2024-01-23 18:49:54 +03:00
|
|
|
BYTE* buffer = NULL;
|
2018-08-20 11:55:29 +03:00
|
|
|
BYTE result = FALSE;
|
2012-07-23 07:23:23 +04:00
|
|
|
|
2018-08-20 11:55:29 +03:00
|
|
|
if (!User || !NtHash)
|
|
|
|
return FALSE;
|
2012-07-23 07:23:23 +04:00
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
if (!(buffer = (BYTE*)malloc(UserLength + DomainLength)))
|
2018-08-20 11:55:29 +03:00
|
|
|
return FALSE;
|
2012-07-23 07:23:23 +04:00
|
|
|
|
|
|
|
/* Concatenate(UpperCase(User), Domain) */
|
|
|
|
CopyMemory(buffer, User, UserLength);
|
2019-11-06 17:24:51 +03:00
|
|
|
CharUpperBuffW((LPWSTR)buffer, UserLength / 2);
|
2012-08-23 09:18:47 +04:00
|
|
|
|
|
|
|
if (DomainLength > 0)
|
|
|
|
{
|
|
|
|
CopyMemory(&buffer[UserLength], Domain, DomainLength);
|
|
|
|
}
|
2012-07-23 07:23:23 +04:00
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
/* Compute the HMAC-MD5 hash of the above value using the NTLMv1 hash as the key, the result is
|
|
|
|
* the NTLMv2 hash */
|
2018-08-22 14:16:31 +03:00
|
|
|
if (!winpr_HMAC(WINPR_MD_MD5, NtHashV1, 16, buffer, UserLength + DomainLength, NtHash,
|
2020-04-23 02:04:33 +03:00
|
|
|
WINPR_MD5_DIGEST_LENGTH))
|
2018-08-20 11:55:29 +03:00
|
|
|
goto out_fail;
|
2012-07-23 07:23:23 +04:00
|
|
|
|
2018-08-20 11:55:29 +03:00
|
|
|
result = TRUE;
|
|
|
|
out_fail:
|
2012-07-23 07:23:23 +04:00
|
|
|
free(buffer);
|
2016-02-24 23:45:09 +03:00
|
|
|
return result;
|
2012-07-23 07:23:23 +04:00
|
|
|
}
|
|
|
|
|
2018-08-20 11:55:29 +03:00
|
|
|
BOOL NTOWFv2FromHashA(BYTE* NtHashV1, LPSTR User, UINT32 UserLength, LPSTR Domain,
|
|
|
|
UINT32 DomainLength, BYTE* NtHash)
|
2012-07-23 07:23:23 +04:00
|
|
|
{
|
|
|
|
LPWSTR UserW = NULL;
|
|
|
|
LPWSTR DomainW = NULL;
|
2018-08-20 11:55:29 +03:00
|
|
|
BOOL result = FALSE;
|
2022-10-28 09:09:27 +03:00
|
|
|
size_t userCharLength = 0;
|
|
|
|
size_t domainCharLength = 0;
|
2018-08-20 11:55:29 +03:00
|
|
|
if (!NtHash)
|
|
|
|
return FALSE;
|
|
|
|
|
2022-10-28 09:09:27 +03:00
|
|
|
UserW = ConvertUtf8NToWCharAlloc(User, UserLength, &userCharLength);
|
|
|
|
DomainW = ConvertUtf8NToWCharAlloc(Domain, DomainLength, &domainCharLength);
|
2012-07-23 07:23:23 +04:00
|
|
|
|
2015-04-03 17:21:01 +03:00
|
|
|
if (!UserW || !DomainW)
|
|
|
|
goto out_fail;
|
|
|
|
|
2023-07-28 10:42:01 +03:00
|
|
|
if (!NTOWFv2FromHashW(NtHashV1, UserW, (UINT32)userCharLength * sizeof(WCHAR), DomainW,
|
|
|
|
(UINT32)domainCharLength * sizeof(WCHAR), NtHash))
|
2018-08-20 11:55:29 +03:00
|
|
|
goto out_fail;
|
|
|
|
|
|
|
|
result = TRUE;
|
2015-04-03 17:21:01 +03:00
|
|
|
out_fail:
|
2012-07-23 07:23:23 +04:00
|
|
|
free(UserW);
|
|
|
|
free(DomainW);
|
2018-08-20 11:55:29 +03:00
|
|
|
return result;
|
2012-07-23 07:23:23 +04:00
|
|
|
}
|