20 #include <winpr/config.h>
22 #include "schannel_openssl.h"
26 #include <winpr/crt.h>
27 #include <winpr/sspi.h>
28 #include <winpr/ssl.h>
29 #include <winpr/print.h>
30 #include <winpr/crypto.h>
32 #include <openssl/ssl.h>
33 #include <openssl/err.h>
34 #include <openssl/bio.h>
36 #define LIMIT_INTMAX(a) ((a) > INT32_MAX) ? INT32_MAX : (int)(a)
38 struct S_SCHANNEL_OPENSSL
49 #include "../../log.h"
50 #define TAG WINPR_TAG("sspi.schannel")
52 static char* openssl_get_ssl_error_string(
int ssl_error)
56 case SSL_ERROR_ZERO_RETURN:
57 return "SSL_ERROR_ZERO_RETURN";
59 case SSL_ERROR_WANT_READ:
60 return "SSL_ERROR_WANT_READ";
62 case SSL_ERROR_WANT_WRITE:
63 return "SSL_ERROR_WANT_WRITE";
65 case SSL_ERROR_SYSCALL:
66 return "SSL_ERROR_SYSCALL";
69 return "SSL_ERROR_SSL";
74 return "SSL_ERROR_UNKNOWN";
77 static void schannel_context_cleanup(SCHANNEL_OPENSSL* context)
79 WINPR_ASSERT(context);
81 free(context->ReadBuffer);
82 context->ReadBuffer = NULL;
84 if (context->bioWrite)
85 BIO_free_all(context->bioWrite);
86 context->bioWrite = NULL;
89 BIO_free_all(context->bioRead);
90 context->bioRead = NULL;
93 SSL_free(context->ssl);
97 SSL_CTX_free(context->ctx);
101 static const SSL_METHOD* get_method(BOOL server)
105 #if OPENSSL_VERSION_NUMBER < 0x10100000L || defined(LIBRESSL_VERSION_NUMBER)
106 return SSLv23_server_method();
108 return TLS_server_method();
113 #if OPENSSL_VERSION_NUMBER < 0x10100000L || defined(LIBRESSL_VERSION_NUMBER)
114 return SSLv23_client_method();
116 return TLS_client_method();
120 int schannel_openssl_client_init(SCHANNEL_OPENSSL* context)
124 context->ctx = SSL_CTX_new(get_method(FALSE));
128 WLog_ERR(TAG,
"SSL_CTX_new failed");
141 #ifdef SSL_OP_NO_COMPRESSION
142 options |= SSL_OP_NO_COMPRESSION;
150 options |= SSL_OP_TLS_BLOCK_PADDING_BUG;
157 options |= SSL_OP_DONT_INSERT_EMPTY_FRAGMENTS;
158 SSL_CTX_set_options(context->ctx, WINPR_ASSERTING_INT_CAST(uint64_t, options));
159 context->ssl = SSL_new(context->ctx);
163 WLog_ERR(TAG,
"SSL_new failed");
167 context->bioRead = BIO_new(BIO_s_mem());
169 if (!context->bioRead)
171 WLog_ERR(TAG,
"BIO_new failed");
175 status = BIO_set_write_buf_size(context->bioRead, SCHANNEL_CB_MAX_TOKEN);
179 WLog_ERR(TAG,
"BIO_set_write_buf_size on bioRead failed");
183 context->bioWrite = BIO_new(BIO_s_mem());
185 if (!context->bioWrite)
187 WLog_ERR(TAG,
"BIO_new failed");
191 status = BIO_set_write_buf_size(context->bioWrite, SCHANNEL_CB_MAX_TOKEN);
195 WLog_ERR(TAG,
"BIO_set_write_buf_size on bioWrite failed");
199 status = BIO_make_bio_pair(context->bioRead, context->bioWrite);
203 WLog_ERR(TAG,
"BIO_make_bio_pair failed");
207 SSL_set_bio(context->ssl, context->bioRead, context->bioWrite);
208 context->ReadBuffer = (BYTE*)malloc(SCHANNEL_CB_MAX_TOKEN);
210 if (!context->ReadBuffer)
212 WLog_ERR(TAG,
"Failed to allocate ReadBuffer");
216 context->WriteBuffer = (BYTE*)malloc(SCHANNEL_CB_MAX_TOKEN);
218 if (!context->WriteBuffer)
220 WLog_ERR(TAG,
"Failed to allocate ReadBuffer");
226 schannel_context_cleanup(context);
230 int schannel_openssl_server_init(SCHANNEL_OPENSSL* context)
233 unsigned long options = 0;
235 context->ctx = SSL_CTX_new(get_method(TRUE));
239 WLog_ERR(TAG,
"SSL_CTX_new failed");
249 options |= SSL_OP_NO_SSLv2;
259 #ifdef SSL_OP_NO_COMPRESSION
260 options |= SSL_OP_NO_COMPRESSION;
268 options |= SSL_OP_TLS_BLOCK_PADDING_BUG;
275 options |= SSL_OP_DONT_INSERT_EMPTY_FRAGMENTS;
276 SSL_CTX_set_options(context->ctx, options);
278 #if defined(WITH_DEBUG_SCHANNEL)
279 if (SSL_CTX_use_RSAPrivateKey_file(context->ctx,
"/tmp/localhost.key", SSL_FILETYPE_PEM) <= 0)
281 WLog_ERR(TAG,
"SSL_CTX_use_RSAPrivateKey_file failed");
286 context->ssl = SSL_new(context->ctx);
290 WLog_ERR(TAG,
"SSL_new failed");
294 if (SSL_use_certificate_file(context->ssl,
"/tmp/localhost.crt", SSL_FILETYPE_PEM) <= 0)
296 WLog_ERR(TAG,
"SSL_use_certificate_file failed");
300 context->bioRead = BIO_new(BIO_s_mem());
302 if (!context->bioRead)
304 WLog_ERR(TAG,
"BIO_new failed");
308 status = BIO_set_write_buf_size(context->bioRead, SCHANNEL_CB_MAX_TOKEN);
312 WLog_ERR(TAG,
"BIO_set_write_buf_size failed for bioRead");
316 context->bioWrite = BIO_new(BIO_s_mem());
318 if (!context->bioWrite)
320 WLog_ERR(TAG,
"BIO_new failed");
324 status = BIO_set_write_buf_size(context->bioWrite, SCHANNEL_CB_MAX_TOKEN);
328 WLog_ERR(TAG,
"BIO_set_write_buf_size failed for bioWrite");
332 status = BIO_make_bio_pair(context->bioRead, context->bioWrite);
336 WLog_ERR(TAG,
"BIO_make_bio_pair failed");
340 SSL_set_bio(context->ssl, context->bioRead, context->bioWrite);
341 context->ReadBuffer = (BYTE*)malloc(SCHANNEL_CB_MAX_TOKEN);
343 if (!context->ReadBuffer)
345 WLog_ERR(TAG,
"Failed to allocate memory for ReadBuffer");
349 context->WriteBuffer = (BYTE*)malloc(SCHANNEL_CB_MAX_TOKEN);
351 if (!context->WriteBuffer)
353 WLog_ERR(TAG,
"Failed to allocate memory for WriteBuffer");
359 schannel_context_cleanup(context);
363 SECURITY_STATUS schannel_openssl_client_process_tokens(SCHANNEL_OPENSSL* context,
371 if (!context->connected)
375 if (pInput->cBuffers < 1)
376 return SEC_E_INVALID_TOKEN;
378 pBuffer = sspi_FindSecBuffer(pInput, SECBUFFER_TOKEN);
381 return SEC_E_INVALID_TOKEN;
385 BIO_write(context->bioRead, pBuffer->pvBuffer, LIMIT_INTMAX(pBuffer->cbBuffer));
387 return SEC_E_INVALID_TOKEN;
390 status = SSL_connect(context->ssl);
394 ssl_error = SSL_get_error(context->ssl, status);
395 WLog_ERR(TAG,
"SSL_connect error: %s", openssl_get_ssl_error_string(ssl_error));
399 context->connected = TRUE;
402 status = BIO_read(context->bioWrite, context->ReadBuffer, SCHANNEL_CB_MAX_TOKEN);
404 if (pOutput->cBuffers < 1)
405 return SEC_E_INVALID_TOKEN;
407 pBuffer = sspi_FindSecBuffer(pOutput, SECBUFFER_TOKEN);
410 return SEC_E_INVALID_TOKEN;
414 if (pBuffer->cbBuffer < WINPR_ASSERTING_INT_CAST(uint32_t, status))
415 return SEC_E_INSUFFICIENT_MEMORY;
417 CopyMemory(pBuffer->pvBuffer, context->ReadBuffer,
418 WINPR_ASSERTING_INT_CAST(uint32_t, status));
419 pBuffer->cbBuffer = WINPR_ASSERTING_INT_CAST(uint32_t, status);
420 return (context->connected) ? SEC_E_OK : SEC_I_CONTINUE_NEEDED;
424 pBuffer->cbBuffer = 0;
425 return (context->connected) ? SEC_E_OK : SEC_I_CONTINUE_NEEDED;
432 SECURITY_STATUS schannel_openssl_server_process_tokens(SCHANNEL_OPENSSL* context,
440 if (!context->connected)
442 if (pInput->cBuffers < 1)
443 return SEC_E_INVALID_TOKEN;
445 pBuffer = sspi_FindSecBuffer(pInput, SECBUFFER_TOKEN);
448 return SEC_E_INVALID_TOKEN;
451 status = BIO_write(context->bioRead, pBuffer->pvBuffer, LIMIT_INTMAX(pBuffer->cbBuffer));
453 status = SSL_accept(context->ssl);
457 ssl_error = SSL_get_error(context->ssl, status);
458 WLog_ERR(TAG,
"SSL_accept error: %s", openssl_get_ssl_error_string(ssl_error));
459 return SEC_E_INVALID_TOKEN;
463 context->connected = TRUE;
466 status = BIO_read(context->bioWrite, context->ReadBuffer, SCHANNEL_CB_MAX_TOKEN);
469 ssl_error = SSL_get_error(context->ssl, status);
470 WLog_ERR(TAG,
"BIO_read: %s", openssl_get_ssl_error_string(ssl_error));
471 return SEC_E_INVALID_TOKEN;
474 if (pOutput->cBuffers < 1)
475 return SEC_E_INVALID_TOKEN;
477 pBuffer = sspi_FindSecBuffer(pOutput, SECBUFFER_TOKEN);
480 return SEC_E_INVALID_TOKEN;
484 if (pBuffer->cbBuffer < WINPR_ASSERTING_INT_CAST(uint32_t, status))
485 return SEC_E_INSUFFICIENT_MEMORY;
487 CopyMemory(pBuffer->pvBuffer, context->ReadBuffer,
488 WINPR_ASSERTING_INT_CAST(uint32_t, status));
489 pBuffer->cbBuffer = WINPR_ASSERTING_INT_CAST(uint32_t, status);
490 return (context->connected) ? SEC_E_OK : SEC_I_CONTINUE_NEEDED;
494 pBuffer->cbBuffer = 0;
495 return (context->connected) ? SEC_E_OK : SEC_I_CONTINUE_NEEDED;
502 SECURITY_STATUS schannel_openssl_encrypt_message(SCHANNEL_OPENSSL* context,
PSecBufferDesc pMessage)
509 pStreamHeaderBuffer = sspi_FindSecBuffer(pMessage, SECBUFFER_STREAM_HEADER);
510 pStreamBodyBuffer = sspi_FindSecBuffer(pMessage, SECBUFFER_DATA);
511 pStreamTrailerBuffer = sspi_FindSecBuffer(pMessage, SECBUFFER_STREAM_TRAILER);
513 if ((!pStreamHeaderBuffer) || (!pStreamBodyBuffer) || (!pStreamTrailerBuffer))
514 return SEC_E_INVALID_TOKEN;
516 status = SSL_write(context->ssl, pStreamBodyBuffer->pvBuffer,
517 LIMIT_INTMAX(pStreamBodyBuffer->cbBuffer));
521 ssl_error = SSL_get_error(context->ssl, status);
522 WLog_ERR(TAG,
"SSL_write: %s", openssl_get_ssl_error_string(ssl_error));
526 status = BIO_read(context->bioWrite, context->ReadBuffer, SCHANNEL_CB_MAX_TOKEN);
530 size_t ustatus = (size_t)status;
535 (pStreamHeaderBuffer->cbBuffer > ustatus) ? ustatus : pStreamHeaderBuffer->cbBuffer;
536 CopyMemory(pStreamHeaderBuffer->pvBuffer, &context->ReadBuffer[offset], length);
539 length = (pStreamBodyBuffer->cbBuffer > ustatus) ? ustatus : pStreamBodyBuffer->cbBuffer;
540 CopyMemory(pStreamBodyBuffer->pvBuffer, &context->ReadBuffer[offset], length);
544 (pStreamTrailerBuffer->cbBuffer > ustatus) ? ustatus : pStreamTrailerBuffer->cbBuffer;
545 CopyMemory(pStreamTrailerBuffer->pvBuffer, &context->ReadBuffer[offset], length);
551 SECURITY_STATUS schannel_openssl_decrypt_message(SCHANNEL_OPENSSL* context,
PSecBufferDesc pMessage)
558 pBuffer = sspi_FindSecBuffer(pMessage, SECBUFFER_DATA);
561 return SEC_E_INVALID_TOKEN;
564 status = BIO_write(context->bioRead, pBuffer->pvBuffer, LIMIT_INTMAX(pBuffer->cbBuffer));
566 status = SSL_read(context->ssl, pBuffer->pvBuffer, LIMIT_INTMAX(pBuffer->cbBuffer));
570 ssl_error = SSL_get_error(context->ssl, status);
571 WLog_ERR(TAG,
"SSL_read: %s", openssl_get_ssl_error_string(ssl_error));
575 buffer = pBuffer->pvBuffer;
576 pMessage->pBuffers[0].BufferType = SECBUFFER_STREAM_HEADER;
577 pMessage->pBuffers[0].cbBuffer = 5;
578 pMessage->pBuffers[1].BufferType = SECBUFFER_DATA;
579 pMessage->pBuffers[1].pvBuffer = buffer;
580 pMessage->pBuffers[1].cbBuffer = WINPR_ASSERTING_INT_CAST(uint32_t, length);
581 pMessage->pBuffers[2].BufferType = SECBUFFER_STREAM_TRAILER;
582 pMessage->pBuffers[2].cbBuffer = 36;
583 pMessage->pBuffers[3].BufferType = SECBUFFER_EMPTY;
584 pMessage->pBuffers[3].cbBuffer = 0;
588 SCHANNEL_OPENSSL* schannel_openssl_new(
void)
590 SCHANNEL_OPENSSL* context = NULL;
591 context = (SCHANNEL_OPENSSL*)calloc(1,
sizeof(SCHANNEL_OPENSSL));
595 winpr_InitializeSSL(WINPR_SSL_INIT_DEFAULT);
596 context->connected = FALSE;
602 void schannel_openssl_free(SCHANNEL_OPENSSL* context)
606 free(context->ReadBuffer);
607 free(context->WriteBuffer);
614 int schannel_openssl_client_init(SCHANNEL_OPENSSL* context)
619 int schannel_openssl_server_init(SCHANNEL_OPENSSL* context)
624 SECURITY_STATUS schannel_openssl_client_process_tokens(SCHANNEL_OPENSSL* context,
631 SECURITY_STATUS schannel_openssl_server_process_tokens(SCHANNEL_OPENSSL* context,
638 SECURITY_STATUS schannel_openssl_encrypt_message(SCHANNEL_OPENSSL* context,
PSecBufferDesc pMessage)
643 SECURITY_STATUS schannel_openssl_decrypt_message(SCHANNEL_OPENSSL* context,
PSecBufferDesc pMessage)
648 SCHANNEL_OPENSSL* schannel_openssl_new(
void)
653 void schannel_openssl_free(SCHANNEL_OPENSSL* context)