/*- * Copyright (c) 1997 Gabor Kincses <gabor@acm.org> * 1997 - 2001 Brian Somers <brian@Awfulhak.org> * based on work by Eric Rosenquist * Strata Software Limited. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * * $FreeBSD: src/usr.sbin/ppp/chap_ms.c,v 1.20.26.1 2010/12/21 17:10:29 kensmith Exp $ */ #include <ctype.h> #ifdef __FreeBSD__ #include <openssl/des.h> #include <sha.h> #else #include <sys/types.h> #include <stdlib.h> #ifdef __NetBSD__ #include <openssl/des.h> #else #include <des.h> #endif #include <openssl/sha.h> #endif #include <md4.h> #include <string.h> #include "chap_ms.h" /* * Documentation & specifications: * * MS-CHAP (CHAP80) rfc2433 * MS-CHAP-V2 (CHAP81) rfc2759 * MPPE key management draft-ietf-pppext-mppe-keys-02.txt */ static char SHA1_Pad1[40] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; static char SHA1_Pad2[40] = {0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2, 0xF2}; /* unused, for documentation only */ /* only NTResp is filled in for FreeBSD */ struct MS_ChapResponse { u_char LANManResp[24]; u_char NTResp[24]; u_char UseNT; /* If 1, ignore the LANMan response field */ }; static u_char Get7Bits(u_char *input, int startBit) { register unsigned int word; word = (unsigned)input[startBit / 8] << 8; word |= (unsigned)input[startBit / 8 + 1]; word >>= 15 - (startBit % 8 + 7); return word & 0xFE; } /* IN 56 bit DES key missing parity bits OUT 64 bit DES key with parity bits added */ static void MakeKey(u_char *key, u_char *des_key) { des_key[0] = Get7Bits(key, 0); des_key[1] = Get7Bits(key, 7); des_key[2] = Get7Bits(key, 14); des_key[3] = Get7Bits(key, 21); des_key[4] = Get7Bits(key, 28); des_key[5] = Get7Bits(key, 35); des_key[6] = Get7Bits(key, 42); des_key[7] = Get7Bits(key, 49); des_set_odd_parity((des_cblock *)des_key); } static void /* IN 8 octets IN 7 octest OUT 8 octets */ DesEncrypt(u_char *clear, u_char *key, u_char *cipher) { des_cblock des_key; des_key_schedule key_schedule; MakeKey(key, des_key); des_set_key(&des_key, key_schedule); des_ecb_encrypt((des_cblock *)clear, (des_cblock *)cipher, key_schedule, 1); } static void /* IN 8 octets IN 16 octets OUT 24 octets */ ChallengeResponse(u_char *challenge, u_char *pwHash, u_char *response) { char ZPasswordHash[21]; memset(ZPasswordHash, '\0', sizeof ZPasswordHash); memcpy(ZPasswordHash, pwHash, 16); DesEncrypt(challenge, ZPasswordHash + 0, response + 0); DesEncrypt(challenge, ZPasswordHash + 7, response + 8); DesEncrypt(challenge, ZPasswordHash + 14, response + 16); } void NtPasswordHash(char *key, int keylen, char *hash) { MD4_CTX MD4context; MD4Init(&MD4context); MD4Update(&MD4context, key, keylen); MD4Final(hash, &MD4context); } void HashNtPasswordHash(char *hash, char *hashhash) { MD4_CTX MD4context; MD4Init(&MD4context); MD4Update(&MD4context, hash, 16); MD4Final(hashhash, &MD4context); } static void ChallengeHash(char *PeerChallenge, char *AuthenticatorChallenge, char *UserName, char *Challenge) { SHA_CTX Context; char Digest[SHA_DIGEST_LENGTH]; char *Name; Name = strrchr(UserName, '\\'); if(NULL == Name) Name = UserName; else Name++; SHA1_Init(&Context); SHA1_Update(&Context, PeerChallenge, 16); SHA1_Update(&Context, AuthenticatorChallenge, 16); SHA1_Update(&Context, Name, strlen(Name)); SHA1_Final(Digest, &Context); memcpy(Challenge, Digest, 8); } void GenerateNTResponse(char *AuthenticatorChallenge, char *PeerChallenge, char *UserName, char *Password, int PasswordLen, char *Response) { char Challenge[8]; char PasswordHash[16]; ChallengeHash(PeerChallenge, AuthenticatorChallenge, UserName, Challenge); NtPasswordHash(Password, PasswordLen, PasswordHash); ChallengeResponse(Challenge, PasswordHash, Response); } #ifndef __FreeBSD__ #define LENGTH 20 static char * SHA1_End(SHA_CTX *ctx, char *buf) { int i; unsigned char digest[LENGTH]; static const char hex[]="0123456789abcdef"; if (!buf) buf = malloc(2*LENGTH + 1); if (!buf) return 0; SHA1_Final(digest, ctx); for (i = 0; i < LENGTH; i++) { buf[i+i] = hex[digest[i] >> 4]; buf[i+i+1] = hex[digest[i] & 0x0f]; } buf[i+i] = '\0'; return buf; } #endif void GenerateAuthenticatorResponse(char *Password, int PasswordLen, char *NTResponse, char *PeerChallenge, char *AuthenticatorChallenge, char *UserName, char *AuthenticatorResponse) { SHA_CTX Context; char PasswordHash[16]; char PasswordHashHash[16]; char Challenge[8]; u_char Digest[SHA_DIGEST_LENGTH]; int i; /* * "Magic" constants used in response generation */ char Magic1[39] = {0x4D, 0x61, 0x67, 0x69, 0x63, 0x20, 0x73, 0x65, 0x72, 0x76, 0x65, 0x72, 0x20, 0x74, 0x6F, 0x20, 0x63, 0x6C, 0x69, 0x65, 0x6E, 0x74, 0x20, 0x73, 0x69, 0x67, 0x6E, 0x69, 0x6E, 0x67, 0x20, 0x63, 0x6F, 0x6E, 0x73, 0x74, 0x61, 0x6E, 0x74}; char Magic2[41] = {0x50, 0x61, 0x64, 0x20, 0x74, 0x6F, 0x20, 0x6D, 0x61, 0x6B, 0x65, 0x20, 0x69, 0x74, 0x20, 0x64, 0x6F, 0x20, 0x6D, 0x6F, 0x72, 0x65, 0x20, 0x74, 0x68, 0x61, 0x6E, 0x20, 0x6F, 0x6E, 0x65, 0x20, 0x69, 0x74, 0x65, 0x72, 0x61, 0x74, 0x69, 0x6F, 0x6E}; /* * Hash the password with MD4 */ NtPasswordHash(Password, PasswordLen, PasswordHash); /* * Now hash the hash */ HashNtPasswordHash(PasswordHash, PasswordHashHash); SHA1_Init(&Context); SHA1_Update(&Context, PasswordHashHash, 16); SHA1_Update(&Context, NTResponse, 24); SHA1_Update(&Context, Magic1, 39); SHA1_Final(Digest, &Context); ChallengeHash(PeerChallenge, AuthenticatorChallenge, UserName, Challenge); SHA1_Init(&Context); SHA1_Update(&Context, Digest, 20); SHA1_Update(&Context, Challenge, 8); SHA1_Update(&Context, Magic2, 41); /* * Encode the value of 'Digest' as "S=" followed by * 40 ASCII hexadecimal digits and return it in * AuthenticatorResponse. * For example, * "S=0123456789ABCDEF0123456789ABCDEF01234567" */ AuthenticatorResponse[0] = 'S'; AuthenticatorResponse[1] = '='; SHA1_End(&Context, AuthenticatorResponse + 2); for (i=2; i<42; i++) AuthenticatorResponse[i] = toupper(AuthenticatorResponse[i]); } void GetMasterKey(char *PasswordHashHash, char *NTResponse, char *MasterKey) { char Digest[SHA_DIGEST_LENGTH]; SHA_CTX Context; static char Magic1[27] = {0x54, 0x68, 0x69, 0x73, 0x20, 0x69, 0x73, 0x20, 0x74, 0x68, 0x65, 0x20, 0x4d, 0x50, 0x50, 0x45, 0x20, 0x4d, 0x61, 0x73, 0x74, 0x65, 0x72, 0x20, 0x4b, 0x65, 0x79}; SHA1_Init(&Context); SHA1_Update(&Context, PasswordHashHash, 16); SHA1_Update(&Context, NTResponse, 24); SHA1_Update(&Context, Magic1, 27); SHA1_Final(Digest, &Context); memcpy(MasterKey, Digest, 16); } void GetAsymetricStartKey(char *MasterKey, char *SessionKey, int SessionKeyLength, int IsSend, int IsServer) { char Digest[SHA_DIGEST_LENGTH]; SHA_CTX Context; char *s; static char Magic2[84] = {0x4f, 0x6e, 0x20, 0x74, 0x68, 0x65, 0x20, 0x63, 0x6c, 0x69, 0x65, 0x6e, 0x74, 0x20, 0x73, 0x69, 0x64, 0x65, 0x2c, 0x20, 0x74, 0x68, 0x69, 0x73, 0x20, 0x69, 0x73, 0x20, 0x74, 0x68, 0x65, 0x20, 0x73, 0x65, 0x6e, 0x64, 0x20, 0x6b, 0x65, 0x79, 0x3b, 0x20, 0x6f, 0x6e, 0x20, 0x74, 0x68, 0x65, 0x20, 0x73, 0x65, 0x72, 0x76, 0x65, 0x72, 0x20, 0x73, 0x69, 0x64, 0x65, 0x2c, 0x20, 0x69, 0x74, 0x20, 0x69, 0x73, 0x20, 0x74, 0x68, 0x65, 0x20, 0x72, 0x65, 0x63, 0x65, 0x69, 0x76, 0x65, 0x20, 0x6b, 0x65, 0x79, 0x2e}; static char Magic3[84] = {0x4f, 0x6e, 0x20, 0x74, 0x68, 0x65, 0x20, 0x63, 0x6c, 0x69, 0x65, 0x6e, 0x74, 0x20, 0x73, 0x69, 0x64, 0x65, 0x2c, 0x20, 0x74, 0x68, 0x69, 0x73, 0x20, 0x69, 0x73, 0x20, 0x74, 0x68, 0x65, 0x20, 0x72, 0x65, 0x63, 0x65, 0x69, 0x76, 0x65, 0x20, 0x6b, 0x65, 0x79, 0x3b, 0x20, 0x6f, 0x6e, 0x20, 0x74, 0x68, 0x65, 0x20, 0x73, 0x65, 0x72, 0x76, 0x65, 0x72, 0x20, 0x73, 0x69, 0x64, 0x65, 0x2c, 0x20, 0x69, 0x74, 0x20, 0x69, 0x73, 0x20, 0x74, 0x68, 0x65, 0x20, 0x73, 0x65, 0x6e, 0x64, 0x20, 0x6b, 0x65, 0x79, 0x2e}; if (IsSend) { if (IsServer) { s = Magic3; } else { s = Magic2; } } else { if (IsServer) { s = Magic2; } else { s = Magic3; } } SHA1_Init(&Context); SHA1_Update(&Context, MasterKey, 16); SHA1_Update(&Context, SHA1_Pad1, 40); SHA1_Update(&Context, s, 84); SHA1_Update(&Context, SHA1_Pad2, 40); SHA1_Final(Digest, &Context); memcpy(SessionKey, Digest, SessionKeyLength); } void GetNewKeyFromSHA(char *StartKey, char *SessionKey, long SessionKeyLength, char *InterimKey) { SHA_CTX Context; char Digest[SHA_DIGEST_LENGTH]; SHA1_Init(&Context); SHA1_Update(&Context, StartKey, SessionKeyLength); SHA1_Update(&Context, SHA1_Pad1, 40); SHA1_Update(&Context, SessionKey, SessionKeyLength); SHA1_Update(&Context, SHA1_Pad2, 40); SHA1_Final(Digest, &Context); memcpy(InterimKey, Digest, SessionKeyLength); } #if 0 static void Get_Key(char *InitialSessionKey, char *CurrentSessionKey, int LengthOfDesiredKey) { SHA_CTX Context; char Digest[SHA_DIGEST_LENGTH]; SHA1_Init(&Context); SHA1_Update(&Context, InitialSessionKey, LengthOfDesiredKey); SHA1_Update(&Context, SHA1_Pad1, 40); SHA1_Update(&Context, CurrentSessionKey, LengthOfDesiredKey); SHA1_Update(&Context, SHA1_Pad2, 40); SHA1_Final(Digest, &Context); memcpy(CurrentSessionKey, Digest, LengthOfDesiredKey); } #endif /* passwordHash 16-bytes MD4 hashed password challenge 8-bytes peer CHAP challenge since passwordHash is in a 24-byte buffer, response is written in there */ void mschap_NT(char *passwordHash, char *challenge) { u_char response[24]; ChallengeResponse(challenge, passwordHash, response); memcpy(passwordHash, response, 24); passwordHash[24] = 1; /* NT-style response */ } void mschap_LANMan(char *digest, char *challenge, char *secret) { static u_char salt[] = "KGS!@#$%"; /* RASAPI32.dll */ char SECRET[14], *ptr, *end; u_char hash[16]; end = SECRET + sizeof SECRET; for (ptr = SECRET; *secret && ptr < end; ptr++, secret++) *ptr = toupper(*secret); if (ptr < end) memset(ptr, '\0', end - ptr); DesEncrypt(salt, SECRET, hash); DesEncrypt(salt, SECRET + 7, hash + 8); ChallengeResponse(challenge, hash, digest); }