|  | /* | 
|  | *  Copyright 2004 Hans Leidekker | 
|  | * | 
|  | *  Based on LMHash.c from libcifs | 
|  | * | 
|  | *  Copyright (C) 2004 by Christopher R. Hertel | 
|  | * | 
|  | *  This library is free software; you can redistribute it and/or | 
|  | *  modify it under the terms of the GNU Lesser General Public | 
|  | *  License as published by the Free Software Foundation; either | 
|  | *  version 2.1 of the License, or (at your option) any later version. | 
|  | * | 
|  | *  This library is distributed in the hope that it will be useful, | 
|  | *  but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
|  | *  Lesser General Public License for more details. | 
|  | * | 
|  | *  You should have received a copy of the GNU Lesser General Public | 
|  | *  License along with this library; if not, write to the Free Software | 
|  | *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA | 
|  | */ | 
|  |  | 
|  | #include <stdarg.h> | 
|  |  | 
|  | #include "ntstatus.h" | 
|  | #define WIN32_NO_STATUS | 
|  | #include "windef.h" | 
|  | #include "winternl.h" | 
|  |  | 
|  | #include "crypt.h" | 
|  |  | 
|  | static const unsigned char CRYPT_LMhash_Magic[8] = | 
|  | { 'K', 'G', 'S', '!', '@', '#', '$', '%' }; | 
|  |  | 
|  | static void CRYPT_LMhash( unsigned char *dst, const unsigned char *pwd, const int len ) | 
|  | { | 
|  | int i, max = 14; | 
|  | unsigned char tmp_pwd[14] = { 0,0,0,0,0,0,0,0,0,0,0,0,0,0 }; | 
|  |  | 
|  | max = len > max ? max : len; | 
|  |  | 
|  | for (i = 0; i < max; i++) | 
|  | tmp_pwd[i] = pwd[i]; | 
|  |  | 
|  | CRYPT_DEShash( dst, tmp_pwd, CRYPT_LMhash_Magic ); | 
|  | CRYPT_DEShash( &dst[8], &tmp_pwd[7], CRYPT_LMhash_Magic ); | 
|  | } | 
|  |  | 
|  | NTSTATUS WINAPI SystemFunction006( LPCSTR password, LPSTR hash ) | 
|  | { | 
|  | CRYPT_LMhash( (unsigned char*)hash, (const unsigned char*)password, strlen(password) ); | 
|  |  | 
|  | return STATUS_SUCCESS; | 
|  | } | 
|  |  | 
|  | /****************************************************************************** | 
|  | * SystemFunction008  [ADVAPI32.@] | 
|  | * | 
|  | * Creates a LM response from a challenge and a password hash | 
|  | * | 
|  | * PARAMS | 
|  | *   challenge  [I] Challenge from authentication server | 
|  | *   hash       [I] NTLM hash (from SystemFunction006) | 
|  | *   response   [O] response to send back to the server | 
|  | * | 
|  | * RETURNS | 
|  | *  Success: STATUS_SUCCESS | 
|  | *  Failure: STATUS_UNSUCCESSFUL | 
|  | * | 
|  | * NOTES | 
|  | *  see http://davenport.sourceforge.net/ntlm.html#theLmResponse | 
|  | * | 
|  | */ | 
|  | NTSTATUS WINAPI SystemFunction008(const BYTE *challenge, const BYTE *hash, LPBYTE response) | 
|  | { | 
|  | BYTE key[7*3]; | 
|  |  | 
|  | if (!challenge || !response) | 
|  | return STATUS_UNSUCCESSFUL; | 
|  |  | 
|  | memset(key, 0, sizeof key); | 
|  | memcpy(key, hash, 0x10); | 
|  |  | 
|  | CRYPT_DEShash(response, key, challenge); | 
|  | CRYPT_DEShash(response+8, key+7, challenge); | 
|  | CRYPT_DEShash(response+16, key+14, challenge); | 
|  |  | 
|  | return STATUS_SUCCESS; | 
|  | } | 
|  |  | 
|  | /****************************************************************************** | 
|  | * SystemFunction009  [ADVAPI32.@] | 
|  | * | 
|  | * Seems to do the same as SystemFunction008 ... | 
|  | */ | 
|  | NTSTATUS WINAPI SystemFunction009(const BYTE *challenge, const BYTE *hash, LPBYTE response) | 
|  | { | 
|  | return SystemFunction008(challenge, hash, response); | 
|  | } | 
|  |  | 
|  | /****************************************************************************** | 
|  | * SystemFunction001  [ADVAPI32.@] | 
|  | * | 
|  | * Encrypts a single block of data using DES | 
|  | * | 
|  | * PARAMS | 
|  | *   data    [I] data to encrypt    (8 bytes) | 
|  | *   key     [I] key data           (7 bytes) | 
|  | *   output  [O] the encrypted data (8 bytes) | 
|  | * | 
|  | * RETURNS | 
|  | *  Success: STATUS_SUCCESS | 
|  | *  Failure: STATUS_UNSUCCESSFUL | 
|  | * | 
|  | */ | 
|  | NTSTATUS WINAPI SystemFunction001(const BYTE *data, const BYTE *key, LPBYTE output) | 
|  | { | 
|  | if (!data || !output) | 
|  | return STATUS_UNSUCCESSFUL; | 
|  | CRYPT_DEShash(output, key, data); | 
|  | return STATUS_SUCCESS; | 
|  | } | 
|  |  | 
|  | /****************************************************************************** | 
|  | * SystemFunction002  [ADVAPI32.@] | 
|  | * | 
|  | * Decrypts a single block of data using DES | 
|  | * | 
|  | * PARAMS | 
|  | *   data    [I] data to decrypt    (8 bytes) | 
|  | *   key     [I] key data           (7 bytes) | 
|  | *   output  [O] the decrypted data (8 bytes) | 
|  | * | 
|  | * RETURNS | 
|  | *  Success: STATUS_SUCCESS | 
|  | *  Failure: STATUS_UNSUCCESSFUL | 
|  | * | 
|  | */ | 
|  | NTSTATUS WINAPI SystemFunction002(const BYTE *data, const BYTE *key, LPBYTE output) | 
|  | { | 
|  | if (!data || !output) | 
|  | return STATUS_UNSUCCESSFUL; | 
|  | CRYPT_DESunhash(output, key, data); | 
|  | return STATUS_SUCCESS; | 
|  | } | 
|  |  | 
|  | /****************************************************************************** | 
|  | * SystemFunction003  [ADVAPI32.@] | 
|  | * | 
|  | * Hashes a key using DES and a fixed datablock | 
|  | * | 
|  | * PARAMS | 
|  | *   key     [I] key data    (7 bytes) | 
|  | *   output  [O] hashed key  (8 bytes) | 
|  | * | 
|  | * RETURNS | 
|  | *  Success: STATUS_SUCCESS | 
|  | *  Failure: STATUS_UNSUCCESSFUL | 
|  | * | 
|  | */ | 
|  | NTSTATUS WINAPI SystemFunction003(const BYTE *key, LPBYTE output) | 
|  | { | 
|  | if (!output) | 
|  | return STATUS_UNSUCCESSFUL; | 
|  | CRYPT_DEShash(output, key, CRYPT_LMhash_Magic); | 
|  | return STATUS_SUCCESS; | 
|  | } | 
|  |  | 
|  | /****************************************************************************** | 
|  | * SystemFunction004  [ADVAPI32.@] | 
|  | * | 
|  | * Encrypts a block of data with DES in ECB mode, preserving the length | 
|  | * | 
|  | * PARAMS | 
|  | *   data    [I] data to encrypt | 
|  | *   key     [I] key data (up to 7 bytes) | 
|  | *   output  [O] buffer to receive encrypted data | 
|  | * | 
|  | * RETURNS | 
|  | *  Success: STATUS_SUCCESS | 
|  | *  Failure: STATUS_BUFFER_TOO_SMALL     if the output buffer is too small | 
|  | *  Failure: STATUS_INVALID_PARAMETER_2  if the key is zero length | 
|  | * | 
|  | * NOTES | 
|  | *  Encrypt buffer size should be input size rounded up to 8 bytes | 
|  | *   plus an extra 8 bytes. | 
|  | */ | 
|  | NTSTATUS WINAPI SystemFunction004(const struct ustring *in, | 
|  | const struct ustring *key, | 
|  | struct ustring *out) | 
|  | { | 
|  | union { | 
|  | unsigned char uc[8]; | 
|  | unsigned int  ui[2]; | 
|  | } data; | 
|  | unsigned char deskey[7]; | 
|  | unsigned int crypt_len, ofs; | 
|  |  | 
|  | if (key->Length<=0) | 
|  | return STATUS_INVALID_PARAMETER_2; | 
|  |  | 
|  | crypt_len = ((in->Length+7)&~7); | 
|  | if (out->MaximumLength < (crypt_len+8)) | 
|  | return STATUS_BUFFER_TOO_SMALL; | 
|  |  | 
|  | data.ui[0] = in->Length; | 
|  | data.ui[1] = 1; | 
|  |  | 
|  | if (key->Length<sizeof deskey) | 
|  | { | 
|  | memset(deskey, 0, sizeof deskey); | 
|  | memcpy(deskey, key->Buffer, key->Length); | 
|  | } | 
|  | else | 
|  | memcpy(deskey, key->Buffer, sizeof deskey); | 
|  |  | 
|  | CRYPT_DEShash(out->Buffer, deskey, data.uc); | 
|  |  | 
|  | for(ofs=0; ofs<(crypt_len-8); ofs+=8) | 
|  | CRYPT_DEShash(out->Buffer+8+ofs, deskey, in->Buffer+ofs); | 
|  |  | 
|  | memset(data.uc, 0, sizeof data.uc); | 
|  | memcpy(data.uc, in->Buffer+ofs, in->Length +8-crypt_len); | 
|  | CRYPT_DEShash(out->Buffer+8+ofs, deskey, data.uc); | 
|  |  | 
|  | out->Length = crypt_len+8; | 
|  |  | 
|  | return STATUS_SUCCESS; | 
|  | } | 
|  |  | 
|  | /****************************************************************************** | 
|  | * SystemFunction005  [ADVAPI32.@] | 
|  | * | 
|  | * Decrypts a block of data with DES in ECB mode | 
|  | * | 
|  | * PARAMS | 
|  | *   data    [I] data to decrypt | 
|  | *   key     [I] key data (up to 7 bytes) | 
|  | *   output  [O] buffer to receive decrypted data | 
|  | * | 
|  | * RETURNS | 
|  | *  Success: STATUS_SUCCESS | 
|  | *  Failure: STATUS_BUFFER_TOO_SMALL     if the output buffer is too small | 
|  | *  Failure: STATUS_INVALID_PARAMETER_2  if the key is zero length | 
|  | * | 
|  | */ | 
|  | NTSTATUS WINAPI SystemFunction005(const struct ustring *in, | 
|  | const struct ustring *key, | 
|  | struct ustring *out) | 
|  | { | 
|  | union { | 
|  | unsigned char uc[8]; | 
|  | unsigned int  ui[2]; | 
|  | } data; | 
|  | unsigned char deskey[7]; | 
|  | unsigned int ofs, crypt_len; | 
|  |  | 
|  | if (key->Length<=0) | 
|  | return STATUS_INVALID_PARAMETER_2; | 
|  |  | 
|  | if (key->Length<sizeof deskey) | 
|  | { | 
|  | memset(deskey, 0, sizeof deskey); | 
|  | memcpy(deskey, key->Buffer, key->Length); | 
|  | } | 
|  | else | 
|  | memcpy(deskey, key->Buffer, sizeof deskey); | 
|  |  | 
|  | CRYPT_DESunhash(data.uc, deskey, in->Buffer); | 
|  |  | 
|  | if (data.ui[1] != 1) | 
|  | return STATUS_UNKNOWN_REVISION; | 
|  |  | 
|  | crypt_len = data.ui[0]; | 
|  | if (crypt_len > out->MaximumLength) | 
|  | return STATUS_BUFFER_TOO_SMALL; | 
|  |  | 
|  | for (ofs=0; (ofs+8)<crypt_len; ofs+=8) | 
|  | CRYPT_DESunhash(out->Buffer+ofs, deskey, in->Buffer+ofs+8); | 
|  |  | 
|  | if (ofs<crypt_len) | 
|  | { | 
|  | CRYPT_DESunhash(data.uc, deskey, in->Buffer+ofs+8); | 
|  | memcpy(out->Buffer+ofs, data.uc, crypt_len-ofs); | 
|  | } | 
|  |  | 
|  | out->Length = crypt_len; | 
|  |  | 
|  | return STATUS_SUCCESS; | 
|  | } | 
|  |  | 
|  | /****************************************************************************** | 
|  | * SystemFunction012  [ADVAPI32.@] | 
|  | * SystemFunction014  [ADVAPI32.@] | 
|  | * SystemFunction016  [ADVAPI32.@] | 
|  | * SystemFunction018  [ADVAPI32.@] | 
|  | * SystemFunction020  [ADVAPI32.@] | 
|  | * SystemFunction022  [ADVAPI32.@] | 
|  | * | 
|  | * Encrypts two DES blocks with two keys | 
|  | * | 
|  | * PARAMS | 
|  | *   data    [I] data to encrypt (16 bytes) | 
|  | *   key     [I] key data (two lots of 7 bytes) | 
|  | *   output  [O] buffer to receive encrypted data (16 bytes) | 
|  | * | 
|  | * RETURNS | 
|  | *  Success: STATUS_SUCCESS | 
|  | *  Failure: STATUS_UNSUCCESSFUL  if the input or output buffer is NULL | 
|  | */ | 
|  | NTSTATUS WINAPI SystemFunction012(const BYTE *in, const BYTE *key, LPBYTE out) | 
|  | { | 
|  | if (!in || !out) | 
|  | return STATUS_UNSUCCESSFUL; | 
|  |  | 
|  | CRYPT_DEShash(out, key, in); | 
|  | CRYPT_DEShash(out+8, key+7, in+8); | 
|  | return STATUS_SUCCESS; | 
|  | } | 
|  |  | 
|  | /****************************************************************************** | 
|  | * SystemFunction013  [ADVAPI32.@] | 
|  | * SystemFunction015  [ADVAPI32.@] | 
|  | * SystemFunction017  [ADVAPI32.@] | 
|  | * SystemFunction019  [ADVAPI32.@] | 
|  | * SystemFunction021  [ADVAPI32.@] | 
|  | * SystemFunction023  [ADVAPI32.@] | 
|  | * | 
|  | * Decrypts two DES blocks with two keys | 
|  | * | 
|  | * PARAMS | 
|  | *   data    [I] data to decrypt (16 bytes) | 
|  | *   key     [I] key data (two lots of 7 bytes) | 
|  | *   output  [O] buffer to receive decrypted data (16 bytes) | 
|  | * | 
|  | * RETURNS | 
|  | *  Success: STATUS_SUCCESS | 
|  | *  Failure: STATUS_UNSUCCESSFUL  if the input or output buffer is NULL | 
|  | */ | 
|  | NTSTATUS WINAPI SystemFunction013(const BYTE *in, const BYTE *key, LPBYTE out) | 
|  | { | 
|  | if (!in || !out) | 
|  | return STATUS_UNSUCCESSFUL; | 
|  |  | 
|  | CRYPT_DESunhash(out, key, in); | 
|  | CRYPT_DESunhash(out+8, key+7, in+8); | 
|  | return STATUS_SUCCESS; | 
|  | } | 
|  |  | 
|  | /****************************************************************************** | 
|  | * SystemFunction024  [ADVAPI32.@] | 
|  | * | 
|  | * Encrypts two DES blocks with a 32 bit key... | 
|  | * | 
|  | * PARAMS | 
|  | *   data    [I] data to encrypt (16 bytes) | 
|  | *   key     [I] key data (4 bytes) | 
|  | *   output  [O] buffer to receive encrypted data (16 bytes) | 
|  | * | 
|  | * RETURNS | 
|  | *  Success: STATUS_SUCCESS | 
|  | */ | 
|  | NTSTATUS WINAPI SystemFunction024(const BYTE *in, const BYTE *key, LPBYTE out) | 
|  | { | 
|  | BYTE deskey[0x10]; | 
|  |  | 
|  | memcpy(deskey, key, 4); | 
|  | memcpy(deskey+4, key, 4); | 
|  | memcpy(deskey+8, key, 4); | 
|  | memcpy(deskey+12, key, 4); | 
|  |  | 
|  | CRYPT_DEShash(out, deskey, in); | 
|  | CRYPT_DEShash(out+8, deskey+7, in+8); | 
|  |  | 
|  | return STATUS_SUCCESS; | 
|  | } | 
|  |  | 
|  | /****************************************************************************** | 
|  | * SystemFunction025  [ADVAPI32.@] | 
|  | * | 
|  | * Decrypts two DES blocks with a 32 bit key... | 
|  | * | 
|  | * PARAMS | 
|  | *   data    [I] data to encrypt (16 bytes) | 
|  | *   key     [I] key data (4 bytes) | 
|  | *   output  [O] buffer to receive encrypted data (16 bytes) | 
|  | * | 
|  | * RETURNS | 
|  | *  Success: STATUS_SUCCESS | 
|  | */ | 
|  | NTSTATUS WINAPI SystemFunction025(const BYTE *in, const BYTE *key, LPBYTE out) | 
|  | { | 
|  | BYTE deskey[0x10]; | 
|  |  | 
|  | memcpy(deskey, key, 4); | 
|  | memcpy(deskey+4, key, 4); | 
|  | memcpy(deskey+8, key, 4); | 
|  | memcpy(deskey+12, key, 4); | 
|  |  | 
|  | CRYPT_DESunhash(out, deskey, in); | 
|  | CRYPT_DESunhash(out+8, deskey+7, in+8); | 
|  |  | 
|  | return STATUS_SUCCESS; | 
|  | } |