| /* | 
 |  *  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]; | 
 |     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]; | 
 |     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; | 
 | } |