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.
337 lines
7.1 KiB
337 lines
7.1 KiB
// Base64Coder.cpp: implementation of the Base64Coder class.
|
|
//
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
#include <stdafx.hxx>
|
|
#include "Base64Coder.h"
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
// Standard foo for file name aliasing. This code block must be after
|
|
// all includes of VSS header files.
|
|
//
|
|
#ifdef VSS_FILE_ALIAS
|
|
#undef VSS_FILE_ALIAS
|
|
#endif
|
|
#define VSS_FILE_ALIAS "BUEB64CC"
|
|
//
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
// Digits...
|
|
static char Base64Digits[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
|
|
|
|
|
|
BOOL Base64Coder::m_Init = FALSE;
|
|
char Base64Coder::m_DecodeTable[256];
|
|
|
|
#ifndef PAGESIZE
|
|
#define PAGESIZE 4096
|
|
#endif
|
|
|
|
#ifndef ROUNDTOPAGE
|
|
#define ROUNDTOPAGE(a) (((a/4096)+1)*4096)
|
|
#endif
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// Construction/Destruction
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
Base64Coder::Base64Coder() :
|
|
m_pDBuffer(NULL),
|
|
m_pEBuffer(NULL),
|
|
m_nDBufLen(0),
|
|
m_nEBufLen(0)
|
|
{
|
|
}
|
|
|
|
Base64Coder::~Base64Coder()
|
|
{
|
|
if(m_pDBuffer != NULL)
|
|
delete [] m_pDBuffer;
|
|
|
|
if(m_pEBuffer != NULL)
|
|
delete [] m_pEBuffer;
|
|
}
|
|
|
|
BYTE *Base64Coder::DecodedMessage() const
|
|
{
|
|
return m_pDBuffer;
|
|
}
|
|
|
|
LPCWSTR Base64Coder::EncodedMessage() const
|
|
{
|
|
return m_pEBuffer;
|
|
}
|
|
|
|
void Base64Coder::AllocEncode(UINT nSize)
|
|
{
|
|
if(m_nEBufLen < nSize)
|
|
{
|
|
if(m_pEBuffer != NULL)
|
|
delete [] m_pEBuffer;
|
|
|
|
m_nEBufLen = ROUNDTOPAGE(nSize);
|
|
m_pEBuffer = new WCHAR[m_nEBufLen];
|
|
if (m_pEBuffer == NULL)
|
|
{
|
|
// reset allocated size to 0
|
|
m_nEBufLen = 0;
|
|
throw(E_OUTOFMEMORY);
|
|
}
|
|
}
|
|
|
|
::ZeroMemory(m_pEBuffer, m_nEBufLen*sizeof(WCHAR));
|
|
m_nEDataLen = 0;
|
|
}
|
|
|
|
void Base64Coder::AllocDecode(UINT nSize)
|
|
{
|
|
if(m_nDBufLen < nSize)
|
|
{
|
|
if(m_pDBuffer != NULL)
|
|
delete [] m_pDBuffer;
|
|
|
|
m_nDBufLen = ROUNDTOPAGE(nSize);
|
|
m_pDBuffer = new BYTE[m_nDBufLen];
|
|
if (m_pDBuffer == NULL)
|
|
{
|
|
// reset allocated size to 0
|
|
m_nDBufLen = 0;
|
|
throw(E_OUTOFMEMORY);
|
|
}
|
|
}
|
|
|
|
::ZeroMemory(m_pDBuffer, m_nDBufLen);
|
|
|
|
m_nDDataLen = 0;
|
|
}
|
|
|
|
void Base64Coder::SetEncodeBuffer(const LPCWSTR pBuffer, UINT nBufLen)
|
|
{
|
|
UINT i = 0;
|
|
|
|
AllocEncode(nBufLen);
|
|
while(i < nBufLen)
|
|
{
|
|
if(!_IsBadMimeChar(pBuffer[i]))
|
|
{
|
|
m_pEBuffer[m_nEDataLen] = pBuffer[i];
|
|
m_nEDataLen++;
|
|
}
|
|
|
|
i++;
|
|
}
|
|
}
|
|
|
|
void Base64Coder::SetDecodeBuffer(const BYTE *pBuffer, UINT nBufLen)
|
|
{
|
|
AllocDecode(nBufLen + sizeof(nBufLen));
|
|
::CopyMemory(m_pDBuffer, &nBufLen, sizeof(nBufLen));
|
|
::CopyMemory(m_pDBuffer + sizeof(nBufLen), pBuffer, nBufLen);
|
|
m_nDDataLen = nBufLen + sizeof(nBufLen);
|
|
}
|
|
|
|
void Base64Coder::Encode(const BYTE *pBuffer, UINT nBufLen)
|
|
{
|
|
SetDecodeBuffer(pBuffer, nBufLen);
|
|
// include length
|
|
nBufLen += sizeof(nBufLen);
|
|
AllocEncode(nBufLen * 2);
|
|
|
|
TempBucket Raw;
|
|
UINT nIndex = 0;
|
|
|
|
while((nIndex + 3) <= nBufLen)
|
|
{
|
|
Raw.Clear();
|
|
::CopyMemory(Raw.nData, m_pDBuffer + nIndex, 3);
|
|
Raw.nSize = 3;
|
|
_EncodeToBuffer(Raw, m_pEBuffer + m_nEDataLen);
|
|
nIndex += 3;
|
|
m_nEDataLen += 4;
|
|
}
|
|
|
|
if(nBufLen > nIndex)
|
|
{
|
|
Raw.Clear();
|
|
Raw.nSize = (BYTE) (nBufLen - nIndex);
|
|
::CopyMemory(Raw.nData, m_pDBuffer + nIndex, nBufLen - nIndex);
|
|
_EncodeToBuffer(Raw, m_pEBuffer + m_nEDataLen);
|
|
m_nEDataLen += 4;
|
|
}
|
|
}
|
|
|
|
void Base64Coder::Decode(const LPCWSTR pBuffer)
|
|
{
|
|
UINT dwBufLen = (UINT) wcslen(pBuffer);
|
|
|
|
if(!Base64Coder::m_Init)
|
|
_Init();
|
|
|
|
SetEncodeBuffer(pBuffer, dwBufLen);
|
|
|
|
AllocDecode(dwBufLen);
|
|
|
|
TempBucket Raw;
|
|
|
|
UINT nIndex = 0;
|
|
|
|
while((nIndex + 4) <= m_nEDataLen)
|
|
{
|
|
Raw.Clear();
|
|
Raw.nData[0] = Base64Coder::m_DecodeTable[m_pEBuffer[nIndex]];
|
|
Raw.nData[1] = Base64Coder::m_DecodeTable[m_pEBuffer[nIndex + 1]];
|
|
Raw.nData[2] = Base64Coder::m_DecodeTable[m_pEBuffer[nIndex + 2]];
|
|
Raw.nData[3] = Base64Coder::m_DecodeTable[m_pEBuffer[nIndex + 3]];
|
|
|
|
if(Raw.nData[2] == 255)
|
|
Raw.nData[2] = 0;
|
|
if(Raw.nData[3] == 255)
|
|
Raw.nData[3] = 0;
|
|
|
|
Raw.nSize = 4;
|
|
_DecodeToBuffer(Raw, m_pDBuffer + m_nDDataLen);
|
|
nIndex += 4;
|
|
m_nDDataLen += 3;
|
|
}
|
|
|
|
// If nIndex < m_nEDataLen, then we got a decode message without padding.
|
|
// We may want to throw some kind of warning here, but we are still required
|
|
// to handle the decoding as if it was properly padded.
|
|
if(nIndex < m_nEDataLen)
|
|
{
|
|
Raw.Clear();
|
|
for(UINT i = nIndex; i < m_nEDataLen; i++)
|
|
{
|
|
Raw.nData[i - nIndex] = Base64Coder::m_DecodeTable[m_pEBuffer[i]];
|
|
Raw.nSize++;
|
|
if(Raw.nData[i - nIndex] == 255)
|
|
Raw.nData[i - nIndex] = 0;
|
|
}
|
|
|
|
_DecodeToBuffer(Raw, m_pDBuffer + m_nDDataLen);
|
|
m_nDDataLen += (m_nEDataLen - nIndex);
|
|
}
|
|
}
|
|
|
|
UINT Base64Coder::_DecodeToBuffer(const TempBucket &Decode, BYTE *pBuffer)
|
|
{
|
|
TempBucket Data;
|
|
UINT nCount = 0;
|
|
|
|
_DecodeRaw(Data, Decode);
|
|
|
|
for(int i = 0; i < 3; i++)
|
|
{
|
|
pBuffer[i] = Data.nData[i];
|
|
if(pBuffer[i] != 255)
|
|
nCount++;
|
|
}
|
|
|
|
return nCount;
|
|
}
|
|
|
|
|
|
void Base64Coder::_EncodeToBuffer(const TempBucket &Decode, LPWSTR pBuffer)
|
|
{
|
|
TempBucket Data;
|
|
|
|
_EncodeRaw(Data, Decode);
|
|
|
|
for(int i = 0; i < 4; i++)
|
|
pBuffer[i] = Base64Digits[Data.nData[i]];
|
|
|
|
switch(Decode.nSize)
|
|
{
|
|
case 1:
|
|
pBuffer[2] = '=';
|
|
case 2:
|
|
pBuffer[3] = '=';
|
|
}
|
|
}
|
|
|
|
void Base64Coder::_DecodeRaw(TempBucket &Data, const TempBucket &Decode)
|
|
{
|
|
BYTE nTemp;
|
|
|
|
Data.nData[0] = Decode.nData[0];
|
|
Data.nData[0] <<= 2;
|
|
|
|
nTemp = Decode.nData[1];
|
|
nTemp >>= 4;
|
|
nTemp &= 0x03;
|
|
Data.nData[0] |= nTemp;
|
|
|
|
Data.nData[1] = Decode.nData[1];
|
|
Data.nData[1] <<= 4;
|
|
|
|
nTemp = Decode.nData[2];
|
|
nTemp >>= 2;
|
|
nTemp &= 0x0F;
|
|
Data.nData[1] |= nTemp;
|
|
|
|
Data.nData[2] = Decode.nData[2];
|
|
Data.nData[2] <<= 6;
|
|
nTemp = Decode.nData[3];
|
|
nTemp &= 0x3F;
|
|
Data.nData[2] |= nTemp;
|
|
}
|
|
|
|
void Base64Coder::_EncodeRaw(TempBucket &Data, const TempBucket &Decode)
|
|
{
|
|
BYTE nTemp;
|
|
|
|
Data.nData[0] = Decode.nData[0];
|
|
Data.nData[0] >>= 2;
|
|
|
|
Data.nData[1] = Decode.nData[0];
|
|
Data.nData[1] <<= 4;
|
|
nTemp = Decode.nData[1];
|
|
nTemp >>= 4;
|
|
Data.nData[1] |= nTemp;
|
|
Data.nData[1] &= 0x3F;
|
|
|
|
Data.nData[2] = Decode.nData[1];
|
|
Data.nData[2] <<= 2;
|
|
|
|
nTemp = Decode.nData[2];
|
|
nTemp >>= 6;
|
|
|
|
Data.nData[2] |= nTemp;
|
|
Data.nData[2] &= 0x3F;
|
|
|
|
Data.nData[3] = Decode.nData[2];
|
|
Data.nData[3] &= 0x3F;
|
|
}
|
|
|
|
BOOL Base64Coder::_IsBadMimeChar(WCHAR nData)
|
|
{
|
|
switch(nData)
|
|
{
|
|
case L'\r': case L'\n': case L'\t': case L' ' :
|
|
case L'\b': case L'\a': case L'\f': case L'\v':
|
|
return TRUE;
|
|
default:
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
void Base64Coder::_Init()
|
|
{
|
|
// Initialize Decoding table.
|
|
int i;
|
|
|
|
for(i = 0; i < 256; i++)
|
|
Base64Coder::m_DecodeTable[i] = -2;
|
|
|
|
for(i = 0; i < 64; i++)
|
|
{
|
|
Base64Coder::m_DecodeTable[Base64Digits[i]] = (char) i;
|
|
Base64Coder::m_DecodeTable[Base64Digits[i]|0x80] = (char) i;
|
|
}
|
|
|
|
Base64Coder::m_DecodeTable['='] = -1;
|
|
Base64Coder::m_DecodeTable['='|0x80] = -1;
|
|
|
|
Base64Coder::m_Init = TRUE;
|
|
}
|
|
|