Iconv390TransService.cpp
上传用户:huihehuasu
上传日期:2007-01-10
资源大小:6948k
文件大小:19k
源码类别:

xml/soap/webservice

开发平台:

C/C++

  1. /*
  2.  * The Apache Software License, Version 1.1
  3.  *
  4.  * Copyright (c) 1999-2000 The Apache Software Foundation.  All rights
  5.  * reserved.
  6.  *
  7.  * Redistribution and use in source and binary forms, with or without
  8.  * modification, are permitted provided that the following conditions
  9.  * are met:
  10.  *
  11.  * 1. Redistributions of source code must retain the above copyright
  12.  *    notice, this list of conditions and the following disclaimer.
  13.  *
  14.  * 2. Redistributions in binary form must reproduce the above copyright
  15.  *    notice, this list of conditions and the following disclaimer in
  16.  *    the documentation and/or other materials provided with the
  17.  *    distribution.
  18.  *
  19.  * 3. The end-user documentation included with the redistribution,
  20.  *    if any, must include the following acknowledgment:
  21.  *       "This product includes software developed by the
  22.  *        Apache Software Foundation (http://www.apache.org/)."
  23.  *    Alternately, this acknowledgment may appear in the software itself,
  24.  *    if and wherever such third-party acknowledgments normally appear.
  25.  *
  26.  * 4. The names "Xerces" and "Apache Software Foundation" must
  27.  *    not be used to endorse or promote products derived from this
  28.  *    software without prior written permission. For written
  29.  *    permission, please contact apache@apache.org.
  30.  *
  31.  * 5. Products derived from this software may not be called "Apache",
  32.  *    nor may "Apache" appear in their name, without prior written
  33.  *    permission of the Apache Software Foundation.
  34.  *
  35.  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED
  36.  * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
  37.  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  38.  * DISCLAIMED.  IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR
  39.  * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  40.  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  41.  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
  42.  * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  43.  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
  44.  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
  45.  * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  46.  * SUCH DAMAGE.
  47.  * ====================================================================
  48.  *
  49.  * This software consists of voluntary contributions made by many
  50.  * individuals on behalf of the Apache Software Foundation, and was
  51.  * originally based on software copyright (c) 1999, International
  52.  * Business Machines, Inc., http://www.ibm.com .  For more information
  53.  * on the Apache Software Foundation, please see
  54.  * <http://www.apache.org/>.
  55.  */
  56. /*
  57.  * $Id: Iconv390TransService.cpp,v 1.12 2001/07/27 14:52:49 tng Exp $
  58.  */
  59. // ---------------------------------------------------------------------------
  60. //  Includes
  61. // ---------------------------------------------------------------------------
  62. #include <util/XMLUniDefs.hpp>
  63. #include <util/XMLUni.hpp>
  64. #include "Iconv390TransService.hpp"
  65. #include <wchar.h>
  66. #include <string.h>
  67. #include <stdlib.h>
  68. #include <stdio.h>
  69. #ifdef OS390BATCH
  70. #include <unistd.h>
  71. #endif
  72. #include <ctype.h>
  73. //
  74. //  Cannot use the OS/390 c/c++ towupper and towlower functions in the
  75. //  Unicode environment. We will use mytowupper and mytowlower here.
  76. //
  77. #undef towupper
  78. #undef towlower
  79. #define towupper mytowupper
  80. #define towlower mytowlower
  81. // ---------------------------------------------------------------------------
  82. //  Local, const data
  83. // ---------------------------------------------------------------------------
  84. static const int gTempBuffArraySize = 1024;
  85. static const XMLCh  gMyServiceId[] =
  86. {
  87.     chLatin_I, chLatin_C, chLatin_o, chLatin_n, chLatin_v, chNull
  88. };
  89. // ---------------------------------------------------------------------------
  90. //  gUnicodeToIBM037XlatTable
  91. //      This is the translation table for Unicode to ibm-037. This table
  92. //      contains 255 entries.
  93. // ---------------------------------------------------------------------------
  94. static const XMLByte gUnicodeToIBM037XlatTable[256] =
  95. {
  96.         0x00, 0x01, 0x02, 0x03, 0x37, 0x2D, 0x2E, 0x2F
  97.     ,   0x16, 0x05, 0x25, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F
  98.     ,   0x10, 0x11, 0x12, 0x13, 0x3C, 0x3D, 0x32, 0x26
  99.     ,   0x18, 0x19, 0x3F, 0x27, 0x1C, 0x1D, 0x1E, 0x1F
  100.     ,   0x40, 0x5A, 0x7F, 0x7B, 0x5B, 0x6C, 0x50, 0x7D
  101.     ,   0x4D, 0x5D, 0x5C, 0x4E, 0x6B, 0x60, 0x4B, 0x61
  102.     ,   0xF0, 0xF1, 0xF2, 0xF3, 0xF4, 0xF5, 0xF6, 0xF7
  103.     ,   0xF8, 0xF9, 0x7A, 0x5E, 0x4C, 0x7E, 0x6E, 0x6F
  104.     ,   0x7C, 0xC1, 0xC2, 0xC3, 0xC4, 0xC5, 0xC6, 0xC7
  105.     ,   0xC8, 0xC9, 0xD1, 0xD2, 0xD3, 0xD4, 0xD5, 0xD6
  106.     ,   0xD7, 0xD8, 0xD9, 0xE2, 0xE3, 0xE4, 0xE5, 0xE6
  107.     ,   0xE7, 0xE8, 0xE9, 0xBA, 0xE0, 0xBB, 0xB0, 0x6D
  108.     ,   0x79, 0x81, 0x82, 0x83, 0x84, 0x85, 0x86, 0x87
  109.     ,   0x88, 0x89, 0x91, 0x92, 0x93, 0x94, 0x95, 0x96
  110.     ,   0x97, 0x98, 0x99, 0xA2, 0xA3, 0xA4, 0xA5, 0xA6
  111.     ,   0xA7, 0xA8, 0xA9, 0xC0, 0x4F, 0xD0, 0xA1, 0x07
  112.     ,   0x20, 0x21, 0x22, 0x23, 0x24, 0x15, 0x06, 0x17
  113.     ,   0x28, 0x29, 0x2A, 0x2B, 0x2C, 0x09, 0x0A, 0x1B
  114.     ,   0x30, 0x31, 0x1A, 0x33, 0x34, 0x35, 0x36, 0x08
  115.     ,   0x38, 0x39, 0x3A, 0x3B, 0x04, 0x14, 0x3E, 0xFF
  116.     ,   0x41, 0xAA, 0x4A, 0xB1, 0x9F, 0xB2, 0x6A, 0xB5
  117.     ,   0xBD, 0xB4, 0x9A, 0x8A, 0x5F, 0xCA, 0xAF, 0xBC
  118.     ,   0x90, 0x8F, 0xEA, 0xFA, 0xBE, 0xA0, 0xB6, 0xB3
  119.     ,   0x9D, 0xDA, 0x9B, 0x8B, 0xB7, 0xB8, 0xB9, 0xAB
  120.     ,   0x64, 0x65, 0x62, 0x66, 0x63, 0x67, 0x9E, 0x68
  121.     ,   0x74, 0x71, 0x72, 0x73, 0x78, 0x75, 0x76, 0x77
  122.     ,   0xAC, 0x69, 0xED, 0xEE, 0xEB, 0xEF, 0xEC, 0xBF
  123.     ,   0x80, 0xFD, 0xFE, 0xFB, 0xFC, 0xAD, 0xAE, 0x59
  124.     ,   0x44, 0x45, 0x42, 0x46, 0x43, 0x47, 0x9C, 0x48
  125.     ,   0x54, 0x51, 0x52, 0x53, 0x58, 0x55, 0x56, 0x57
  126.     ,   0x8C, 0x49, 0xCD, 0xCE, 0xCB, 0xCF, 0xCC, 0xE1
  127.     ,   0x70, 0xDD, 0xDE, 0xDB, 0xDC, 0x8D, 0x8E, 0xDF
  128. };
  129. iconvconverter * converterList;
  130. XMLMutex  converterListMutex;
  131. iconvconverter * addConverter(const char* const EncodingName
  132.                              ,XMLTransService::Codes& resValue)
  133. {
  134.     XMLMutexLock lockConverterlist(&converterListMutex);
  135.     iconvconverter *tconv=converterList;
  136.     while ( (tconv) &&
  137.             (strcmp(tconv->name,EncodingName)) )
  138.       tconv = tconv->nextconverter;
  139.     if (tconv) {
  140.       tconv->usecount++;
  141.     }
  142.     else {
  143.       tconv = new iconvconverter;
  144.       strcpy(tconv->name,EncodingName);
  145.       tconv->usecount=1;
  146.       tconv->fIconv390Descriptor = iconv_open("UCS-2",EncodingName);
  147.       if (tconv->fIconv390Descriptor == (iconv_t)(-1)) {
  148.          resValue = XMLTransService::UnsupportedEncoding;
  149.          delete tconv;
  150.          return 0;
  151.       }
  152.       tconv->nextconverter = converterList;
  153.       converterList = tconv;
  154.     }
  155.     return tconv;
  156. }
  157. void removeConverter(iconvconverter* const converter)
  158. {
  159.     iconvconverter *pconv,*tconv;
  160.     tconv = 0;
  161.     if (converter) {
  162.       XMLMutexLock lockConverterlist(&converterListMutex);
  163.       if (--converter->usecount==0) {
  164.         tconv = converterList;
  165.         pconv = (iconvconverter*)&converterList;
  166.         while ( (tconv) && (tconv!=converter) ) {
  167.           pconv=tconv;
  168.           tconv=tconv->nextconverter;
  169.         }
  170.         pconv->nextconverter=tconv->nextconverter;
  171.       }
  172.     }
  173.     if (tconv) {
  174.       iconv_close(tconv->fIconv390Descriptor);
  175.       delete tconv;
  176.     }
  177. }
  178. // ---------------------------------------------------------------------------
  179. //  Local methods
  180. // ---------------------------------------------------------------------------
  181. static unsigned int  getWideCharLength(const XMLCh* const src)
  182. {
  183.     if (!src)
  184.         return 0;
  185.     unsigned int len = 0;
  186.     const XMLCh* pTmp = src;
  187.     while (*pTmp++)
  188.         len++;
  189.     return len;
  190. }
  191. // ---------------------------------------------------------------------------
  192. //  Iconv390TransService: Constructors and Destructor
  193. // ---------------------------------------------------------------------------
  194. Iconv390TransService::Iconv390TransService()
  195. {
  196. }
  197. Iconv390TransService::~Iconv390TransService()
  198. {
  199. }
  200. // ---------------------------------------------------------------------------
  201. //  Iconv390TransService: The virtual transcoding service API
  202. // ---------------------------------------------------------------------------
  203. int Iconv390TransService::compareIString(  const   XMLCh* const    comp1
  204.                                            , const XMLCh* const    comp2)
  205. {
  206.     const XMLCh* cptr1 = comp1;
  207.     const XMLCh* cptr2 = comp2;
  208.     while ( (*cptr1 != 0) && (*cptr2 != 0) )
  209.     {
  210.         wint_t wch1 = towupper(*cptr1);
  211.         wint_t wch2 = towupper(*cptr2);
  212.         if (wch1 != wch2)
  213.             break;
  214.         cptr1++;
  215.         cptr2++;
  216.     }
  217.     return (int) ( towupper(*cptr1) - towupper(*cptr2) );
  218. }
  219. int Iconv390TransService::compareNIString( const   XMLCh* const    comp1
  220.                                            , const XMLCh* const    comp2
  221.                                            , const unsigned int    maxChars)
  222. {
  223.     unsigned int  n = 0;
  224.     const XMLCh* cptr1 = comp1;
  225.     const XMLCh* cptr2 = comp2;
  226.     while (true && maxChars)
  227.     {
  228.         wint_t wch1 = towupper(*cptr1);
  229.         wint_t wch2 = towupper(*cptr2);
  230.         if (wch1 != wch2)
  231.             return (int) (wch1 - wch2);
  232.         // If either ended, then both ended, so equal
  233.         if (!*cptr1 || !*cptr2)
  234.             break;
  235.         cptr1++;
  236.         cptr2++;
  237.         //  Bump the count of chars done. If it equals the count then we
  238.         //  are equal for the requested count, so break out and return
  239.         //  equal.
  240.         n++;
  241.         if (n == maxChars)
  242.             break;
  243.     }
  244.     return 0;
  245. }
  246. const XMLCh* Iconv390TransService::getId() const
  247. {
  248.     return gMyServiceId;
  249. }
  250. bool Iconv390TransService::isSpace(const XMLCh toCheck) const
  251. {
  252.     return (iswspace(toCheck) != 0);
  253. }
  254. XMLLCPTranscoder* Iconv390TransService::makeNewLCPTranscoder()
  255. {
  256.     XMLTransService::Codes resValue;
  257.     // native MVS default code page is IBM-037
  258.     iconvconverter *tconv=addConverter("IBM-037",resValue);
  259.     if (tconv == 0) {
  260.         return 0;
  261.     }
  262.     return new Iconv390LCPTranscoder(tconv);
  263. }
  264. bool Iconv390TransService::supportsSrcOfs() const
  265. {
  266.     return true;
  267. }
  268. // ---------------------------------------------------------------------------
  269. //  Iconv390TransService: The protected virtual transcoding service API
  270. // ---------------------------------------------------------------------------
  271. XMLTranscoder*
  272. Iconv390TransService::makeNewXMLTranscoder(const   XMLCh* const            encodingName
  273.                                         ,       XMLTransService::Codes& resValue
  274.                                         , const unsigned int            )
  275. {
  276.     //
  277.     //  Translate the input encodingName from Unicode XMLCh format into
  278.     //  ibm-037 char format via the lookup table.
  279.     //
  280.     char charEncodingName[256];
  281.     const XMLCh*  srcPtr = encodingName;
  282.     char*         outPtr = charEncodingName;
  283.     while (*srcPtr != 0)
  284.         *outPtr++ = toupper(gUnicodeToIBM037XlatTable[*srcPtr++]);
  285.     *outPtr=0;
  286.     iconvconverter *tconv=addConverter(charEncodingName,resValue);
  287.     return new Iconv390Transcoder(tconv, encodingName, 0);
  288. }
  289. void Iconv390TransService::upperCase(XMLCh* const toUpperCase) const
  290. {
  291.     XMLCh* outPtr = toUpperCase;
  292.     while (*outPtr != 0) {
  293. if ((*outPtr >= 0x61) && (*outPtr <= 0x7A))
  294.     *outPtr = *outPtr - 0x20;
  295. outPtr++;
  296.     }
  297. }
  298. // ---------------------------------------------------------------------------
  299. unsigned int Iconv390LCPTranscoder::calcRequiredSize(const char* const srcText)
  300. {
  301.     unsigned int retVal;
  302.     if (!srcText)
  303.         return 0;
  304. #ifdef OS390BATCH
  305.     //
  306.     // Cannot use mbstowcs in a non-POSIX environment(???).
  307.     //
  308.     if (!__isPosixOn()) {
  309.         const unsigned charLen = mblen(srcText, MB_CUR_MAX);
  310.         if (charLen == -1)
  311.             return 0;
  312.         else {
  313.             if (charLen != 0)
  314.                 retVal = strlen(srcText)/charLen;
  315.             else
  316.                 retVal = charLen;
  317.         }
  318.     }
  319.     else
  320. #endif
  321.         retVal = ::mbstowcs(NULL, srcText, 0);
  322.     if (retVal == -1)
  323.         return 0;
  324.     return retVal;
  325. }
  326. unsigned int Iconv390LCPTranscoder::calcRequiredSize(const XMLCh* const srcText)
  327. {
  328.     if (!srcText)
  329.         return 0;
  330.     unsigned int  wLent = getWideCharLength(srcText);
  331.     wchar_t       tmpWideCharArr[gTempBuffArraySize];
  332.     wchar_t*      allocatedArray = 0;
  333.     wchar_t*      wideCharBuf = 0;
  334.     if (wLent >= gTempBuffArraySize)
  335.         wideCharBuf = allocatedArray = new wchar_t[wLent + 1];
  336.     else
  337.         wideCharBuf = tmpWideCharArr;
  338.     for (unsigned int i = 0; i < wLent; i++)
  339.     {
  340.         wideCharBuf[i] = srcText[i];
  341.     }
  342.     wideCharBuf[wLent] = 0x00;
  343.     const unsigned int retVal = ::wcstombs(NULL, wideCharBuf, 0);
  344.     delete [] allocatedArray;
  345.     if (retVal == -1)
  346.         return 0;
  347.     return retVal;
  348. }
  349. char* Iconv390LCPTranscoder::transcode(const XMLCh* const toTranscode)
  350. {
  351.     if (!toTranscode)
  352.         return 0;
  353.     char* retVal = 0;
  354.     if (*toTranscode)
  355.     {
  356.         unsigned int  wLent = getWideCharLength(toTranscode);
  357. //
  358. //  Translate the input from Unicode XMLCh format into
  359. //  ibm-037 char format via the lookup table.
  360. //
  361.         retVal = new char[wLent + 1];
  362.         const XMLCh *srcPtr = toTranscode;
  363.         char *outPtr = retVal;
  364. while (*srcPtr != 0)
  365.     *outPtr++ = gUnicodeToIBM037XlatTable[*srcPtr++];
  366. *outPtr=0;
  367.     }
  368.     else
  369.     {
  370.         retVal = new char[1];
  371.         retVal[0] = 0;
  372.     }
  373.     return retVal;
  374. }
  375. bool Iconv390LCPTranscoder::transcode( const   XMLCh* const    toTranscode
  376.                                     ,       char* const     toFill
  377.                                     , const unsigned int    maxBytes)
  378. {
  379.     // Watch for a couple of pyscho corner cases
  380.     if (!toTranscode || !maxBytes)
  381.     {
  382.         toFill[0] = 0;
  383.         return true;
  384.     }
  385.     if (!*toTranscode)
  386.     {
  387.         toFill[0] = 0;
  388.         return true;
  389.     }
  390.     const XMLCh *srcPtr = toTranscode;
  391.     char *outPtr = toFill;
  392.     int bytectr = maxBytes;
  393.     while (bytectr-- && *srcPtr)
  394.        *outPtr++ = gUnicodeToIBM037XlatTable[*srcPtr++];
  395.     *outPtr=0;
  396.     return true;
  397. }
  398. XMLCh* Iconv390LCPTranscoder::transcode(const char* const toTranscode)
  399. {
  400.     if (!toTranscode)
  401.         return 0;
  402.     XMLCh* retVal = 0;
  403.     if (*toTranscode)
  404.     {
  405.         const unsigned int len = calcRequiredSize(toTranscode);
  406.         if (len == 0)
  407.         {
  408.             retVal = new XMLCh[1];
  409.             retVal[0] = 0;
  410.             return retVal;
  411.         }
  412.         wchar_t       tmpWideCharArr[gTempBuffArraySize];
  413.         wchar_t*      allocatedArray = 0;
  414.         wchar_t*      wideCharBuf = 0;
  415.         retVal = new XMLCh[len + 1];
  416.         size_t retCode;
  417.         char *tmpInPtr = (char*) toTranscode;
  418.         char *tmpOutPtr = (char*) retVal;
  419.         size_t inByteLeft = len;
  420.         size_t outByteLeft = len*2;
  421.         {
  422.          XMLMutexLock lockConverter(&converter->fMutex);
  423.          retCode = iconv(converter->fIconv390Descriptor, &tmpInPtr, &inByteLeft, &tmpOutPtr, &outByteLeft);
  424.         }
  425.         if (retCode == -1) {
  426.             delete [] retVal;
  427.             return 0;
  428.         }
  429.         retVal[len] = 0x00;
  430.         delete [] allocatedArray;
  431.     }
  432.     else
  433.     {
  434.         retVal = new XMLCh[1];
  435.         retVal[0] = 0;
  436.     }
  437.     return retVal;
  438. }
  439. bool Iconv390LCPTranscoder::transcode( const   char* const     toTranscode
  440.                                     ,       XMLCh* const    toFill
  441.                                     , const unsigned int    maxChars)
  442. {
  443.     // Check for a couple of psycho corner cases
  444.     if (!toTranscode || !maxChars)
  445.     {
  446.         toFill[0] = 0;
  447.         return true;
  448.     }
  449.     if (!*toTranscode)
  450.     {
  451.         toFill[0] = 0;
  452.         return true;
  453.     }
  454.     size_t retCode;
  455.     char *tmpInPtr = (char*) toTranscode;
  456.     char *tmpOutPtr = (char*) toFill;
  457.     size_t inByteLeft = maxChars;
  458.     size_t outByteLeft = maxChars*2;
  459.     {
  460.      XMLMutexLock lockConverter(&converter->fMutex);
  461.      retCode = iconv(converter->fIconv390Descriptor, &tmpInPtr, &inByteLeft, &tmpOutPtr, &outByteLeft);
  462.     }
  463.     if ( (retCode == -1) && (outByteLeft!=0) ) {
  464.         return false;
  465.     }
  466.     toFill[maxChars] = 0x00;
  467.     return true;
  468. }
  469. // ---------------------------------------------------------------------------
  470. //  Iconv390LCPTranscoder: Constructors and Destructor
  471. // ---------------------------------------------------------------------------
  472. Iconv390LCPTranscoder::Iconv390LCPTranscoder()
  473. {
  474. }
  475. Iconv390LCPTranscoder::Iconv390LCPTranscoder(iconvconverter_t* const toAdopt) :
  476.         converter (toAdopt)
  477. {
  478. }
  479. Iconv390LCPTranscoder::~Iconv390LCPTranscoder()
  480. {
  481.     removeConverter(converter);
  482.     converter=0;
  483. }
  484. // ---------------------------------------------------------------------------
  485. //  Iconv390Transcoder: Constructors and Destructor
  486. // ---------------------------------------------------------------------------
  487. Iconv390Transcoder::Iconv390Transcoder(const  XMLCh* const    encodingName
  488.                                 , const unsigned int    blockSize) :
  489.     XMLTranscoder(encodingName, blockSize)
  490. {
  491. }
  492. Iconv390Transcoder::Iconv390Transcoder(iconvconverter_t* const toAdopt
  493. ,const XMLCh* const encodingName
  494. ,const unsigned int blockSize) :
  495.   XMLTranscoder(encodingName, blockSize)
  496.        ,converter (toAdopt)
  497. {
  498. }
  499. Iconv390Transcoder::~Iconv390Transcoder()
  500. {
  501.     removeConverter(converter);
  502.     converter=0;
  503. }
  504. // ---------------------------------------------------------------------------
  505. //  Iconv390Transcoder: Implementation of the virtual transcoder API
  506. // ---------------------------------------------------------------------------
  507. XMLCh Iconv390Transcoder::transcodeOne(const   XMLByte* const     srcData
  508.                                     , const unsigned int    srcBytes
  509.                                     ,       unsigned int&   bytesEaten)
  510. {
  511.     wchar_t  toFill;
  512.     size_t retCode;
  513.     char *tmpInPtr = (char*) srcData;
  514.     char *tmpOutPtr = (char*)&toFill;
  515.     size_t inByteLeft = srcBytes;
  516.     size_t outByteLeft = 2;
  517.     {
  518.      XMLMutexLock lockConverter(&converter->fMutex);
  519.      retCode = iconv(converter->fIconv390Descriptor, &tmpInPtr, &inByteLeft, &tmpOutPtr, &outByteLeft);
  520.     }
  521.     if (retCode == -1) {
  522.         bytesEaten = 0;
  523.         return 0;
  524.     }
  525.     int eaten = srcBytes-inByteLeft;
  526.     if (eaten == -1)
  527.     {
  528.         bytesEaten = 0;
  529.         return 0;
  530.     }
  531.     // Return the bytes we ate and the resulting char.
  532.     bytesEaten = eaten;
  533.     return toFill;
  534. }
  535. unsigned int
  536. Iconv390Transcoder::transcodeXML(  const   XMLByte* const             srcData
  537.                                 , const unsigned int            srcCount
  538.                                 ,       XMLCh* const            toFill
  539.                                 , const unsigned int            maxChars
  540.                                 ,       unsigned int&           bytesEaten
  541. ,       unsigned char* const    charSizes)
  542. {
  543.     //
  544.     //  For this one, because we have to maintain the offset table, we have
  545.     //  to do them one char at a time until we run out of source data.
  546.     //
  547.     unsigned int countIn = 0;
  548.     unsigned int countOut = 0;
  549.     size_t retCode;
  550.     char *tmpInPtr = (char *) srcData;
  551.     char *tmpOutPtr = (char *) toFill;
  552.     size_t inByteLeft = srcCount;
  553.     size_t outByteLeft = maxChars*2;
  554.     {
  555.      XMLMutexLock lockConverter(&converter->fMutex);
  556.      retCode = iconv(converter->fIconv390Descriptor, &tmpInPtr, &inByteLeft, &tmpOutPtr, &outByteLeft);
  557.     }
  558.     if (retCode == -1)
  559.     {
  560. return 0;
  561.     }
  562.     // Give back the counts of eaten and transcoded
  563.     bytesEaten = srcCount-inByteLeft;
  564.     return maxChars-outByteLeft/2;
  565. }