Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

1833 lines
47 KiB

//+---------------------------------------------------------------------------
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1992 - 1995.
//
// File: ctxtapi.c
//
// Contents:
//
// Classes:
//
// Functions:
//
// History: 10-02-96 RichardW Created
//
//----------------------------------------------------------------------------
#include "sslp.h"
#include <ssl2msg.h>
#include <ssl3msg.h>
#include <ssl2prot.h>
#include <sslcache.h>
#include <lsasecpk.h>
NTSTATUS NTAPI
SslPurgeSessionCache(
IN PLSA_CLIENT_REQUEST ClientRequest,
IN PVOID ProtocolSubmitBuffer,
IN PVOID ClientBufferBase,
IN ULONG SubmitBufferSize,
OUT PVOID *ProtocolReturnBuffer,
OUT PULONG ReturnBufferLength,
OUT PNTSTATUS ProtocolStatus);
NTSTATUS NTAPI
SslSessionCacheInfo(
IN PLSA_CLIENT_REQUEST ClientRequest,
IN PVOID ProtocolSubmitBuffer,
IN PVOID ClientBufferBase,
IN ULONG SubmitBufferSize,
OUT PVOID *ProtocolReturnBuffer,
OUT PULONG ReturnBufferLength,
OUT PNTSTATUS ProtocolStatus);
NTSTATUS NTAPI
SslGetPerfmonInfo(
IN PLSA_CLIENT_REQUEST ClientRequest,
IN PVOID ProtocolSubmitBuffer,
IN PVOID ClientBufferBase,
IN ULONG SubmitBufferSize,
OUT PVOID *ProtocolReturnBuffer,
OUT PULONG ReturnBufferLength,
OUT PNTSTATUS ProtocolStatus);
SECURITY_STATUS SEC_ENTRY
SpInitLsaModeContext(
LSA_SEC_HANDLE dwCredHandle,
LSA_SEC_HANDLE dwCtxtHandle,
PSECURITY_STRING pszTargetName,
ULONG fContextReq,
ULONG TargetDataRep,
PSecBufferDesc pInput,
PLSA_SEC_HANDLE pdwNewContext,
PSecBufferDesc pOutput,
PULONG pfContextAttr,
PTimeStamp ptsExpiry,
PBYTE pfMapContext,
PSecBuffer pContextData)
{
PSPContext pContext = NULL;
PSPCredentialGroup pCred = NULL;
SPBuffer CommOut;
SPBuffer CommIn;
PSecBuffer pInToken = NULL;
PSecBuffer pOutToken = NULL;
PSecBuffer pExtra = NULL;
DWORD fAttr = ISC_RET_REPLAY_DETECT |
ISC_RET_SEQUENCE_DETECT |
ISC_RET_CONFIDENTIALITY |
ISC_RET_STREAM;
DWORD fSchContext = CONTEXT_FLAG_CLIENT;
NTSTATUS Status;
ANSI_STRING String;
int i;
SP_STATUS pctRet = PCT_ERR_OK;
#if DBG
DebugLog((DEB_TRACE, "SpInitLsaModeContext(0x%x, 0x%x, 0x%x, 0x%x, 0x%x, 0x%x, 0x%x, 0x%x, 0x%x, 0x%x, 0x%x, 0x%x)\n",
dwCredHandle, dwCtxtHandle, pszTargetName, fContextReq, TargetDataRep,
pInput, pdwNewContext, pOutput, pfContextAttr, ptsExpiry, pfMapContext,
pContextData));
if(pszTargetName)
{
DebugLog((DEB_TRACE, "pszTargetName:<%ls>\n",pszTargetName->Buffer));
}
#endif
/* These flags are never allowed */
if (fContextReq & (ISC_REQ_DELEGATE | ISC_REQ_PROMPT_FOR_CREDS ))
{
TRACE_EXIT( SpInitLsaModeContext, SEC_E_UNSUPPORTED_FUNCTION );
return SP_LOG_RESULT( SEC_E_UNSUPPORTED_FUNCTION );
}
/* Initialize output buffer locations */
for (i = 0; i < (int)pOutput->cBuffers; i++ )
{
switch( ( pOutput->pBuffers[i].BufferType ) & (~(SECBUFFER_ATTRMASK)))
{
case SECBUFFER_EMPTY:
if(!pOutToken && (fContextReq & ISC_REQ_ALLOCATE_MEMORY))
pOutToken = &pOutput->pBuffers[i];
break;
case SECBUFFER_TOKEN:
pOutToken = &pOutput->pBuffers[i];
break;
case SECBUFFER_DATA:
case SECBUFFER_STREAM_HEADER:
case SECBUFFER_STREAM_TRAILER:
default:
break;
}
}
if ( pOutToken == NULL )
{
TRACE_EXIT( SpInitLsaModeContext, SEC_E_INVALID_TOKEN );
return SP_LOG_RESULT( SEC_E_INVALID_TOKEN );
}
pOutToken->BufferType = SECBUFFER_TOKEN;
if ( fContextReq & ISC_REQ_ALLOCATE_MEMORY )
{
CommOut.pvBuffer = NULL;
CommOut.cbBuffer = 0;
CommOut.cbData = 0;
fAttr |= ISC_RET_ALLOCATED_MEMORY;
pOutToken->pvBuffer = NULL;
pOutToken->cbBuffer = 0;
}
else
{
if ( pOutToken->pvBuffer == NULL )
{
TRACE_EXIT( SpInitLsaModeContext, SEC_E_INSUFFICIENT_MEMORY );
return SP_LOG_RESULT( SEC_E_INSUFFICIENT_MEMORY );
}
}
CommOut.pvBuffer = pOutToken->pvBuffer;
CommOut.cbBuffer = pOutToken->cbBuffer;
CommOut.cbData = 0;
if ( fContextReq & (ISC_REQ_EXTENDED_ERROR) )
{
fAttr |= ISC_RET_EXTENDED_ERROR;
fSchContext |= CONTEXT_FLAG_EXT_ERR;
}
if ( fContextReq & (ISC_REQ_CONNECTION) )
{
fAttr |= ISC_REQ_CONNECTION;
fSchContext |= CONTEXT_FLAG_CONNECTION_MODE;
}
if ( fContextReq & (ISC_REQ_MUTUAL_AUTH) )
{
// Validate the server certificate.
fAttr |= ISC_RET_MUTUAL_AUTH;
fSchContext |= CONTEXT_FLAG_MUTUAL_AUTH;
fSchContext &= ~CONTEXT_FLAG_MANUAL_CRED_VALIDATION;
if ( fContextReq & (ISC_REQ_MANUAL_CRED_VALIDATION))
{
// These flags are mutually exclusive
return SP_LOG_RESULT( SEC_E_UNSUPPORTED_FUNCTION );
}
}
else
{
if ( fContextReq & (ISC_REQ_MANUAL_CRED_VALIDATION))
{
fAttr |= ISC_RET_MANUAL_CRED_VALIDATION;
fSchContext |= CONTEXT_FLAG_MANUAL_CRED_VALIDATION;
}
// Turn off automatic credential validation if so specified in registry.
if(g_fManualCredValidation)
{
fSchContext |= CONTEXT_FLAG_MANUAL_CRED_VALIDATION;
}
}
if ( fContextReq & (ISC_REQ_USE_SUPPLIED_CREDS))
{
fAttr |= ISC_REQ_USE_SUPPLIED_CREDS;
fSchContext |= CONTEXT_FLAG_NO_INCOMPLETE_CRED_MSG;
}
if( pfContextAttr )
{
*pfContextAttr = fAttr;
}
if ( dwCtxtHandle == 0 )
{
pContext = SPContextCreate( pszTargetName->Buffer );
if(pContext == NULL)
{
TRACE_EXIT( SpInitLsaModeContext, SEC_E_INSUFFICIENT_MEMORY );
return SP_LOG_RESULT( SEC_E_INSUFFICIENT_MEMORY );
}
pContext->Flags |= fSchContext;
pCred = (PSPCredentialGroup) dwCredHandle ;
if(pctRet == PCT_ERR_OK)
{
pctRet = SPContextSetCredentials(pContext, pCred);
}
if(pctRet != PCT_ERR_OK)
{
SPContextDelete(pContext);
*pdwNewContext = 0;
TRACE_EXIT( SpInitLsaModeContext, PctTranslateError( pctRet ));
return SP_LOG_RESULT( PctTranslateError(pctRet) );
}
pctRet = pContext->InitiateHello( pContext, &CommOut, TRUE);
if ( (CommOut.cbBuffer == 0) &&
(pctRet == PCT_INT_BUFF_TOO_SMALL) )
{
SPContextDelete(pContext);
TRACE_EXIT( SpInitLsaModeContext, SEC_E_INSUFFICIENT_MEMORY );
return SP_LOG_RESULT( SEC_E_INSUFFICIENT_MEMORY );
}
if ( pctRet != PCT_ERR_OK )
{
SPContextDelete(pContext);
*pdwNewContext = 0;
TRACE_EXIT( SpInitLsaModeContext, PctTranslateError( pctRet) );
return SP_LOG_RESULT( PctTranslateError(pctRet) );
}
if ( fContextReq & ISC_REQ_ALLOCATE_MEMORY )
{
//
// Easy: The caller asked for us to allocate memory for them, so
// let the LSA do it.
//
pOutToken->pvBuffer = CommOut.pvBuffer ;
}
else
{
//
// The caller has a buffer that we're supposed to use. Make sure we
// can fit.
//
if ( (ULONG) CommOut.cbBuffer <= pOutToken->cbBuffer )
{
RtlCopyMemory( pOutToken->pvBuffer,
CommOut.pvBuffer,
CommOut.cbBuffer );
}
else
{
DebugLog(( DEB_TRACE, "Supplied buffer is too small\n" ));
SPContextDelete( pContext );
TRACE_EXIT( SpInitLsaModeContext, SEC_E_INSUFFICIENT_MEMORY );
return SP_LOG_RESULT( SEC_E_INSUFFICIENT_MEMORY );
}
}
*pdwNewContext = (LSA_SEC_HANDLE) pContext ;
pOutToken->pvBuffer = CommOut.pvBuffer;
pOutToken->cbBuffer = CommOut.cbData;
#if DBG
DebugLog((
DEB_TRACE,
"Output: type:0x%8.8x, pv:0x%8.8x, cb:0x%x\n",
pOutToken->BufferType,
pOutToken->pvBuffer,
pOutToken->cbBuffer));
if(pOutToken->pvBuffer)
{
DBG_HEX_STRING(DEB_BUFFERS, pOutToken->pvBuffer, pOutToken->cbBuffer);
}
#endif
}
else
{
/* Initialize input buffer locations */
for (i = 0; i < (int)pInput->cBuffers; i++ )
{
switch( (pInput->pBuffers[i].BufferType & (~SECBUFFER_ATTRMASK)) )
{
case SECBUFFER_TOKEN:
case SECBUFFER_TOKEN | SECBUFFER_READONLY:
pInToken = &pInput->pBuffers[i];
break;
case SECBUFFER_EMPTY:
if(!pInToken)
{
pInToken = &pInput->pBuffers[i];
}
else if(!pExtra)
{
pExtra = &pInput->pBuffers[i];
}
break;
case SECBUFFER_DATA:
case SECBUFFER_STREAM_HEADER:
case SECBUFFER_STREAM_TRAILER:
default:
break;
}
}
#if DBG
if(pInToken)
{
DebugLog((
DEB_TRACE,
"Input: type:0x%8.8x, pv:0x%8.8x, cb:0x%x\n",
pInToken->BufferType,
pInToken->pvBuffer,
pInToken->cbBuffer));
if(pInToken->pvBuffer)
{
DBG_HEX_STRING(DEB_BUFFERS, pInToken->pvBuffer, pInToken->cbBuffer);
}
}
if(pExtra)
{
DebugLog((
DEB_TRACE,
"Extra: type:0x%8.8x, pv:0x%8.8x, cb:0x%x\n",
pExtra->BufferType,
pExtra->pvBuffer,
pExtra->cbBuffer));
}
#endif
if(pInToken == NULL)
{
CommIn.pvBuffer = NULL;
CommIn.cbBuffer = 0;
CommIn.cbData = 0;
}
else
{
CommIn.pvBuffer = pInToken->pvBuffer;
CommIn.cbBuffer = pInToken->cbBuffer;
CommIn.cbData = pInToken->cbBuffer;
}
pContext = (PSPContext) dwCtxtHandle ;
if ( dwCredHandle )
{
pCred = (PSPCredentialGroup) dwCredHandle ;
}
if( pContext == NULL || pCred == NULL )
{
TRACE_EXIT( SpInitLsaModeContext, SEC_E_INVALID_HANDLE );
return SP_LOG_RESULT( SEC_E_INVALID_HANDLE );
}
pContext->Flags |= fSchContext;
pctRet = SPContextSetCredentials(pContext, pCred);
if(pctRet == PCT_ERR_OK)
{
// HACKHACK - adjust SSL3/TLS1 state
if(pContext->State == SSL3_STATE_RENEGOTIATE)
{
pContext->State = SSL3_STATE_GEN_HELLO_REQUEST;
}
pctRet = pContext->ProtocolHandler( pContext,
&CommIn,
&CommOut);
}
if(pctRet == PCT_INT_INCOMPLETE_MSG)
{
if(pExtra)
{
pExtra->BufferType = SECBUFFER_MISSING | SECBUFFER_UNMAPPED ;
pExtra->cbBuffer = CommIn.cbData - pInToken->cbBuffer;
pExtra->pvBuffer = NULL ;
}
}
else
{
pOutToken->pvBuffer = CommOut.pvBuffer;
pOutToken->cbBuffer = CommOut.cbData;
}
if(pctRet == PCT_INT_BUFF_TOO_SMALL)
{
pOutToken->BufferType |= SECBUFFER_UNMAPPED;
}
#if DBG
if(pOutToken)
{
DebugLog((
DEB_TRACE,
"Output: type:0x%8.8x, pv:0x%8.8x, cb:0x%x\n",
pOutToken->BufferType,
pOutToken->pvBuffer,
pOutToken->cbBuffer));
if(pOutToken->pvBuffer)
{
DBG_HEX_STRING(DEB_BUFFERS, pOutToken->pvBuffer, pOutToken->cbBuffer);
}
}
if(pExtra)
{
DebugLog((
DEB_TRACE,
"Extra: type:0x%8.8x, pv:0x%8.8x, cb:0x%x\n",
pExtra->BufferType,
pExtra->pvBuffer,
pExtra->cbBuffer));
}
#endif
if(!SP_FATAL(pctRet))
{
*pdwNewContext = dwCtxtHandle ;
}
if(PCT_ERR_OK != pctRet)
{
TRACE_EXIT( SpInitLsaModeContext, PctTranslateError( pctRet ));
return SP_LOG_RESULT( PctTranslateError(pctRet) );
}
if(pInToken)
{
if(CommIn.cbData < pInToken->cbBuffer && pExtra)
{
pExtra->BufferType = SECBUFFER_EXTRA | SECBUFFER_UNMAPPED ;
pExtra->cbBuffer = pInToken->cbBuffer - CommIn.cbData;
pExtra->pvBuffer = NULL ;
}
}
}
if ( (pContext->State == SP_STATE_CONNECTED) &&
( (pContext->Flags & CONTEXT_FLAG_MAPPED) == 0 ) )
{
//
// Need to map the context back down to the user process. It
// doesn't get any scarier than this:
//
*pfMapContext = TRUE ;
DebugOut(( DEB_TRACE, "Mapping context to usermode\n" ));
pctRet = SPContextSerialize(pContext,
SslRelocateToken,
(PUCHAR *) &pContextData->pvBuffer,
&pContextData->cbBuffer,
TRUE);
if(PCT_ERR_OK != pctRet)
{
return SP_LOG_RESULT( PctTranslateError(pctRet) );
}
pContext->Flags |= CONTEXT_FLAG_MAPPED ;
LogHandshakeInfoEvent(pContext->RipeZombie->fProtocol,
pContext->pCipherInfo,
pContext->pHashInfo,
pContext->pKeyExchInfo,
pContext->RipeZombie->dwExchStrength);
}
if(ptsExpiry != NULL)
{
if(pContext->RipeZombie->pRemoteCert != NULL)
{
ptsExpiry->QuadPart = *((LONGLONG *)&pContext->RipeZombie->pRemoteCert->pCertInfo->NotAfter);
}
else
{
ptsExpiry->QuadPart = MAXTIMEQUADPART;
}
}
if(pContext->State == SP_STATE_CONNECTED ||
pContext->State == SP_STATE_SHUTDOWN)
{
return SEC_E_OK;
}
else
{
return SEC_I_CONTINUE_NEEDED;
}
}
SECURITY_STATUS SEC_ENTRY
SpMoveContextToUser( ULONG dwCtxtHandle,
PSecBuffer pContextBuffer)
{
return(SEC_E_UNSUPPORTED_FUNCTION);
}
SECURITY_STATUS
SEC_ENTRY
SpDeleteContext(
LSA_SEC_HANDLE dwCtxtHandle)
{
PSPContext pContext ;
DebugLog((DEB_TRACE, "SpDeleteContext(0x%x)\n", dwCtxtHandle));
pContext = (PSPContext) dwCtxtHandle ;
SPContextDelete( pContext );
return( SEC_E_OK );
}
SECURITY_STATUS
SEC_ENTRY
SpApplyControlToken(
LSA_SEC_HANDLE dwCtxtHandle,
PSecBufferDesc pInput)
{
PSPContext pContext;
PSecBuffer Buffer ;
DWORD cbState;
DebugLog((DEB_TRACE, "SpApplyControlToken(0x%x, 0x%x)\n", dwCtxtHandle, pInput));
pContext = (PSPContext) dwCtxtHandle ;
if ( pInput->cBuffers != 1 )
{
return SP_LOG_RESULT(SEC_E_INVALID_TOKEN);
}
Buffer = pInput->pBuffers ;
if(Buffer->cbBuffer < sizeof(DWORD))
{
return SP_LOG_RESULT(SEC_E_UNSUPPORTED_FUNCTION);
}
switch(*(DWORD *)Buffer->pvBuffer)
{
case SCHANNEL_RENEGOTIATE:
{
PDWORD RedoData;
PBYTE pbReadKey;
DWORD cbReadKey;
DebugLog((DEB_TRACE, "SCHANNEL_RENEGOTIATE\n"));
if(Buffer->cbBuffer < sizeof(DWORD) * 2)
{
return SP_LOG_RESULT(SEC_E_INVALID_TOKEN);
}
RedoData = (DWORD *)Buffer->pvBuffer;
if(RedoData[1] != SSL3_STATE_RENEGOTIATE &&
RedoData[1] != PCT1_STATE_RENEGOTIATE)
{
return SP_LOG_RESULT(SEC_E_UNSUPPORTED_FUNCTION);
}
pContext->State = RedoData[1];
return SEC_E_OK;
}
case SCHANNEL_SHUTDOWN:
DebugLog((DEB_TRACE, "SCHANNEL_SHUTDOWN\n"));
pContext->State = SP_STATE_SHUTDOWN_PENDING;
return SEC_E_OK;
case SCHANNEL_ALERT:
{
SCHANNEL_ALERT_TOKEN *pAlertToken;
DebugLog((DEB_TRACE, "SCHANNEL_TLS1_ALERT\n"));
if(Buffer->cbBuffer < sizeof(SCHANNEL_ALERT_TOKEN))
{
return SP_LOG_RESULT(SEC_E_INVALID_TOKEN);
}
pAlertToken = (SCHANNEL_ALERT_TOKEN *)Buffer->pvBuffer;
// Alerts are only supported in SSL3 and TLS1
if(!(pContext->RipeZombie->fProtocol & SP_PROT_SSL3TLS1))
{
return SP_LOG_RESULT(SEC_E_UNSUPPORTED_FUNCTION);
}
// Reality check alert values.
if(pAlertToken->dwAlertType >= 0x100 ||
pAlertToken->dwAlertNumber >= 0x100)
{
return SP_LOG_RESULT(SEC_E_INVALID_TOKEN);
}
SetTls1Alert(pContext,
(BYTE)pAlertToken->dwAlertType,
(BYTE)pAlertToken->dwAlertNumber);
#if DBG
DebugLog((DEB_TRACE,
"AlertLevel:0x%x, AlertNumber:0x%x\n",
pAlertToken->dwAlertType,
pAlertToken->dwAlertNumber));
#endif
return SEC_E_OK;
}
case SCHANNEL_SESSION:
{
SCHANNEL_SESSION_TOKEN *pSessionToken;
SECURITY_STATUS Status = SEC_E_UNSUPPORTED_FUNCTION;
DebugLog((DEB_TRACE, "SCHANNEL_SESSION\n"));
if(pContext->RipeZombie == NULL)
{
return SP_LOG_RESULT(SEC_E_INVALID_HANDLE);
}
if(Buffer->cbBuffer < sizeof(SCHANNEL_SESSION_TOKEN))
{
return SP_LOG_RESULT(SEC_E_INVALID_TOKEN);
}
pSessionToken = (SCHANNEL_SESSION_TOKEN *)Buffer->pvBuffer;
if(pSessionToken->dwFlags & SSL_SESSION_DISABLE_RECONNECTS)
{
// Disable reconnects
pContext->RipeZombie->ZombieJuju = FALSE;
Status = SEC_E_OK;
}
if(pSessionToken->dwFlags & SSL_SESSION_ENABLE_RECONNECTS)
{
// Enable reconnects
if(pContext->RipeZombie->DeferredJuju)
{
pContext->RipeZombie->ZombieJuju = TRUE;
pContext->RipeZombie->DeferredJuju = FALSE;
Status = SEC_E_OK;
}
}
return Status;
}
default:
return SP_LOG_RESULT(SEC_E_UNSUPPORTED_FUNCTION);
}
}
VOID
SEC_ENTRY
SpLogonTerminated(PLUID pLogonId)
{
return;
}
SECURITY_STATUS SEC_ENTRY
SpAcceptLsaModeContext(
LSA_SEC_HANDLE dwCredHandle,
LSA_SEC_HANDLE dwCtxtHandle,
PSecBufferDesc pInput,
ULONG fContextReq,
ULONG TargetDataRep,
PLSA_SEC_HANDLE pdwNewContext,
PSecBufferDesc pOutput,
PULONG pfContextAttr,
PTimeStamp ptsExpiry,
PBYTE pfMapContext,
PSecBuffer pContextData)
{
PSPContext pContext = NULL;
PSPCredentialGroup pCred = NULL;
SPBuffer CommOut;
SPBuffer CommIn;
PSecBuffer pInToken = NULL;
PSecBuffer pOutToken = NULL;
PSecBuffer pExtra = NULL;
unsigned long fAttr = ASC_RET_REPLAY_DETECT |
ASC_RET_SEQUENCE_DETECT |
ASC_RET_CONFIDENTIALITY |
ASC_RET_STREAM;
DWORD fSchContext = 0;
int i;
SP_STATUS pctRet = PCT_ERR_OK;
TRACE_ENTER( SpAcceptLsaModeContext );
#if DBG
DebugLog((DEB_TRACE, "SpAcceptLsaModeContext(0x%x, 0x%x, 0x%x, 0x%x, 0x%x, 0x%x, 0x%x, 0x%x, 0x%x, 0x%x, 0x%x)\n",
dwCredHandle, dwCtxtHandle, pInput, fContextReq, TargetDataRep, pdwNewContext,
pOutput, pfContextAttr, ptsExpiry, pfMapContext, pContextData));
#endif
// This flag is never allowed
if(fContextReq & ASC_REQ_DELEGATE)
{
TRACE_EXIT( SpAcceptLsaModeContext, SEC_E_UNSUPPORTED_FUNCTION );
return SEC_E_UNSUPPORTED_FUNCTION;
}
if ( fContextReq & ASC_REQ_MUTUAL_AUTH )
{
fSchContext |= CONTEXT_FLAG_MUTUAL_AUTH;
}
/* Initialize input buffer locations */
for (i = 0; i < (int)pInput->cBuffers; i++ )
{
switch( pInput->pBuffers[i].BufferType & (~(SECBUFFER_ATTRMASK)) )
{
case SECBUFFER_TOKEN:
case SECBUFFER_TOKEN | SECBUFFER_READONLY:
pInToken = &pInput->pBuffers[i];
break;
case SECBUFFER_EMPTY:
if(!pInToken)
{
pInToken = &pInput->pBuffers[i];
}
else if(!pExtra)
{
pExtra = &pInput->pBuffers[i];
}
break;
case SECBUFFER_DATA:
case SECBUFFER_STREAM_HEADER:
case SECBUFFER_STREAM_TRAILER:
default:
break;
}
}
#if DBG
if(pInToken)
{
DebugLog((
DEB_TRACE,
"Input: type:0x%8.8x, pv:0x%8.8x, cb:0x%x\n",
pInToken->BufferType,
pInToken->pvBuffer,
pInToken->cbBuffer));
if(pInToken->pvBuffer)
{
DBG_HEX_STRING(DEB_BUFFERS, pInToken->pvBuffer, pInToken->cbBuffer);
}
}
if(pExtra)
{
DebugLog((
DEB_TRACE,
"Extra: type:0x%8.8x, pv:0x%8.8x, cb:0x%x\n",
pExtra->BufferType,
pExtra->pvBuffer,
pExtra->cbBuffer));
}
#endif
/* Initialize output buffer locations */
for (i = 0; i < (int) pOutput->cBuffers; i++ )
{
switch( pOutput->pBuffers[i].BufferType & (~(SECBUFFER_ATTRMASK)) )
{
case SECBUFFER_EMPTY:
if(!pOutToken && (fContextReq & ASC_REQ_ALLOCATE_MEMORY))
pOutToken = &pOutput->pBuffers[i];
break;
case SECBUFFER_TOKEN:
pOutToken = &pOutput->pBuffers[i];
break;
case SECBUFFER_DATA:
case SECBUFFER_STREAM_HEADER:
case SECBUFFER_STREAM_TRAILER:
default:
break;
}
}
if(pOutToken == NULL)
{
TRACE_EXIT( SpAcceptLsaModeContext, SEC_E_INVALID_TOKEN );
return SEC_E_INVALID_TOKEN;
}
if ( !pExtra )
{
DebugOut(( DEB_TRACE, " Warning - no Empty security buffer\n"));
}
pOutToken->BufferType = SECBUFFER_TOKEN;
if(pInToken && pInToken->BufferType == SECBUFFER_TOKEN)
{
CommIn.pvBuffer = pInToken->pvBuffer;
CommIn.cbBuffer = pInToken->cbBuffer;
CommIn.cbData = pInToken->cbBuffer;
}
else
{
CommIn.pvBuffer = NULL;
CommIn.cbBuffer = 0;
CommIn.cbData = 0;
}
if (fContextReq & ASC_REQ_ALLOCATE_MEMORY)
{
fAttr |= ASC_RET_ALLOCATED_MEMORY;
pOutToken->pvBuffer = NULL;
pOutToken->cbBuffer = 0;
}
CommOut.pvBuffer = pOutToken->pvBuffer;
CommOut.cbBuffer = pOutToken->cbBuffer;
CommOut.cbData = 0;
if (fContextReq & (ASC_REQ_EXTENDED_ERROR))
{
fAttr |= ASC_RET_EXTENDED_ERROR;
fSchContext |= CONTEXT_FLAG_EXT_ERR;
}
if ( fContextReq & (ASC_REQ_CONNECTION) )
{
fAttr |= ASC_RET_CONNECTION;
fSchContext |= CONTEXT_FLAG_CONNECTION_MODE;
}
if (pfContextAttr)
{
*pfContextAttr = fAttr;
}
if ( dwCtxtHandle == 0 )
{
pContext = SPContextCreate(NULL);
if (pContext == NULL)
{
TRACE_EXIT( SpAcceptLsaModeContext, SEC_E_INSUFFICIENT_MEMORY );
return SEC_E_INSUFFICIENT_MEMORY;
}
}
else
{
pContext = (PSPContext) dwCtxtHandle ;
}
if ( dwCredHandle == 0 )
{
pCred = NULL ;
}
else
{
pCred = (PSPCredentialGroup) dwCredHandle ;
}
if ( (pContext == NULL) || (pCred == NULL) )
{
if ( dwCtxtHandle == 0 )
{
SPContextDelete( pContext );
}
TRACE_EXIT( SpAcceptLsaModeContext, SEC_E_INVALID_HANDLE );
return( SEC_E_INVALID_HANDLE );
}
pctRet = SPContextSetCredentials(pContext, pCred);
pContext->Flags |= fSchContext;
if ( pctRet == PCT_ERR_OK )
{
pctRet = pContext->ProtocolHandler( pContext, &CommIn, &CommOut);
}
if ( dwCtxtHandle == 0 )
{
if ( pctRet != PCT_ERR_OK )
{
SPContextDelete( pContext );
}
else
{
*pdwNewContext = (LSA_SEC_HANDLE) pContext ;
}
}
else
{
*pdwNewContext = (LSA_SEC_HANDLE) pContext ;
}
if (CommOut.cbData == 0 && pctRet == PCT_INT_BUFF_TOO_SMALL)
{
TRACE_EXIT( SpAcceptLsaModeContext, SEC_E_INSUFFICIENT_MEMORY );
return SEC_E_INSUFFICIENT_MEMORY;
}
if (pctRet == PCT_INT_INCOMPLETE_MSG)
{
if(pExtra)
{
pExtra->BufferType = SECBUFFER_MISSING | SECBUFFER_UNMAPPED ;
pExtra->cbBuffer = CommIn.cbData - pInToken->cbBuffer;
pExtra->pvBuffer = NULL ;
DebugOut(( DEB_TRACE, "Incomplete message, needs %d more bytes\n",
pExtra->cbBuffer ));
}
else
{
DebugOut(( DEB_TRACE, "No Empty buffer for returning missing info!\n" ));
}
}
else
{
pOutToken->pvBuffer = CommOut.pvBuffer;
pOutToken->cbBuffer = CommOut.cbData;
}
if(pctRet == PCT_INT_BUFF_TOO_SMALL)
{
pOutToken->BufferType |= SECBUFFER_UNMAPPED;
}
if(pOutToken->cbBuffer == 0)
{
// Don't return an output token if the output buffer is
// empty. Also, make sure that the extended error flag is
// turned off.
pOutToken->BufferType = SECBUFFER_EMPTY;
if(pfContextAttr)
{
*pfContextAttr &= ~ASC_RET_EXTENDED_ERROR;
}
}
if (PCT_ERR_OK != pctRet)
{
TRACE_EXIT( SpAcceptLsaModeContext, PctTranslateError( pctRet ) );
return PctTranslateError(pctRet);
}
if(pInToken)
{
if (CommIn.cbData < pInToken->cbBuffer && pExtra)
{
pExtra->BufferType = SECBUFFER_EXTRA | SECBUFFER_UNMAPPED ;
pExtra->cbBuffer = pInToken->cbBuffer - CommIn.cbData;
pExtra->pvBuffer = NULL ;
DebugOut(( DEB_TRACE, "Extra data, needs to be mapped back: %d\n", pExtra->cbBuffer ));
}
}
if ( (pContext->State == SP_STATE_CONNECTED) &&
( (pContext->Flags & CONTEXT_FLAG_MAPPED) == 0 ) )
{
//
// Need to map the context back down to the user process. It
// doesn't get any scarier than this:
//
*pfMapContext = TRUE ;
DebugOut(( DEB_TRACE, "Mapping context to usermode\n" ));
pctRet = SPContextSerialize(pContext,
SslRelocateToken,
(PUCHAR *) &pContextData->pvBuffer,
&pContextData->cbBuffer,
TRUE);
if (PCT_ERR_OK != pctRet)
{
TRACE_EXIT( SpAcceptLsaModeContext, PctTranslateError( pctRet ) );
return PctTranslateError(pctRet);
}
pContext->Flags |= CONTEXT_FLAG_MAPPED ;
LogHandshakeInfoEvent(pContext->RipeZombie->fProtocol,
pContext->pCipherInfo,
pContext->pHashInfo,
pContext->pKeyExchInfo,
pContext->RipeZombie->dwExchStrength);
}
#if DBG
if(pOutToken)
{
DebugLog((
DEB_TRACE,
"Output: type:0x%8.8x, pv:0x%8.8x, cb:0x%x\n",
pOutToken->BufferType,
pOutToken->pvBuffer,
pOutToken->cbBuffer));
if(pOutToken->pvBuffer)
{
DBG_HEX_STRING(DEB_BUFFERS, pOutToken->pvBuffer, pOutToken->cbBuffer);
}
}
if(pExtra)
{
DebugLog((
DEB_TRACE,
"Extra: type:0x%8.8x, pv:0x%8.8x, cb:0x%x\n",
pExtra->BufferType,
pExtra->pvBuffer,
pExtra->cbBuffer));
}
if(pContext->State == SP_STATE_CONNECTED)
{
DebugLog((DEB_TRACE, "Server handshake complete\n"));
}
#endif
if(ptsExpiry != NULL)
{
if(pContext->RipeZombie->pRemoteCert != NULL)
{
ptsExpiry->QuadPart = *((LONGLONG *)&pContext->RipeZombie->pRemoteCert->pCertInfo->NotAfter);
}
else
{
ptsExpiry->QuadPart = MAXTIMEQUADPART;
}
}
if(pContext->State == SP_STATE_CONNECTED &&
pContext->RipeZombie->hLocator)
{
// Certificate mapping was successful.
*pfContextAttr |= ASC_RET_MUTUAL_AUTH;
}
if(pContext->State == SP_STATE_CONNECTED ||
pContext->State == SP_STATE_SHUTDOWN)
{
return SEC_E_OK;
}
else
{
return SEC_I_CONTINUE_NEEDED;
}
}
NTSTATUS
SpCallPackage(
IN PLSA_CLIENT_REQUEST ClientRequest,
IN PVOID ProtocolSubmitBuffer,
IN PVOID ClientBufferBase,
IN ULONG SubmitBufferLength,
OUT PVOID *ProtocolReturnBuffer,
OUT PULONG ReturnBufferLength,
OUT PNTSTATUS ProtocolStatus
)
{
PULONG Request ;
if ( !ProtocolSubmitBuffer )
{
return SEC_E_UNSUPPORTED_FUNCTION ;
}
if(!SchannelInit(FALSE))
{
return SP_LOG_RESULT(SEC_E_UNSUPPORTED_FUNCTION);
}
Request = (PULONG) ProtocolSubmitBuffer ;
if ( *Request == SSL_LOOKUP_CERT_MESSAGE )
{
return SslDoClientRequest(
ClientRequest,
ProtocolSubmitBuffer,
ClientBufferBase,
SubmitBufferLength,
ProtocolReturnBuffer,
ReturnBufferLength,
ProtocolStatus );
}
else if ( *Request == SSL_LOOKUP_EXTERNAL_CERT_MESSAGE &&
ClientBufferBase == ProtocolSubmitBuffer)
{
// This function is only allowed to be called from the
// lsass.exe process.
return SslMapExternalCredential(
ClientRequest,
ProtocolSubmitBuffer,
ClientBufferBase,
SubmitBufferLength,
ProtocolReturnBuffer,
ReturnBufferLength,
ProtocolStatus );
}
else if( *Request == SSL_CACHE_INFO_MESSAGE )
{
return SslSessionCacheInfo(
ClientRequest,
ProtocolSubmitBuffer,
ClientBufferBase,
SubmitBufferLength,
ProtocolReturnBuffer,
ReturnBufferLength,
ProtocolStatus );
}
else if( *Request == SSL_PURGE_CACHE_MESSAGE )
{
return SslPurgeSessionCache(
ClientRequest,
ProtocolSubmitBuffer,
ClientBufferBase,
SubmitBufferLength,
ProtocolReturnBuffer,
ReturnBufferLength,
ProtocolStatus );
}
return( SEC_E_UNSUPPORTED_FUNCTION );
}
NTSTATUS
SpCallPackageUntrusted(
IN PLSA_CLIENT_REQUEST ClientRequest,
IN PVOID ProtocolSubmitBuffer,
IN PVOID ClientBufferBase,
IN ULONG SubmitBufferLength,
OUT PVOID *ProtocolReturnBuffer,
OUT PULONG ReturnBufferLength,
OUT PNTSTATUS ProtocolStatus
)
{
ULONG MessageType;
//
// Get the messsage type from the protocol submit buffer.
//
if(SubmitBufferLength < sizeof(ULONG))
{
return STATUS_INVALID_PARAMETER;
}
MessageType = *((ULONG *)(ProtocolSubmitBuffer));
if(!SchannelInit(FALSE))
{
return SP_LOG_RESULT(SEC_E_UNSUPPORTED_FUNCTION);
}
//
// Allow the dispatch routines to only set the return buffer information
// on success conditions.
//
*ProtocolReturnBuffer = NULL;
*ReturnBufferLength = 0;
//
// Process message as appropriate.
//
switch(MessageType)
{
case SSL_PURGE_CACHE_MESSAGE:
return SslPurgeSessionCache(
ClientRequest,
ProtocolSubmitBuffer,
ClientBufferBase,
SubmitBufferLength,
ProtocolReturnBuffer,
ReturnBufferLength,
ProtocolStatus);
case SSL_CACHE_INFO_MESSAGE:
return SslSessionCacheInfo(
ClientRequest,
ProtocolSubmitBuffer,
ClientBufferBase,
SubmitBufferLength,
ProtocolReturnBuffer,
ReturnBufferLength,
ProtocolStatus);
case SSL_PERFMON_INFO_MESSAGE:
return SslGetPerfmonInfo(
ClientRequest,
ProtocolSubmitBuffer,
ClientBufferBase,
SubmitBufferLength,
ProtocolReturnBuffer,
ReturnBufferLength,
ProtocolStatus);
default:
return( SEC_E_UNSUPPORTED_FUNCTION );
}
}
NTSTATUS NTAPI
SslPurgeSessionCache(
IN PLSA_CLIENT_REQUEST ClientRequest,
IN PVOID ProtocolSubmitBuffer,
IN PVOID ClientBufferBase,
IN ULONG SubmitBufferSize,
OUT PVOID *ProtocolReturnBuffer,
OUT PULONG ReturnBufferLength,
OUT PNTSTATUS ProtocolStatus)
{
NTSTATUS Status;
SECPKG_CALL_INFO CallInfo;
SECPKG_CLIENT_INFO ClientInfo;
PLUID LogonId;
SSL_PURGE_SESSION_CACHE_REQUEST PurgeRequest;
PSSL_PURGE_SESSION_CACHE_REQUEST pPurgeRequest;
//
// Verify the request.
//
DebugLog((DEB_TRACE, "Purging session cache\n"));
if(!LsaTable->GetCallInfo(&CallInfo))
{
Status = STATUS_INTERNAL_ERROR;
goto Cleanup;
}
if(CallInfo.Attributes & SECPKG_CALL_WOWCLIENT)
{
PSSL_PURGE_SESSION_CACHE_REQUEST_WOW64 pRequest;
if(SubmitBufferSize < sizeof(SSL_PURGE_SESSION_CACHE_REQUEST_WOW64))
{
Status = STATUS_INVALID_PARAMETER;
goto Cleanup;
}
pRequest = (PSSL_PURGE_SESSION_CACHE_REQUEST_WOW64)ProtocolSubmitBuffer;
memset(&PurgeRequest, 0, sizeof(PurgeRequest));
PurgeRequest.MessageType = pRequest->MessageType;
PurgeRequest.LogonId = pRequest->LogonId;
PurgeRequest.Flags = pRequest->Flags;
PurgeRequest.ServerName.Length = pRequest->ServerName.Length;
PurgeRequest.ServerName.MaximumLength = pRequest->ServerName.MaximumLength;
PurgeRequest.ServerName.Buffer = (PVOID) UlongToPtr(pRequest->ServerName.Buffer);
pPurgeRequest = &PurgeRequest;
}
else
{
if (SubmitBufferSize < sizeof(SSL_PURGE_SESSION_CACHE_REQUEST))
{
Status = STATUS_INVALID_PARAMETER;
goto Cleanup;
}
pPurgeRequest = (PSSL_PURGE_SESSION_CACHE_REQUEST) ProtocolSubmitBuffer;
}
//
// Normalize the strings
//
NULL_RELOCATE_ONE(&pPurgeRequest->ServerName);
//
// Find the callers logon id & TCB status
//
Status = LsaTable->GetClientInfo(&ClientInfo);
if (!NT_SUCCESS(Status))
{
goto Cleanup;
}
//
// Verify the caller has TCB privilege if they want to purge someone
// else's session cache entries.
//
if(!RtlIsZeroLuid(&pPurgeRequest->LogonId) ||
(pPurgeRequest->Flags & SSL_PURGE_CLIENT_ALL_ENTRIES) ||
(pPurgeRequest->Flags & SSL_PURGE_SERVER_ALL_ENTRIES))
{
if(!ClientInfo.HasTcbPrivilege)
{
Status = STATUS_PRIVILEGE_NOT_HELD;
goto Cleanup;
}
}
//
// If the caller did not provide a logon id, use the caller's logon id.
//
if(RtlIsZeroLuid(&pPurgeRequest->LogonId))
{
LogonId = &ClientInfo.LogonId;
}
else
{
LogonId = &pPurgeRequest->LogonId;
}
//
// Purge the requested cache entries.
//
Status = SPCachePurgeEntries(LogonId,
ClientInfo.ProcessID,
pPurgeRequest->ServerName.Buffer,
pPurgeRequest->Flags);
*ProtocolReturnBuffer = NULL;
*ReturnBufferLength = 0;
Cleanup:
*ProtocolStatus = Status;
return(STATUS_SUCCESS);
}
NTSTATUS NTAPI
SslSessionCacheInfo(
IN PLSA_CLIENT_REQUEST ClientRequest,
IN PVOID ProtocolSubmitBuffer,
IN PVOID ClientBufferBase,
IN ULONG SubmitBufferSize,
OUT PVOID *ProtocolReturnBuffer,
OUT PULONG ReturnBufferLength,
OUT PNTSTATUS ProtocolStatus)
{
NTSTATUS Status;
SECPKG_CALL_INFO CallInfo;
SECPKG_CLIENT_INFO ClientInfo;
PLUID LogonId;
SSL_SESSION_CACHE_INFO_REQUEST InfoRequest;
PSSL_SESSION_CACHE_INFO_REQUEST pInfoRequest;
PSSL_SESSION_CACHE_INFO_RESPONSE pInfoResponse = NULL;
DWORD cbInfoResponse;
PVOID pvClient;
*ProtocolReturnBuffer = NULL;
*ReturnBufferLength = 0;
//
// Verify the request.
//
if(!LsaTable->GetCallInfo(&CallInfo))
{
Status = STATUS_INTERNAL_ERROR;
goto Cleanup;
}
if(CallInfo.Attributes & SECPKG_CALL_WOWCLIENT)
{
PSSL_SESSION_CACHE_INFO_REQUEST_WOW64 pRequest;
if(SubmitBufferSize < sizeof(SSL_SESSION_CACHE_INFO_REQUEST_WOW64))
{
Status = STATUS_INVALID_PARAMETER;
goto Cleanup;
}
pRequest = (PSSL_SESSION_CACHE_INFO_REQUEST_WOW64)ProtocolSubmitBuffer;
memset(&InfoRequest, 0, sizeof(InfoRequest));
InfoRequest.MessageType = pRequest->MessageType;
InfoRequest.LogonId = pRequest->LogonId;
InfoRequest.Flags = pRequest->Flags;
InfoRequest.ServerName.Length = pRequest->ServerName.Length;
InfoRequest.ServerName.MaximumLength = pRequest->ServerName.MaximumLength;
InfoRequest.ServerName.Buffer = (PVOID) UlongToPtr(pRequest->ServerName.Buffer);
pInfoRequest = &InfoRequest;
}
else
{
if (SubmitBufferSize < sizeof(SSL_SESSION_CACHE_INFO_REQUEST))
{
Status = STATUS_INVALID_PARAMETER;
goto Cleanup;
}
pInfoRequest = (PSSL_SESSION_CACHE_INFO_REQUEST)ProtocolSubmitBuffer;
}
//
// Normalize the strings
//
NULL_RELOCATE_ONE(&pInfoRequest->ServerName);
//
// Find the callers logon id & TCB status
//
Status = LsaTable->GetClientInfo(&ClientInfo);
if (!NT_SUCCESS(Status))
{
goto Cleanup;
}
//
// If the caller did not provide a logon id, use the caller's logon id.
//
if ( RtlIsZeroLuid( &pInfoRequest->LogonId ) )
{
LogonId = &ClientInfo.LogonId;
}
else
{
//
// Verify the caller has TCB privilege if they want access to someone
// else's session cache.
//
if (!ClientInfo.HasTcbPrivilege)
{
Status = STATUS_PRIVILEGE_NOT_HELD;
goto Cleanup;
}
LogonId = &pInfoRequest->LogonId;
}
pInfoResponse = SPExternalAlloc(sizeof(SSL_SESSION_CACHE_INFO_RESPONSE));
if(pInfoResponse == NULL)
{
Status = STATUS_NO_MEMORY;
goto Cleanup;
}
Status = SPCacheGetInfo(LogonId,
pInfoRequest->ServerName.Buffer,
pInfoRequest->Flags,
pInfoResponse);
if(!NT_SUCCESS(Status))
{
goto Cleanup;
}
cbInfoResponse = sizeof(SSL_SESSION_CACHE_INFO_RESPONSE);
//
// Copy the response data to the client process.
//
Status = LsaTable->AllocateClientBuffer(
NULL,
cbInfoResponse,
&pvClient);
if(!NT_SUCCESS(Status))
{
goto Cleanup;
}
Status = LsaTable->CopyToClientBuffer(
NULL,
cbInfoResponse,
pvClient,
pInfoResponse);
if(!NT_SUCCESS(Status))
{
LsaTable->FreeClientBuffer(NULL, pvClient);
goto Cleanup;
}
*ProtocolReturnBuffer = pvClient;
*ReturnBufferLength = cbInfoResponse;
Cleanup:
if(pInfoResponse)
{
SPExternalFree(pInfoResponse);
}
*ProtocolStatus = Status;
return(STATUS_SUCCESS);
}
NTSTATUS NTAPI
SslGetPerfmonInfo(
IN PLSA_CLIENT_REQUEST ClientRequest,
IN PVOID ProtocolSubmitBuffer,
IN PVOID ClientBufferBase,
IN ULONG SubmitBufferSize,
OUT PVOID *ProtocolReturnBuffer,
OUT PULONG ReturnBufferLength,
OUT PNTSTATUS ProtocolStatus)
{
NTSTATUS Status;
PSSL_PERFMON_INFO_REQUEST pInfoRequest;
PSSL_PERFMON_INFO_RESPONSE pInfoResponse = NULL;
DWORD cbInfoResponse;
PVOID pvClient;
*ProtocolReturnBuffer = NULL;
*ReturnBufferLength = 0;
//
// Verify the request.
//
if (SubmitBufferSize < sizeof(SSL_PERFMON_INFO_REQUEST))
{
Status = STATUS_INVALID_PARAMETER;
goto Cleanup;
}
pInfoRequest = (PSSL_PERFMON_INFO_REQUEST)ProtocolSubmitBuffer;
pInfoResponse = SPExternalAlloc(sizeof(SSL_PERFMON_INFO_RESPONSE));
if(pInfoResponse == NULL)
{
Status = STATUS_NO_MEMORY;
goto Cleanup;
}
Status = SPCacheGetPerfmonInfo(pInfoRequest->Flags,
pInfoResponse);
if(!NT_SUCCESS(Status))
{
goto Cleanup;
}
cbInfoResponse = sizeof(SSL_PERFMON_INFO_RESPONSE);
//
// Copy the response data to the client process.
//
Status = LsaTable->AllocateClientBuffer(
NULL,
cbInfoResponse,
&pvClient);
if(!NT_SUCCESS(Status))
{
goto Cleanup;
}
Status = LsaTable->CopyToClientBuffer(
NULL,
cbInfoResponse,
pvClient,
pInfoResponse);
if(!NT_SUCCESS(Status))
{
LsaTable->FreeClientBuffer(NULL, pvClient);
goto Cleanup;
}
*ProtocolReturnBuffer = pvClient;
*ReturnBufferLength = cbInfoResponse;
Cleanup:
if(pInfoResponse)
{
SPExternalFree(pInfoResponse);
}
*ProtocolStatus = Status;
return(STATUS_SUCCESS);
}
NTSTATUS
SpCallPackagePassthrough(
IN PLSA_CLIENT_REQUEST ClientRequest,
IN PVOID ProtocolSubmitBuffer,
IN PVOID ClientBufferBase,
IN ULONG SubmitBufferLength,
OUT PVOID *ProtocolReturnBuffer,
OUT PULONG ReturnBufferLength,
OUT PNTSTATUS ProtocolStatus
)
{
//
// NOTE: if other sensitive request types are to be supported,
// this routine should filter them out prior to calling SpCallPackage.
// This is required because untrusted code has the opportunity for
// making genericpassthrough requests.
//
PULONG Request ;
if ( !ProtocolSubmitBuffer )
{
return SEC_E_UNSUPPORTED_FUNCTION ;
}
Request = (PULONG) ProtocolSubmitBuffer ;
if ( *Request != SSL_LOOKUP_CERT_MESSAGE )
return SEC_E_UNSUPPORTED_FUNCTION;
return SpCallPackage(
ClientRequest,
ProtocolSubmitBuffer,
ClientBufferBase,
SubmitBufferLength,
ProtocolReturnBuffer,
ReturnBufferLength,
ProtocolStatus
);
}
SECURITY_STATUS SEC_ENTRY
SpShutdown(void)
{
return(SEC_E_UNSUPPORTED_FUNCTION);
}
SECURITY_STATUS SEC_ENTRY
SpSystemLogon( PSECURITY_STRING pName,
DWORD cbKey,
PBYTE pbKey,
DWORD * pdwHandle,
PTimeStamp ptsExpiry)
{
return(SEC_E_UNSUPPORTED_FUNCTION);
}
SECURITY_STATUS SEC_ENTRY
SpGetUserInfo( PLUID pLogonId,
ULONG fFlags,
PSecurityUserData * ppUserInfo)
{
return(SEC_E_UNSUPPORTED_FUNCTION);
}
//+---------------------------------------------------------------------------
//
// Function: SpSaveCredentials
//
// Synopsis: Store credentials (not supported)
//
// Arguments: [dwCredHandle] --
// [CredType] --
// [pCredentials] --
//
//
// History: 7-26-96 RichardW Created
//
// Notes:
//
//----------------------------------------------------------------------------
SECURITY_STATUS SEC_ENTRY
SpSaveCredentials( LSA_SEC_HANDLE dwCredHandle,
PSecBuffer pCredentials)
{
return(SEC_E_UNSUPPORTED_FUNCTION);
}
//+---------------------------------------------------------------------------
//
// Function: SpGetCredentials
//
// Synopsis: Get Credentials (not supported)
//
// Arguments: [dwCredHandle] --
// [CredType] --
// [pCredentials] --
//
// History: 7-26-96 RichardW Created
//
// Notes:
//
//----------------------------------------------------------------------------
SECURITY_STATUS SEC_ENTRY
SpGetCredentials( LSA_SEC_HANDLE dwCredHandle,
PSecBuffer pCredentials)
{
return(SEC_E_UNSUPPORTED_FUNCTION);
}
//+---------------------------------------------------------------------------
//
// Function: SpDeleteCredentials
//
// Synopsis: Delete stored creds (not supported)
//
// Arguments: [dwCredHandle] --
// [CredType] --
// [pKey] --
//
// History: 7-26-96 RichardW Created
//
// Notes:
//
//----------------------------------------------------------------------------
SECURITY_STATUS SEC_ENTRY
SpDeleteCredentials(LSA_SEC_HANDLE dwCredHandle,
PSecBuffer pKey)
{
return(SEC_E_UNSUPPORTED_FUNCTION);
}