xref: /aoo4110/main/sal/textenc/convertbig5hkscs.c (revision b1cdbd2c)
1*b1cdbd2cSJim Jagielski /**************************************************************
2*b1cdbd2cSJim Jagielski  *
3*b1cdbd2cSJim Jagielski  * Licensed to the Apache Software Foundation (ASF) under one
4*b1cdbd2cSJim Jagielski  * or more contributor license agreements.  See the NOTICE file
5*b1cdbd2cSJim Jagielski  * distributed with this work for additional information
6*b1cdbd2cSJim Jagielski  * regarding copyright ownership.  The ASF licenses this file
7*b1cdbd2cSJim Jagielski  * to you under the Apache License, Version 2.0 (the
8*b1cdbd2cSJim Jagielski  * "License"); you may not use this file except in compliance
9*b1cdbd2cSJim Jagielski  * with the License.  You may obtain a copy of the License at
10*b1cdbd2cSJim Jagielski  *
11*b1cdbd2cSJim Jagielski  *   http://www.apache.org/licenses/LICENSE-2.0
12*b1cdbd2cSJim Jagielski  *
13*b1cdbd2cSJim Jagielski  * Unless required by applicable law or agreed to in writing,
14*b1cdbd2cSJim Jagielski  * software distributed under the License is distributed on an
15*b1cdbd2cSJim Jagielski  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16*b1cdbd2cSJim Jagielski  * KIND, either express or implied.  See the License for the
17*b1cdbd2cSJim Jagielski  * specific language governing permissions and limitations
18*b1cdbd2cSJim Jagielski  * under the License.
19*b1cdbd2cSJim Jagielski  *
20*b1cdbd2cSJim Jagielski  *************************************************************/
21*b1cdbd2cSJim Jagielski 
22*b1cdbd2cSJim Jagielski 
23*b1cdbd2cSJim Jagielski 
24*b1cdbd2cSJim Jagielski #include "convertbig5hkscs.h"
25*b1cdbd2cSJim Jagielski #include "context.h"
26*b1cdbd2cSJim Jagielski #include "converter.h"
27*b1cdbd2cSJim Jagielski #include "tenchelp.h"
28*b1cdbd2cSJim Jagielski #include "unichars.h"
29*b1cdbd2cSJim Jagielski #include "osl/diagnose.h"
30*b1cdbd2cSJim Jagielski #include "rtl/alloc.h"
31*b1cdbd2cSJim Jagielski #include "rtl/textcvt.h"
32*b1cdbd2cSJim Jagielski #include "sal/types.h"
33*b1cdbd2cSJim Jagielski 
34*b1cdbd2cSJim Jagielski typedef struct
35*b1cdbd2cSJim Jagielski {
36*b1cdbd2cSJim Jagielski     sal_Int32 m_nRow; /* 0--255; 0 means none */
37*b1cdbd2cSJim Jagielski } ImplBig5HkscsToUnicodeContext;
38*b1cdbd2cSJim Jagielski 
ImplCreateBig5HkscsToUnicodeContext(void)39*b1cdbd2cSJim Jagielski void * ImplCreateBig5HkscsToUnicodeContext(void)
40*b1cdbd2cSJim Jagielski {
41*b1cdbd2cSJim Jagielski     void * pContext
42*b1cdbd2cSJim Jagielski         = rtl_allocateMemory(sizeof (ImplBig5HkscsToUnicodeContext));
43*b1cdbd2cSJim Jagielski     ((ImplBig5HkscsToUnicodeContext *) pContext)->m_nRow = 0;
44*b1cdbd2cSJim Jagielski     return pContext;
45*b1cdbd2cSJim Jagielski }
46*b1cdbd2cSJim Jagielski 
ImplResetBig5HkscsToUnicodeContext(void * pContext)47*b1cdbd2cSJim Jagielski void ImplResetBig5HkscsToUnicodeContext(void * pContext)
48*b1cdbd2cSJim Jagielski {
49*b1cdbd2cSJim Jagielski     if (pContext)
50*b1cdbd2cSJim Jagielski         ((ImplBig5HkscsToUnicodeContext *) pContext)->m_nRow = 0;
51*b1cdbd2cSJim Jagielski }
52*b1cdbd2cSJim Jagielski 
ImplConvertBig5HkscsToUnicode(ImplTextConverterData const * pData,void * pContext,sal_Char const * pSrcBuf,sal_Size nSrcBytes,sal_Unicode * pDestBuf,sal_Size nDestChars,sal_uInt32 nFlags,sal_uInt32 * pInfo,sal_Size * pSrcCvtBytes)53*b1cdbd2cSJim Jagielski sal_Size ImplConvertBig5HkscsToUnicode(ImplTextConverterData const * pData,
54*b1cdbd2cSJim Jagielski                                        void * pContext,
55*b1cdbd2cSJim Jagielski                                        sal_Char const * pSrcBuf,
56*b1cdbd2cSJim Jagielski                                        sal_Size nSrcBytes,
57*b1cdbd2cSJim Jagielski                                        sal_Unicode * pDestBuf,
58*b1cdbd2cSJim Jagielski                                        sal_Size nDestChars,
59*b1cdbd2cSJim Jagielski                                        sal_uInt32 nFlags,
60*b1cdbd2cSJim Jagielski                                        sal_uInt32 * pInfo,
61*b1cdbd2cSJim Jagielski                                        sal_Size * pSrcCvtBytes)
62*b1cdbd2cSJim Jagielski {
63*b1cdbd2cSJim Jagielski     sal_uInt16 const * pBig5Hkscs2001Data
64*b1cdbd2cSJim Jagielski         = ((ImplBig5HkscsConverterData const *) pData)->
65*b1cdbd2cSJim Jagielski               m_pBig5Hkscs2001ToUnicodeData;
66*b1cdbd2cSJim Jagielski     sal_Int32 const * pBig5Hkscs2001RowOffsets
67*b1cdbd2cSJim Jagielski         = ((ImplBig5HkscsConverterData const *) pData)->
68*b1cdbd2cSJim Jagielski               m_pBig5Hkscs2001ToUnicodeRowOffsets;
69*b1cdbd2cSJim Jagielski     ImplDBCSToUniLeadTab const * pBig5Data
70*b1cdbd2cSJim Jagielski         = ((ImplBig5HkscsConverterData const *) pData)->
71*b1cdbd2cSJim Jagielski               m_pBig5ToUnicodeData;
72*b1cdbd2cSJim Jagielski     sal_Int32 nRow = 0;
73*b1cdbd2cSJim Jagielski     sal_uInt32 nInfo = 0;
74*b1cdbd2cSJim Jagielski     sal_Size nConverted = 0;
75*b1cdbd2cSJim Jagielski     sal_Unicode * pDestBufPtr = pDestBuf;
76*b1cdbd2cSJim Jagielski     sal_Unicode * pDestBufEnd = pDestBuf + nDestChars;
77*b1cdbd2cSJim Jagielski 
78*b1cdbd2cSJim Jagielski     if (pContext)
79*b1cdbd2cSJim Jagielski         nRow = ((ImplBig5HkscsToUnicodeContext *) pContext)->m_nRow;
80*b1cdbd2cSJim Jagielski 
81*b1cdbd2cSJim Jagielski     for (; nConverted < nSrcBytes; ++nConverted)
82*b1cdbd2cSJim Jagielski     {
83*b1cdbd2cSJim Jagielski         sal_Bool bUndefined = sal_True;
84*b1cdbd2cSJim Jagielski         sal_uInt32 nChar = *(sal_uChar const *) pSrcBuf++;
85*b1cdbd2cSJim Jagielski         if (nRow == 0)
86*b1cdbd2cSJim Jagielski             if (nChar < 0x80)
87*b1cdbd2cSJim Jagielski                 if (pDestBufPtr != pDestBufEnd)
88*b1cdbd2cSJim Jagielski                     *pDestBufPtr++ = (sal_Unicode) nChar;
89*b1cdbd2cSJim Jagielski                 else
90*b1cdbd2cSJim Jagielski                     goto no_output;
91*b1cdbd2cSJim Jagielski             else if (nChar >= 0x81 && nChar <= 0xFE)
92*b1cdbd2cSJim Jagielski                 nRow = nChar;
93*b1cdbd2cSJim Jagielski             else
94*b1cdbd2cSJim Jagielski             {
95*b1cdbd2cSJim Jagielski                 bUndefined = sal_False;
96*b1cdbd2cSJim Jagielski                 goto bad_input;
97*b1cdbd2cSJim Jagielski             }
98*b1cdbd2cSJim Jagielski         else
99*b1cdbd2cSJim Jagielski             if ((nChar >= 0x40 && nChar <= 0x7E)
100*b1cdbd2cSJim Jagielski                 || (nChar >= 0xA1 && nChar <= 0xFE))
101*b1cdbd2cSJim Jagielski             {
102*b1cdbd2cSJim Jagielski                 sal_uInt32 nUnicode = 0xFFFF;
103*b1cdbd2cSJim Jagielski                 sal_Int32 nOffset = pBig5Hkscs2001RowOffsets[nRow];
104*b1cdbd2cSJim Jagielski                 sal_uInt32 nFirst=0;
105*b1cdbd2cSJim Jagielski                 sal_uInt32 nLast=0;
106*b1cdbd2cSJim Jagielski                 if (nOffset != -1)
107*b1cdbd2cSJim Jagielski                 {
108*b1cdbd2cSJim Jagielski                     sal_uInt32 nFirstLast = pBig5Hkscs2001Data[nOffset++];
109*b1cdbd2cSJim Jagielski                     nFirst = nFirstLast & 0xFF;
110*b1cdbd2cSJim Jagielski                     nLast = nFirstLast >> 8;
111*b1cdbd2cSJim Jagielski                     if (nChar >= nFirst && nChar <= nLast)
112*b1cdbd2cSJim Jagielski                         nUnicode
113*b1cdbd2cSJim Jagielski                             = pBig5Hkscs2001Data[nOffset + (nChar - nFirst)];
114*b1cdbd2cSJim Jagielski                 }
115*b1cdbd2cSJim Jagielski                 if (nUnicode == 0xFFFF)
116*b1cdbd2cSJim Jagielski                 {
117*b1cdbd2cSJim Jagielski                     sal_uInt32 nFirst = pBig5Data[nRow].mnTrailStart;
118*b1cdbd2cSJim Jagielski                     if (nChar >= nFirst
119*b1cdbd2cSJim Jagielski                         && nChar <= pBig5Data[nRow].mnTrailEnd)
120*b1cdbd2cSJim Jagielski                     {
121*b1cdbd2cSJim Jagielski                         nUnicode
122*b1cdbd2cSJim Jagielski                             = pBig5Data[nRow].mpToUniTrailTab[nChar - nFirst];
123*b1cdbd2cSJim Jagielski                         if (nUnicode == 0)
124*b1cdbd2cSJim Jagielski                             nUnicode = 0xFFFF;
125*b1cdbd2cSJim Jagielski                         OSL_VERIFY(!ImplIsHighSurrogate(nUnicode));
126*b1cdbd2cSJim Jagielski                     }
127*b1cdbd2cSJim Jagielski                 }
128*b1cdbd2cSJim Jagielski                 if (nUnicode == 0xFFFF)
129*b1cdbd2cSJim Jagielski                 {
130*b1cdbd2cSJim Jagielski                     ImplDBCSEUDCData const * p
131*b1cdbd2cSJim Jagielski                         = ((ImplBig5HkscsConverterData const *) pData)->
132*b1cdbd2cSJim Jagielski                               m_pEudcData;
133*b1cdbd2cSJim Jagielski                     sal_uInt32 nCount
134*b1cdbd2cSJim Jagielski                         = ((ImplBig5HkscsConverterData const *) pData)->
135*b1cdbd2cSJim Jagielski                               m_nEudcCount;
136*b1cdbd2cSJim Jagielski                     sal_uInt32 i;
137*b1cdbd2cSJim Jagielski                     for (i = 0; i < nCount; ++i)
138*b1cdbd2cSJim Jagielski                     {
139*b1cdbd2cSJim Jagielski                         if (nRow >= p->mnLeadStart && nRow <= p->mnLeadEnd)
140*b1cdbd2cSJim Jagielski                         {
141*b1cdbd2cSJim Jagielski                             if (nChar < p->mnTrail1Start)
142*b1cdbd2cSJim Jagielski                                 break;
143*b1cdbd2cSJim Jagielski                             if (nChar <= p->mnTrail1End)
144*b1cdbd2cSJim Jagielski                             {
145*b1cdbd2cSJim Jagielski                                 nUnicode
146*b1cdbd2cSJim Jagielski                                     = p->mnUniStart
147*b1cdbd2cSJim Jagielski                                           + (nRow - p->mnLeadStart)
148*b1cdbd2cSJim Jagielski                                                 * p->mnTrailRangeCount
149*b1cdbd2cSJim Jagielski                                           + (nChar - p->mnTrail1Start);
150*b1cdbd2cSJim Jagielski                                 break;
151*b1cdbd2cSJim Jagielski                             }
152*b1cdbd2cSJim Jagielski                             if (p->mnTrailCount < 2
153*b1cdbd2cSJim Jagielski                                 || nChar < p->mnTrail2Start)
154*b1cdbd2cSJim Jagielski                                 break;
155*b1cdbd2cSJim Jagielski                             if (nChar <= p->mnTrail2End)
156*b1cdbd2cSJim Jagielski                             {
157*b1cdbd2cSJim Jagielski                                 nUnicode
158*b1cdbd2cSJim Jagielski                                     = p->mnUniStart
159*b1cdbd2cSJim Jagielski                                           + (nRow - p->mnLeadStart)
160*b1cdbd2cSJim Jagielski                                                 * p->mnTrailRangeCount
161*b1cdbd2cSJim Jagielski                                           + (nChar - p->mnTrail2Start)
162*b1cdbd2cSJim Jagielski                                           + (p->mnTrail1End - p->mnTrail1Start
163*b1cdbd2cSJim Jagielski                                                  + 1);
164*b1cdbd2cSJim Jagielski                                 break;
165*b1cdbd2cSJim Jagielski                             }
166*b1cdbd2cSJim Jagielski                             if (p->mnTrailCount < 3
167*b1cdbd2cSJim Jagielski                                 || nChar < p->mnTrail3Start)
168*b1cdbd2cSJim Jagielski                                 break;
169*b1cdbd2cSJim Jagielski                             if (nChar <= p->mnTrail3End)
170*b1cdbd2cSJim Jagielski                             {
171*b1cdbd2cSJim Jagielski                                 nUnicode
172*b1cdbd2cSJim Jagielski                                     = p->mnUniStart
173*b1cdbd2cSJim Jagielski                                           + (nRow - p->mnLeadStart)
174*b1cdbd2cSJim Jagielski                                                 * p->mnTrailRangeCount
175*b1cdbd2cSJim Jagielski                                           + (nChar - p->mnTrail3Start)
176*b1cdbd2cSJim Jagielski                                           + (p->mnTrail1End - p->mnTrail1Start
177*b1cdbd2cSJim Jagielski                                                  + 1)
178*b1cdbd2cSJim Jagielski                                           + (p->mnTrail2End - p->mnTrail2Start
179*b1cdbd2cSJim Jagielski                                                  + 1);
180*b1cdbd2cSJim Jagielski                                 break;
181*b1cdbd2cSJim Jagielski                             }
182*b1cdbd2cSJim Jagielski                             break;
183*b1cdbd2cSJim Jagielski                         }
184*b1cdbd2cSJim Jagielski                         ++p;
185*b1cdbd2cSJim Jagielski                     }
186*b1cdbd2cSJim Jagielski                     OSL_VERIFY(!ImplIsHighSurrogate(nUnicode));
187*b1cdbd2cSJim Jagielski                 }
188*b1cdbd2cSJim Jagielski                 if (nUnicode == 0xFFFF)
189*b1cdbd2cSJim Jagielski                     goto bad_input;
190*b1cdbd2cSJim Jagielski                 if (ImplIsHighSurrogate(nUnicode))
191*b1cdbd2cSJim Jagielski                     if (pDestBufEnd - pDestBufPtr >= 2)
192*b1cdbd2cSJim Jagielski                     {
193*b1cdbd2cSJim Jagielski                         nOffset += nLast - nFirst + 1;
194*b1cdbd2cSJim Jagielski                         nFirst = pBig5Hkscs2001Data[nOffset++];
195*b1cdbd2cSJim Jagielski                         *pDestBufPtr++ = (sal_Unicode) nUnicode;
196*b1cdbd2cSJim Jagielski                         *pDestBufPtr++
197*b1cdbd2cSJim Jagielski                             = (sal_Unicode) pBig5Hkscs2001Data[
198*b1cdbd2cSJim Jagielski                                                 nOffset + (nChar - nFirst)];
199*b1cdbd2cSJim Jagielski                     }
200*b1cdbd2cSJim Jagielski                     else
201*b1cdbd2cSJim Jagielski                         goto no_output;
202*b1cdbd2cSJim Jagielski                 else
203*b1cdbd2cSJim Jagielski                     if (pDestBufPtr != pDestBufEnd)
204*b1cdbd2cSJim Jagielski                         *pDestBufPtr++ = (sal_Unicode) nUnicode;
205*b1cdbd2cSJim Jagielski                     else
206*b1cdbd2cSJim Jagielski                         goto no_output;
207*b1cdbd2cSJim Jagielski                 nRow = 0;
208*b1cdbd2cSJim Jagielski             }
209*b1cdbd2cSJim Jagielski             else
210*b1cdbd2cSJim Jagielski             {
211*b1cdbd2cSJim Jagielski                 bUndefined = sal_False;
212*b1cdbd2cSJim Jagielski                 goto bad_input;
213*b1cdbd2cSJim Jagielski             }
214*b1cdbd2cSJim Jagielski         continue;
215*b1cdbd2cSJim Jagielski 
216*b1cdbd2cSJim Jagielski     bad_input:
217*b1cdbd2cSJim Jagielski         switch (ImplHandleBadInputTextToUnicodeConversion(
218*b1cdbd2cSJim Jagielski                     bUndefined, sal_True, 0, nFlags, &pDestBufPtr, pDestBufEnd,
219*b1cdbd2cSJim Jagielski                     &nInfo))
220*b1cdbd2cSJim Jagielski         {
221*b1cdbd2cSJim Jagielski         case IMPL_BAD_INPUT_STOP:
222*b1cdbd2cSJim Jagielski             nRow = 0;
223*b1cdbd2cSJim Jagielski             break;
224*b1cdbd2cSJim Jagielski 
225*b1cdbd2cSJim Jagielski         case IMPL_BAD_INPUT_CONTINUE:
226*b1cdbd2cSJim Jagielski             nRow = 0;
227*b1cdbd2cSJim Jagielski             continue;
228*b1cdbd2cSJim Jagielski 
229*b1cdbd2cSJim Jagielski         case IMPL_BAD_INPUT_NO_OUTPUT:
230*b1cdbd2cSJim Jagielski             goto no_output;
231*b1cdbd2cSJim Jagielski         }
232*b1cdbd2cSJim Jagielski         break;
233*b1cdbd2cSJim Jagielski 
234*b1cdbd2cSJim Jagielski     no_output:
235*b1cdbd2cSJim Jagielski         --pSrcBuf;
236*b1cdbd2cSJim Jagielski         nInfo |= RTL_TEXTTOUNICODE_INFO_DESTBUFFERTOSMALL;
237*b1cdbd2cSJim Jagielski         break;
238*b1cdbd2cSJim Jagielski     }
239*b1cdbd2cSJim Jagielski 
240*b1cdbd2cSJim Jagielski     if (nRow != 0
241*b1cdbd2cSJim Jagielski         && (nInfo & (RTL_TEXTTOUNICODE_INFO_ERROR
242*b1cdbd2cSJim Jagielski                          | RTL_TEXTTOUNICODE_INFO_DESTBUFFERTOSMALL))
243*b1cdbd2cSJim Jagielski                == 0)
244*b1cdbd2cSJim Jagielski     {
245*b1cdbd2cSJim Jagielski         if ((nFlags & RTL_TEXTTOUNICODE_FLAGS_FLUSH) == 0)
246*b1cdbd2cSJim Jagielski             nInfo |= RTL_TEXTTOUNICODE_INFO_SRCBUFFERTOSMALL;
247*b1cdbd2cSJim Jagielski         else
248*b1cdbd2cSJim Jagielski             switch (ImplHandleBadInputTextToUnicodeConversion(
249*b1cdbd2cSJim Jagielski                         sal_False, sal_True, 0, nFlags, &pDestBufPtr,
250*b1cdbd2cSJim Jagielski                         pDestBufEnd, &nInfo))
251*b1cdbd2cSJim Jagielski             {
252*b1cdbd2cSJim Jagielski             case IMPL_BAD_INPUT_STOP:
253*b1cdbd2cSJim Jagielski             case IMPL_BAD_INPUT_CONTINUE:
254*b1cdbd2cSJim Jagielski                 nRow = 0;
255*b1cdbd2cSJim Jagielski                 break;
256*b1cdbd2cSJim Jagielski 
257*b1cdbd2cSJim Jagielski             case IMPL_BAD_INPUT_NO_OUTPUT:
258*b1cdbd2cSJim Jagielski                 nInfo |= RTL_TEXTTOUNICODE_INFO_DESTBUFFERTOSMALL;
259*b1cdbd2cSJim Jagielski                 break;
260*b1cdbd2cSJim Jagielski             }
261*b1cdbd2cSJim Jagielski     }
262*b1cdbd2cSJim Jagielski 
263*b1cdbd2cSJim Jagielski     if (pContext)
264*b1cdbd2cSJim Jagielski         ((ImplBig5HkscsToUnicodeContext *) pContext)->m_nRow = nRow;
265*b1cdbd2cSJim Jagielski     if (pInfo)
266*b1cdbd2cSJim Jagielski         *pInfo = nInfo;
267*b1cdbd2cSJim Jagielski     if (pSrcCvtBytes)
268*b1cdbd2cSJim Jagielski         *pSrcCvtBytes = nConverted;
269*b1cdbd2cSJim Jagielski 
270*b1cdbd2cSJim Jagielski     return pDestBufPtr - pDestBuf;
271*b1cdbd2cSJim Jagielski }
272*b1cdbd2cSJim Jagielski 
ImplConvertUnicodeToBig5Hkscs(ImplTextConverterData const * pData,void * pContext,sal_Unicode const * pSrcBuf,sal_Size nSrcChars,sal_Char * pDestBuf,sal_Size nDestBytes,sal_uInt32 nFlags,sal_uInt32 * pInfo,sal_Size * pSrcCvtChars)273*b1cdbd2cSJim Jagielski sal_Size ImplConvertUnicodeToBig5Hkscs(ImplTextConverterData const * pData,
274*b1cdbd2cSJim Jagielski                                        void * pContext,
275*b1cdbd2cSJim Jagielski                                        sal_Unicode const * pSrcBuf,
276*b1cdbd2cSJim Jagielski                                        sal_Size nSrcChars,
277*b1cdbd2cSJim Jagielski                                        sal_Char * pDestBuf,
278*b1cdbd2cSJim Jagielski                                        sal_Size nDestBytes,
279*b1cdbd2cSJim Jagielski                                        sal_uInt32 nFlags,
280*b1cdbd2cSJim Jagielski                                        sal_uInt32 * pInfo,
281*b1cdbd2cSJim Jagielski                                        sal_Size * pSrcCvtChars)
282*b1cdbd2cSJim Jagielski {
283*b1cdbd2cSJim Jagielski     sal_uInt16 const * pBig5Hkscs2001Data
284*b1cdbd2cSJim Jagielski         = ((ImplBig5HkscsConverterData const *) pData)->
285*b1cdbd2cSJim Jagielski               m_pUnicodeToBig5Hkscs2001Data;
286*b1cdbd2cSJim Jagielski     sal_Int32 const * pBig5Hkscs2001PageOffsets
287*b1cdbd2cSJim Jagielski         = ((ImplBig5HkscsConverterData const *) pData)->
288*b1cdbd2cSJim Jagielski               m_pUnicodeToBig5Hkscs2001PageOffsets;
289*b1cdbd2cSJim Jagielski     sal_Int32 const * pBig5Hkscs2001PlaneOffsets
290*b1cdbd2cSJim Jagielski         = ((ImplBig5HkscsConverterData const *) pData)->
291*b1cdbd2cSJim Jagielski               m_pUnicodeToBig5Hkscs2001PlaneOffsets;
292*b1cdbd2cSJim Jagielski     ImplUniToDBCSHighTab const * pBig5Data
293*b1cdbd2cSJim Jagielski         = ((ImplBig5HkscsConverterData const *) pData)->
294*b1cdbd2cSJim Jagielski               m_pUnicodeToBig5Data;
295*b1cdbd2cSJim Jagielski     sal_Unicode nHighSurrogate = 0;
296*b1cdbd2cSJim Jagielski     sal_uInt32 nInfo = 0;
297*b1cdbd2cSJim Jagielski     sal_Size nConverted = 0;
298*b1cdbd2cSJim Jagielski     sal_Char * pDestBufPtr = pDestBuf;
299*b1cdbd2cSJim Jagielski     sal_Char * pDestBufEnd = pDestBuf + nDestBytes;
300*b1cdbd2cSJim Jagielski 
301*b1cdbd2cSJim Jagielski     if (pContext)
302*b1cdbd2cSJim Jagielski         nHighSurrogate
303*b1cdbd2cSJim Jagielski             = ((ImplUnicodeToTextContext *) pContext)->m_nHighSurrogate;
304*b1cdbd2cSJim Jagielski 
305*b1cdbd2cSJim Jagielski     for (; nConverted < nSrcChars; ++nConverted)
306*b1cdbd2cSJim Jagielski     {
307*b1cdbd2cSJim Jagielski         sal_Bool bUndefined = sal_True;
308*b1cdbd2cSJim Jagielski         sal_uInt32 nChar = *pSrcBuf++;
309*b1cdbd2cSJim Jagielski         if (nHighSurrogate == 0)
310*b1cdbd2cSJim Jagielski         {
311*b1cdbd2cSJim Jagielski             if (ImplIsHighSurrogate(nChar))
312*b1cdbd2cSJim Jagielski             {
313*b1cdbd2cSJim Jagielski                 nHighSurrogate = (sal_Unicode) nChar;
314*b1cdbd2cSJim Jagielski                 continue;
315*b1cdbd2cSJim Jagielski             }
316*b1cdbd2cSJim Jagielski         }
317*b1cdbd2cSJim Jagielski         else if (ImplIsLowSurrogate(nChar))
318*b1cdbd2cSJim Jagielski             nChar = ImplCombineSurrogates(nHighSurrogate, nChar);
319*b1cdbd2cSJim Jagielski         else
320*b1cdbd2cSJim Jagielski         {
321*b1cdbd2cSJim Jagielski             bUndefined = sal_False;
322*b1cdbd2cSJim Jagielski             goto bad_input;
323*b1cdbd2cSJim Jagielski         }
324*b1cdbd2cSJim Jagielski 
325*b1cdbd2cSJim Jagielski         if (ImplIsLowSurrogate(nChar) || ImplIsNoncharacter(nChar))
326*b1cdbd2cSJim Jagielski         {
327*b1cdbd2cSJim Jagielski             bUndefined = sal_False;
328*b1cdbd2cSJim Jagielski             goto bad_input;
329*b1cdbd2cSJim Jagielski         }
330*b1cdbd2cSJim Jagielski 
331*b1cdbd2cSJim Jagielski         if (nChar < 0x80)
332*b1cdbd2cSJim Jagielski             if (pDestBufPtr != pDestBufEnd)
333*b1cdbd2cSJim Jagielski                 *pDestBufPtr++ = (sal_Char) nChar;
334*b1cdbd2cSJim Jagielski             else
335*b1cdbd2cSJim Jagielski                 goto no_output;
336*b1cdbd2cSJim Jagielski         else
337*b1cdbd2cSJim Jagielski         {
338*b1cdbd2cSJim Jagielski             sal_uInt32 nBytes = 0;
339*b1cdbd2cSJim Jagielski             sal_Int32 nOffset = pBig5Hkscs2001PlaneOffsets[nChar >> 16];
340*b1cdbd2cSJim Jagielski             if (nOffset != -1)
341*b1cdbd2cSJim Jagielski             {
342*b1cdbd2cSJim Jagielski                 nOffset
343*b1cdbd2cSJim Jagielski                     = pBig5Hkscs2001PageOffsets[nOffset + ((nChar & 0xFF00)
344*b1cdbd2cSJim Jagielski                                                                >> 8)];
345*b1cdbd2cSJim Jagielski                 if (nOffset != -1)
346*b1cdbd2cSJim Jagielski                 {
347*b1cdbd2cSJim Jagielski                     sal_uInt32 nFirstLast = pBig5Hkscs2001Data[nOffset++];
348*b1cdbd2cSJim Jagielski                     sal_uInt32 nFirst = nFirstLast & 0xFF;
349*b1cdbd2cSJim Jagielski                     sal_uInt32 nLast = nFirstLast >> 8;
350*b1cdbd2cSJim Jagielski                     sal_uInt32 nIndex = nChar & 0xFF;
351*b1cdbd2cSJim Jagielski                     if (nIndex >= nFirst && nIndex <= nLast)
352*b1cdbd2cSJim Jagielski                     {
353*b1cdbd2cSJim Jagielski                         nBytes
354*b1cdbd2cSJim Jagielski                             = pBig5Hkscs2001Data[nOffset + (nIndex - nFirst)];
355*b1cdbd2cSJim Jagielski                     }
356*b1cdbd2cSJim Jagielski                 }
357*b1cdbd2cSJim Jagielski             }
358*b1cdbd2cSJim Jagielski             if (nBytes == 0)
359*b1cdbd2cSJim Jagielski             {
360*b1cdbd2cSJim Jagielski                 sal_uInt32 nIndex1 = nChar >> 8;
361*b1cdbd2cSJim Jagielski                 if (nIndex1 < 0x100)
362*b1cdbd2cSJim Jagielski                 {
363*b1cdbd2cSJim Jagielski                     sal_uInt32 nIndex2 = nChar & 0xFF;
364*b1cdbd2cSJim Jagielski                     sal_uInt32 nFirst = pBig5Data[nIndex1].mnLowStart;
365*b1cdbd2cSJim Jagielski                     if (nIndex2 >= nFirst
366*b1cdbd2cSJim Jagielski                         && nIndex2 <= pBig5Data[nIndex1].mnLowEnd)
367*b1cdbd2cSJim Jagielski                         nBytes = pBig5Data[nIndex1].
368*b1cdbd2cSJim Jagielski                                      mpToUniTrailTab[nIndex2 - nFirst];
369*b1cdbd2cSJim Jagielski                 }
370*b1cdbd2cSJim Jagielski             }
371*b1cdbd2cSJim Jagielski             if (nBytes == 0)
372*b1cdbd2cSJim Jagielski             {
373*b1cdbd2cSJim Jagielski                 ImplDBCSEUDCData const * p
374*b1cdbd2cSJim Jagielski                     = ((ImplBig5HkscsConverterData const *) pData)->
375*b1cdbd2cSJim Jagielski                           m_pEudcData;
376*b1cdbd2cSJim Jagielski                 sal_uInt32 nCount
377*b1cdbd2cSJim Jagielski                     = ((ImplBig5HkscsConverterData const *) pData)->
378*b1cdbd2cSJim Jagielski                           m_nEudcCount;
379*b1cdbd2cSJim Jagielski                 sal_uInt32 i;
380*b1cdbd2cSJim Jagielski                 for (i = 0; i < nCount; ++i) {
381*b1cdbd2cSJim Jagielski                     if (nChar >= p->mnUniStart && nChar <= p->mnUniEnd)
382*b1cdbd2cSJim Jagielski                     {
383*b1cdbd2cSJim Jagielski                         sal_uInt32 nIndex = nChar - p->mnUniStart;
384*b1cdbd2cSJim Jagielski                         sal_uInt32 nLeadOff = nIndex / p->mnTrailRangeCount;
385*b1cdbd2cSJim Jagielski                         sal_uInt32 nTrailOff = nIndex % p->mnTrailRangeCount;
386*b1cdbd2cSJim Jagielski                         sal_uInt32 nSize;
387*b1cdbd2cSJim Jagielski                         nBytes = (p->mnLeadStart + nLeadOff) << 8;
388*b1cdbd2cSJim Jagielski                         nSize = p->mnTrail1End - p->mnTrail1Start + 1;
389*b1cdbd2cSJim Jagielski                         if (nTrailOff < nSize)
390*b1cdbd2cSJim Jagielski                         {
391*b1cdbd2cSJim Jagielski                             nBytes |= p->mnTrail1Start + nTrailOff;
392*b1cdbd2cSJim Jagielski                             break;
393*b1cdbd2cSJim Jagielski                         }
394*b1cdbd2cSJim Jagielski                         nTrailOff -= nSize;
395*b1cdbd2cSJim Jagielski                         nSize = p->mnTrail2End - p->mnTrail2Start + 1;
396*b1cdbd2cSJim Jagielski                         if (nTrailOff < nSize)
397*b1cdbd2cSJim Jagielski                         {
398*b1cdbd2cSJim Jagielski                             nBytes |= p->mnTrail2Start + nTrailOff;
399*b1cdbd2cSJim Jagielski                             break;
400*b1cdbd2cSJim Jagielski                         }
401*b1cdbd2cSJim Jagielski                         nTrailOff -= nSize;
402*b1cdbd2cSJim Jagielski                         nBytes |= p->mnTrail3Start + nTrailOff;
403*b1cdbd2cSJim Jagielski                         break;
404*b1cdbd2cSJim Jagielski                     }
405*b1cdbd2cSJim Jagielski                     ++p;
406*b1cdbd2cSJim Jagielski                 }
407*b1cdbd2cSJim Jagielski             }
408*b1cdbd2cSJim Jagielski             if (nBytes == 0)
409*b1cdbd2cSJim Jagielski                 goto bad_input;
410*b1cdbd2cSJim Jagielski             if (pDestBufEnd - pDestBufPtr >= 2)
411*b1cdbd2cSJim Jagielski             {
412*b1cdbd2cSJim Jagielski                 *pDestBufPtr++ = (sal_Char) (nBytes >> 8);
413*b1cdbd2cSJim Jagielski                 *pDestBufPtr++ = (sal_Char) (nBytes & 0xFF);
414*b1cdbd2cSJim Jagielski             }
415*b1cdbd2cSJim Jagielski             else
416*b1cdbd2cSJim Jagielski                 goto no_output;
417*b1cdbd2cSJim Jagielski         }
418*b1cdbd2cSJim Jagielski         nHighSurrogate = 0;
419*b1cdbd2cSJim Jagielski         continue;
420*b1cdbd2cSJim Jagielski 
421*b1cdbd2cSJim Jagielski     bad_input:
422*b1cdbd2cSJim Jagielski         switch (ImplHandleBadInputUnicodeToTextConversion(bUndefined,
423*b1cdbd2cSJim Jagielski                                                           nChar,
424*b1cdbd2cSJim Jagielski                                                           nFlags,
425*b1cdbd2cSJim Jagielski                                                           &pDestBufPtr,
426*b1cdbd2cSJim Jagielski                                                           pDestBufEnd,
427*b1cdbd2cSJim Jagielski                                                           &nInfo,
428*b1cdbd2cSJim Jagielski                                                           NULL,
429*b1cdbd2cSJim Jagielski                                                           0,
430*b1cdbd2cSJim Jagielski                                                           NULL))
431*b1cdbd2cSJim Jagielski         {
432*b1cdbd2cSJim Jagielski         case IMPL_BAD_INPUT_STOP:
433*b1cdbd2cSJim Jagielski             nHighSurrogate = 0;
434*b1cdbd2cSJim Jagielski             break;
435*b1cdbd2cSJim Jagielski 
436*b1cdbd2cSJim Jagielski         case IMPL_BAD_INPUT_CONTINUE:
437*b1cdbd2cSJim Jagielski             nHighSurrogate = 0;
438*b1cdbd2cSJim Jagielski             continue;
439*b1cdbd2cSJim Jagielski 
440*b1cdbd2cSJim Jagielski         case IMPL_BAD_INPUT_NO_OUTPUT:
441*b1cdbd2cSJim Jagielski             goto no_output;
442*b1cdbd2cSJim Jagielski         }
443*b1cdbd2cSJim Jagielski         break;
444*b1cdbd2cSJim Jagielski 
445*b1cdbd2cSJim Jagielski     no_output:
446*b1cdbd2cSJim Jagielski         --pSrcBuf;
447*b1cdbd2cSJim Jagielski         nInfo |= RTL_UNICODETOTEXT_INFO_DESTBUFFERTOSMALL;
448*b1cdbd2cSJim Jagielski         break;
449*b1cdbd2cSJim Jagielski     }
450*b1cdbd2cSJim Jagielski 
451*b1cdbd2cSJim Jagielski     if (nHighSurrogate != 0
452*b1cdbd2cSJim Jagielski         && (nInfo & (RTL_UNICODETOTEXT_INFO_ERROR
453*b1cdbd2cSJim Jagielski                          | RTL_UNICODETOTEXT_INFO_DESTBUFFERTOSMALL))
454*b1cdbd2cSJim Jagielski                == 0)
455*b1cdbd2cSJim Jagielski     {
456*b1cdbd2cSJim Jagielski         if ((nFlags & RTL_UNICODETOTEXT_FLAGS_FLUSH) != 0)
457*b1cdbd2cSJim Jagielski             nInfo |= RTL_UNICODETOTEXT_INFO_SRCBUFFERTOSMALL;
458*b1cdbd2cSJim Jagielski         else
459*b1cdbd2cSJim Jagielski             switch (ImplHandleBadInputUnicodeToTextConversion(sal_False,
460*b1cdbd2cSJim Jagielski                                                               0,
461*b1cdbd2cSJim Jagielski                                                               nFlags,
462*b1cdbd2cSJim Jagielski                                                               &pDestBufPtr,
463*b1cdbd2cSJim Jagielski                                                               pDestBufEnd,
464*b1cdbd2cSJim Jagielski                                                               &nInfo,
465*b1cdbd2cSJim Jagielski                                                               NULL,
466*b1cdbd2cSJim Jagielski                                                               0,
467*b1cdbd2cSJim Jagielski                                                               NULL))
468*b1cdbd2cSJim Jagielski             {
469*b1cdbd2cSJim Jagielski             case IMPL_BAD_INPUT_STOP:
470*b1cdbd2cSJim Jagielski             case IMPL_BAD_INPUT_CONTINUE:
471*b1cdbd2cSJim Jagielski                 nHighSurrogate = 0;
472*b1cdbd2cSJim Jagielski                 break;
473*b1cdbd2cSJim Jagielski 
474*b1cdbd2cSJim Jagielski             case IMPL_BAD_INPUT_NO_OUTPUT:
475*b1cdbd2cSJim Jagielski                 nInfo |= RTL_UNICODETOTEXT_INFO_DESTBUFFERTOSMALL;
476*b1cdbd2cSJim Jagielski                 break;
477*b1cdbd2cSJim Jagielski             }
478*b1cdbd2cSJim Jagielski     }
479*b1cdbd2cSJim Jagielski 
480*b1cdbd2cSJim Jagielski     if (pContext)
481*b1cdbd2cSJim Jagielski         ((ImplUnicodeToTextContext *) pContext)->m_nHighSurrogate
482*b1cdbd2cSJim Jagielski             = nHighSurrogate;
483*b1cdbd2cSJim Jagielski     if (pInfo)
484*b1cdbd2cSJim Jagielski         *pInfo = nInfo;
485*b1cdbd2cSJim Jagielski     if (pSrcCvtChars)
486*b1cdbd2cSJim Jagielski         *pSrcCvtChars = nConverted;
487*b1cdbd2cSJim Jagielski 
488*b1cdbd2cSJim Jagielski     return pDestBufPtr - pDestBuf;
489*b1cdbd2cSJim Jagielski }
490