📄 dbx_rpcc.c
字号:
/****************************************************************************
* *
* cryptlib Database RPC Client Interface *
* Copyright Peter Gutmann 1996-2006 *
* *
****************************************************************************/
#include <ctype.h>
#include <stdarg.h>
#if defined( INC_ALL )
#include "crypt.h"
#include "keyset.h"
#include "dbms.h"
#include "rpc.h"
#else
#include "crypt.h"
#include "keyset/keyset.h"
#include "keyset/dbms.h"
#include "misc/rpc.h"
#endif /* Compiler-specific includes */
#ifdef USE_DBMS
/****************************************************************************
* *
* Network Database Interface Routines *
* *
****************************************************************************/
#ifdef USE_DATABASE_PLUGIN
#ifdef USE_RPCAPI
static void netEncodeError( BYTE *buffer, const int status )
{
putMessageType( buffer, COMMAND_RESULT, 0, 1, 0 );
putMessageLength( buffer + COMMAND_WORDSIZE, COMMAND_WORDSIZE );
putMessageWord( buffer + COMMAND_WORD1_OFFSET, status );
}
void netProcessCommand( void *stateInfo, BYTE *buffer )
{
DBMS_STATE_INFO *dbmsInfo = ( DBMS_STATE_INFO * ) stateInfo;
COMMAND_INFO cmd;
int length, status;
memset( &cmd, 0, sizeof( COMMAND_INFO ) );
/* Get the messge information from the header */
getMessageType( buffer, cmd.type, cmd.flags,
cmd.noArgs, cmd.noStrArgs );
length = getMessageLength( buffer + COMMAND_WORDSIZE );
if( cmd.type == DBX_COMMAND_OPEN )
{
NET_CONNECT_INFO connectInfo;
BYTE *bufPtr = buffer + COMMAND_FIXED_DATA_SIZE + COMMAND_WORDSIZE;
int nameLen;
/* Get the length of the server name and null-terminate it */
nameLen = getMessageWord( bufPtr );
bufPtr += COMMAND_WORDSIZE;
bufPtr[ nameLen ] = '\0';
/* Connect to the plugin */
initNetConnectInfo( &connectInfo, DEFAULTUSER_OBJECT_HANDLE,
CRYPT_ERROR, CRYPT_ERROR, NET_OPTION_HOSTNAME );
connectInfo.name = bufPtr;
status = sNetConnect( &dbmsInfo->stream, STREAM_PROTOCOL_TCPIP,
&connectInfo, dbmsInfo->errorMessage,
&dbmsInfo->errorCode );
if( cryptStatusError( status ) )
{
netEncodeError( buffer, status );
return;
}
}
/* Send the command to the plugin and read back the response */
status = swrite( &dbmsInfo->stream, buffer,
COMMAND_FIXED_DATA_SIZE + COMMAND_WORDSIZE + length );
if( cryptStatusOK( status ) )
status = sread( &dbmsInfo->stream, buffer, COMMAND_FIXED_DATA_SIZE );
if( !cryptStatusError( status ) )
{
/* Perform a consistency check on the returned data */
getMessageType( buffer, cmd.type, cmd.flags,
cmd.noArgs, cmd.noStrArgs );
length = getMessageLength( buffer + COMMAND_WORDSIZE );
if( !dbxCheckCommandInfo( &cmd, length ) || \
cmd.type != COMMAND_RESULT )
status = CRYPT_ERROR_BADDATA;
}
if( !cryptStatusError( status ) )
/* Read the rest of the message */
status = sread( &dbmsInfo->stream, buffer + COMMAND_FIXED_DATA_SIZE,
length );
/* If it's a close command, terminate the connection to the plugin. We
don't do any error checking once we get this far since there's not
much that we can still do at this point */
if( cmd.type == DBX_COMMAND_CLOSE )
sNetDisconnect( &dbmsInfo->stream );
else
if( cryptStatusError( status ) )
netEncodeError( buffer, status );
}
#else
int initDispatchNet( DBMS_INFO *dbmsInfo )
{
return( CRYPT_ERROR );
}
#endif /* USE_RPCAPI */
#endif /* USE_DATABASE_PLUGIN */
/****************************************************************************
* *
* Database RPC Routines *
* *
****************************************************************************/
/* Dispatch functions for various database types. ODBC is the native keyset
for Windows and (if possible) Unix, a cryptlib-native plugin is the
fallback for Unix, and the rest are only accessible via database network
plugins */
#ifdef USE_ODBC
void odbcProcessCommand( void *stateInfo, BYTE *buffer );
#define initDispatchODBC( dbmsInfo ) \
( dbmsInfo->dispatchFunction = odbcProcessCommand ) != NULL
#else
#define initDispatchODBC( dbmsInfo ) CRYPT_ERROR
#endif /* USE_ODBC */
#if defined( USE_DATABASE )
void databaseProcessCommand( void *stateInfo, BYTE *buffer );
#define initDispatchDatabase( dbmsInfo ) \
( dbmsInfo->dispatchFunction = databaseProcessCommand ) != NULL
#else
#define initDispatchDatabase( dbmsInfo ) CRYPT_ERROR
#endif /* General database interface */
#ifdef USE_DATABASE_PLUGIN
int initDispatchNet( DBMS_INFO *dbmsInfo );
#else
#define initDispatchNet( dbmsInfo ) CRYPT_ERROR
#endif /* USE_DATABASE_PLUGIN */
/* Make sure that we can fit the largest possible SQL query into the RPC
buffer */
#if MAX_SQL_QUERY_SIZE + 256 >= DBX_IO_BUFSIZE
#error Database RPC buffer size is too small, increase DBX_IO_BUFSIZE and rebuild
#endif /* SQL query size larger than RPC buffer size */
/* Dispatch data to the back-end */
static int dispatchCommand( COMMAND_INFO *cmd, void *stateInfo,
DISPATCH_FUNCTION dispatchFunction )
{
COMMAND_INFO sentCmd = *cmd;
BYTE buffer[ DBX_IO_BUFSIZE + 8 ], *bufPtr = buffer;
BYTE header[ COMMAND_FIXED_DATA_SIZE + 8 ];
const int payloadLength = ( cmd->noArgs * COMMAND_WORDSIZE ) + \
( cmd->noStrArgs * COMMAND_WORDSIZE ) + \
cmd->strArgLen[ 0 ] + cmd->strArgLen[ 1 ] + \
cmd->strArgLen[ 2 ];
long resultLength;
int i;
assert( payloadLength + 32 < DBX_IO_BUFSIZE );
assert( dispatchFunction != NULL );
/* Clear the return value */
memset( cmd, 0, sizeof( COMMAND_INFO ) );
/* Write the header and message fields to the buffer */
putMessageType( bufPtr, sentCmd.type, sentCmd.flags,
sentCmd.noArgs, sentCmd.noStrArgs );
putMessageLength( bufPtr + COMMAND_WORDSIZE, payloadLength );
bufPtr += COMMAND_FIXED_DATA_SIZE;
for( i = 0; i < sentCmd.noArgs; i++ )
{
putMessageWord( bufPtr, sentCmd.arg[ i ] );
bufPtr += COMMAND_WORDSIZE;
}
for( i = 0; i < sentCmd.noStrArgs; i++ )
{
const int argLength = sentCmd.strArgLen[ i ];
putMessageWord( bufPtr, argLength );
if( argLength > 0 )
memcpy( bufPtr + COMMAND_WORDSIZE, sentCmd.strArg[ i ],
argLength );
bufPtr += COMMAND_WORDSIZE + argLength;
}
/* Send the command to the server and read back the server's message
header */
dispatchFunction( stateInfo, buffer );
memcpy( header, buffer, COMMAND_FIXED_DATA_SIZE );
/* Process the fixed message header and make sure that it's valid */
getMessageType( header, cmd->type, cmd->flags,
cmd->noArgs, cmd->noStrArgs );
resultLength = getMessageLength( header + COMMAND_WORDSIZE );
if( !dbxCheckCommandInfo( cmd, resultLength ) || \
cmd->type != COMMAND_RESULT )
return( CRYPT_ERROR );
if( ( cmd->noStrArgs && cmd->strArgLen[ 0 ] ) && \
( sentCmd.type != DBX_COMMAND_QUERY && \
sentCmd.type != DBX_COMMAND_GETERRORINFO ) )
/* Only these commands can return data */
return( CRYPT_ERROR );
/* Read the rest of the server's message */
bufPtr = buffer + COMMAND_FIXED_DATA_SIZE;
for( i = 0; i < cmd->noArgs; i++ )
{
cmd->arg[ i ] = getMessageWord( bufPtr );
bufPtr += COMMAND_WORDSIZE;
}
for( i = 0; i < cmd->noStrArgs; i++ )
{
cmd->strArgLen[ i ] = getMessageWord( bufPtr );
cmd->strArg[ i ] = bufPtr + COMMAND_WORDSIZE;
bufPtr += COMMAND_WORDSIZE + cmd->strArgLen[ i ];
}
/* The first value returned is the status code, if it's nonzero return
it to the caller, otherwise move the other values down */
if( cryptStatusError( cmd->arg[ 0 ] ) )
return( cmd->arg[ 0 ] );
assert( cryptStatusOK( cmd->arg[ 0 ] ) );
for( i = 1; i < cmd->noArgs; i++ )
cmd->arg[ i - 1 ] = cmd->arg[ i ];
cmd->arg[ i ] = 0;
cmd->noArgs--;
/* Copy any string arg data back to the caller */
if( cmd->noStrArgs && cmd->strArgLen[ 0 ] )
{
const int maxBufSize = ( sentCmd.type == DBX_COMMAND_QUERY ) ? \
MAX_QUERY_RESULT_SIZE : MAX_ERRMSG_SIZE;
const int argIndex = sentCmd.noStrArgs;
memcpy( sentCmd.strArg[ argIndex ], cmd->strArg[ 0 ],
min( cmd->strArgLen[ 0 ], maxBufSize ) );
cmd->strArg[ 0 ] = sentCmd.strArg[ argIndex ];
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -