2020-09-30 16:53:55 +02:00

352 lines
6.2 KiB
C++

/*++
Copyright (c) 2001 Microsoft Corporation
Module Name:
sacio.cpp
Abstract:
This module implements the SAC IoHandler functionality.
The SAC IoHandler implements a write buffer so that the
# of calls to teh SAC driver are reduced.
Author:
Brian Guarraci (briangu), 2001
Revision History:
--*/
#include "sacio.h"
//
// Enable the write buffer
//
#define USE_SEND_BUFFER 1
//
// The size of the write buffer
//
#define SEND_BUFFER_SIZE 8192
CSacIoHandler::CSacIoHandler(
VOID
)
/*++
Routine Description:
Constructor
Arguments:
None
Return Value:
N/A
--*/
{
//
// We don't yet have a SAC channel object
//
mySacChannel = NULL;
//
// Create a send buffer
//
mySendBufferIndex = 0;
mySendBuffer = new BYTE[SEND_BUFFER_SIZE];
}
CSacIoHandler::~CSacIoHandler(
)
/*++
Routine Description:
Destructor
Arguments:
N/A
Return Value:
N/A
--*/
{
if (mySacChannel) {
delete mySacChannel;
}
delete [] mySendBuffer;
}
CSacIoHandler*
CSacIoHandler::Construct(
IN SAC_CHANNEL_OPEN_ATTRIBUTES Attributes
)
/*++
Routine Description:
static constructor - this does the real construction
Arguments:
Attributes - attributes of the SAC channel to create
Return Value:
On success, a pointer to the new IoHandler
On failure, NULL
--*/
{
CSacIoHandler *IoHandler;
//
// Create a new SAC IoHandler
//
IoHandler = new CSacIoHandler();
//
// Attempt to open a SAC channel
//
IoHandler->mySacChannel = EMSCmdChannel::Construct(Attributes);
//
// If we failed to open the SAC channel,
// then destroy the IoHandler and notify the caller
// that we failed by returning null
//
if (IoHandler->mySacChannel == NULL) {
delete IoHandler;
return NULL;
}
return IoHandler;
}
BOOL
CSacIoHandler::Write(
IN PBYTE Buffer,
IN ULONG BufferSize
)
/*++
Routine Description:
This routine impelements a buffered write IoHandler operation.
Arguments:
Buffer - the data to send
BufferSize - the size of the buffer in bytes
Return Value:
TRUE - success
FALSE - otherwise
--*/
{
#if USE_SEND_BUFFER
//
// If the requested buffer to send is bigger than the
// remaining local buffer, send the local buffer.
//
if (mySendBufferIndex + BufferSize > SEND_BUFFER_SIZE) {
if (! Flush()) {
return FALSE;
}
}
// ASSERT(mySendBufferIndex + BufferSize <= SEND_BUFFER_SIZE);
//
// Copy the incoming buffer into our local buffer
//
RtlCopyMemory(
&mySendBuffer[mySendBufferIndex],
Buffer,
BufferSize
);
//
// Account for the added buffer contents
//
mySendBufferIndex += BufferSize;
// ASSERT(mySendBufferIndex % sizeof(WCHAR) == 0);
//
// we succeeded
//
return TRUE;
#else
//
// Send the local buffer to the SAC channel
//
bSuccess = mySacChannel->Write(
(PCWCHAR)Buffer,
BufferSize
);
//
// Reset the local buffer index
//
mySendBufferIndex = 0;
return bSuccess;
#endif
}
BOOL
CSacIoHandler::Flush(
VOID
)
/*++
Routine Description:
This routine implements the Flush IoHandler method.
If there is any data stored in the write buffer, it
is flushed to the SAC channel.
Arguments:
None
Return Value:
TRUE - success
FALSE - otherwise
--*/
{
#if USE_SEND_BUFFER
BOOL bSuccess;
//
// default: we succeeded
//
bSuccess = TRUE;
#if 0
TCHAR Buffer[1024];
wsprintf(Buffer, TEXT("buffsize=%d\r\n"), mySendBufferIndex);
OutputDebugString(Buffer);
#endif
//
// Send the local buffer to the SAC channel
//
if (mySendBufferIndex > 0) {
bSuccess = mySacChannel->Write(
mySendBuffer,
mySendBufferIndex
);
//
// Reset the local buffer index
//
mySendBufferIndex = 0;
}
return bSuccess;
#else
return TRUE;
#endif
}
BOOL
CSacIoHandler::Read(
OUT PBYTE Buffer,
IN ULONG BufferSize,
OUT PULONG ByteCount
)
/*++
Routine Description:
This routine implements the Read IoHandler method.
Arguments:
Buffer - on success, contains the read data
BufferSize - the size of the read buffer in bytes
ByteCount - on success, contains the # of bytes read
Return Value:
TRUE - success
FALSE - otherwise
--*/
{
if (!mySacChannel) {
return FALSE;
}
//
// Read data from the channel
//
return mySacChannel->Read(
Buffer,
BufferSize,
ByteCount
);
}
//
// Determine if the ioHandler has new data to read
//
BOOL
CSacIoHandler::HasNewData(
OUT PBOOL InputWaiting
)
/*++
Routine Description:
This routine impelements the HasNewData IoHandler method.
Arguments:
InputWaiting - on success, contains the status of the channel's
input buffer.
Return Value:
TRUE - success
FALSE - otherwise
--*/
{
//
// Determine if the channel has new data
//
return mySacChannel->HasNewData(InputWaiting);
}