perplex.cpp
上传用户:dangjiwu
上传日期:2013-07-19
资源大小:42019k
文件大小:21k
源码类别:

Symbian

开发平台:

Visual C++

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