1 // Compiles with Visual Studio 2008 for Windows 2 3 // This C example is designed as more of a guide than a library to be plugged into an application 4 // That module required a couple of major re-writes and is available upon request 5 // The Basic example has tips to the direction you should take 6 // This will work with connections on port 587 that upgrade a plain text session to an encrypted session with STARTTLS as covered here. 7 8 // TLSclient.c - SSPI Schannel gmail TLS connection example 9 10 #define _CRT_SECURE_NO_WARNINGS 11 12 #define SECURITY_WIN32 13 #define IO_BUFFER_SIZE 0x10000 14 #define DLL_NAME TEXT("Secur32.dll") 15 #define NT4_DLL_NAME TEXT("Security.dll") 16 17 #include <stdio.h> 18 #include <stdlib.h> 19 #include <windows.h> 20 #include <winsock.h> 21 #include <wincrypt.h> 22 #include <wintrust.h> 23 #include <schannel.h> 24 #include <security.h> 25 #include <sspi.h> 26 27 #pragma comment(lib, "WSock32.Lib") 28 #pragma comment(lib, "Crypt32.Lib") 29 #pragma comment(lib, "user32.lib") 30 #pragma comment(lib, "MSVCRTD.lib") 31 32 // Globals. 33 BOOL fVerbose = FALSE; // FALSE; // TRUE; 34 35 36 INT iPortNumber = 465; // gmail TLS 37 LPCSTR pszServerName = "smtp.163.com"; // DNS name of server 38 LPCSTR pszUser = 0; // if specified, a certificate in "MY" store is searched for 39 40 DWORD dwProtocol = SP_PROT_TLS1; // SP_PROT_TLS1; // SP_PROT_PCT1; SP_PROT_SSL2; SP_PROT_SSL3; 0=default 41 ALG_ID aiKeyExch = 0; // = default; CALG_DH_EPHEM; CALG_RSA_KEYX; 42 43 BOOL fUseProxy = FALSE; 44 LPCSTR pszProxyServer = "proxy"; 45 INT iProxyPort = 80; 46 47 HCERTSTORE hMyCertStore = NULL; 48 HMODULE g_hSecurity = NULL; 49 50 SCHANNEL_CRED SchannelCred; 51 PSecurityFunctionTable g_pSSPI; 52 53 54 55 /*****************************************************************************/ 56 static void DisplayWinVerifyTrustError(DWORD Status) 57 { 58 LPCSTR pszName = NULL; 59 60 switch (Status) 61 { 62 case CERT_E_EXPIRED: pszName = "CERT_E_EXPIRED"; break; 63 case CERT_E_VALIDITYPERIODNESTING: pszName = "CERT_E_VALIDITYPERIODNESTING"; break; 64 case CERT_E_ROLE: pszName = "CERT_E_ROLE"; break; 65 case CERT_E_PATHLENCONST: pszName = "CERT_E_PATHLENCONST"; break; 66 case CERT_E_CRITICAL: pszName = "CERT_E_CRITICAL"; break; 67 case CERT_E_PURPOSE: pszName = "CERT_E_PURPOSE"; break; 68 case CERT_E_ISSUERCHAINING: pszName = "CERT_E_ISSUERCHAINING"; break; 69 case CERT_E_MALFORMED: pszName = "CERT_E_MALFORMED"; break; 70 case CERT_E_UNTRUSTEDROOT: pszName = "CERT_E_UNTRUSTEDROOT"; break; 71 case CERT_E_CHAINING: pszName = "CERT_E_CHAINING"; break; 72 case TRUST_E_FAIL: pszName = "TRUST_E_FAIL"; break; 73 case CERT_E_REVOKED: pszName = "CERT_E_REVOKED"; break; 74 case CERT_E_UNTRUSTEDTESTROOT: pszName = "CERT_E_UNTRUSTEDTESTROOT"; break; 75 case CERT_E_REVOCATION_FAILURE: pszName = "CERT_E_REVOCATION_FAILURE"; break; 76 case CERT_E_CN_NO_MATCH: pszName = "CERT_E_CN_NO_MATCH"; break; 77 case CERT_E_WRONG_USAGE: pszName = "CERT_E_WRONG_USAGE"; break; 78 default: pszName = "(unknown)"; break; 79 } 80 printf("Error 0x%x (%s) returned by CertVerifyCertificateChainPolicy! ", Status, pszName); 81 } 82 83 84 /*****************************************************************************/ 85 static void DisplayWinSockError(DWORD ErrCode) 86 { 87 LPCSTR pszName = NULL; // http://www.sockets.com/err_lst1.htm#WSANO_DATA 88 89 switch (ErrCode) // http://msdn.microsoft.com/en-us/library/ms740668(VS.85).aspx 90 { 91 case 10035: pszName = "WSAEWOULDBLOCK "; break; 92 case 10036: pszName = "WSAEINPROGRESS "; break; 93 case 10037: pszName = "WSAEALREADY "; break; 94 case 10038: pszName = "WSAENOTSOCK "; break; 95 case 10039: pszName = "WSAEDESTADDRREQ "; break; 96 case 10040: pszName = "WSAEMSGSIZE "; break; 97 case 10041: pszName = "WSAEPROTOTYPE "; break; 98 case 10042: pszName = "WSAENOPROTOOPT "; break; 99 case 10043: pszName = "WSAEPROTONOSUPPORT"; break; 100 case 10044: pszName = "WSAESOCKTNOSUPPORT"; break; 101 case 10045: pszName = "WSAEOPNOTSUPP "; break; 102 case 10046: pszName = "WSAEPFNOSUPPORT "; break; 103 case 10047: pszName = "WSAEAFNOSUPPORT "; break; 104 case 10048: pszName = "WSAEADDRINUSE "; break; 105 case 10049: pszName = "WSAEADDRNOTAVAIL "; break; 106 case 10050: pszName = "WSAENETDOWN "; break; 107 case 10051: pszName = "WSAENETUNREACH "; break; 108 case 10052: pszName = "WSAENETRESET "; break; 109 case 10053: pszName = "WSAECONNABORTED "; break; 110 case 10054: pszName = "WSAECONNRESET "; break; 111 case 10055: pszName = "WSAENOBUFS "; break; 112 case 10056: pszName = "WSAEISCONN "; break; 113 case 10057: pszName = "WSAENOTCONN "; break; 114 case 10058: pszName = "WSAESHUTDOWN "; break; 115 case 10059: pszName = "WSAETOOMANYREFS "; break; 116 case 10060: pszName = "WSAETIMEDOUT "; break; 117 case 10061: pszName = "WSAECONNREFUSED "; break; 118 case 10062: pszName = "WSAELOOP "; break; 119 case 10063: pszName = "WSAENAMETOOLONG "; break; 120 case 10064: pszName = "WSAEHOSTDOWN "; break; 121 case 10065: pszName = "WSAEHOSTUNREACH "; break; 122 case 10066: pszName = "WSAENOTEMPTY "; break; 123 case 10067: pszName = "WSAEPROCLIM "; break; 124 case 10068: pszName = "WSAEUSERS "; break; 125 case 10069: pszName = "WSAEDQUOT "; break; 126 case 10070: pszName = "WSAESTALE "; break; 127 case 10071: pszName = "WSAEREMOTE "; break; 128 case 10091: pszName = "WSASYSNOTREADY "; break; 129 case 10092: pszName = "WSAVERNOTSUPPORTED"; break; 130 case 10093: pszName = "WSANOTINITIALISED "; break; 131 case 11001: pszName = "WSAHOST_NOT_FOUND "; break; 132 case 11002: pszName = "WSATRY_AGAIN "; break; 133 case 11003: pszName = "WSANO_RECOVERY "; break; 134 case 11004: pszName = "WSANO_DATA "; break; 135 } 136 printf("Error 0x%x (%s) ", ErrCode, pszName); 137 } 138 139 /*****************************************************************************/ 140 static void DisplaySECError(DWORD ErrCode) 141 { 142 LPCSTR pszName = NULL; // WinError.h 143 144 switch (ErrCode) 145 { 146 case SEC_E_BUFFER_TOO_SMALL: 147 pszName = "SEC_E_BUFFER_TOO_SMALL - The message buffer is too small. Used with the Digest SSP."; 148 break; 149 150 case SEC_E_CRYPTO_SYSTEM_INVALID: 151 pszName = "SEC_E_CRYPTO_SYSTEM_INVALID - The cipher chosen for the security context is not supported. Used with the Digest SSP."; 152 break; 153 case SEC_E_INCOMPLETE_MESSAGE: 154 pszName = "SEC_E_INCOMPLETE_MESSAGE - The data in the input buffer is incomplete. The application needs to read more data from the server and call DecryptMessage (General) again."; 155 break; 156 157 case SEC_E_INVALID_HANDLE: 158 pszName = "SEC_E_INVALID_HANDLE - A context handle that is not valid was specified in the phContext parameter. Used with the Digest and Schannel SSPs."; 159 break; 160 161 case SEC_E_INVALID_TOKEN: 162 pszName = "SEC_E_INVALID_TOKEN - The buffers are of the wrong type or no buffer of type SECBUFFER_DATA was found. Used with the Schannel SSP."; 163 break; 164 165 case SEC_E_MESSAGE_ALTERED: 166 pszName = "SEC_E_MESSAGE_ALTERED - The message has been altered. Used with the Digest and Schannel SSPs."; 167 break; 168 169 case SEC_E_OUT_OF_SEQUENCE: 170 pszName = "SEC_E_OUT_OF_SEQUENCE - The message was not received in the correct sequence."; 171 break; 172 173 case SEC_E_QOP_NOT_SUPPORTED: 174 pszName = "SEC_E_QOP_NOT_SUPPORTED - Neither confidentiality nor integrity are supported by the security context. Used with the Digest SSP."; 175 break; 176 177 case SEC_I_CONTEXT_EXPIRED: 178 pszName = "SEC_I_CONTEXT_EXPIRED - The message sender has finished using the connection and has initiated a shutdown."; 179 break; 180 181 case SEC_I_RENEGOTIATE: 182 pszName = "SEC_I_RENEGOTIATE - The remote party requires a new handshake sequence or the application has just initiated a shutdown."; 183 break; 184 185 case SEC_E_ENCRYPT_FAILURE: 186 pszName = "SEC_E_ENCRYPT_FAILURE - The specified data could not be encrypted."; 187 break; 188 189 case SEC_E_DECRYPT_FAILURE: 190 pszName = "SEC_E_DECRYPT_FAILURE - The specified data could not be decrypted."; 191 break; 192 193 } 194 printf("Error 0x%x %s ", ErrCode, pszName); 195 } 196 197 198 199 /*****************************************************************************/ 200 static void DisplayCertChain(PCCERT_CONTEXT pServerCert, BOOL fLocal) 201 { 202 CHAR szName[1000]; 203 PCCERT_CONTEXT pCurrentCert, pIssuerCert; 204 DWORD dwVerificationFlags; 205 206 printf(" "); 207 208 // display leaf name 209 if (!CertNameToStr(pServerCert->dwCertEncodingType, 210 &pServerCert->pCertInfo->Subject, 211 CERT_X500_NAME_STR | CERT_NAME_STR_NO_PLUS_FLAG, 212 szName, sizeof(szName))) 213 { 214 printf("**** Error 0x%x building subject name ", GetLastError()); 215 } 216 217 if (fLocal) printf("Client subject: %s ", szName); 218 else printf("Server subject: %s ", szName); 219 220 if (!CertNameToStr(pServerCert->dwCertEncodingType, 221 &pServerCert->pCertInfo->Issuer, 222 CERT_X500_NAME_STR | CERT_NAME_STR_NO_PLUS_FLAG, 223 szName, sizeof(szName))) 224 { 225 printf("**** Error 0x%x building issuer name ", GetLastError()); 226 } 227 228 if (fLocal) printf("Client issuer: %s ", szName); 229 else printf("Server issuer: %s ", szName); 230 231 232 // display certificate chain 233 pCurrentCert = pServerCert; 234 while (pCurrentCert != NULL) 235 { 236 dwVerificationFlags = 0; 237 pIssuerCert = CertGetIssuerCertificateFromStore(pServerCert->hCertStore, pCurrentCert, NULL, &dwVerificationFlags); 238 if (pIssuerCert == NULL) 239 { 240 if (pCurrentCert != pServerCert) CertFreeCertificateContext(pCurrentCert); 241 break; 242 } 243 244 if (!CertNameToStr(pIssuerCert->dwCertEncodingType, 245 &pIssuerCert->pCertInfo->Subject, 246 CERT_X500_NAME_STR | CERT_NAME_STR_NO_PLUS_FLAG, 247 szName, sizeof(szName))) 248 { 249 printf("**** Error 0x%x building subject name ", GetLastError()); 250 } 251 252 printf("CA subject: %s ", szName); 253 254 if (!CertNameToStr(pIssuerCert->dwCertEncodingType, 255 &pIssuerCert->pCertInfo->Issuer, 256 CERT_X500_NAME_STR | CERT_NAME_STR_NO_PLUS_FLAG, 257 szName, sizeof(szName))) 258 { 259 printf("**** Error 0x%x building issuer name ", GetLastError()); 260 } 261 262 printf("CA issuer: %s ", szName); 263 264 if (pCurrentCert != pServerCert) CertFreeCertificateContext(pCurrentCert); 265 pCurrentCert = pIssuerCert; 266 pIssuerCert = NULL; 267 } 268 } 269 270 /*****************************************************************************/ 271 static void DisplayConnectionInfo(CtxtHandle *phContext) 272 { 273 274 SECURITY_STATUS Status; 275 SecPkgContext_ConnectionInfo ConnectionInfo; 276 277 Status = g_pSSPI->QueryContextAttributes(phContext, SECPKG_ATTR_CONNECTION_INFO, (PVOID)&ConnectionInfo); 278 if (Status != SEC_E_OK) { printf("Error 0x%x querying connection info ", Status); return; } 279 280 printf(" "); 281 282 switch (ConnectionInfo.dwProtocol) 283 { 284 case SP_PROT_TLS1_CLIENT: 285 printf("Protocol: TLS1 "); 286 break; 287 288 case SP_PROT_SSL3_CLIENT: 289 printf("Protocol: SSL3 "); 290 break; 291 292 case SP_PROT_PCT1_CLIENT: 293 printf("Protocol: PCT "); 294 break; 295 296 case SP_PROT_SSL2_CLIENT: 297 printf("Protocol: SSL2 "); 298 break; 299 300 default: 301 printf("Protocol: 0x%x ", ConnectionInfo.dwProtocol); 302 } 303 304 switch (ConnectionInfo.aiCipher) 305 { 306 case CALG_RC4: 307 printf("Cipher: RC4 "); 308 break; 309 310 case CALG_3DES: 311 printf("Cipher: Triple DES "); 312 break; 313 314 case CALG_RC2: 315 printf("Cipher: RC2 "); 316 break; 317 318 case CALG_DES: 319 case CALG_CYLINK_MEK: 320 printf("Cipher: DES "); 321 break; 322 323 case CALG_SKIPJACK: 324 printf("Cipher: Skipjack "); 325 break; 326 327 default: 328 printf("Cipher: 0x%x ", ConnectionInfo.aiCipher); 329 } 330 331 printf("Cipher strength: %d ", ConnectionInfo.dwCipherStrength); 332 333 switch (ConnectionInfo.aiHash) 334 { 335 case CALG_MD5: 336 printf("Hash: MD5 "); 337 break; 338 339 case CALG_SHA: 340 printf("Hash: SHA "); 341 break; 342 343 default: 344 printf("Hash: 0x%x ", ConnectionInfo.aiHash); 345 } 346 347 printf("Hash strength: %d ", ConnectionInfo.dwHashStrength); 348 349 switch (ConnectionInfo.aiExch) 350 { 351 case CALG_RSA_KEYX: 352 case CALG_RSA_SIGN: 353 printf("Key exchange: RSA "); 354 break; 355 356 case CALG_KEA_KEYX: 357 printf("Key exchange: KEA "); 358 break; 359 360 case CALG_DH_EPHEM: 361 printf("Key exchange: DH Ephemeral "); 362 break; 363 364 default: 365 printf("Key exchange: 0x%x ", ConnectionInfo.aiExch); 366 } 367 368 printf("Key exchange strength: %d ", ConnectionInfo.dwExchStrength); 369 } 370 371 372 /*****************************************************************************/ 373 static void PrintHexDump(DWORD length, PBYTE buffer) 374 { 375 DWORD i, count, index; 376 CHAR rgbDigits[] = "0123456789abcdef"; 377 CHAR rgbLine[100]; 378 char cbLine; 379 380 for (index = 0; length; length -= count, buffer += count, index += count) 381 { 382 count = (length > 16) ? 16 : length; 383 sprintf(rgbLine, "%4.4x ", index); 384 cbLine = 6; 385 386 for (i = 0; i < count; i++) 387 { 388 rgbLine[cbLine++] = rgbDigits[buffer[i] >> 4]; 389 rgbLine[cbLine++] = rgbDigits[buffer[i] & 0x0f]; 390 if (i == 7) rgbLine[cbLine++] = ':'; 391 else rgbLine[cbLine++] = ' '; 392 } 393 for (; i < 16; i++) 394 { 395 rgbLine[cbLine++] = ' '; 396 rgbLine[cbLine++] = ' '; 397 rgbLine[cbLine++] = ' '; 398 } 399 rgbLine[cbLine++] = ' '; 400 401 for (i = 0; i < count; i++) 402 { 403 if (buffer[i] < 32 || buffer[i] > 126 || buffer[i] == '%') rgbLine[cbLine++] = '.'; 404 else rgbLine[cbLine++] = buffer[i]; 405 } 406 rgbLine[cbLine++] = 0; 407 printf("%s ", rgbLine); 408 } 409 } 410 411 /*****************************************************************************/ 412 static void PrintText(DWORD length, PBYTE buffer) // handle unprintable charaters 413 { 414 int i; // 415 416 printf(" "); // "length = %d bytes ", length); 417 for (i = 0; i < (int)length; i++) 418 { 419 if (buffer[i] == 10 || buffer[i] == 13) 420 printf("%c", (char)buffer[i]); 421 else if (buffer[i] < 32 || buffer[i] > 126 || buffer[i] == '%') 422 printf("%c", '.'); 423 else 424 printf("%c", (char)buffer[i]); 425 } 426 printf(" "); 427 } 428 429 430 431 /*****************************************************************************/ 432 static void WriteDataToFile(PSTR pszData, PBYTE pbData, DWORD cbData) 433 { 434 FILE *file; 435 436 file = fopen(pszData, "wb"); 437 if (file == NULL) 438 { 439 printf("**** Error opening file '%s' ", pszData); return; 440 } 441 442 if (fwrite(pbData, 1, cbData, file) != cbData) 443 { 444 printf("**** Error writing to file "); return; 445 } 446 447 fclose(file); 448 } 449 450 451 452 453 454 /*****************************************************************************/ 455 BOOL LoadSecurityLibrary(void) // load SSPI.DLL, set up a special table - PSecurityFunctionTable 456 { 457 INIT_SECURITY_INTERFACE pInitSecurityInterface; 458 // QUERY_CREDENTIALS_ATTRIBUTES_FN pQueryCredentialsAttributes; 459 OSVERSIONINFO VerInfo; 460 UCHAR lpszDLL[MAX_PATH]; 461 462 463 // Find out which security DLL to use, depending on 464 // whether we are on Win2K, NT or Win9x 465 VerInfo.dwOSVersionInfoSize = sizeof(OSVERSIONINFO); 466 467 if (!GetVersionEx(&VerInfo)) return FALSE; 468 469 if (VerInfo.dwPlatformId == VER_PLATFORM_WIN32_NT && VerInfo.dwMajorVersion == 4) 470 { 471 strcpy((char*)lpszDLL, NT4_DLL_NAME); // NT4_DLL_NAME TEXT("Security.dll") 472 } 473 else if (VerInfo.dwPlatformId == VER_PLATFORM_WIN32_WINDOWS || 474 VerInfo.dwPlatformId == VER_PLATFORM_WIN32_NT) 475 { 476 strcpy((char*)lpszDLL, DLL_NAME); // DLL_NAME TEXT("Secur32.dll") 477 } 478 else 479 { 480 printf("System not recognized "); return FALSE; 481 } 482 483 484 // Load Security DLL 485 g_hSecurity = LoadLibrary((char*)lpszDLL); 486 if (g_hSecurity == NULL) { printf("Error 0x%x loading %s. ", GetLastError(), lpszDLL); return FALSE; } 487 488 pInitSecurityInterface = (INIT_SECURITY_INTERFACE)GetProcAddress(g_hSecurity, "InitSecurityInterfaceA"); 489 if (pInitSecurityInterface == NULL) { printf("Error 0x%x reading InitSecurityInterface entry point. ", GetLastError()); return FALSE; } 490 491 g_pSSPI = pInitSecurityInterface(); // call InitSecurityInterfaceA(void); 492 if (g_pSSPI == NULL) { printf("Error 0x%x reading security interface. ", GetLastError()); return FALSE; } 493 494 return TRUE; // and PSecurityFunctionTable 495 } 496 497 498 /*****************************************************************************/ 499 void UnloadSecurityLibrary(void) 500 { 501 FreeLibrary(g_hSecurity); 502 g_hSecurity = NULL; 503 } 504 505 506 /*****************************************************************************/ 507 static DWORD VerifyServerCertificate(PCCERT_CONTEXT pServerCert, PSTR pszServerName, DWORD dwCertFlags) 508 { 509 HTTPSPolicyCallbackData polHttps; 510 CERT_CHAIN_POLICY_PARA PolicyPara; 511 CERT_CHAIN_POLICY_STATUS PolicyStatus; 512 CERT_CHAIN_PARA ChainPara; 513 PCCERT_CHAIN_CONTEXT pChainContext = NULL; 514 DWORD cchServerName, Status; 515 LPSTR rgszUsages[] = { (char*)szOID_PKIX_KP_SERVER_AUTH, 516 (char*)szOID_SERVER_GATED_CRYPTO, 517 (char*)szOID_SGC_NETSCAPE }; 518 519 DWORD cUsages = sizeof(rgszUsages) / sizeof(LPSTR); 520 521 PWSTR pwszServerName = NULL; 522 523 524 if (pServerCert == NULL) 525 { 526 Status = SEC_E_WRONG_PRINCIPAL; goto cleanup; 527 } 528 529 // Convert server name to unicode. 530 if (pszServerName == NULL || strlen(pszServerName) == 0) 531 { 532 Status = SEC_E_WRONG_PRINCIPAL; goto cleanup; 533 } 534 535 cchServerName = MultiByteToWideChar(CP_ACP, 0, pszServerName, -1, NULL, 0); 536 pwszServerName = (PWSTR)LocalAlloc(LMEM_FIXED, cchServerName * sizeof(WCHAR)); 537 if (pwszServerName == NULL) 538 { 539 Status = SEC_E_INSUFFICIENT_MEMORY; goto cleanup; 540 } 541 542 cchServerName = MultiByteToWideChar(CP_ACP, 0, pszServerName, -1, pwszServerName, cchServerName); 543 if (cchServerName == 0) 544 { 545 Status = SEC_E_WRONG_PRINCIPAL; goto cleanup; 546 } 547 548 549 // Build certificate chain. 550 ZeroMemory(&ChainPara, sizeof(ChainPara)); 551 ChainPara.cbSize = sizeof(ChainPara); 552 ChainPara.RequestedUsage.dwType = USAGE_MATCH_TYPE_OR; 553 ChainPara.RequestedUsage.Usage.cUsageIdentifier = cUsages; 554 ChainPara.RequestedUsage.Usage.rgpszUsageIdentifier = rgszUsages; 555 556 if (!CertGetCertificateChain(NULL, 557 pServerCert, 558 NULL, 559 pServerCert->hCertStore, 560 &ChainPara, 561 0, 562 NULL, 563 &pChainContext)) 564 { 565 Status = GetLastError(); 566 printf("Error 0x%x returned by CertGetCertificateChain! ", Status); 567 goto cleanup; 568 } 569 570 571 // Validate certificate chain. 572 ZeroMemory(&polHttps, sizeof(HTTPSPolicyCallbackData)); 573 polHttps.cbStruct = sizeof(HTTPSPolicyCallbackData); 574 polHttps.dwAuthType = AUTHTYPE_SERVER; 575 polHttps.fdwChecks = dwCertFlags; 576 polHttps.pwszServerName = pwszServerName; 577 578 memset(&PolicyPara, 0, sizeof(PolicyPara)); 579 PolicyPara.cbSize = sizeof(PolicyPara); 580 PolicyPara.pvExtraPolicyPara = &polHttps; 581 582 memset(&PolicyStatus, 0, sizeof(PolicyStatus)); 583 PolicyStatus.cbSize = sizeof(PolicyStatus); 584 585 if (!CertVerifyCertificateChainPolicy(CERT_CHAIN_POLICY_SSL, 586 pChainContext, 587 &PolicyPara, 588 &PolicyStatus)) 589 { 590 Status = GetLastError(); 591 printf("Error 0x%x returned by CertVerifyCertificateChainPolicy! ", Status); 592 goto cleanup; 593 } 594 595 if (PolicyStatus.dwError) 596 { 597 Status = PolicyStatus.dwError; 598 DisplayWinVerifyTrustError(Status); 599 goto cleanup; 600 } 601 602 Status = SEC_E_OK; 603 604 605 cleanup: 606 if (pChainContext) CertFreeCertificateChain(pChainContext); 607 if (pwszServerName) LocalFree(pwszServerName); 608 609 return Status; 610 } 611 612 613 /*****************************************************************************/ 614 static SECURITY_STATUS CreateCredentials(LPSTR pszUser, PCredHandle phCreds) 615 { // in out 616 TimeStamp tsExpiry; 617 SECURITY_STATUS Status; 618 DWORD cSupportedAlgs = 0; 619 ALG_ID rgbSupportedAlgs[16]; 620 PCCERT_CONTEXT pCertContext = NULL; 621 622 623 // Open the "MY" certificate store, where IE stores client certificates. 624 // Windows maintains 4 stores -- MY, CA, ROOT, SPC. 625 if (hMyCertStore == NULL) 626 { 627 hMyCertStore = CertOpenSystemStore(0, "MY"); 628 if (!hMyCertStore) 629 { 630 printf("**** Error 0x%x returned by CertOpenSystemStore ", GetLastError()); 631 return SEC_E_NO_CREDENTIALS; 632 } 633 } 634 635 636 // If a user name is specified, then attempt to find a client 637 // certificate. Otherwise, just create a NULL credential. 638 if (pszUser) 639 { 640 // Find client certificate. Note that this sample just searches for a 641 // certificate that contains the user name somewhere in the subject name. 642 // A real application should be a bit less casual. 643 pCertContext = CertFindCertificateInStore(hMyCertStore, // hCertStore 644 X509_ASN_ENCODING, // dwCertEncodingType 645 0, // dwFindFlags 646 CERT_FIND_SUBJECT_STR_A,// dwFindType 647 pszUser, // *pvFindPara 648 NULL); // pPrevCertContext 649 650 651 if (pCertContext == NULL) 652 { 653 printf("**** Error 0x%x returned by CertFindCertificateInStore ", GetLastError()); 654 if (GetLastError() == CRYPT_E_NOT_FOUND) printf("CRYPT_E_NOT_FOUND - property doesn't exist "); 655 return SEC_E_NO_CREDENTIALS; 656 } 657 } 658 659 660 // Build Schannel credential structure. Currently, this sample only 661 // specifies the protocol to be used (and optionally the certificate, 662 // of course). Real applications may wish to specify other parameters as well. 663 ZeroMemory(&SchannelCred, sizeof(SchannelCred)); 664 665 SchannelCred.dwVersion = SCHANNEL_CRED_VERSION; 666 if (pCertContext) 667 { 668 SchannelCred.cCreds = 1; 669 SchannelCred.paCred = &pCertContext; 670 } 671 672 SchannelCred.grbitEnabledProtocols = dwProtocol; 673 674 if (aiKeyExch) rgbSupportedAlgs[cSupportedAlgs++] = aiKeyExch; 675 676 if (cSupportedAlgs) 677 { 678 SchannelCred.cSupportedAlgs = cSupportedAlgs; 679 SchannelCred.palgSupportedAlgs = rgbSupportedAlgs; 680 } 681 682 SchannelCred.dwFlags |= SCH_CRED_NO_DEFAULT_CREDS; 683 684 // The SCH_CRED_MANUAL_CRED_VALIDATION flag is specified because 685 // this sample verifies the server certificate manually. 686 // Applications that expect to run on WinNT, Win9x, or WinME 687 // should specify this flag and also manually verify the server 688 // certificate. Applications running on newer versions of Windows can 689 // leave off this flag, in which case the InitializeSecurityContext 690 // function will validate the server certificate automatically. 691 SchannelCred.dwFlags |= SCH_CRED_MANUAL_CRED_VALIDATION; 692 693 694 // Create an SSPI credential. 695 Status = g_pSSPI->AcquireCredentialsHandleA(NULL, // Name of principal 696 (char*)UNISP_NAME_A, // Name of package 697 SECPKG_CRED_OUTBOUND, // Flags indicating use 698 NULL, // Pointer to logon ID 699 &SchannelCred, // Package specific data 700 NULL, // Pointer to GetKey() func 701 NULL, // Value to pass to GetKey() 702 phCreds, // (out) Cred Handle 703 &tsExpiry); // (out) Lifetime (optional) 704 705 if (Status != SEC_E_OK) printf("**** Error 0x%x returned by AcquireCredentialsHandle ", Status); 706 707 // cleanup: Free the certificate context. Schannel has already made its own copy. 708 if (pCertContext) CertFreeCertificateContext(pCertContext); 709 710 return Status; 711 } 712 713 /*****************************************************************************/ 714 static INT ConnectToServer(LPSTR pszServerName, INT iPortNumber, SOCKET * pSocket) 715 { // in in out 716 SOCKET Socket; 717 struct sockaddr_in sin; 718 struct hostent *hp; 719 720 721 Socket = socket(PF_INET, SOCK_STREAM, 0); 722 if (Socket == INVALID_SOCKET) 723 { 724 printf("**** Error %d creating socket ", WSAGetLastError()); 725 DisplayWinSockError(WSAGetLastError()); 726 return WSAGetLastError(); 727 } 728 729 730 if (fUseProxy) 731 { 732 sin.sin_family = AF_INET; 733 sin.sin_port = ntohs((u_short)iProxyPort); 734 if ((hp = gethostbyname(pszProxyServer)) == NULL) 735 { 736 printf("**** Error %d returned by gethostbyname using Proxy ", WSAGetLastError()); 737 DisplayWinSockError(WSAGetLastError()); 738 return WSAGetLastError(); 739 } 740 else 741 memcpy(&sin.sin_addr, hp->h_addr, 4); 742 } 743 744 else // No proxy used 745 { 746 sin.sin_family = AF_INET; 747 sin.sin_port = htons((u_short)iPortNumber); 748 if ((hp = gethostbyname(pszServerName)) == NULL) 749 { 750 printf("**** Error returned by gethostbyname "); 751 DisplayWinSockError(WSAGetLastError()); 752 return WSAGetLastError(); 753 } 754 else 755 memcpy(&sin.sin_addr, hp->h_addr, 4); 756 } 757 758 759 if (connect(Socket, (struct sockaddr *)&sin, sizeof(sin)) == SOCKET_ERROR) 760 { 761 printf("**** Error %d connecting to "%s" (%s) ", WSAGetLastError(), pszServerName, inet_ntoa(sin.sin_addr)); 762 closesocket(Socket); 763 DisplayWinSockError(WSAGetLastError()); 764 return WSAGetLastError(); 765 } 766 767 768 if (fUseProxy) 769 { 770 BYTE pbMessage[200]; 771 DWORD cbMessage; 772 773 // Build message for proxy server 774 strcpy((char*)pbMessage, "CONNECT "); 775 strcat((char*)pbMessage, pszServerName); 776 strcat((char*)pbMessage, ":"); 777 _itoa(iPortNumber, (char*)pbMessage + strlen((char*)pbMessage), 10); 778 strcat((char*)pbMessage, " HTTP/1.0 User-Agent: webclient "); 779 cbMessage = (DWORD)strlen((char*)pbMessage); 780 781 // Send message to proxy server 782 if (send(Socket, (char*)pbMessage, cbMessage, 0) == SOCKET_ERROR) 783 { 784 printf("**** Error %d sending message to proxy! ", WSAGetLastError()); 785 DisplayWinSockError(WSAGetLastError()); 786 return WSAGetLastError(); 787 } 788 789 // Receive message from proxy server 790 cbMessage = recv(Socket, (char*)pbMessage, 200, 0); 791 if (cbMessage == SOCKET_ERROR) 792 { 793 printf("**** Error %d receiving message from proxy ", WSAGetLastError()); 794 DisplayWinSockError(WSAGetLastError()); 795 return WSAGetLastError(); 796 } 797 // this sample is limited but in normal use it 798 // should continue to receive until CR LF CR LF is received 799 } 800 *pSocket = Socket; 801 802 return SEC_E_OK; 803 } 804 805 /*****************************************************************************/ 806 static LONG DisconnectFromServer(SOCKET Socket, PCredHandle phCreds, CtxtHandle * phContext) 807 { 808 PBYTE pbMessage; 809 DWORD dwType, dwSSPIFlags, dwSSPIOutFlags, cbMessage, cbData, Status; 810 SecBufferDesc OutBuffer; 811 SecBuffer OutBuffers[1]; 812 TimeStamp tsExpiry; 813 814 815 dwType = SCHANNEL_SHUTDOWN; // Notify schannel that we are about to close the connection. 816 817 OutBuffers[0].pvBuffer = &dwType; 818 OutBuffers[0].BufferType = SECBUFFER_TOKEN; 819 OutBuffers[0].cbBuffer = sizeof(dwType); 820 821 OutBuffer.cBuffers = 1; 822 OutBuffer.pBuffers = OutBuffers; 823 OutBuffer.ulVersion = SECBUFFER_VERSION; 824 825 Status = g_pSSPI->ApplyControlToken(phContext, &OutBuffer); 826 if (FAILED(Status)) { printf("**** Error 0x%x returned by ApplyControlToken ", Status); goto cleanup; } 827 828 829 // Build an SSL close notify message. 830 dwSSPIFlags = ISC_REQ_SEQUENCE_DETECT | 831 ISC_REQ_REPLAY_DETECT | 832 ISC_REQ_CONFIDENTIALITY | 833 ISC_RET_EXTENDED_ERROR | 834 ISC_REQ_ALLOCATE_MEMORY | 835 ISC_REQ_STREAM; 836 837 OutBuffers[0].pvBuffer = NULL; 838 OutBuffers[0].BufferType = SECBUFFER_TOKEN; 839 OutBuffers[0].cbBuffer = 0; 840 841 OutBuffer.cBuffers = 1; 842 OutBuffer.pBuffers = OutBuffers; 843 OutBuffer.ulVersion = SECBUFFER_VERSION; 844 845 Status = g_pSSPI->InitializeSecurityContextA(phCreds, 846 phContext, 847 NULL, 848 dwSSPIFlags, 849 0, 850 SECURITY_NATIVE_DREP, 851 NULL, 852 0, 853 phContext, 854 &OutBuffer, 855 &dwSSPIOutFlags, 856 &tsExpiry); 857 858 if (FAILED(Status)) { printf("**** Error 0x%x returned by InitializeSecurityContext ", Status); goto cleanup; } 859 860 pbMessage = (PBYTE)OutBuffers[0].pvBuffer; 861 cbMessage = OutBuffers[0].cbBuffer; 862 863 864 // Send the close notify message to the server. 865 if (pbMessage != NULL && cbMessage != 0) 866 { 867 cbData = send(Socket, (char*)pbMessage, cbMessage, 0); 868 if (cbData == SOCKET_ERROR || cbData == 0) 869 { 870 Status = WSAGetLastError(); 871 printf("**** Error %d sending close notify ", Status); 872 DisplayWinSockError(WSAGetLastError()); 873 goto cleanup; 874 } 875 printf("Sending Close Notify "); 876 printf("%d bytes of handshake data sent ", cbData); 877 if (fVerbose) { PrintHexDump(cbData, pbMessage); printf(" "); } 878 g_pSSPI->FreeContextBuffer(pbMessage); // Free output buffer. 879 } 880 881 882 cleanup: 883 g_pSSPI->DeleteSecurityContext(phContext); // Free the security context. 884 closesocket(Socket); // Close the socket. 885 886 return Status; 887 } 888 889 890 891 /*****************************************************************************/ 892 static void GetNewClientCredentials(CredHandle *phCreds, CtxtHandle *phContext) 893 { 894 895 CredHandle hCreds; 896 SecPkgContext_IssuerListInfoEx IssuerListInfo; 897 PCCERT_CHAIN_CONTEXT pChainContext; 898 CERT_CHAIN_FIND_BY_ISSUER_PARA FindByIssuerPara; 899 PCCERT_CONTEXT pCertContext; 900 TimeStamp tsExpiry; 901 SECURITY_STATUS Status; 902 903 904 // Read list of trusted issuers from schannel. 905 Status = g_pSSPI->QueryContextAttributes(phContext, SECPKG_ATTR_ISSUER_LIST_EX, (PVOID)&IssuerListInfo); 906 if (Status != SEC_E_OK) { printf("Error 0x%x querying issuer list info ", Status); return; } 907 908 // Enumerate the client certificates. 909 ZeroMemory(&FindByIssuerPara, sizeof(FindByIssuerPara)); 910 911 FindByIssuerPara.cbSize = sizeof(FindByIssuerPara); 912 FindByIssuerPara.pszUsageIdentifier = szOID_PKIX_KP_CLIENT_AUTH; 913 FindByIssuerPara.dwKeySpec = 0; 914 FindByIssuerPara.cIssuer = IssuerListInfo.cIssuers; 915 FindByIssuerPara.rgIssuer = IssuerListInfo.aIssuers; 916 917 pChainContext = NULL; 918 919 while (TRUE) 920 { // Find a certificate chain. 921 pChainContext = CertFindChainInStore(hMyCertStore, 922 X509_ASN_ENCODING, 923 0, 924 CERT_CHAIN_FIND_BY_ISSUER, 925 &FindByIssuerPara, 926 pChainContext); 927 if (pChainContext == NULL) { printf("Error 0x%x finding cert chain ", GetLastError()); break; } 928 929 printf(" certificate chain found "); 930 931 // Get pointer to leaf certificate context. 932 pCertContext = pChainContext->rgpChain[0]->rgpElement[0]->pCertContext; 933 934 // Create schannel credential. 935 SchannelCred.dwVersion = SCHANNEL_CRED_VERSION; 936 SchannelCred.cCreds = 1; 937 SchannelCred.paCred = &pCertContext; 938 939 Status = g_pSSPI->AcquireCredentialsHandleA(NULL, // Name of principal 940 (char*)UNISP_NAME_A, // Name of package 941 SECPKG_CRED_OUTBOUND, // Flags indicating use 942 NULL, // Pointer to logon ID 943 &SchannelCred, // Package specific data 944 NULL, // Pointer to GetKey() func 945 NULL, // Value to pass to GetKey() 946 &hCreds, // (out) Cred Handle 947 &tsExpiry); // (out) Lifetime (optional) 948 949 if (Status != SEC_E_OK) { printf("**** Error 0x%x returned by AcquireCredentialsHandle ", Status); continue; } 950 951 printf(" new schannel credential created "); 952 953 g_pSSPI->FreeCredentialsHandle(phCreds); // Destroy the old credentials. 954 955 *phCreds = hCreds; 956 957 } 958 } 959 960 /*****************************************************************************/ 961 static SECURITY_STATUS ClientHandshakeLoop(SOCKET Socket, // in 962 PCredHandle phCreds, // in 963 CtxtHandle * phContext, // in, out 964 BOOL fDoInitialRead, // in 965 SecBuffer * pExtraData) // out 966 967 { 968 969 SecBufferDesc OutBuffer, InBuffer; 970 SecBuffer InBuffers[2], OutBuffers[1]; 971 DWORD dwSSPIFlags, dwSSPIOutFlags, cbData, cbIoBuffer; 972 TimeStamp tsExpiry; 973 SECURITY_STATUS scRet; 974 PUCHAR IoBuffer; 975 BOOL fDoRead; 976 977 978 dwSSPIFlags = ISC_REQ_SEQUENCE_DETECT | ISC_REQ_REPLAY_DETECT | ISC_REQ_CONFIDENTIALITY | 979 ISC_RET_EXTENDED_ERROR | ISC_REQ_ALLOCATE_MEMORY | ISC_REQ_STREAM; 980 981 982 // Allocate data buffer. 983 IoBuffer = (PUCHAR)LocalAlloc(LMEM_FIXED, IO_BUFFER_SIZE); 984 if (IoBuffer == NULL) { printf("**** Out of memory (1) "); return SEC_E_INTERNAL_ERROR; } 985 cbIoBuffer = 0; 986 fDoRead = fDoInitialRead; 987 988 989 990 // Loop until the handshake is finished or an error occurs. 991 scRet = SEC_I_CONTINUE_NEEDED; 992 993 while (scRet == SEC_I_CONTINUE_NEEDED || 994 scRet == SEC_E_INCOMPLETE_MESSAGE || 995 scRet == SEC_I_INCOMPLETE_CREDENTIALS) 996 { 997 if (0 == cbIoBuffer || scRet == SEC_E_INCOMPLETE_MESSAGE) // Read data from server. 998 { 999 if (fDoRead) 1000 { 1001 cbData = recv(Socket, (char*)IoBuffer + cbIoBuffer, IO_BUFFER_SIZE - cbIoBuffer, 0); 1002 if (cbData == SOCKET_ERROR) 1003 { 1004 printf("**** Error %d reading data from server ", WSAGetLastError()); 1005 scRet = SEC_E_INTERNAL_ERROR; 1006 break; 1007 } 1008 else if (cbData == 0) 1009 { 1010 printf("**** Server unexpectedly disconnected "); 1011 scRet = SEC_E_INTERNAL_ERROR; 1012 break; 1013 } 1014 printf("%d bytes of handshake data received ", cbData); 1015 if (fVerbose) { PrintHexDump(cbData, IoBuffer + cbIoBuffer); printf(" "); } 1016 cbIoBuffer += cbData; 1017 } 1018 else 1019 fDoRead = TRUE; 1020 } 1021 1022 1023 1024 // Set up the input buffers. Buffer 0 is used to pass in data 1025 // received from the server. Schannel will consume some or all 1026 // of this. Leftover data (if any) will be placed in buffer 1 and 1027 // given a buffer type of SECBUFFER_EXTRA. 1028 InBuffers[0].pvBuffer = IoBuffer; 1029 InBuffers[0].cbBuffer = cbIoBuffer; 1030 InBuffers[0].BufferType = SECBUFFER_TOKEN; 1031 1032 InBuffers[1].pvBuffer = NULL; 1033 InBuffers[1].cbBuffer = 0; 1034 InBuffers[1].BufferType = SECBUFFER_EMPTY; 1035 1036 InBuffer.cBuffers = 2; 1037 InBuffer.pBuffers = InBuffers; 1038 InBuffer.ulVersion = SECBUFFER_VERSION; 1039 1040 1041 // Set up the output buffers. These are initialized to NULL 1042 // so as to make it less likely we'll attempt to free random 1043 // garbage later. 1044 OutBuffers[0].pvBuffer = NULL; 1045 OutBuffers[0].BufferType = SECBUFFER_TOKEN; 1046 OutBuffers[0].cbBuffer = 0; 1047 1048 OutBuffer.cBuffers = 1; 1049 OutBuffer.pBuffers = OutBuffers; 1050 OutBuffer.ulVersion = SECBUFFER_VERSION; 1051 1052 1053 // Call InitializeSecurityContext. 1054 scRet = g_pSSPI->InitializeSecurityContextA(phCreds, 1055 phContext, 1056 NULL, 1057 dwSSPIFlags, 1058 0, 1059 SECURITY_NATIVE_DREP, 1060 &InBuffer, 1061 0, 1062 NULL, 1063 &OutBuffer, 1064 &dwSSPIOutFlags, 1065 &tsExpiry); 1066 1067 1068 // If InitializeSecurityContext was successful (or if the error was 1069 // one of the special extended ones), send the contends of the output 1070 // buffer to the server. 1071 if (scRet == SEC_E_OK || 1072 scRet == SEC_I_CONTINUE_NEEDED || 1073 FAILED(scRet) && (dwSSPIOutFlags & ISC_RET_EXTENDED_ERROR)) 1074 { 1075 if (OutBuffers[0].cbBuffer != 0 && OutBuffers[0].pvBuffer != NULL) 1076 { 1077 cbData = send(Socket, (char*)OutBuffers[0].pvBuffer, OutBuffers[0].cbBuffer, 0); 1078 if (cbData == SOCKET_ERROR || cbData == 0) 1079 { 1080 printf("**** Error %d sending data to server (2) ", WSAGetLastError()); 1081 DisplayWinSockError(WSAGetLastError()); 1082 g_pSSPI->FreeContextBuffer(OutBuffers[0].pvBuffer); 1083 g_pSSPI->DeleteSecurityContext(phContext); 1084 return SEC_E_INTERNAL_ERROR; 1085 } 1086 printf("%d bytes of handshake data sent ", cbData); 1087 if (fVerbose) { PrintHexDump(cbData, (PBYTE)OutBuffers[0].pvBuffer); printf(" "); } 1088 1089 // Free output buffer. 1090 g_pSSPI->FreeContextBuffer(OutBuffers[0].pvBuffer); 1091 OutBuffers[0].pvBuffer = NULL; 1092 } 1093 } 1094 1095 1096 1097 // If InitializeSecurityContext returned SEC_E_INCOMPLETE_MESSAGE, 1098 // then we need to read more data from the server and try again. 1099 if (scRet == SEC_E_INCOMPLETE_MESSAGE) continue; 1100 1101 1102 // If InitializeSecurityContext returned SEC_E_OK, then the 1103 // handshake completed successfully. 1104 if (scRet == SEC_E_OK) 1105 { 1106 // If the "extra" buffer contains data, this is encrypted application 1107 // protocol layer stuff. It needs to be saved. The application layer 1108 // will later decrypt it with DecryptMessage. 1109 printf("Handshake was successful "); 1110 1111 if (InBuffers[1].BufferType == SECBUFFER_EXTRA) 1112 { 1113 pExtraData->pvBuffer = LocalAlloc(LMEM_FIXED, InBuffers[1].cbBuffer); 1114 if (pExtraData->pvBuffer == NULL) { printf("**** Out of memory (2) "); return SEC_E_INTERNAL_ERROR; } 1115 1116 MoveMemory(pExtraData->pvBuffer, 1117 IoBuffer + (cbIoBuffer - InBuffers[1].cbBuffer), 1118 InBuffers[1].cbBuffer); 1119 1120 pExtraData->cbBuffer = InBuffers[1].cbBuffer; 1121 pExtraData->BufferType = SECBUFFER_TOKEN; 1122 1123 printf("%d bytes of app data was bundled with handshake data ", pExtraData->cbBuffer); 1124 } 1125 else 1126 { 1127 pExtraData->pvBuffer = NULL; 1128 pExtraData->cbBuffer = 0; 1129 pExtraData->BufferType = SECBUFFER_EMPTY; 1130 } 1131 break; // Bail out to quit 1132 } 1133 1134 1135 1136 // Check for fatal error. 1137 if (FAILED(scRet)) { printf("**** Error 0x%x returned by InitializeSecurityContext (2) ", scRet); break; } 1138 1139 // If InitializeSecurityContext returned SEC_I_INCOMPLETE_CREDENTIALS, 1140 // then the server just requested client authentication. 1141 if (scRet == SEC_I_INCOMPLETE_CREDENTIALS) 1142 { 1143 // Busted. The server has requested client authentication and 1144 // the credential we supplied didn't contain a client certificate. 1145 // This function will read the list of trusted certificate 1146 // authorities ("issuers") that was received from the server 1147 // and attempt to find a suitable client certificate that 1148 // was issued by one of these. If this function is successful, 1149 // then we will connect using the new certificate. Otherwise, 1150 // we will attempt to connect anonymously (using our current credentials). 1151 GetNewClientCredentials(phCreds, phContext); 1152 1153 // Go around again. 1154 fDoRead = FALSE; 1155 scRet = SEC_I_CONTINUE_NEEDED; 1156 continue; 1157 } 1158 1159 // Copy any leftover data from the "extra" buffer, and go around again. 1160 if (InBuffers[1].BufferType == SECBUFFER_EXTRA) 1161 { 1162 MoveMemory(IoBuffer, IoBuffer + (cbIoBuffer - InBuffers[1].cbBuffer), InBuffers[1].cbBuffer); 1163 cbIoBuffer = InBuffers[1].cbBuffer; 1164 } 1165 else 1166 cbIoBuffer = 0; 1167 } 1168 1169 // Delete the security context in the case of a fatal error. 1170 if (FAILED(scRet)) g_pSSPI->DeleteSecurityContext(phContext); 1171 LocalFree(IoBuffer); 1172 1173 return scRet; 1174 } 1175 1176 1177 /*****************************************************************************/ 1178 static SECURITY_STATUS PerformClientHandshake(SOCKET Socket, // in 1179 PCredHandle phCreds, // in 1180 LPSTR pszServerName, // in 1181 CtxtHandle * phContext, // out 1182 SecBuffer * pExtraData) // out 1183 { 1184 1185 SecBufferDesc OutBuffer; 1186 SecBuffer OutBuffers[1]; 1187 DWORD dwSSPIFlags, dwSSPIOutFlags, cbData; 1188 TimeStamp tsExpiry; 1189 SECURITY_STATUS scRet; 1190 1191 1192 dwSSPIFlags = ISC_REQ_SEQUENCE_DETECT | ISC_REQ_REPLAY_DETECT | ISC_REQ_CONFIDENTIALITY | 1193 ISC_RET_EXTENDED_ERROR | ISC_REQ_ALLOCATE_MEMORY | ISC_REQ_STREAM; 1194 1195 1196 // Initiate a ClientHello message and generate a token. 1197 OutBuffers[0].pvBuffer = NULL; 1198 OutBuffers[0].BufferType = SECBUFFER_TOKEN; 1199 OutBuffers[0].cbBuffer = 0; 1200 1201 OutBuffer.cBuffers = 1; 1202 OutBuffer.pBuffers = OutBuffers; 1203 OutBuffer.ulVersion = SECBUFFER_VERSION; 1204 1205 scRet = g_pSSPI->InitializeSecurityContextA(phCreds, 1206 NULL, 1207 pszServerName, 1208 dwSSPIFlags, 1209 0, 1210 SECURITY_NATIVE_DREP, 1211 NULL, 1212 0, 1213 phContext, 1214 &OutBuffer, 1215 &dwSSPIOutFlags, 1216 &tsExpiry); 1217 1218 if (scRet != SEC_I_CONTINUE_NEEDED) { printf("**** Error %d returned by InitializeSecurityContext (1) ", scRet); return scRet; } 1219 1220 // Send response to server if there is one. 1221 if (OutBuffers[0].cbBuffer != 0 && OutBuffers[0].pvBuffer != NULL) 1222 { 1223 cbData = send(Socket, (CHAR*)OutBuffers[0].pvBuffer, OutBuffers[0].cbBuffer, 0); 1224 if (cbData == SOCKET_ERROR || cbData == 0) 1225 { 1226 printf("**** Error %d sending data to server (1) ", WSAGetLastError()); 1227 g_pSSPI->FreeContextBuffer(OutBuffers[0].pvBuffer); 1228 g_pSSPI->DeleteSecurityContext(phContext); 1229 return SEC_E_INTERNAL_ERROR; 1230 } 1231 printf("%d bytes of handshake data sent ", cbData); 1232 if (fVerbose) { PrintHexDump(cbData, (PBYTE)OutBuffers[0].pvBuffer); printf(" "); } 1233 g_pSSPI->FreeContextBuffer(OutBuffers[0].pvBuffer); // Free output buffer. 1234 OutBuffers[0].pvBuffer = NULL; 1235 } 1236 1237 return ClientHandshakeLoop(Socket, phCreds, phContext, TRUE, pExtraData); 1238 } 1239 1240 1241 1242 /*****************************************************************************/ 1243 static DWORD EncryptSend(SOCKET Socket, CtxtHandle * phContext, PBYTE pbIoBuffer, SecPkgContext_StreamSizes Sizes) 1244 // http://msdn.microsoft.com/en-us/library/aa375378(VS.85).aspx 1245 // The encrypted message is encrypted in place, overwriting the original contents of its buffer. 1246 { 1247 SECURITY_STATUS scRet; // unsigned long cbBuffer; // Size of the buffer, in bytes 1248 SecBufferDesc Message; // unsigned long BufferType; // Type of the buffer (below) 1249 SecBuffer Buffers[4]; // void SEC_FAR * pvBuffer; // Pointer to the buffer 1250 DWORD cbMessage, cbData; 1251 PBYTE pbMessage; 1252 1253 1254 pbMessage = pbIoBuffer + Sizes.cbHeader; // Offset by "header size" 1255 cbMessage = (DWORD)strlen((const char*)pbMessage); 1256 printf("Sending %d bytes of plaintext:", cbMessage); PrintText(cbMessage, pbMessage); 1257 if (fVerbose) { PrintHexDump(cbMessage, pbMessage); printf(" "); } 1258 1259 1260 // Encrypt the HTTP request. 1261 Buffers[0].pvBuffer = pbIoBuffer; // Pointer to buffer 1 1262 Buffers[0].cbBuffer = Sizes.cbHeader; // length of header 1263 Buffers[0].BufferType = SECBUFFER_STREAM_HEADER; // Type of the buffer 1264 1265 Buffers[1].pvBuffer = pbMessage; // Pointer to buffer 2 1266 Buffers[1].cbBuffer = cbMessage; // length of the message 1267 Buffers[1].BufferType = SECBUFFER_DATA; // Type of the buffer 1268 1269 Buffers[2].pvBuffer = pbMessage + cbMessage; // Pointer to buffer 3 1270 Buffers[2].cbBuffer = Sizes.cbTrailer; // length of the trailor 1271 Buffers[2].BufferType = SECBUFFER_STREAM_TRAILER; // Type of the buffer 1272 1273 Buffers[3].pvBuffer = SECBUFFER_EMPTY; // Pointer to buffer 4 1274 Buffers[3].cbBuffer = SECBUFFER_EMPTY; // length of buffer 4 1275 Buffers[3].BufferType = SECBUFFER_EMPTY; // Type of the buffer 4 1276 1277 1278 Message.ulVersion = SECBUFFER_VERSION; // Version number 1279 Message.cBuffers = 4; // Number of buffers - must contain four SecBuffer structures. 1280 Message.pBuffers = Buffers; // Pointer to array of buffers 1281 scRet = g_pSSPI->EncryptMessage(phContext, 0, &Message, 0); // must contain four SecBuffer structures. 1282 if (FAILED(scRet)) { printf("**** Error 0x%x returned by EncryptMessage ", scRet); return scRet; } 1283 1284 1285 // Send the encrypted data to the server. len flags 1286 cbData = send(Socket, (const char*)pbIoBuffer, Buffers[0].cbBuffer + Buffers[1].cbBuffer + Buffers[2].cbBuffer, 0); 1287 1288 printf("%d bytes of encrypted data sent ", cbData); 1289 if (fVerbose) { PrintHexDump(cbData, pbIoBuffer); printf(" "); } 1290 1291 return cbData; // send( Socket, pbIoBuffer, Sizes.cbHeader + strlen(pbMessage) + Sizes.cbTrailer, 0 ); 1292 1293 } 1294 1295 1296 /*****************************************************************************/ 1297 static SECURITY_STATUS ReadDecrypt(SOCKET Socket, PCredHandle phCreds, CtxtHandle * phContext, PBYTE pbIoBuffer, DWORD cbIoBufferLength) 1298 1299 // calls recv() - blocking socket read 1300 // http://msdn.microsoft.com/en-us/library/ms740121(VS.85).aspx 1301 1302 // The encrypted message is decrypted in place, overwriting the original contents of its buffer. 1303 // http://msdn.microsoft.com/en-us/library/aa375211(VS.85).aspx 1304 1305 { 1306 SecBuffer ExtraBuffer; 1307 SecBuffer *pDataBuffer, *pExtraBuffer; 1308 1309 SECURITY_STATUS scRet; // unsigned long cbBuffer; // Size of the buffer, in bytes 1310 SecBufferDesc Message; // unsigned long BufferType; // Type of the buffer (below) 1311 SecBuffer Buffers[4]; // void SEC_FAR * pvBuffer; // Pointer to the buffer 1312 1313 DWORD cbIoBuffer, cbData, length; 1314 PBYTE buff; 1315 int i; 1316 1317 1318 1319 // Read data from server until done. 1320 cbIoBuffer = 0; 1321 scRet = 0; 1322 while (TRUE) // Read some data. 1323 { 1324 if (cbIoBuffer == 0 || scRet == SEC_E_INCOMPLETE_MESSAGE) // get the data 1325 { 1326 cbData = recv(Socket, (char*)pbIoBuffer + cbIoBuffer, cbIoBufferLength - cbIoBuffer, 0); 1327 if (cbData == SOCKET_ERROR) 1328 { 1329 printf("**** Error %d reading data from server ", WSAGetLastError()); 1330 scRet = SEC_E_INTERNAL_ERROR; 1331 break; 1332 } 1333 else if (cbData == 0) // Server disconnected. 1334 { 1335 if (cbIoBuffer) 1336 { 1337 printf("**** Server unexpectedly disconnected "); 1338 scRet = SEC_E_INTERNAL_ERROR; 1339 return scRet; 1340 } 1341 else 1342 break; // All Done 1343 } 1344 else // success 1345 { 1346 printf("%d bytes of (encrypted) application data received ", cbData); 1347 if (fVerbose) { PrintHexDump(cbData, pbIoBuffer + cbIoBuffer); printf(" "); } 1348 cbIoBuffer += cbData; 1349 } 1350 } 1351 1352 1353 // Decrypt the received data. 1354 Buffers[0].pvBuffer = pbIoBuffer; 1355 Buffers[0].cbBuffer = cbIoBuffer; 1356 Buffers[0].BufferType = SECBUFFER_DATA; // Initial Type of the buffer 1 1357 Buffers[1].BufferType = SECBUFFER_EMPTY; // Initial Type of the buffer 2 1358 Buffers[2].BufferType = SECBUFFER_EMPTY; // Initial Type of the buffer 3 1359 Buffers[3].BufferType = SECBUFFER_EMPTY; // Initial Type of the buffer 4 1360 1361 Message.ulVersion = SECBUFFER_VERSION; // Version number 1362 Message.cBuffers = 4; // Number of buffers - must contain four SecBuffer structures. 1363 Message.pBuffers = Buffers; // Pointer to array of buffers 1364 scRet = g_pSSPI->DecryptMessage(phContext, &Message, 0, NULL); 1365 if (scRet == SEC_I_CONTEXT_EXPIRED) break; // Server signalled end of session 1366 // if( scRet == SEC_E_INCOMPLETE_MESSAGE - Input buffer has partial encrypted record, read more 1367 if (scRet != SEC_E_OK && 1368 scRet != SEC_I_RENEGOTIATE && 1369 scRet != SEC_I_CONTEXT_EXPIRED) 1370 { 1371 printf("**** DecryptMessage "); 1372 DisplaySECError((DWORD)scRet); 1373 return scRet; 1374 } 1375 1376 1377 1378 // Locate data and (optional) extra buffers. 1379 pDataBuffer = NULL; 1380 pExtraBuffer = NULL; 1381 for (i = 1; i < 4; i++) 1382 { 1383 if (pDataBuffer == NULL && Buffers[i].BufferType == SECBUFFER_DATA) pDataBuffer = &Buffers[i]; 1384 if (pExtraBuffer == NULL && Buffers[i].BufferType == SECBUFFER_EXTRA) pExtraBuffer = &Buffers[i]; 1385 } 1386 1387 1388 // Display the decrypted data. 1389 if (pDataBuffer) 1390 { 1391 length = pDataBuffer->cbBuffer; 1392 if (length) // check if last two chars are CR LF 1393 { 1394 buff = (PBYTE)pDataBuffer->pvBuffer; // printf( "n-2= %d, n-1= %d ", buff[length-2], buff[length-1] ); 1395 printf("Decrypted data: %d bytes", length); PrintText(length, buff); 1396 if (fVerbose) { PrintHexDump(length, buff); printf(" "); } 1397 if (buff[length - 2] == 13 && buff[length - 1] == 10) break; // printf("Found CRLF "); 1398 } 1399 } 1400 1401 1402 1403 // Move any "extra" data to the input buffer. 1404 if (pExtraBuffer) 1405 { 1406 MoveMemory(pbIoBuffer, pExtraBuffer->pvBuffer, pExtraBuffer->cbBuffer); 1407 cbIoBuffer = pExtraBuffer->cbBuffer; // printf("cbIoBuffer= %d ", cbIoBuffer); 1408 } 1409 else 1410 cbIoBuffer = 0; 1411 1412 1413 // The server wants to perform another handshake sequence. 1414 if (scRet == SEC_I_RENEGOTIATE) 1415 { 1416 printf("Server requested renegotiate! "); 1417 scRet = ClientHandshakeLoop(Socket, phCreds, phContext, FALSE, &ExtraBuffer); 1418 if (scRet != SEC_E_OK) return scRet; 1419 1420 if (ExtraBuffer.pvBuffer) // Move any "extra" data to the input buffer. 1421 { 1422 MoveMemory(pbIoBuffer, ExtraBuffer.pvBuffer, ExtraBuffer.cbBuffer); 1423 cbIoBuffer = ExtraBuffer.cbBuffer; 1424 } 1425 } 1426 } // Loop till CRLF is found at the end of the data 1427 1428 return SEC_E_OK; 1429 } 1430 1431 1432 1433 /*****************************************************************************/ 1434 static SECURITY_STATUS SMTPsession(SOCKET Socket, // in 1435 PCredHandle phCreds, // in 1436 CtxtHandle * phContext) // in 1437 { 1438 SecPkgContext_StreamSizes Sizes; // unsigned long cbBuffer; // Size of the buffer, in bytes 1439 SECURITY_STATUS scRet; // unsigned long BufferType; // Type of the buffer (below) 1440 PBYTE pbIoBuffer; // void SEC_FAR * pvBuffer; // Pointer to the buffer 1441 DWORD cbIoBufferLength, cbData; 1442 1443 1444 // Read stream encryption properties. 1445 scRet = g_pSSPI->QueryContextAttributes(phContext, SECPKG_ATTR_STREAM_SIZES, &Sizes); 1446 if (scRet != SEC_E_OK) 1447 { 1448 printf("**** Error 0x%x reading SECPKG_ATTR_STREAM_SIZES ", scRet); return scRet; 1449 } 1450 1451 1452 // Create a buffer. 1453 cbIoBufferLength = Sizes.cbHeader + Sizes.cbMaximumMessage + Sizes.cbTrailer; 1454 pbIoBuffer = (PBYTE)LocalAlloc(LMEM_FIXED, cbIoBufferLength); 1455 if (pbIoBuffer == NULL) { printf("**** Out of memory (2) "); return SEC_E_INTERNAL_ERROR; } 1456 1457 1458 // Receive a Response 1459 scRet = ReadDecrypt(Socket, phCreds, phContext, pbIoBuffer, cbIoBufferLength); 1460 if (scRet != SEC_E_OK) return scRet; 1461 1462 1463 // Build the request - must be < maximum message size 1464 sprintf((char*)pbIoBuffer + Sizes.cbHeader, "%s", "EHLO "); // message begins after the header 1465 1466 1467 // Send a request. 1468 cbData = EncryptSend(Socket, phContext, pbIoBuffer, Sizes); 1469 if (cbData == SOCKET_ERROR || cbData == 0) 1470 { 1471 printf("**** Error %d sending data to server (3) ", WSAGetLastError()); return SEC_E_INTERNAL_ERROR; 1472 } 1473 1474 1475 // Receive a Response 1476 scRet = ReadDecrypt(Socket, phCreds, phContext, pbIoBuffer, cbIoBufferLength); 1477 if (scRet != SEC_E_OK) return scRet; 1478 1479 1480 1481 1482 // Build the request - must be < maximum message size 1483 sprintf((char*)pbIoBuffer + Sizes.cbHeader, "%s", "QUIT "); // message begins after the header 1484 1485 1486 // Send a request. 1487 cbData = EncryptSend(Socket, phContext, pbIoBuffer, Sizes); 1488 if (cbData == SOCKET_ERROR || cbData == 0) 1489 { 1490 printf("**** Error %d sending data to server (3) ", WSAGetLastError()); return SEC_E_INTERNAL_ERROR; 1491 } 1492 1493 1494 // Receive a Response 1495 scRet = ReadDecrypt(Socket, phCreds, phContext, pbIoBuffer, cbIoBufferLength); 1496 if (scRet != SEC_E_OK) return scRet; 1497 1498 1499 return SEC_E_OK; 1500 } 1501 1502 1503 /*****************************************************************************/ 1504 void _cdecl main(int argc, char *argv[]) 1505 { 1506 WSADATA WsaData; 1507 SOCKET Socket = INVALID_SOCKET; 1508 1509 CredHandle hClientCreds; 1510 CtxtHandle hContext; 1511 BOOL fCredsInitialized = FALSE; 1512 BOOL fContextInitialized = FALSE; 1513 1514 SecBuffer ExtraData; 1515 SECURITY_STATUS Status; 1516 1517 PCCERT_CONTEXT pRemoteCertContext = NULL; 1518 1519 1520 1521 if (!LoadSecurityLibrary()) 1522 { 1523 printf("Error initializing the security library "); goto cleanup; 1524 } // 1525 printf("----- SSPI Initialized "); 1526 1527 1528 // Initialize the WinSock subsystem. 1529 if (WSAStartup(0x0101, &WsaData) == SOCKET_ERROR) // Winsock.h 1530 { 1531 printf("Error %d returned by WSAStartup ", GetLastError()); goto cleanup; 1532 } // 1533 printf("----- WinSock Initialized "); 1534 1535 1536 // Create credentials. 1537 if (CreateCredentials((char*)pszUser, &hClientCreds)) 1538 { 1539 printf("Error creating credentials "); goto cleanup; 1540 } 1541 fCredsInitialized = TRUE; // 1542 printf("----- Credentials Initialized "); 1543 1544 1545 // Connect to server. 1546 if (ConnectToServer((char*)pszServerName, iPortNumber, &Socket)) 1547 { 1548 printf("Error connecting to server "); goto cleanup; 1549 } // 1550 printf("----- Connectd To Server "); 1551 1552 1553 1554 // Perform handshake 1555 if (PerformClientHandshake(Socket, &hClientCreds, (char*)pszServerName, &hContext, &ExtraData)) 1556 { 1557 printf("Error performing handshake "); goto cleanup; 1558 } 1559 fContextInitialized = TRUE; // 1560 printf("----- Client Handshake Performed "); 1561 1562 1563 // Authenticate server's credentials. Get server's certificate. 1564 Status = g_pSSPI->QueryContextAttributes(&hContext, SECPKG_ATTR_REMOTE_CERT_CONTEXT, (PVOID)&pRemoteCertContext); 1565 if (Status != SEC_E_OK) 1566 { 1567 printf("Error 0x%x querying remote certificate ", Status); goto cleanup; 1568 } // 1569 printf("----- Server Credentials Authenticated "); 1570 1571 1572 // Display server certificate chain. 1573 DisplayCertChain(pRemoteCertContext, FALSE); // 1574 printf("----- Certificate Chain Displayed "); 1575 1576 1577 // Attempt to validate server certificate. 1578 Status = VerifyServerCertificate(pRemoteCertContext, (char*)pszServerName, 0); 1579 if (Status) { printf("**** Error 0x%x authenticating server credentials! ", Status); goto cleanup; } 1580 // The server certificate did not validate correctly. At this point, we cannot tell 1581 // if we are connecting to the correct server, or if we are connecting to a 1582 // "man in the middle" attack server - Best to just abort the connection. 1583 printf("----- Server Certificate Verified "); 1584 1585 1586 1587 // Free the server certificate context. 1588 CertFreeCertificateContext(pRemoteCertContext); 1589 pRemoteCertContext = NULL; // 1590 printf("----- Server certificate context released "); 1591 1592 1593 // Display connection info. 1594 DisplayConnectionInfo(&hContext); // 1595 printf("----- Secure Connection Info "); 1596 1597 1598 1599 // Send Request, recover response. LPSTR pszRequest = "EHLO"; 1600 if (SMTPsession(Socket, &hClientCreds, &hContext)) 1601 { 1602 printf("Error SMTP Session "); goto cleanup; 1603 } // 1604 printf("----- SMTP session Complete "); 1605 1606 1607 // Send a close_notify alert to the server and close down the connection. 1608 if (DisconnectFromServer(Socket, &hClientCreds, &hContext)) 1609 { 1610 printf("Error disconnecting from server "); goto cleanup; 1611 } 1612 fContextInitialized = FALSE; 1613 Socket = INVALID_SOCKET; // 1614 printf("----- Disconnected From Server "); 1615 1616 1617 1618 1619 cleanup: // 1620 printf("----- Begin Cleanup "); 1621 1622 // Free the server certificate context. 1623 if (pRemoteCertContext) 1624 { 1625 CertFreeCertificateContext(pRemoteCertContext); 1626 pRemoteCertContext = NULL; 1627 } 1628 1629 // Free SSPI context handle. 1630 if (fContextInitialized) 1631 { 1632 g_pSSPI->DeleteSecurityContext(&hContext); 1633 fContextInitialized = FALSE; 1634 } 1635 1636 // Free SSPI credentials handle. 1637 if (fCredsInitialized) 1638 { 1639 g_pSSPI->FreeCredentialsHandle(&hClientCreds); 1640 fCredsInitialized = FALSE; 1641 } 1642 1643 // Close socket. 1644 if (Socket != INVALID_SOCKET) closesocket(Socket); 1645 1646 // Shutdown WinSock subsystem. 1647 WSACleanup(); 1648 1649 // Close "MY" certificate store. 1650 if (hMyCertStore) CertCloseStore(hMyCertStore, 0); 1651 1652 UnloadSecurityLibrary(); 1653 1654 1655 printf("----- All Done ----- "); 1656 1657 } 1658 1659
PS:会笑的人,运气通常都会比别人好。