Windows2003-3790/inetsrv/pop3/smtp/storedriver/pop3dropdir.cpp
2020-09-30 16:53:55 +02:00

563 lines
22 KiB
C++

// POP3DropDir.cpp: implementation of the POP3DropDir class.
//
//////////////////////////////////////////////////////////////////////
#include "stdafx.h"
#include "POP3DropDir.h"
#include "SimpleDriver.h"
#include "CSimpleDriver.h"
#include "mailmsgprops.h"
#include <POP3Events.h>
#include <POP3Server.h>
#include <POP3RegKeys.h>
VOID POP3DropDirReadCompletion( PFIO_CONTEXT pContext, PFH_OVERLAPPED lpo, DWORD cbRead, DWORD dwCompletionStatus );
VOID POP3DropDirWriteCompletion( PFIO_CONTEXT pContext, PFH_OVERLAPPED lpo, DWORD cbWritten, DWORD dwCompletionStatus );
//////////////////////////////////////////////////////////////////////
// Construction/Destruction
//////////////////////////////////////////////////////////////////////
CPOP3DropDir::CPOP3DropDir( IMailMsgProperties *pIMailMsgProperties, DWORD dwRecipCount, DWORD *pdwRecipIndexes, IMailMsgNotify *pIMailMsgNotify ) :
m_pIMailMsgProperties(pIMailMsgProperties), m_pIMailMsgNotify(pIMailMsgNotify), m_pIMailMsgBind(NULL), m_pIMailMsgRecipients(NULL),
m_pdwRecipIndexes(NULL), m_dwRecipCount(dwRecipCount), m_dwRecipCurrent(0), m_hr( S_OK ), m_enumBWS( NA )
{
ZeroMemory( &m_OverlappedRead, sizeof( m_OverlappedRead ));
ZeroMemory( &m_OverlappedWrite, sizeof( m_OverlappedWrite ));
m_sStoreFileName[0] = 0;
m_PFIOContextWrite = NULL;
if ( NULL != m_pIMailMsgProperties )
m_pIMailMsgProperties->AddRef();
if ( NULL != m_pIMailMsgNotify )
m_pIMailMsgNotify->AddRef();
if ( 0 != dwRecipCount )
{
m_pdwRecipIndexes = new DWORD[dwRecipCount];
if ( NULL != m_pdwRecipIndexes )
memcpy( m_pdwRecipIndexes, pdwRecipIndexes, sizeof( DWORD ) * dwRecipCount );
else
m_hr = E_OUTOFMEMORY;
}
else
m_hr = E_UNEXPECTED;
if ( NULL == CSimpleDriver::s_pStoreDriver )
m_hr = E_POINTER;
}
CPOP3DropDir::~CPOP3DropDir()
{
if ( SUCCEEDED( m_hr ))
MailboxAndContextCleanup( false ); // No value in checking return codes
else
{
MailboxAndContextCleanup( true ); // No value in checking return codes
CSimpleDriver::s_pStoreDriver->LogEvent( LOGTYPE_ERR_WARNING, POP3_SMTPSINK_MESSAGEDELIVERY_FAILED, m_hr );
}
if ( NULL != m_pdwRecipIndexes )
delete [] m_pdwRecipIndexes;
if ( NULL != m_pIMailMsgRecipients )
m_pIMailMsgRecipients->Release();
if ( NULL != m_pIMailMsgBind )
{
m_pIMailMsgBind->ReleaseContext();
m_pIMailMsgBind->Release();
}
if ( NULL != m_pIMailMsgProperties )
m_pIMailMsgProperties->Release();
if ( NULL != m_pIMailMsgNotify )
{
m_pIMailMsgNotify->Notify( m_hr );
m_pIMailMsgNotify->Release();
}
}
//////////////////////////////////////////////////////////////////////
// Implementation - public
//////////////////////////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////
// DoLocalDelivery, public
//
// Purpose:
// Initiate the asynchronous delivery of the current recipient
//
// Returns: MAILMSG_S_PENDING on success, an appropriate HRESULT otherwise
HRESULT CPOP3DropDir::DoLocalDelivery()
{
HRESULT hr = m_hr;
// May contain error from constructor
if ( S_OK == hr ) // Get the FIO_CONTEXT for the mail message
{
hr = m_pIMailMsgProperties->QueryInterface( IID_IMailMsgBind, reinterpret_cast<LPVOID*>( &m_pIMailMsgBind ));
if ( S_OK == hr )
hr = m_pIMailMsgBind->GetBinding( &m_PFIOContextRead, NULL );
}
if ( S_OK == hr )
{
hr = m_pIMailMsgProperties->QueryInterface(IID_IMailMsgRecipients, reinterpret_cast<LPVOID*>( &m_pIMailMsgRecipients));
if ( S_OK == hr ) // About to go Async (CopyMailToDropDir), don't use member variables anymore
{
do
{
hr = CopyMailToDropDir();
}
while ( ERROR_INVALID_MESSAGEDEST == hr );
}
else
CSimpleDriver::s_pStoreDriver->LogEvent( LOGTYPE_ERR_CRITICAL, POP3_SMTPSINK_QI_MAILMSGRECIPIENTS_FAILED, hr );
}
if ( S_OK == hr )
return MAILMSG_S_PENDING;
else if ( FAILED( hr ))
SetHr( hr );
// There was an error so we don't need the Notify interface, release it right now so we don't do it incorrectly in the destrustructor
if ( NULL != m_pIMailMsgNotify )
{
m_pIMailMsgNotify->Release();
m_pIMailMsgNotify = NULL;
}
return hr;
}
//////////////////////////////////////////////////////////////////////
// Implementation - private
//////////////////////////////////////////////////////////////////////
HRESULT CPOP3DropDir::CopyMailToDropDir()
{
HRESULT hr = E_FAIL;
HANDLE hf = INVALID_HANDLE_VALUE;
DWORD dwRecipFlags, dwRC;
IMailMsgRecipients *pIMailMsgRecipients;
SYSTEMTIME st;
char szRecpEmailName[sizeof(m_sRecipEmailName)/sizeof(WCHAR)];
GetLocalTime( &st );
if ( m_dwRecipCurrent < m_dwRecipCount )
{
hr = m_pIMailMsgRecipients->GetStringA(m_pdwRecipIndexes[m_dwRecipCurrent], IMMPID_RP_ADDRESS_SMTP, sizeof(szRecpEmailName), szRecpEmailName);
if ( S_OK == hr )
{
if ( 0 == MultiByteToWideChar( CP_ACP, 0, szRecpEmailName, -1, m_sRecipEmailName, sizeof(m_sRecipEmailName)/sizeof(WCHAR) ))
hr = HRESULT_FROM_WIN32( GetLastError() );
else if ( m_mailboxX.OpenMailBox( m_sRecipEmailName ))
{
swprintf( m_sStoreFileName, L"%s%04u%02u%02u%02u%02u%02u%04u%08x.eml",
MAILBOX_PREFIX_W, st.wYear, st.wMonth, st.wDay, st.wHour, st.wMinute, st.wSecond, st.wMilliseconds, InterlockedIncrement( reinterpret_cast<PLONG>( &CSimpleDriver::s_dwCounter )) );
dwRC = CSimpleDriver::s_pStoreDriver->m_AdjustTokenPrivilegesX.SetToken();
if ( ERROR_SUCCESS != dwRC )
{
CSimpleDriver::s_pStoreDriver->LogEvent( LOGTYPE_ERR_CRITICAL, POP3_SMTPSINK_SETTHREADTOKEN_FAILED, dwRC );
hr = HRESULT_FROM_WIN32( dwRC );
}
hf = m_mailboxX.CreateMail( m_sStoreFileName, FILE_FLAG_OVERLAPPED|FILE_FLAG_SEQUENTIAL_SCAN );
dwRC = CSimpleDriver::s_pStoreDriver->m_AdjustTokenPrivilegesX.ResetToken(); // Safe to call even if SetToken failed
if ( ERROR_SUCCESS != dwRC )
{
CSimpleDriver::s_pStoreDriver->LogEvent( LOGTYPE_ERR_CRITICAL, POP3_SMTPSINK_RESETTHREADTOKEN_FAILED, dwRC );
m_hr = HRESULT_FROM_WIN32( dwRC );
}
if ( INVALID_HANDLE_VALUE != hf )
{
m_PFIOContextWrite = AssociateFile( hf );
if ( NULL != m_PFIOContextWrite )
{ // Okay let's start the Asynchronous operations - Read first
m_i64ReadOffset = m_i64WriteOffset = 0;
hr = ReadFile( m_sBuffer, PRIVATE_OPTIMAL_BUFFER_SIZE );
}
else
{
CSimpleDriver::s_pStoreDriver->LogEvent( LOGTYPE_ERR_CRITICAL, POP3_SMTPSINK_ASSOCIATEFILE_FAILED );
CloseHandle( hf ); // Will get deleted in desstructor
hr = E_FAIL;
}
}
else
{
hr = HRESULT_FROM_WIN32( GetLastError() );
if ( HRESULT_FROM_WIN32( ERROR_DISK_FULL ) == hr )
{
CSimpleDriver::s_pStoreDriver->LogEvent( LOGTYPE_INFORMATION, POP3_SMTPSINK_MESSAGEDELIERY_FAILED_OUTOFDISK, ERROR_DISK_FULL );
hr = ERROR_INVALID_MESSAGEDEST; // Quota (or really a disk) failure continue with next recipient
}
else
{
if ( S_OK == hr ) hr = HRESULT_FROM_WIN32(ERROR_BUFFER_OVERFLOW);
CSimpleDriver::s_pStoreDriver->LogEvent( LOGTYPE_ERR_CRITICAL, POP3_SMTPSINK_CREATEMAIL_FAILED, hr );
}
}
}
else
hr = ERROR_INVALID_MESSAGEDEST; // Continue with next recipient!
if ( S_OK != hr )
{
MarkRecipient( RP_FAILED ); // No value in checking return codes
MailboxAndContextCleanup( true ); // No value in checking return codes
m_dwRecipCurrent++; // Completion routine will re-initiate process for next recipient.
if ( ERROR_INVALID_MESSAGEDEST == hr )
{
if ( isAllRecipientsProcessed() )
hr = S_FALSE;
}
}
}
else
CSimpleDriver::s_pStoreDriver->LogEvent( LOGTYPE_ERR_CRITICAL, POP3_SMTPSINK_GET_IMMPID_RP_ADDRESS_SMTP_FAILED, hr );
}
return hr;
}
BYTE g_Matrix[256] = { 0,0,0,0,0,0,0,0,0,0,2,0,0,3,0,0, // 00-0F
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, // 10-1F
0,0,0,0,0,0,0,0,0,0,0,0,0,0,1,0, // 20-2F
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, // 30-3F
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, // 40-4F
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, // 50-5F
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, // 60-6F
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, // 70-7F
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, // 80-8F
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, // 90-9F
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, // A0-AF
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, // B0-BF
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, // C0-CF
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, // D0-DF
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, // E0-EF
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0 // F0-FF
};
HRESULT CPOP3DropDir::DotStuffBuffer( LPVOID *ppbBuffer, LPDWORD pdwSize )
{
if ( NULL == ppbBuffer || NULL == pdwSize )
return E_INVALIDARG;
if ( 0 == *pdwSize )
return E_INVALIDARG;
DWORD dwSize = *pdwSize;
bool bStuff = false;
CHAR *pbBuffer = static_cast<CHAR*>( *ppbBuffer );
CHAR *pToEnd = m_sBuffer + sizeof( m_sBuffer );
CHAR *pFromEnd = pbBuffer + dwSize;
CHAR *pToStart, *pFromStart = NULL;
CHAR *pChar;
for ( pChar = pbBuffer + dwSize - 3; pChar > pbBuffer + 2; pChar -= 3 )
{
if ( g_Matrix[static_cast<BYTE>(*pChar)] )
{ // Found a '\r' '\n' or '.'
pChar += g_Matrix[static_cast<BYTE>(*pChar)] - 1; // Start at the Dot
if ( (0xd == *(pChar-2)) && (0xa == *(pChar-1)) && (0x2e == *(pChar)) )
{ // Stuff Required
pFromStart = pChar; // Start copy at the Dot
pToStart = pToEnd - ( pFromEnd - pFromStart );
memmove( pToStart, pFromStart, pFromEnd - pFromStart );
pToStart--;
*pToStart = 0x2e; // Dot stuff
dwSize++;
// Reset the pointers
pFromEnd = pFromStart;
pToEnd = pToStart;
}
}
}
if ( NULL != pFromStart ) // Have we Dot stuffed anything yet?
*pFromStart = 0x0; // yes.
pChar = pbBuffer;
if ( g_Matrix[static_cast<BYTE>(*pChar)] )
{
if ( (BufferWrapSequence::CR == m_enumBWS) && (0xa == *pChar) && (0x2e == *(pChar+1)) )
pFromStart = pChar + 1; // Start at the Dot
else if ( (BufferWrapSequence::CRLF == m_enumBWS) && (0x2e == *pChar) )
pFromStart = pChar; // Start at the Dot
else if ( (0xd == *pChar) && (0xa == *(pChar+1)) && (0x2e == *(pChar+2)) )
pFromStart = pChar+2; // Start at the Dot
}
if ( NULL != pFromStart && 0x0 != *pFromStart )
{ // 1 more stuff required
pToStart = pToEnd - ( pFromEnd - pFromStart );
memmove( pToStart, pFromStart, pFromEnd - pFromStart );
pToStart--;
*pToStart = 0x2e; // Dot stuff
dwSize++;
// Reset the pointers
pFromEnd = pFromStart;
pToEnd = pToStart;
}
if ( NULL != pFromStart )
{ // Need to move the beginning of the buffer
pFromEnd = pFromStart;
pFromStart = pbBuffer;
if ( pFromStart != pFromEnd )
{
pToStart = pToEnd - ( pFromEnd - pFromStart );
memmove( pToStart, pFromStart, pFromEnd - pFromStart );
}
*ppbBuffer = pToStart;
*pdwSize = dwSize;
}
// Update the Buffer Wrap Sequence
if ( 0x0d == *( static_cast<CHAR*>(*ppbBuffer) + dwSize - 1))
m_enumBWS = BufferWrapSequence::CR;
else if ( 0x0d == *( static_cast<CHAR*>(*ppbBuffer) + dwSize - 2) && 0x0a == *( static_cast<CHAR*>(*ppbBuffer) + dwSize - 1) )
m_enumBWS = BufferWrapSequence::CRLF;
else
m_enumBWS = BufferWrapSequence::NA;
return S_OK;
}
HRESULT CPOP3DropDir::MailboxAndContextCleanup( bool bDeleteMailFile )
{
HRESULT hr = S_OK;
if ( NULL != m_PFIOContextWrite )
{
if ( ERROR_SUCCESS != m_mailboxX.CloseMail( m_PFIOContextWrite->m_hFile,FILE_FLAG_OVERLAPPED ))
{
CSimpleDriver::s_pStoreDriver->LogEvent( LOGTYPE_ERR_CRITICAL, POP3_SMTPSINK_CLOSEMAIL_FAILED );
hr = E_FAIL;
}
ReleaseContext( m_PFIOContextWrite ); // no return void!
m_PFIOContextWrite = NULL;
}
if ( bDeleteMailFile && 0 != m_sStoreFileName[0] )
m_mailboxX.DeleteMail( m_sStoreFileName );
m_mailboxX.CloseMailBox(); // no return void!
return hr;
}
HRESULT CPOP3DropDir::MarkRecipient( DWORD dwMark )
{
HRESULT hr;
DWORD dwRecipFlags;
hr = m_pIMailMsgRecipients->GetDWORD( m_pdwRecipIndexes[m_dwRecipCurrent], IMMPID_RP_RECIPIENT_FLAGS, &dwRecipFlags );
if ( S_OK == hr )
{
dwRecipFlags |= dwMark; // mark the recipient as failed
hr = m_pIMailMsgRecipients->PutDWORD( m_pdwRecipIndexes[m_dwRecipCurrent], IMMPID_RP_RECIPIENT_FLAGS, dwRecipFlags );
if ( S_OK != hr )
CSimpleDriver::s_pStoreDriver->LogEvent( LOGTYPE_ERR_WARNING, POP3_SMTPSINK_PUT_IMMPID_RP_RECIPIENT_FLAGS_FAILED, hr );
}
else
CSimpleDriver::s_pStoreDriver->LogEvent( LOGTYPE_ERR_WARNING, POP3_SMTPSINK_GET_IMMPID_RP_RECIPIENT_FLAGS_FAILED, hr );
return hr;
}
HRESULT CPOP3DropDir::ReadFile( IN LPVOID pBuffer, IN DWORD cbSize )
{
if ( NULL == pBuffer || 0 == cbSize )
return E_INVALIDARG;
HRESULT hr = S_OK;
m_OverlappedRead.Overlapped.Offset = LODWORD( m_i64ReadOffset );
m_OverlappedRead.Overlapped.OffsetHigh = HIDWORD( m_i64ReadOffset );
m_OverlappedRead.Overlapped.pfnCompletion = POP3DropDirReadCompletion;
m_OverlappedRead.ThisPtr = (PVOID)this;
BOOL bRC = FIOReadFile( m_PFIOContextRead, pBuffer, cbSize, &m_OverlappedRead.Overlapped);
if( !bRC )
{
DWORD dwErr = GetLastError();
if ( ERROR_IO_PENDING != dwErr )
{
hr = HRESULT_FROM_WIN32( dwErr );
if ( dwErr == ERROR_HANDLE_EOF )
{ // Cleanup
HRESULT hr2;
hr2 = MarkRecipient( RP_DELIVERED );
if ( S_OK != hr2 )
hr = hr2;
hr2 = MailboxAndContextCleanup( false );
if ( S_OK != hr2 )
hr = hr2;
m_dwRecipCurrent++; // Completion routine will re-initiate process for next recipient.
}
else
{
CSimpleDriver::s_pStoreDriver->LogEvent( LOGTYPE_ERR_CRITICAL, POP3_SMTPSINK_READFILE_FAILED, dwErr );
// Cleanup
MarkRecipient( RP_FAILED ); // No value in checking return codes
MailboxAndContextCleanup( true ); // No value in checking return codes
m_dwRecipCurrent++; // Completion routine will re-initiate process for next recipient.
}
}
}
return hr;
}
HRESULT CPOP3DropDir::ReadFileCompletion( DWORD cbSize, DWORD dwErr, PFH_OVERLAPPED lpo )
{
HRESULT hr = S_OK;
if (( NO_ERROR != dwErr ) && ( ERROR_HANDLE_EOF != dwErr ))
{
hr = HRESULT_FROM_WIN32( dwErr );
CSimpleDriver::s_pStoreDriver->LogEvent( LOGTYPE_ERR_WARNING, POP3_SMTPSINK_READFILE_FAILED, dwErr );
// Cleanup
MarkRecipient( RP_FAILED ); // No value in checking return codes
MailboxAndContextCleanup( true ); // No value in checking return codes
m_dwRecipCurrent++; // Completion routine will re-initiate process for next recipient.
}
else
{
if ( ERROR_HANDLE_EOF == dwErr && 0 == cbSize )
{ // Cleanup
HRESULT hr2;
hr2 = MarkRecipient( RP_DELIVERED );
if ( S_OK != hr2 )
hr = hr2;
hr2 = MailboxAndContextCleanup( (S_OK == hr) ? false : true );
if ( S_OK != hr2 )
hr = hr2;
m_dwRecipCurrent++; // Completion routine will re-initiate process for next recipient.
if ( S_OK == hr )
hr = HRESULT_FROM_WIN32( ERROR_HANDLE_EOF ); // Already done reading the file, nothing else to write
}
else
{
m_i64ReadOffset += cbSize;
// Asynchronous operation - Write
hr = WriteFile( m_sBuffer, cbSize );
}
}
return hr;
}
HRESULT CPOP3DropDir::WriteFile( IN LPVOID pBuffer, IN DWORD cbSize )
{
if ( NULL == pBuffer || 0 == cbSize )
return E_INVALIDARG;
HRESULT hr = S_OK;
hr = DotStuffBuffer( &pBuffer, &cbSize );
if ( S_OK == hr )
{
m_OverlappedWrite.Overlapped.Offset = LODWORD( m_i64WriteOffset );
m_OverlappedWrite.Overlapped.OffsetHigh = HIDWORD( m_i64WriteOffset );
m_OverlappedWrite.Overlapped.pfnCompletion = POP3DropDirWriteCompletion;
m_OverlappedWrite.ThisPtr = (PVOID)this;
BOOL bRC = FIOWriteFile( m_PFIOContextWrite, pBuffer, cbSize, &m_OverlappedWrite.Overlapped);
if( !bRC )
{
DWORD dwErr = GetLastError();
if( dwErr != ERROR_IO_PENDING )
{
hr = HRESULT_FROM_WIN32( dwErr );
if ( ERROR_DISK_FULL == dwErr )
CSimpleDriver::s_pStoreDriver->LogEvent( LOGTYPE_INFORMATION, POP3_SMTPSINK_MESSAGEDELIERY_FAILED_OUTOFDISK, dwErr );
else
CSimpleDriver::s_pStoreDriver->LogEvent( LOGTYPE_ERR_CRITICAL, POP3_SMTPSINK_WRITEFILE_FAILED, dwErr );
// Cleanup
MarkRecipient( RP_FAILED ); // No value in checking return codes
MailboxAndContextCleanup( true ); // No value in checking return codes
m_dwRecipCurrent++; // Completion routine will re-initiate process for next recipient.
}
}
}
return hr;
}
HRESULT CPOP3DropDir::WriteFileCompletion( DWORD cbSize, DWORD dwErr, PFH_OVERLAPPED lpo )
{
HRESULT hr = S_OK;
if (( dwErr != NO_ERROR ))
{
hr = HRESULT_FROM_WIN32( dwErr );
if ( ERROR_DISK_FULL == dwErr )
CSimpleDriver::s_pStoreDriver->LogEvent( LOGTYPE_INFORMATION, POP3_SMTPSINK_MESSAGEDELIERY_FAILED_OUTOFDISK, dwErr );
else
CSimpleDriver::s_pStoreDriver->LogEvent( LOGTYPE_ERR_CRITICAL, POP3_SMTPSINK_WRITEFILE_FAILED, dwErr );
// Cleanup
MarkRecipient( RP_FAILED ); // No value in checking return codes
MailboxAndContextCleanup( true ); // No value in checking return codes
m_dwRecipCurrent++; // Completion routine will re-initiate process for next recipient.
}
else
{
m_i64WriteOffset += cbSize;
// Asynchronous operation - Read
hr = ReadFile( m_sBuffer, PRIVATE_OPTIMAL_BUFFER_SIZE );
}
return hr;
}
//////////////////////////////////////////////////////////////////////
// Other
//////////////////////////////////////////////////////////////////////
VOID POP3DropDirReadCompletion( PFIO_CONTEXT pContext, PFH_OVERLAPPED lpo, DWORD cbRead, DWORD dwCompletionStatus )
{
if ( NULL == lpo ) return;
POP3DROPDIR_OVERLAPPED *p = reinterpret_cast<POP3DROPDIR_OVERLAPPED*>( lpo );
CPOP3DropDir *pThis = static_cast<CPOP3DropDir*>( p->ThisPtr );
HRESULT hr = pThis->ReadFileCompletion( cbRead, dwCompletionStatus, lpo );
if ( S_OK != hr )
{
if ( HRESULT_FROM_WIN32( ERROR_HANDLE_EOF ) == hr )
{
if ( pThis->isAllRecipientsProcessed( )) // More recipients to process?
{
hr = S_OK;
delete pThis; // Were done!
}
else
hr = pThis->NextRecipientCopyMailToDropDir();
}
if ( S_OK != hr )
{
pThis->SetHr( hr ); // Unexpected completion, set the error
delete pThis; // Were done!
}
}
}
VOID POP3DropDirWriteCompletion( PFIO_CONTEXT pContext, PFH_OVERLAPPED lpo, DWORD cbWritten, DWORD dwCompletionStatus )
{
if ( NULL == lpo ) return;
POP3DROPDIR_OVERLAPPED *p = reinterpret_cast<POP3DROPDIR_OVERLAPPED*>( lpo );
CPOP3DropDir *pThis = static_cast<CPOP3DropDir*>( p->ThisPtr );
HRESULT hr = pThis->WriteFileCompletion( cbWritten, dwCompletionStatus, lpo );
if ( S_OK != hr )
{
if ( pThis->isAllRecipientsProcessed( )) // More recipients to process?
{
hr = S_OK;
delete pThis; // Were done!
}
else
hr = pThis->NextRecipientCopyMailToDropDir();
if ( S_OK != hr )
{
pThis->SetHr( hr ); // Unexpected completion, set the error
delete pThis; // Were done!
}
}
}