perplex.cpp
上传用户:zhongxx05
上传日期:2007-06-06
资源大小:33641k
文件大小:20k
源码类别:

Symbian

开发平台:

C/C++

  1. /* ***** BEGIN LICENSE BLOCK ***** 
  2.  * Version: RCSL 1.0/RPSL 1.0 
  3.  *  
  4.  * Portions Copyright (c) 1995-2002 RealNetworks, Inc. All Rights Reserved. 
  5.  *      
  6.  * The contents of this file, and the files included with this file, are 
  7.  * subject to the current version of the RealNetworks Public Source License 
  8.  * Version 1.0 (the "RPSL") available at 
  9.  * http://www.helixcommunity.org/content/rpsl unless you have licensed 
  10.  * the file under the RealNetworks Community Source License Version 1.0 
  11.  * (the "RCSL") available at http://www.helixcommunity.org/content/rcsl, 
  12.  * in which case the RCSL will apply. You may also obtain the license terms 
  13.  * directly from RealNetworks.  You may not use this file except in 
  14.  * compliance with the RPSL or, if you have a valid RCSL with RealNetworks 
  15.  * applicable to this file, the RCSL.  Please see the applicable RPSL or 
  16.  * RCSL for the rights, obligations and limitations governing use of the 
  17.  * contents of the file.  
  18.  *  
  19.  * This file is part of the Helix DNA Technology. RealNetworks is the 
  20.  * developer of the Original Code and owns the copyrights in the portions 
  21.  * it created. 
  22.  *  
  23.  * This file, and the files included with this file, is distributed and made 
  24.  * available on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER 
  25.  * EXPRESS OR IMPLIED, AND REALNETWORKS HEREBY DISCLAIMS ALL SUCH WARRANTIES, 
  26.  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY, FITNESS 
  27.  * FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT. 
  28.  * 
  29.  * Technology Compatibility Kit Test Suite(s) Location: 
  30.  *    http://www.helixcommunity.org/content/tck 
  31.  * 
  32.  * Contributor(s): 
  33.  *  
  34.  * ***** END LICENSE BLOCK ***** */ 
  35. //*******************************************************************
  36. //*************************** INCLUDE FILES *************************
  37. //*******************************************************************
  38. #include "perplex.h"
  39. #include "hxassert.h"
  40. #include "hlxclib/string.h"
  41. // #include "hlxclib/stdio.h"
  42. #include "netbyte.h"
  43. #include "hxheap.h"
  44. #ifdef _DEBUG
  45. #undef HX_THIS_FILE
  46. static const char HX_THIS_FILE[] = __FILE__;
  47. #endif
  48. //*******************************************************************
  49. //************************* INTERFACE LISTING ***********************
  50. //*******************************************************************
  51. BEGIN_INTERFACE_LIST(CHXPerplex)
  52.     INTERFACE_LIST_ENTRY(IID_IHXPerplex, IHXPerplex)
  53. END_INTERFACE_LIST
  54. //*******************************************************************
  55. //*************************** CONSTRUCTOR ***************************
  56. //*******************************************************************
  57. CHXPerplex::CHXPerplex()
  58. {
  59. ;
  60. }
  61. //*******************************************************************
  62. //*************************** DESTRUCTOR ****************************
  63. //*******************************************************************
  64. CHXPerplex::~CHXPerplex()
  65. {
  66.   ;
  67. }
  68. //*******************************************************************
  69. //*************************** Perplex() *****************************
  70. //*******************************************************************
  71. STDMETHODIMP CHXPerplex::Perplex(IHXBuffer *pInBuf, IHXBuffer *pOutBuf)
  72. {
  73.     UINT32 FinalLen;     // length of encoded data
  74.     CHXPerplexBuffer pInPBuf;     // Temp buffer to keep code
  75.     // copy the IRMA Input Buffer into the CHXPerplex Input Buffer
  76.     pInPBuf.SafeMemCopy(0, (const void*) pInBuf->GetBuffer(), (UINT32)pInBuf->GetSize());
  77.     // Make sure the input buffer size is a multiple of 4bytes as 
  78.     // needed by Perplex alg.  Padding with '' otherwise.
  79.     UINT32 nAlign = (pInBuf->GetSize()) % sizeof(ULONG32);
  80.     UINT32 Offset = (UINT32)pInBuf->GetSize();
  81.     if (nAlign > 0)
  82.     {
  83. pInPBuf.EnsureValidOffset(Offset+sizeof(ULONG32)-nAlign);
  84. for (; nAlign < sizeof(ULONG32); nAlign++)
  85. {
  86.     pInPBuf.SetUCHAR(Offset++,0);
  87. }
  88.     }
  89.     FinalLen = Offset * Perplex_PER_ULONG32 / sizeof(ULONG32); // calc size of the outout (Perplexed) buffer.
  90.     // alloc mem for final perplexed buffer
  91.     // Add one to length 'cause low level perplex adds
  92.     // a '' to the resulting string
  93.     pOutBuf->SetSize(FinalLen+1);
  94.     if (pOutBuf->GetBuffer() == NULL) return(HXR_FAIL);
  95.     CHXPerplex::DumpToPerplex((char*)(pOutBuf->GetBuffer()), FinalLen+1, pInPBuf.GetPtr(), Offset);
  96.     return(HXR_OK);
  97. }
  98. //*******************************************************************
  99. //*************************** DePerplex() ***************************
  100. //*******************************************************************
  101. // Parameters:
  102. //
  103. // const char* Perplex
  104. // Pointer to a buffer that contains a Perplexidecimal encoding 
  105. //
  106. // UCHAR* Bits
  107. // Pointer to a buffer that will contain decoded bytes of information
  108. //
  109. // int nSize
  110. // Input size in bytes
  111. STDMETHODIMP CHXPerplex::DePerplex(IHXBuffer *pInBuf, IHXBuffer *pOutBuf)
  112. {
  113.     const char* pPerplex    = (const char *)pInBuf->GetBuffer();
  114.     UCHAR* Bits     = NULL;
  115.     UINT32 nSize     = pInBuf->GetSize();
  116.     if (!nSize)
  117.     {
  118. pOutBuf->SetSize(0);
  119. return HXR_OK;
  120.     }
  121.     
  122.     nSize--;  // -1 is for the extra character added in Perplex
  123.     UINT32 ndxBits;
  124.     UINT32 ndxPerplex = 0;
  125.     ULONG32 temp32;
  126.     UINT32      ulOutSize = 2*nSize+100;
  127.     pOutBuf->SetSize(ulOutSize);     // Make it bigger since the real size isn't known yet
  128.     Bits = (UCHAR *)pOutBuf->GetBuffer();
  129.     for (ndxBits = 0; ndxPerplex < nSize; )
  130.     {
  131. temp32 = FromPerplex(&pPerplex[ndxPerplex]);
  132. ndxPerplex+=Perplex_PER_ULONG32;
  133. if (ndxBits+4 <= ulOutSize) memcpy(&Bits[ndxBits],&temp32,sizeof(temp32)); /* Flawfinder: ignore */
  134. ndxBits+=sizeof(temp32);
  135.     }
  136.     pOutBuf->SetSize(ndxBits);
  137.     return HXR_OK;
  138. }
  139. //*******************************************************************
  140. //************************* MapFromPerplex **************************
  141. //*******************************************************************
  142. // Converts appropriate characters for Perplex encoding into a ULONG32.
  143. // A copy of this is kept in in pnmiscpubwinpnmisc16.h.  If you change 
  144. // this (which is very unlikely), then be sure to change it there.  
  145. static const char zPerplexChars[] =
  146. {
  147. '0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
  148. 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j',
  149. 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't',
  150. 'u', 'v', 'w', 'x', 'y', 'z', 'A', 'B', 'C', 'D',
  151. 'E'
  152. };
  153. UCHAR CHXPerplex::MapFromPerplex(char Perplex)
  154. {
  155.     UCHAR temp = sizeof(zPerplexChars)/sizeof(zPerplexChars[0]);
  156.     for (UCHAR n = 0; n < temp; n++)
  157.     {
  158. if (Perplex == zPerplexChars[n]) return n;
  159.     }
  160.     return 0;
  161. }
  162. //*******************************************************************
  163. //*************************** MapToPerplex **************************
  164. //*******************************************************************
  165. // Converts a digit ordinal to the Perplex digit...
  166. char CHXPerplex::MapToPerplex(UCHAR cPerplex)
  167. {                                                  
  168. #ifdef _DEBUG
  169.     int size_zPerplexChars  = sizeof(zPerplexChars); 
  170.     int size_zPerplexChars0 = sizeof(zPerplexChars[0]);
  171.     HX_ASSERT(cPerplex < size_zPerplexChars/size_zPerplexChars0);
  172. #endif
  173.     return zPerplexChars[cPerplex];
  174. }
  175. //*******************************************************************
  176. //*************************** FromPerplex ***************************
  177. //*******************************************************************
  178. // Converts appropriate characters for Perplex encoding into a ULONG32.
  179. ULONG32 CHXPerplex::FromPerplex(const char* Perplex)
  180. {
  181. ULONG32 value = 0;
  182. ULONG32 PerplexBase = 1;
  183. for (int n = 0; n < Perplex_PER_ULONG32; n++)
  184. {
  185. value += MapFromPerplex(Perplex[n]) * PerplexBase;
  186. PerplexBase *= Perplex_BASE;
  187. }
  188. // Convert to local byte order!
  189. value = DwToHost(value);
  190. return value;
  191. }
  192. //*******************************************************************
  193. //**************************** ToPerplex ****************************
  194. //*******************************************************************
  195. // Converts a ULONG32 into the appropriate characters for Perplex encoding.
  196. void CHXPerplex::ToPerplex(ULONG32 Input, char* pPerplex)
  197. {
  198.     ULONG32 value;
  199.     UCHAR   charValue;
  200.     value = DwToNet(Input); // Convert to net byte order!
  201.     for (int n=0; n < Perplex_PER_ULONG32; n++)
  202.     {
  203.      charValue   = (UCHAR)(value % Perplex_BASE);
  204. pPerplex[n] = MapToPerplex(charValue);
  205. value = value / Perplex_BASE;
  206.     }
  207. }
  208. //*******************************************************************
  209. //*************************** DumpToPerplex *************************
  210. //*******************************************************************
  211. // Parameters:
  212. //
  213. // char* pPerplex
  214. // Pointer to buffer to be filled with Perplexadecimal 
  215. void CHXPerplex::DumpToPerplex(char* pPerplex, UINT32 ulPerplexSize, UCHAR* Bits, UINT32 nSize)
  216. {
  217.     UINT32 ndxBits;
  218.     UINT32 ndxPerplex = 0;
  219.     ULONG32 temp32;
  220.     for (ndxBits = 0; ndxBits < nSize; )
  221.     {
  222. if (ndxBits+4 <= nSize) memcpy(&temp32,&Bits[ndxBits],sizeof(temp32)); /* Flawfinder: ignore */
  223. ndxBits+=sizeof(temp32);
  224. if (ndxPerplex+Perplex_PER_ULONG32 <= ulPerplexSize) ToPerplex(temp32,&pPerplex[ndxPerplex]);
  225. ndxPerplex+=Perplex_PER_ULONG32;
  226.     }
  227.     pPerplex[ndxPerplex] = '';
  228. }
  229. //*******************************************************************
  230. //*****                                                        ******
  231. //*****                                                        ******
  232. //*****                                                        ******
  233. //*****                                                        ******
  234. //*****                                                        ******
  235. //*****                                                        ******
  236. //*****                     CHXPerplexBuffer                   ******
  237. //*****                                                        ******
  238. //*****                                                        ******
  239. //*****                                                        ******
  240. //*****                                                        ******
  241. //*****                                                        ******
  242. //*****                                                        ******
  243. //*******************************************************************
  244. //*******************************************************************
  245. //****************** CHXPerplexBuffer CONSTRUCTOR *******************
  246. //*******************************************************************
  247. CHXPerplexBuffer::CHXPerplexBuffer()
  248. :  m_nSize(0), m_pData(NULL), m_nGrowBy(DEFAULT_GROW_SIZE)
  249. {
  250. }
  251. //*******************************************************************
  252. //****************** CHXPerplexBuffer CONSTRUCTOR *******************
  253. //*******************************************************************
  254. CHXPerplexBuffer::CHXPerplexBuffer(UINT32 nSize, UINT32 nGrowBy)
  255. :  m_nSize(0), m_pData(NULL), m_nGrowBy(nGrowBy)
  256. {
  257. Resize(nSize);
  258. }
  259. //*******************************************************************
  260. //******************* CHXPerplexBuffer DESTRUCTOR *******************
  261. //*******************************************************************
  262. CHXPerplexBuffer::~CHXPerplexBuffer()
  263. {
  264. Free();
  265. }
  266. //*******************************************************************
  267. //****************** CHXPerplexBuffer::IsValidOffset() **************
  268. //*******************************************************************
  269. inline BOOL CHXPerplexBuffer::IsValidOffset(UINT32 n) const
  270. {
  271. if (n<m_nSize)
  272. return(TRUE);
  273. else return(FALSE);
  274. }
  275. //*******************************************************************
  276. //*************** CHXPerplexBuffer::EnsureValidOffset() *************
  277. //*******************************************************************
  278. // make sure that pPos is in the buffer.
  279. // If not, the buffer is automatically resized and contents copied.
  280. // pPos must be > pData.
  281. BOOL CHXPerplexBuffer::EnsureValidOffset(UINT32 n)
  282. {
  283.     if (IsValidOffset(n)==TRUE)
  284.     {
  285. return(TRUE);
  286.     }
  287.     return (this->Resize(n));
  288. }
  289. //*******************************************************************
  290. //********************* CHXPerplexBuffer::Resize() ******************
  291. //*******************************************************************
  292. BOOL CHXPerplexBuffer::Resize(UINT32 nNewSize)
  293. {
  294.     if (nNewSize==0)
  295.     {
  296. Free();
  297. return(TRUE);
  298.     }
  299.     nNewSize = RoundUpToGrowSize(nNewSize);
  300.     UCHAR* pNewBuffer = new UCHAR[nNewSize];
  301.     if (pNewBuffer==NULL) return(FALSE);
  302.     if (m_pData!=NULL)
  303.     {
  304. // copy MIN(oldSize,newSize) elements
  305. memcpy(pNewBuffer,m_pData, (nNewSize<m_nSize) ? (int)nNewSize : (int)m_nSize); /* Flawfinder: ignore */
  306. delete [] m_pData;
  307.     }
  308.     m_pData = pNewBuffer;
  309.     m_nSize = nNewSize;
  310.     return(TRUE);
  311. }
  312. //*******************************************************************
  313. //**************** CHXPerplexBuffer::RoundUpToGrowSize() ************
  314. //*******************************************************************
  315. UINT32 CHXPerplexBuffer::RoundUpToGrowSize(UINT32 nSize)
  316. {
  317.     // check for div-by-zero errors (as long as DEFAULT!=0)
  318.     if (m_nGrowBy==0)
  319.     {
  320. m_nGrowBy = DEFAULT_GROW_SIZE;
  321.     }
  322.     return ( (int)(nSize/m_nGrowBy)+1 ) * m_nGrowBy;
  323. }
  324. //*******************************************************************
  325. //****************** CHXPerplexBuffer::SafeMemCopy() ****************
  326. //*******************************************************************
  327. // copy data into the buffer at the given offset, and if the buffer
  328. // is too small we'll resize the buffer first.
  329. BOOL CHXPerplexBuffer::SafeMemCopy(UINT32 nOffset, const void* data, UINT32 len)
  330. {
  331.     if (EnsureValidOffset(nOffset+len-1)==TRUE)
  332.     {
  333.      memcpy( m_pData+nOffset, data, (int)len ); /* Flawfinder: ignore */
  334. return(TRUE);
  335.     }
  336.     return(FALSE);
  337. }
  338. //*******************************************************************
  339. //********************** CHXPerplexBuffer::Free() *******************
  340. //*******************************************************************
  341. void CHXPerplexBuffer::Free()
  342. {
  343.     if (m_pData!=NULL)
  344.     delete [] m_pData;
  345.     m_pData=NULL;
  346.     m_nSize=0;
  347. }
  348. //*******************************************************************
  349. //************************** FourByteAlign **************************
  350. //*******************************************************************
  351. /*  already in pnmisc.lib
  352. UINT16  FourByteAlign(UINT16 number)
  353. {
  354. return( ((number+3)>>2)<<2 );
  355. }
  356. */
  357. //*******************************************************************
  358. //************************ MapFromMIMEBase64 ************************
  359. //*******************************************************************
  360. //  Converts appropriate characters for MIME-Base64 encoding into a the
  361. //  Base64 ordinal.
  362. // A copy of this is kept in in pnmiscpubwinpnmisc16.h.  If you change 
  363. // this (which is very unlikely), then be sure to change it there.  
  364. static const char zMIMEBase64Chars[] =
  365. {
  366. 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J',
  367. 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T',
  368. 'U', 'V', 'W', 'X', 'Y', 'Z', 'a', 'b', 'c', 'd',
  369. 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n',
  370. 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x',
  371. 'y', 'z', '0', '1', '2', '3', '4', '5', '6', '7',
  372. '8', '9', '+', '/'
  373. };
  374. #define MIME_BASE64_PADDING '='
  375. #define FROM_MIME_BASE64_BYTES_IN 4
  376. #define FROM_MIME_BASE64_BYTES_OUT 3
  377. #define TO_MIME_BASE64_BYTES_IN FROM_MIME_BASE64_BYTES_OUT
  378. #define TO_MIME_BASE64_BYTES_OUT FROM_MIME_BASE64_BYTES_IN
  379. UCHAR CHXPerplex::MapFromMIMEBase64(char MIMEBase64)
  380. {
  381.     for (UCHAR n = 0; n < sizeof(zMIMEBase64Chars)/sizeof(zMIMEBase64Chars[0]); n++)
  382.     {
  383. if (MIMEBase64 == zMIMEBase64Chars[n])
  384. {
  385.     return n;
  386. }
  387.     }
  388.     HX_ASSERT(FALSE);
  389.     return 0;
  390. }
  391. //*******************************************************************
  392. //************************ MapFromMIMEBase64 ************************
  393. //*******************************************************************
  394. // Converts a digit ordinal to the MIMEBase64 digit...
  395. char CHXPerplex::MapToMIMEBase64(UCHAR MIMEBase64)
  396. {
  397.     HX_ASSERT(MIMEBase64 < sizeof(zMIMEBase64Chars)/sizeof(zMIMEBase64Chars[0]));
  398.     return zMIMEBase64Chars[MIMEBase64];
  399. }
  400. //*******************************************************************
  401. //************************ SetFromMIMEBase64 ************************
  402. //*******************************************************************
  403. // Return: int Size in bytes of decoded information
  404. UINT32 CHXPerplex::SetFromMIMEBase64(const char* MIMEBase64, char* Bits, UINT32 nSize)
  405. {
  406.     UINT32 ndxBits = 0;
  407.     UINT32 ndxMIMEBase64 = 0;
  408.     BOOL bDone = FALSE;
  409.     UINT32 ndxTempIn = 0;
  410.     UCHAR tempBitsIn[FROM_MIME_BASE64_BYTES_IN];
  411.     UINT32 padding = 0;
  412.     HX_ASSERT(strlen(MIMEBase64) <= nSize);
  413.     while (!bDone)
  414.     {
  415. HX_ASSERT(ndxMIMEBase64 <= nSize);
  416. HX_ASSERT(ndxBits <= nSize);
  417. // Fill in our "temporary" in buffer with the Base64 characters.
  418. for (ndxTempIn = 0;
  419.      ndxTempIn < FROM_MIME_BASE64_BYTES_IN && (padding == 0);
  420.      ndxTempIn++)
  421. {
  422.     HX_ASSERT(ndxMIMEBase64 <= nSize);
  423.     UCHAR MIMEBase64char = MIMEBase64[ndxMIMEBase64];
  424.     switch (MIMEBase64char)
  425.     {
  426. case MIME_BASE64_PADDING:
  427. case '':
  428. {
  429.     tempBitsIn[ndxTempIn] = 0;
  430.     bDone = TRUE;
  431.     padding = (FROM_MIME_BASE64_BYTES_IN - ndxTempIn);
  432. }
  433. break;
  434. default:
  435. {
  436.     tempBitsIn[ndxTempIn] = MapFromMIMEBase64(MIMEBase64char);
  437. }
  438. break;
  439.     }
  440.     ndxMIMEBase64++;
  441. }
  442. HX_ASSERT(padding == 2 || padding == 1 || padding == 0);
  443. // Map the Base64 in buffer to the the output buffer...
  444. // This should map the 6 pertinate bits of each IN byte, to the
  445. // the correct bits of the OUT byte.
  446. {
  447.     Bits[ndxBits] = (tempBitsIn[0] << 2) + (tempBitsIn[1]>>4);
  448.     ndxBits++;
  449.     if (padding < 2)
  450.     {
  451. Bits[ndxBits] = (tempBitsIn[1] << 4) + (tempBitsIn[2]>>2);
  452. ndxBits++;
  453.     }
  454.     if (padding < 1)
  455.     {
  456. Bits[ndxBits] = (tempBitsIn[2] << 6) + (tempBitsIn[3]);
  457. ndxBits++;
  458.     }
  459. }
  460.     }
  461.     Bits[ndxBits] = '';
  462.     return ndxBits;
  463. }
  464. //*******************************************************************
  465. //************************ DumpToMIMEBase64 *************************
  466. //*******************************************************************
  467. void CHXPerplex::DumpToMIMEBase64(char* MIMEBase64, const char* Bits, UINT32 nSize)
  468. {
  469.     UINT32 ndxBits = 0;
  470.     UINT32 ndxMIMEBase64 = 0;
  471.     BOOL bDone = FALSE;
  472.     UINT32 ndxTempIn = 0;
  473.     UINT32 ndxTempOut = 0;
  474.     UCHAR tempBitsIn [FROM_MIME_BASE64_BYTES_IN];
  475.     UINT32 padding = 0;
  476.     HX_ASSERT(strlen(Bits) <= nSize);
  477.     while (!bDone)
  478.     {
  479. HX_ASSERT(ndxMIMEBase64 <= nSize);
  480. HX_ASSERT(ndxBits <= nSize);
  481. // Fill in our "temporary" out buffer with the 6 bit chunks of the input.
  482. for ( ndxTempIn = 0;
  483. ndxTempIn < TO_MIME_BASE64_BYTES_IN && (padding == 0);
  484. ndxTempIn++)
  485. {
  486.     UCHAR rawChar = Bits[ndxBits];
  487.     if (rawChar != '')
  488.     {
  489. switch (ndxTempIn)
  490. {
  491.     case 0:
  492.     {
  493. tempBitsIn[0] = rawChar >> 2;
  494. tempBitsIn[1] = (rawChar & 0x3) << 4;
  495.     }
  496.     break;
  497.     case 1:
  498.     {
  499. tempBitsIn[1] += rawChar >> 4;
  500. tempBitsIn[2] = (rawChar & 0xF) << 2;
  501.     }
  502.     break;
  503.     case 2:
  504.     {
  505. tempBitsIn[2] += rawChar >> 6;
  506. tempBitsIn[3] = (rawChar & 0x3F);
  507.     }
  508.     break;
  509. }
  510.     }
  511.     else
  512.     {
  513. bDone = TRUE;
  514. padding = (TO_MIME_BASE64_BYTES_IN - ndxTempIn);
  515.     }
  516.     ndxBits++;
  517. }
  518. HX_ASSERT(padding == 2 || padding == 1 || padding == 0);
  519. // Map the Base64 in buffer to the output buffer...
  520. // This should map the 6 pertinate bits of each IN byte, as an
  521. // entire OUT byte
  522. for ( ndxTempOut = 0;
  523. ndxTempOut < TO_MIME_BASE64_BYTES_OUT;
  524. ndxTempOut++)
  525. {
  526.     if (ndxTempOut < (TO_MIME_BASE64_BYTES_OUT-padding))
  527.     {
  528. MIMEBase64[ndxMIMEBase64] = MapToMIMEBase64(tempBitsIn[ndxTempOut]);
  529.     }
  530.     else
  531.     {
  532. MIMEBase64[ndxMIMEBase64] = MIME_BASE64_PADDING;
  533.     }
  534.     ndxMIMEBase64++;
  535. }
  536.     }
  537.     MIMEBase64[ndxMIMEBase64] = '';
  538. }