Files correlati : pdflib Ricompilazione Demo : [ ] Commento : Aggiornata pdflib.dll alla versione 7.0.4 git-svn-id: svn://10.65.10.50/trunk@18580 c028cbd2-c16b-5b4b-a496-9718f37d4682
		
			
				
	
	
		
			2007 lines
		
	
	
		
			64 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			2007 lines
		
	
	
		
			64 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
/*---------------------------------------------------------------------------*
 | 
						|
 |              PDFlib - A library for generating PDF on the fly             |
 | 
						|
 +---------------------------------------------------------------------------+
 | 
						|
 | Copyright (c) 1997-2006 Thomas Merz and PDFlib GmbH. All rights reserved. |
 | 
						|
 +---------------------------------------------------------------------------+
 | 
						|
 |                                                                           |
 | 
						|
 |    This software is subject to the PDFlib license. It is NOT in the       |
 | 
						|
 |    public domain. Extended versions and commercial licenses are           |
 | 
						|
 |    available, please check http://www.pdflib.com.                         |
 | 
						|
 |                                                                           |
 | 
						|
 *---------------------------------------------------------------------------*/
 | 
						|
 | 
						|
/* $Id: pc_unicode.c,v 1.4 2009-03-23 08:51:17 guy Exp $
 | 
						|
 *
 | 
						|
 * PDFlib Unicode converting routines
 | 
						|
 *
 | 
						|
 */
 | 
						|
 | 
						|
#define PC_UNICODE_C
 | 
						|
 | 
						|
#include "pc_util.h"
 | 
						|
 | 
						|
#if defined(WIN32)
 | 
						|
#define WIN32_LEAN_AND_MEAN
 | 
						|
#include <windows.h>
 | 
						|
#endif /* WIN32 */
 | 
						|
 | 
						|
/*
 | 
						|
 *  The following source is based on Unicode's original source
 | 
						|
 *  code ConvertUTF.c. It has been adapted to PDFlib programming
 | 
						|
 *  conventions.
 | 
						|
 *
 | 
						|
 *  The original file had the following notice:
 | 
						|
 *
 | 
						|
 *      Copyright 2001 Unicode, Inc.
 | 
						|
 *
 | 
						|
 *      Limitations on Rights to Redistribute This Code
 | 
						|
 *
 | 
						|
 *      Author: Mark E. Davis, 1994.
 | 
						|
 *      Rev History: Rick McGowan, fixes & updates May 2001.
 | 
						|
 *
 | 
						|
 *
 | 
						|
 *  Functions for conversions between UTF32, UTF-16, and UTF-8.
 | 
						|
 *  These funtions forming a complete set of conversions between
 | 
						|
 *  the three formats. UTF-7 is not included here.
 | 
						|
 *
 | 
						|
 *  Each of these routines takes pointers to input buffers and output
 | 
						|
 *  buffers. The input buffers are const.
 | 
						|
 *
 | 
						|
 *  Each routine converts the text between *sourceStart and sourceEnd,
 | 
						|
 *  putting the result into the buffer between *targetStart and
 | 
						|
 *  targetEnd. Note: the end pointers are *after* the last item: e.g.
 | 
						|
 *  *(sourceEnd - 1) is the last item.
 | 
						|
 *
 | 
						|
 *  The return result indicates whether the conversion was successful,
 | 
						|
 *  and if not, whether the problem was in the source or target buffers.
 | 
						|
 *  (Only the first encountered problem is indicated.)
 | 
						|
 *
 | 
						|
 *  After the conversion, *sourceStart and *targetStart are both
 | 
						|
 *  updated to point to the end of last text successfully converted in
 | 
						|
 *  the respective buffers.
 | 
						|
 *
 | 
						|
 *  Input parameters:
 | 
						|
 *      sourceStart - pointer to a pointer to the source buffer.
 | 
						|
 *              The contents of this are modified on return so that
 | 
						|
 *              it points at the next thing to be converted.
 | 
						|
 *      targetStart - similarly, pointer to pointer to the target buffer.
 | 
						|
 *      sourceEnd, targetEnd - respectively pointers to the ends of the
 | 
						|
 *              two buffers, for overflow checking only.
 | 
						|
 *
 | 
						|
 *  These conversion functions take a pdc_convers_flags argument. When this
 | 
						|
 *  flag is set to strict, both irregular sequences and isolated surrogates
 | 
						|
 *  will cause an error.  When the flag is set to lenient, both irregular
 | 
						|
 *  sequences and isolated surrogates are converted.
 | 
						|
 *
 | 
						|
 *  Whether the flag is strict or lenient, all illegal sequences will cause
 | 
						|
 *  an error return. This includes sequences such as: <F4 90 80 80>, <C0 80>,
 | 
						|
 *  or <A0> in UTF-8, and values above 0x10FFFF in UTF-32. Conformant code
 | 
						|
 *  must check for illegal sequences.
 | 
						|
 *
 | 
						|
 *  When the flag is set to lenient, characters over 0x10FFFF are converted
 | 
						|
 *  to the replacement character; otherwise (when the flag is set to strict)
 | 
						|
 *  they constitute an error.
 | 
						|
 *
 | 
						|
 *  Output parameters:
 | 
						|
 *      The value "sourceIllegal" is returned from some routines if the input
 | 
						|
 *      sequence is malformed.  When "sourceIllegal" is returned, the source
 | 
						|
 *      value will point to the illegal value that caused the problem. E.g.,
 | 
						|
 *      in UTF-8 when a sequence is malformed, it points to the start of the
 | 
						|
 *      malformed sequence.
 | 
						|
 *
 | 
						|
 *  Author: Mark E. Davis, 1994.
 | 
						|
 *  Rev History: Rick McGowan, fixes & updates May 2001.
 | 
						|
 *
 | 
						|
 */
 | 
						|
 | 
						|
/*
 | 
						|
 * The following 4 definitions are compiler-specific.
 | 
						|
 * The C standard does not guarantee that wchar_t has at least
 | 
						|
 * 16 bits, so wchar_t is no less portable than unsigned short!
 | 
						|
 * All should be unsigned values to avoid sign extension during
 | 
						|
 * bit mask & shift operations.
 | 
						|
 */
 | 
						|
 | 
						|
/* Unicode original:
 | 
						|
typedef unsigned long   UTF32;   at least 32 bits
 | 
						|
typedef unsigned short  UTF16;   at least 16 bits
 | 
						|
*/
 | 
						|
 | 
						|
typedef unsigned int    UTF32;  /* 32 bits */
 | 
						|
typedef unsigned short  UTF16;  /* 16 bits */
 | 
						|
typedef unsigned char   UTF8;   /* typically 8 bits */
 | 
						|
 | 
						|
/* Some fundamental constants */
 | 
						|
#define UNI_SUR_HIGH_START      (UTF32)0xD800
 | 
						|
#define UNI_SUR_HIGH_END        (UTF32)0xDBFF
 | 
						|
#define UNI_SUR_LOW_START       (UTF32)0xDC00
 | 
						|
#define UNI_SUR_LOW_END         (UTF32)0xDFFF
 | 
						|
#define UNI_REPLACEMENT_CHAR    (UTF32)0x0000FFFD
 | 
						|
#define UNI_MAX_BMP             (UTF32)0x0000FFFF
 | 
						|
#define UNI_MAX_UTF16           (UTF32)0x0010FFFF
 | 
						|
#define UNI_MAX_UTF32           (UTF32)0x7FFFFFFF
 | 
						|
 | 
						|
static const int halfShift      = 10; /* used for shifting by 10 bits */
 | 
						|
 | 
						|
static const UTF32 halfBase     = 0x0010000UL;
 | 
						|
static const UTF32 halfMask     = 0x3FFUL;
 | 
						|
 | 
						|
 | 
						|
/* --------------------------------------------------------------------- */
 | 
						|
 | 
						|
static pdc_convers_result
 | 
						|
pdc_convertUTF32toUTF16 (
 | 
						|
                UTF32** sourceStart, const UTF32* sourceEnd,
 | 
						|
                UTF16** targetStart, const UTF16* targetEnd,
 | 
						|
                const pdc_convers_flags flags) {
 | 
						|
    pdc_convers_result result = conversionOK;
 | 
						|
    UTF32* source = *sourceStart;
 | 
						|
    UTF16* target = *targetStart;
 | 
						|
    while (source < sourceEnd) {
 | 
						|
        UTF32 ch;
 | 
						|
        if (target >= targetEnd) {
 | 
						|
            result = targetExhausted; break;
 | 
						|
        }
 | 
						|
        ch = *source++;
 | 
						|
        if (ch <= UNI_MAX_BMP) { /* Target is a character <= 0xFFFF */
 | 
						|
            if ((flags == strictConversion) &&
 | 
						|
                (ch >= UNI_SUR_HIGH_START &&
 | 
						|
                 ch <= UNI_SUR_LOW_END)) {
 | 
						|
                --source; /* return to the illegal value itself */
 | 
						|
                result = sourceIllegal;
 | 
						|
                break;
 | 
						|
            } else {
 | 
						|
                *target++ = (UTF16) ch;     /* normal case */
 | 
						|
            }
 | 
						|
        } else if (ch > UNI_MAX_UTF16) {
 | 
						|
            if (flags == strictConversion) {
 | 
						|
                result = sourceIllegal;
 | 
						|
            } else {
 | 
						|
                *target++ = UNI_REPLACEMENT_CHAR;
 | 
						|
            }
 | 
						|
        } else {
 | 
						|
            /* target is a character in range 0xFFFF - 0x10FFFF. */
 | 
						|
            if (target + 1 >= targetEnd) {
 | 
						|
                result = targetExhausted;
 | 
						|
                break;
 | 
						|
            }
 | 
						|
            ch -= halfBase;
 | 
						|
            *target++ = (UTF16) ((ch >> halfShift) + UNI_SUR_HIGH_START);
 | 
						|
            *target++ = (UTF16) ((ch & halfMask) + UNI_SUR_LOW_START);
 | 
						|
        }
 | 
						|
    }
 | 
						|
    *sourceStart = source;
 | 
						|
    *targetStart = target;
 | 
						|
    return result;
 | 
						|
}
 | 
						|
 | 
						|
/* --------------------------------------------------------------------- */
 | 
						|
 | 
						|
static pdc_convers_result
 | 
						|
pdc_convertUTF16toUTF32 (
 | 
						|
                UTF16** sourceStart, UTF16* sourceEnd,
 | 
						|
                UTF32** targetStart, const UTF32* targetEnd,
 | 
						|
                const pdc_convers_flags flags) {
 | 
						|
    pdc_convers_result result = conversionOK;
 | 
						|
    UTF16* source = *sourceStart;
 | 
						|
    UTF32* target = *targetStart;
 | 
						|
    UTF32 ch, ch2;
 | 
						|
    while (source < sourceEnd) {
 | 
						|
        ch = *source++;
 | 
						|
        if (ch >= UNI_SUR_HIGH_START &&
 | 
						|
            ch <= UNI_SUR_HIGH_END &&
 | 
						|
            source < sourceEnd) {
 | 
						|
            ch2 = *source;
 | 
						|
            if (ch2 >= UNI_SUR_LOW_START && ch2 <= UNI_SUR_LOW_END) {
 | 
						|
                ch = ((ch - UNI_SUR_HIGH_START) << halfShift)
 | 
						|
                      + (ch2 - UNI_SUR_LOW_START) + halfBase;
 | 
						|
                ++source;
 | 
						|
            } else if (flags == strictConversion) {
 | 
						|
                /* it's an unpaired high surrogate */
 | 
						|
                --source; /* return to the illegal value itself */
 | 
						|
                result = sourceIllegal;
 | 
						|
                break;
 | 
						|
            }
 | 
						|
        } else if ((flags == strictConversion) &&
 | 
						|
                   (ch >= UNI_SUR_LOW_START &&
 | 
						|
                    ch <= UNI_SUR_LOW_END)) {
 | 
						|
            /* an unpaired low surrogate */
 | 
						|
            --source; /* return to the illegal value itself */
 | 
						|
            result = sourceIllegal;
 | 
						|
            break;
 | 
						|
        }
 | 
						|
        if (target >= targetEnd) {
 | 
						|
            result = targetExhausted;
 | 
						|
            break;
 | 
						|
        }
 | 
						|
        *target++ = ch;
 | 
						|
    }
 | 
						|
    *sourceStart = source;
 | 
						|
    *targetStart = target;
 | 
						|
#ifdef CVTUTF_DEBUG
 | 
						|
if (result == sourceIllegal) {
 | 
						|
    fprintf(stderr, "pdc_convertUTF16toUTF32 illegal seq 0x%04x,%04x\n",
 | 
						|
            ch, ch2);
 | 
						|
    fflush(stderr);
 | 
						|
}
 | 
						|
#endif
 | 
						|
    return result;
 | 
						|
}
 | 
						|
 | 
						|
/* --------------------------------------------------------------------- */
 | 
						|
 | 
						|
/*
 | 
						|
 * Index into the table below with the first byte of a UTF-8 sequence to
 | 
						|
 * get the number of trailing bytes that are supposed to follow it.
 | 
						|
 */
 | 
						|
static const char trailingBytesForUTF8[256] = {
 | 
						|
        0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
 | 
						|
        0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
 | 
						|
        0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
 | 
						|
        0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
 | 
						|
        0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
 | 
						|
        0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
 | 
						|
        1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1, 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
 | 
						|
        2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2, 3,3,3,3,3,3,3,3,4,4,4,4,5,5,5,5
 | 
						|
};
 | 
						|
 | 
						|
#if 0
 | 
						|
static const char
 | 
						|
pdc_get_trailingBytesForUTF8(int i) {
 | 
						|
    return (trailingBytesForUTF8[i]);
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
/*
 | 
						|
 * Magic values subtracted from a buffer value during UTF8 conversion.
 | 
						|
 * This table contains as many values as there might be trailing bytes
 | 
						|
 * in a UTF-8 sequence.
 | 
						|
 */
 | 
						|
static const UTF32 offsetsFromUTF8[6] = {
 | 
						|
    0x00000000UL, 0x00003080UL, 0x000E2080UL,
 | 
						|
    0x03C82080UL, 0xFA082080UL, 0x82082080UL
 | 
						|
};
 | 
						|
 | 
						|
/*
 | 
						|
 * Once the bits are split out into bytes of UTF-8, this is a mask OR-ed
 | 
						|
 * into the first byte, depending on how many bytes follow.  There are
 | 
						|
 * as many entries in this table as there are UTF-8 sequence types.
 | 
						|
 * (I.e., one byte sequence, two byte... six byte sequence.)
 | 
						|
 */
 | 
						|
static const UTF8 firstByteMark[7] = {
 | 
						|
    0x00, 0x00, 0xC0, 0xE0, 0xF0, 0xF8, 0xFC
 | 
						|
};
 | 
						|
 | 
						|
/* --------------------------------------------------------------------- */
 | 
						|
 | 
						|
/* The interface converts a whole buffer to avoid function-call overhead.
 | 
						|
 * Constants have been gathered. Loops & conditionals have been removed as
 | 
						|
 * much as possible for efficiency, in favor of drop-through switches.
 | 
						|
 * (See "Note A" at the bottom of the file for equivalent code.)
 | 
						|
 * If your compiler supports it, the "pdc_islegalUTF8" call can be turned
 | 
						|
 * into an inline function.
 | 
						|
 */
 | 
						|
 | 
						|
/* --------------------------------------------------------------------- */
 | 
						|
 | 
						|
static pdc_convers_result
 | 
						|
pdc_convertUTF16toUTF8 (
 | 
						|
                UTF16** sourceStart, const UTF16* sourceEnd,
 | 
						|
                UTF8** targetStart, const UTF8* targetEnd,
 | 
						|
                const pdc_convers_flags flags) {
 | 
						|
    pdc_convers_result result = conversionOK;
 | 
						|
    UTF16* source = *sourceStart;
 | 
						|
    UTF8* target = *targetStart;
 | 
						|
    while (source < sourceEnd) {
 | 
						|
        UTF32 ch;
 | 
						|
        unsigned short bytesToWrite = 0;
 | 
						|
        const UTF32 byteMask = 0xBF;
 | 
						|
        const UTF32 byteMark = 0x80;
 | 
						|
        ch = *source++;
 | 
						|
        /* If we have a surrogate pair, convert to UTF32 first. */
 | 
						|
        if (ch >= UNI_SUR_HIGH_START &&
 | 
						|
            ch <= UNI_SUR_HIGH_END &&
 | 
						|
            source < sourceEnd) {
 | 
						|
            UTF32 ch2 = *source;
 | 
						|
            if (ch2 >= UNI_SUR_LOW_START &&
 | 
						|
                ch2 <= UNI_SUR_LOW_END) {
 | 
						|
                ch = ((ch - UNI_SUR_HIGH_START) << halfShift)
 | 
						|
                        + (ch2 - UNI_SUR_LOW_START) + halfBase;
 | 
						|
                ++source;
 | 
						|
            } else if (flags == strictConversion) {
 | 
						|
                /* it's an unpaired high surrogate */
 | 
						|
                --source; /* return to the illegal value itself */
 | 
						|
                result = sourceIllegal;
 | 
						|
                break;
 | 
						|
            }
 | 
						|
        } else if ((flags == strictConversion) &&
 | 
						|
                   (ch >= UNI_SUR_LOW_START &&
 | 
						|
                    ch <= UNI_SUR_LOW_END)) {
 | 
						|
            --source; /* return to the illegal value itself */
 | 
						|
            result = sourceIllegal;
 | 
						|
            break;
 | 
						|
        }
 | 
						|
        /* Figure out how many bytes the result will require */
 | 
						|
        if (ch < (UTF32)0x80) {                 bytesToWrite = 1;
 | 
						|
        } else if (ch < (UTF32)0x800) {         bytesToWrite = 2;
 | 
						|
        } else if (ch < (UTF32)0x10000) {       bytesToWrite = 3;
 | 
						|
        } else if (ch < (UTF32)0x200000) {      bytesToWrite = 4;
 | 
						|
        } else {                                bytesToWrite = 2;
 | 
						|
                                                ch = UNI_REPLACEMENT_CHAR;
 | 
						|
        }
 | 
						|
 | 
						|
        target += bytesToWrite;
 | 
						|
        if (target > targetEnd) {
 | 
						|
            target -= bytesToWrite; result = targetExhausted; break;
 | 
						|
        }
 | 
						|
        switch (bytesToWrite) { /* note: everything falls through. */
 | 
						|
            case 4: *--target = (UTF8) ((ch | byteMark) & byteMask); ch >>= 6;
 | 
						|
            case 3: *--target = (UTF8) ((ch | byteMark) & byteMask); ch >>= 6;
 | 
						|
            case 2: *--target = (UTF8) ((ch | byteMark) & byteMask); ch >>= 6;
 | 
						|
            case 1: *--target = (UTF8) (ch | firstByteMark[bytesToWrite]);
 | 
						|
        }
 | 
						|
        target += bytesToWrite;
 | 
						|
    }
 | 
						|
    *sourceStart = source;
 | 
						|
    *targetStart = target;
 | 
						|
    return result;
 | 
						|
}
 | 
						|
 | 
						|
/* --------------------------------------------------------------------- */
 | 
						|
 | 
						|
/*
 | 
						|
 * Utility routine to tell whether a sequence of bytes is legal UTF-8.
 | 
						|
 * This must be called with the length pre-determined by the first byte.
 | 
						|
 * If not calling this from pdc_convertUTF8to*, then the length can be set by:
 | 
						|
 *      length = trailingBytesForUTF8[*source]+1;
 | 
						|
 * and the sequence is illegal right away if there aren't that many bytes
 | 
						|
 * available.
 | 
						|
 * If presented with a length > 4, this returns pdc_false.  The Unicode
 | 
						|
 * definition of UTF-8 goes up to 4-byte sequences.
 | 
						|
 */
 | 
						|
 | 
						|
static pdc_bool
 | 
						|
pdc_islegalUTF8(UTF8 *source, int length) {
 | 
						|
    UTF8 a;
 | 
						|
    UTF8 *srcptr = source+length;
 | 
						|
    switch (length) {
 | 
						|
    default: return pdc_false;
 | 
						|
        /* Everything else falls through when "pdc_true"... */
 | 
						|
    case 4: if ((a = (*--srcptr)) < 0x80 || a > 0xBF) return pdc_false;
 | 
						|
    case 3: if ((a = (*--srcptr)) < 0x80 || a > 0xBF) return pdc_false;
 | 
						|
    case 2: if ((a = (*--srcptr)) > 0xBF) return pdc_false;
 | 
						|
        switch (*source) {
 | 
						|
            /* no fall-through in this inner switch */
 | 
						|
            case 0xE0: if (a < 0xA0) return pdc_false; break;
 | 
						|
            case 0xF0: if (a < 0x90) return pdc_false; break;
 | 
						|
            case 0xF4: if (a > 0x8F) return pdc_false; break;
 | 
						|
            default:  if (a < 0x80) return pdc_false;
 | 
						|
        }
 | 
						|
    case 1: if (*source >= 0x80 && *source < 0xC2) return pdc_false;
 | 
						|
            if (*source > 0xF4) return pdc_false;
 | 
						|
    }
 | 
						|
    return pdc_true;
 | 
						|
}
 | 
						|
 | 
						|
/* --------------------------------------------------------------------- */
 | 
						|
 | 
						|
/*
 | 
						|
 * Exported function to return whether a UTF-8 sequence is legal or not.
 | 
						|
 * This is not used here; it's just exported.
 | 
						|
 */
 | 
						|
#if 0
 | 
						|
static pdc_bool pdc_islegalUTF8sequence(UTF8 *source, UTF8 *sourceEnd) {
 | 
						|
    int length = trailingBytesForUTF8[*source]+1;
 | 
						|
    if (source+length > sourceEnd) {
 | 
						|
        return pdc_false;
 | 
						|
    }
 | 
						|
    return pdc_islegalUTF8(source, length);
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
/* --------------------------------------------------------------------- */
 | 
						|
 | 
						|
static pdc_convers_result
 | 
						|
pdc_convertUTF8toUTF16 (
 | 
						|
                UTF8** sourceStart, UTF8* sourceEnd,
 | 
						|
                UTF16** targetStart, const UTF16* targetEnd,
 | 
						|
                const pdc_convers_flags flags) {
 | 
						|
    pdc_convers_result result = conversionOK;
 | 
						|
    UTF8* source = *sourceStart;
 | 
						|
    UTF16* target = *targetStart;
 | 
						|
    while (source < sourceEnd) {
 | 
						|
        UTF32 ch = 0L;
 | 
						|
        unsigned short extraBytesToRead = trailingBytesForUTF8[*source];
 | 
						|
        if (source + extraBytesToRead >= sourceEnd) {
 | 
						|
            result = sourceExhausted;
 | 
						|
            break;
 | 
						|
        }
 | 
						|
        /* Do this check whether lenient or strict */
 | 
						|
        if (! pdc_islegalUTF8(source, extraBytesToRead+1)) {
 | 
						|
            result = sourceIllegal;
 | 
						|
            break;
 | 
						|
        }
 | 
						|
        /*
 | 
						|
         * The cases all fall through. See "Note A" below.
 | 
						|
         */
 | 
						|
        switch (extraBytesToRead) {
 | 
						|
            case 3: ch += *source++; ch <<= 6;
 | 
						|
            case 2: ch += *source++; ch <<= 6;
 | 
						|
            case 1: ch += *source++; ch <<= 6;
 | 
						|
            case 0: ch += *source++;
 | 
						|
        }
 | 
						|
        ch -= offsetsFromUTF8[extraBytesToRead];
 | 
						|
 | 
						|
        if (target >= targetEnd) {
 | 
						|
            result = targetExhausted;
 | 
						|
            break;
 | 
						|
        }
 | 
						|
        if (ch <= UNI_MAX_BMP) { /* Target is a character <= 0xFFFF */
 | 
						|
            if ((flags == strictConversion) &&
 | 
						|
                (ch >= UNI_SUR_HIGH_START &&
 | 
						|
                 ch <= UNI_SUR_LOW_END)) {
 | 
						|
                --source; /* return to the illegal value itself */
 | 
						|
                result = sourceIllegal;
 | 
						|
                break;
 | 
						|
            } else {
 | 
						|
                *target++ = (UTF16) ch;     /* normal case */
 | 
						|
            }
 | 
						|
        } else if (ch > UNI_MAX_UTF16) {
 | 
						|
            if (flags == strictConversion) {
 | 
						|
                    result = sourceIllegal;
 | 
						|
                    source -= extraBytesToRead; /* return to the start */
 | 
						|
            } else {
 | 
						|
                    *target++ = UNI_REPLACEMENT_CHAR;
 | 
						|
            }
 | 
						|
        } else {
 | 
						|
            /* target is a character in range 0xFFFF - 0x10FFFF. */
 | 
						|
            if (target + 1 >= targetEnd) {
 | 
						|
                    result = targetExhausted;
 | 
						|
                    break;
 | 
						|
            }
 | 
						|
            ch -= halfBase;
 | 
						|
            *target++ = (UTF16) ((ch >> halfShift) + UNI_SUR_HIGH_START);
 | 
						|
            *target++ = (UTF16) ((ch & halfMask) + UNI_SUR_LOW_START);
 | 
						|
        }
 | 
						|
    }
 | 
						|
    *sourceStart = source;
 | 
						|
    *targetStart = target;
 | 
						|
    return result;
 | 
						|
}
 | 
						|
 | 
						|
/* --------------------------------------------------------------------- */
 | 
						|
 | 
						|
static pdc_convers_result
 | 
						|
pdc_convertUTF32toUTF8 (
 | 
						|
                UTF32** sourceStart, const UTF32* sourceEnd,
 | 
						|
                UTF8** targetStart, const UTF8* targetEnd,
 | 
						|
                const pdc_convers_flags flags) {
 | 
						|
    pdc_convers_result result = conversionOK;
 | 
						|
    UTF32* source = *sourceStart;
 | 
						|
    UTF8* target = *targetStart;
 | 
						|
    while (source < sourceEnd) {
 | 
						|
        UTF32 ch;
 | 
						|
        unsigned short bytesToWrite = 0;
 | 
						|
        const UTF32 byteMask = 0x000000BF;
 | 
						|
        const UTF32 byteMark = 0x00000080;
 | 
						|
        ch = *source++;
 | 
						|
        /* surrogates of any stripe are not legal UTF32 characters */
 | 
						|
        if (flags == strictConversion ) {
 | 
						|
            if ((ch >= UNI_SUR_HIGH_START) && (ch <= UNI_SUR_LOW_END)) {
 | 
						|
                --source; /* return to the illegal value itself */
 | 
						|
                result = sourceIllegal;
 | 
						|
                break;
 | 
						|
            }
 | 
						|
        }
 | 
						|
        /* Figure out how many bytes the result will require */
 | 
						|
        if (ch < (UTF32)0x80) {                 bytesToWrite = 1;
 | 
						|
        } else if (ch < (UTF32)0x800) {         bytesToWrite = 2;
 | 
						|
        } else if (ch < (UTF32)0x10000) {       bytesToWrite = 3;
 | 
						|
        } else if (ch < (UTF32)0x200000) {      bytesToWrite = 4;
 | 
						|
        } else {                                bytesToWrite = 2;
 | 
						|
                                                ch = UNI_REPLACEMENT_CHAR;
 | 
						|
        }
 | 
						|
 | 
						|
        target += bytesToWrite;
 | 
						|
        if (target > targetEnd) {
 | 
						|
            target -= bytesToWrite; result = targetExhausted; break;
 | 
						|
        }
 | 
						|
        switch (bytesToWrite) { /* note: everything falls through. */
 | 
						|
            case 4: *--target = (UTF8) ((ch | byteMark) & byteMask); ch >>= 6;
 | 
						|
            case 3: *--target = (UTF8) ((ch | byteMark) & byteMask); ch >>= 6;
 | 
						|
            case 2: *--target = (UTF8) ((ch | byteMark) & byteMask); ch >>= 6;
 | 
						|
            case 1: *--target = (UTF8) (ch | firstByteMark[bytesToWrite]);
 | 
						|
        }
 | 
						|
        target += bytesToWrite;
 | 
						|
    }
 | 
						|
    *sourceStart = source;
 | 
						|
    *targetStart = target;
 | 
						|
    return result;
 | 
						|
}
 | 
						|
 | 
						|
/* --------------------------------------------------------------------- */
 | 
						|
 | 
						|
static pdc_convers_result
 | 
						|
pdc_convertUTF8toUTF32 (
 | 
						|
                UTF8** sourceStart, UTF8* sourceEnd,
 | 
						|
                UTF32** targetStart, const UTF32* targetEnd,
 | 
						|
                const pdc_convers_flags flags) {
 | 
						|
    pdc_convers_result result = conversionOK;
 | 
						|
    UTF8* source = *sourceStart;
 | 
						|
    UTF32* target = *targetStart;
 | 
						|
 | 
						|
    (void) flags;
 | 
						|
 | 
						|
    while (source < sourceEnd) {
 | 
						|
        UTF32 ch = 0;
 | 
						|
        unsigned short extraBytesToRead = trailingBytesForUTF8[*source];
 | 
						|
        if (source + extraBytesToRead >= sourceEnd) {
 | 
						|
            result = sourceExhausted; break;
 | 
						|
        }
 | 
						|
        /* Do this check whether lenient or strict */
 | 
						|
        if (! pdc_islegalUTF8(source, extraBytesToRead+1)) {
 | 
						|
            result = sourceIllegal;
 | 
						|
            break;
 | 
						|
        }
 | 
						|
        /*
 | 
						|
         * The cases all fall through. See "Note A" below.
 | 
						|
         */
 | 
						|
        switch (extraBytesToRead) {
 | 
						|
            case 3: ch += *source++; ch <<= 6;
 | 
						|
            case 2: ch += *source++; ch <<= 6;
 | 
						|
            case 1: ch += *source++; ch <<= 6;
 | 
						|
            case 0: ch += *source++;
 | 
						|
        }
 | 
						|
        ch -= offsetsFromUTF8[extraBytesToRead];
 | 
						|
 | 
						|
        if (target >= targetEnd) {
 | 
						|
            result = targetExhausted;
 | 
						|
            break;
 | 
						|
        }
 | 
						|
        if (ch <= UNI_MAX_UTF32) {
 | 
						|
            *target++ = ch;
 | 
						|
        } else if (ch > UNI_MAX_UTF32) {
 | 
						|
            *target++ = UNI_REPLACEMENT_CHAR;
 | 
						|
        } else {
 | 
						|
            if (target + 1 >= targetEnd) {
 | 
						|
                result = targetExhausted;
 | 
						|
                break;
 | 
						|
            }
 | 
						|
            ch -= halfBase;
 | 
						|
            *target++ = (ch >> halfShift) + UNI_SUR_HIGH_START;
 | 
						|
            *target++ = (ch & halfMask) + UNI_SUR_LOW_START;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    *sourceStart = source;
 | 
						|
    *targetStart = target;
 | 
						|
    return result;
 | 
						|
}
 | 
						|
 | 
						|
/* ---------------------------------------------------------------------
 | 
						|
 | 
						|
        Note A.
 | 
						|
        The fall-through switches in UTF-8 reading code save a
 | 
						|
        temp variable, some decrements & conditionals.  The switches
 | 
						|
        are equivalent to the following loop:
 | 
						|
                {
 | 
						|
                        int tmpBytesToRead = extraBytesToRead+1;
 | 
						|
                        do {
 | 
						|
                                ch += *source++;
 | 
						|
                                --tmpBytesToRead;
 | 
						|
                                if (tmpBytesToRead) ch <<= 6;
 | 
						|
                        } while (tmpBytesToRead > 0);
 | 
						|
                }
 | 
						|
        In UTF-8 writing code, the switches on "bytesToWrite" are
 | 
						|
        similarly unrolled loops.
 | 
						|
 | 
						|
   --------------------------------------------------------------------- */
 | 
						|
 | 
						|
const char *
 | 
						|
pdc_get_textformat(int textformat)
 | 
						|
{
 | 
						|
    return pdc_get_keyword(textformat, pdc_textformat_keylist);
 | 
						|
}
 | 
						|
 | 
						|
static const pdc_keyconn pdc_utfformat_keylist[] =
 | 
						|
{
 | 
						|
    {"8",     pdc_utf8},
 | 
						|
    {"16",    pdc_utf16},
 | 
						|
    {"32",    pdc_utf32},
 | 
						|
    {NULL, 0}
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 *  pdc_convert_string converts a arbitrary encoded string (maybe UTF) to
 | 
						|
 *  another encoded string.
 | 
						|
 *
 | 
						|
 *  The new converted string is allocated and terminated by the required
 | 
						|
 *  number of zeros.
 | 
						|
 *
 | 
						|
 *  The caller is responsible for freeing the resulting string buffer.
 | 
						|
 *
 | 
						|
 *
 | 
						|
 *  LBP: low byte picking
 | 
						|
 *
 | 
						|
 *  Input-Parameter:
 | 
						|
 *
 | 
						|
 *  inutf:      input string format (see pc_unicode.h):
 | 
						|
 *
 | 
						|
 *              pdc_auto:     If codepage != 0:
 | 
						|
 *                                see above.
 | 
						|
 *                            Otherwise:
 | 
						|
 *                            If a BOM is recognized:
 | 
						|
 *                                pdc_utf8 or pdc_utf16xx resp.
 | 
						|
 *                            Otherwise if input encoding <inev> is specified
 | 
						|
 *                            and flag PDC_CONV_FORCEUTF16 not set:
 | 
						|
 *                                pdc_bytes
 | 
						|
 *                            Otherwise:
 | 
						|
 *                                pdc_utf16
 | 
						|
 *
 | 
						|
 *              pdc_auto2:    If input encoding is not specified:
 | 
						|
 *                                pdc_utf16
 | 
						|
 *                            Otherwise after successfull LBP:
 | 
						|
 *                                pdc_auto
 | 
						|
 *                            Otherwise:
 | 
						|
 *                                pdc_utf16
 | 
						|
 *
 | 
						|
 *              pdc_bytes:    8-bit string. Encoding is <inev> if specified.
 | 
						|
 *
 | 
						|
 *              pdc_bytes2:   After successfull LBP:
 | 
						|
 *                                pdc_bytes
 | 
						|
 *                            Otherwise:
 | 
						|
 *                                pdc_utf16
 | 
						|
 *
 | 
						|
 *              pdc_utf8:     UTF-8 formatted string.
 | 
						|
 *
 | 
						|
 *              pdc_ebcdicutf8: EBCDIC-UTF-8 formatted string.
 | 
						|
 *
 | 
						|
 *              pdc_utf16:    If a UTF16 BOM is recognized:
 | 
						|
 *                                pdc_utf16be or pdc_utf16le
 | 
						|
 *                            Otherwise UTF-16 machine byte ordered string.
 | 
						|
 *
 | 
						|
 *              pdc_utf16be   UTF-16 big endian formatted string.
 | 
						|
 *
 | 
						|
 *              pdc_utf16le   UTF-16 little endian formatted string.
 | 
						|
 *
 | 
						|
 *  codepage:   OEM multi byte code-page number. If > 0 and
 | 
						|
 *              <inutf> = pdc_auto, text will be converted to UTF-16.
 | 
						|
 *
 | 
						|
 *  inev:       Encoding vector for input pdc_bytes string.
 | 
						|
 *
 | 
						|
 *  glyphtab:   Mapping table for character reference names
 | 
						|
 *
 | 
						|
 *  tabsize:    Size of mapping table
 | 
						|
 *
 | 
						|
 *  replchar:   Treatment of non resolvable character references:
 | 
						|
 *              >= 0: replacement character
 | 
						|
 *              == text_error: error message
 | 
						|
 *              == text_nocheck: will be ignored
 | 
						|
 *              (see also pdc_charref2unicodelist())
 | 
						|
 *
 | 
						|
 *  instring:   Input string.
 | 
						|
 *
 | 
						|
 *  inlen:      Length of input string in byte.
 | 
						|
 *
 | 
						|
 *  oututf:     Target format for output string.
 | 
						|
 *              pdc_auto, pdc_auto2 and pdc_bytes2 are not supported.
 | 
						|
 *
 | 
						|
 *  outev:      Encoding vector for output pdc_bytes string.
 | 
						|
 *
 | 
						|
 *  flags:      PDC_CONV_FORCEUTF16:
 | 
						|
 *              In the case of <inutf> = pdc_auto[2] and <inev> != NULL
 | 
						|
 *              <inutf> = pdc_utf16 will be forced.
 | 
						|
 *
 | 
						|
 *              PDC_CONV_TRY7BYTES:
 | 
						|
 *              UTF-8 output strings will have no BOM if each byte
 | 
						|
 *              is smaller than x80.
 | 
						|
 *              *oututf: pdc_byte.
 | 
						|
 *
 | 
						|
 *              PDC_CONV_TRYBYTES:
 | 
						|
 *              UTF-UTF-16xx output strings will be converted by LBP
 | 
						|
 *              if each character is smaller than x0100.
 | 
						|
 *              *oututf: pdc_byte.
 | 
						|
 *
 | 
						|
 *              PDC_CONV_WITHBOM:
 | 
						|
 *              UTF-8 or UTF-UTF-16xx output strings will be armed
 | 
						|
 *              with an appropriate BOM.
 | 
						|
 *
 | 
						|
 *              PDC_CONV_NOBOM:
 | 
						|
 *              In UTF-8 or UTF-UTF-16xx output strings any BOM sequence
 | 
						|
 *              will be removed. PDC_CONV_WITHBOM is dominant.
 | 
						|
 *
 | 
						|
 *              PDC_CONV_AUTOBOM:
 | 
						|
 *              BOM sequence will be set automatically if input string
 | 
						|
 *              has a BOM.
 | 
						|
 *
 | 
						|
 *              PDC_CONV_ANALYZE:
 | 
						|
 *              Only analyzing BOMs of input string and dissolving auto
 | 
						|
 *              textformats.
 | 
						|
 *
 | 
						|
 *              PDC_CONV_TMPALLOC
 | 
						|
 *              Temporary memory functions (pdc_malloc_tmp) are used
 | 
						|
 *              rather than pdc_malloc etc.
 | 
						|
 *
 | 
						|
 *              PDC_CONV_HTMLCHAR
 | 
						|
 *              If input encoding vector is specified HTML character
 | 
						|
 *              entities will be substituted.
 | 
						|
 *
 | 
						|
 *              PDC_CONV_NEWALLOC
 | 
						|
 *              Input string must be allocated at first to guarantee
 | 
						|
 *              pointer alignment.
 | 
						|
 *
 | 
						|
 *              PDC_CONV_INFLATE
 | 
						|
 *              Invalid UTF-8 to UTF-16xx conversion will not cause
 | 
						|
 *              an exception but rather an inflated byte string will
 | 
						|
 *              be output.
 | 
						|
 *
 | 
						|
 *              PDC_CONV_ESCSEQU
 | 
						|
 *              Unicode sequences framed by escape character U+001B
 | 
						|
 *              (found in PDF text strings) will be skipped.
 | 
						|
 *
 | 
						|
 *              PDC_CONV_BSSEQU
 | 
						|
 *              Code sequences beginning with backslash '\'
 | 
						|
 *              will be substituted.
 | 
						|
 *
 | 
						|
 *              PDC_CONV_ENCERROR
 | 
						|
 *              If an 8-bit code cannot be converted to Unicode by <inev>
 | 
						|
 *              or a Unicode cannot be converted to an 8-bit code by <outev>
 | 
						|
 *              an error message will be created.
 | 
						|
 *
 | 
						|
 *              PDC_CONV_KEEPLBCHAR
 | 
						|
 *              In the case of PDC_CONV_ENCERROR relevant characters for
 | 
						|
 *              line breaking do not lead to an error message.
 | 
						|
 *
 | 
						|
 *              PDC_CONV_LOGGING
 | 
						|
 *              Enables logging.
 | 
						|
 *
 | 
						|
 *  verbose:    Error messages are put out. Otherwise they are saved only.
 | 
						|
 *
 | 
						|
 *  Output-Parameter:
 | 
						|
 *
 | 
						|
 *  oututf:     Reached format for output string.
 | 
						|
 *
 | 
						|
 *  outstring:  Pointer of allocated output string
 | 
						|
 *
 | 
						|
 *  outlen:     Length of output string.
 | 
						|
 *
 | 
						|
 */
 | 
						|
 | 
						|
#if defined(_MSC_VER) && defined(_MANAGED)
 | 
						|
#pragma unmanaged
 | 
						|
#endif
 | 
						|
int
 | 
						|
pdc_convert_string(pdc_core *pdc,
 | 
						|
                   pdc_text_format inutf, int codepage,
 | 
						|
                   pdc_encodingvector *inev,
 | 
						|
                   pdc_byte *instring, int inlen,
 | 
						|
                   pdc_text_format *oututf_p, pdc_encodingvector *outev,
 | 
						|
                   pdc_byte **outstring, int *outlen, int flags,
 | 
						|
                   pdc_bool verbose)
 | 
						|
{
 | 
						|
    /* text_nocheck: see bug #1664 */
 | 
						|
    return pdc_convert_textstring(pdc, inutf, codepage, inev,
 | 
						|
                   NULL, 0, text_nocheck, instring, inlen, oututf_p, outev,
 | 
						|
                   outstring, outlen, flags, verbose);
 | 
						|
}
 | 
						|
 | 
						|
int
 | 
						|
pdc_convert_textstring(pdc_core *pdc,
 | 
						|
                   pdc_text_format inutf, int codepage,
 | 
						|
                   pdc_encodingvector *inev,
 | 
						|
                   const pdc_glyph_tab *glyphtab, int tabsize, int replchar,
 | 
						|
                   pdc_byte *instring, int inlen,
 | 
						|
                   pdc_text_format *oututf_p, pdc_encodingvector *outev,
 | 
						|
                   pdc_byte **outstring, int *outlen, int flags,
 | 
						|
                   pdc_bool verbose)
 | 
						|
{
 | 
						|
    static const char *fn = "pdc_convert_textstring";
 | 
						|
    pdc_bool logg = flags & PDC_CONV_LOGGING;
 | 
						|
    const char *stemp1 = NULL, *stemp2 = NULL;
 | 
						|
    char sbuf[64];
 | 
						|
    pdc_text_format oututf = *oututf_p;
 | 
						|
    pdc_text_format oututf_s;
 | 
						|
    pdc_ushort *usinstr = (pdc_ushort *) instring;
 | 
						|
    pdc_ushort uv = 0;
 | 
						|
    pdc_byte *instr = NULL;
 | 
						|
    pdc_bool inalloc = pdc_false;
 | 
						|
    pdc_bool hasbom = pdc_false;
 | 
						|
    pdc_bool toswap = pdc_false;
 | 
						|
    int errcode = 0;
 | 
						|
    int i, j, n, len = 0;
 | 
						|
 | 
						|
    (void) glyphtab;
 | 
						|
    (void) tabsize;
 | 
						|
    (void) replchar;
 | 
						|
 | 
						|
    if (logg || pdc_logg_is_enabled(pdc, 5, trc_encoding))
 | 
						|
    {
 | 
						|
        pdc_logg(pdc, "\n");
 | 
						|
        if (!logg)
 | 
						|
            pdc_logg(pdc, "\t\ttext string of length %d will be converted...\n",
 | 
						|
                     inlen);
 | 
						|
        logg = pdc_true;
 | 
						|
    }
 | 
						|
 | 
						|
    if (logg)
 | 
						|
    {
 | 
						|
        pdc_logg(pdc, "\t\tinput textformat for string conversion: %s\n",
 | 
						|
                 pdc_get_keyword(inutf, pdc_textformat_keylist));
 | 
						|
 | 
						|
        if (inev != NULL)
 | 
						|
            pdc_logg(pdc, "\t\tinput encoding: %s\n", inev->apiname);
 | 
						|
 | 
						|
        if (outev != NULL)
 | 
						|
            pdc_logg(pdc, "\t\toutput encoding: %s\n", outev->apiname);
 | 
						|
    }
 | 
						|
 | 
						|
    /* prophylactic */
 | 
						|
    if (!inlen)
 | 
						|
    {
 | 
						|
        instring = (pdc_byte *) ((flags & PDC_CONV_TMPALLOC) ?
 | 
						|
             pdc_calloc_tmp(pdc, 4, fn, NULL, NULL) :
 | 
						|
             pdc_calloc(pdc, 4, fn));
 | 
						|
 | 
						|
        inalloc = pdc_true;
 | 
						|
    }
 | 
						|
    else if ((flags & PDC_CONV_NEWALLOC) ||
 | 
						|
             (flags & PDC_CONV_TMPALLOC) ||
 | 
						|
             (flags & PDC_CONV_BSSEQU))
 | 
						|
    {
 | 
						|
        instr = (pdc_byte *) ((flags & PDC_CONV_TMPALLOC) ?
 | 
						|
             pdc_calloc_tmp(pdc, (size_t) (inlen + 2), fn, NULL, NULL) :
 | 
						|
             pdc_calloc(pdc, (size_t) (inlen + 2), fn));
 | 
						|
        memcpy(instr, instring, (size_t) inlen);
 | 
						|
 | 
						|
        inalloc = pdc_true;
 | 
						|
        instring = instr;
 | 
						|
        instr = NULL;
 | 
						|
        usinstr = (pdc_ushort *) instring;
 | 
						|
    }
 | 
						|
 | 
						|
    switch(inutf)
 | 
						|
    {
 | 
						|
        /* analyzing 2 byte textformat */
 | 
						|
        case pdc_auto2:
 | 
						|
        case pdc_bytes2:
 | 
						|
        if ((inutf == pdc_auto2 &&
 | 
						|
             (inev == NULL || (flags & PDC_CONV_FORCEUTF16))) ||
 | 
						|
            (flags & PDC_CONV_ANALYZE))
 | 
						|
        {
 | 
						|
            inutf = pdc_utf16;
 | 
						|
        }
 | 
						|
        else
 | 
						|
        {
 | 
						|
            if (logg)
 | 
						|
                pdc_logg(pdc, "\t\ttry to pick low bytes\n");
 | 
						|
 | 
						|
            len = inlen / 2;
 | 
						|
            if (2 * len != inlen)
 | 
						|
            {
 | 
						|
                errcode = PDC_E_CONV_ILLUTF16;
 | 
						|
                goto PDC_CONV_ERROR;
 | 
						|
            }
 | 
						|
            for (i = 0; i < len; i++)
 | 
						|
                if (usinstr[i] > PDC_UNICODE_MAXLATIN1)
 | 
						|
                    break;
 | 
						|
 | 
						|
            /* low byte picking */
 | 
						|
            if (i == len)
 | 
						|
            {
 | 
						|
                instr = (pdc_byte *) ((flags & PDC_CONV_TMPALLOC) ?
 | 
						|
                     pdc_calloc_tmp(pdc, (size_t) (len + 2), fn, NULL, NULL) :
 | 
						|
                     pdc_calloc(pdc, (size_t) (len + 2), fn));
 | 
						|
                for (i = 0; i < len; i++)
 | 
						|
                    instr[i] = (pdc_byte) usinstr[i];
 | 
						|
 | 
						|
                if (inalloc)
 | 
						|
                {
 | 
						|
                    if (flags & PDC_CONV_TMPALLOC)
 | 
						|
                        pdc_free_tmp(pdc, instring);
 | 
						|
                    else
 | 
						|
                        pdc_free(pdc, instring);
 | 
						|
                }
 | 
						|
 | 
						|
                inalloc = pdc_true;
 | 
						|
                instring = instr;
 | 
						|
                instr = NULL;
 | 
						|
                inlen = len;
 | 
						|
 | 
						|
                if (inutf == pdc_bytes2)
 | 
						|
                    inutf = pdc_bytes;
 | 
						|
                else
 | 
						|
                    inutf = pdc_auto;
 | 
						|
            }
 | 
						|
            else
 | 
						|
            {
 | 
						|
                inutf = pdc_utf16;
 | 
						|
            }
 | 
						|
        }
 | 
						|
        break;
 | 
						|
 | 
						|
        /* OEM multi byte text strings */
 | 
						|
        case pdc_auto:
 | 
						|
        case pdc_bytes:
 | 
						|
        if (codepage > 0)
 | 
						|
        {
 | 
						|
#if defined(WIN32)
 | 
						|
            if (!(flags & PDC_CONV_ANALYZE) && inlen > 0)
 | 
						|
            {
 | 
						|
                if (logg)
 | 
						|
                    pdc_logg(pdc,
 | 
						|
                        "\t\tconverting according Windows codepage %d\n",
 | 
						|
                        codepage);
 | 
						|
 | 
						|
                len = MultiByteToWideChar((UINT) codepage, (DWORD) 0,
 | 
						|
                                          (LPCSTR) instring, inlen, NULL, 0);
 | 
						|
                if (len == 0)
 | 
						|
                {
 | 
						|
                    DWORD lasterror = GetLastError();
 | 
						|
 | 
						|
                    stemp1 = pdc_errprintf(pdc, "cp%d", codepage);
 | 
						|
                    if (lasterror == ERROR_INVALID_PARAMETER)
 | 
						|
                    {
 | 
						|
                        errcode = PDC_E_CONV_UNSUPP_MBTEXTFORM;
 | 
						|
                    }
 | 
						|
                    else
 | 
						|
                    {
 | 
						|
                        errcode = PDC_E_CONV_ILL_MBTEXTSTRING;
 | 
						|
                    }
 | 
						|
                    goto PDC_CONV_ERROR;
 | 
						|
                }
 | 
						|
 | 
						|
                len *= 2;
 | 
						|
                instr = (pdc_byte *) ((flags & PDC_CONV_TMPALLOC) ?
 | 
						|
                             pdc_calloc_tmp(pdc, (size_t) (len + 2), fn,
 | 
						|
                                            NULL, NULL) :
 | 
						|
                             pdc_calloc(pdc, (size_t) (len + 2), fn));
 | 
						|
                MultiByteToWideChar((UINT) codepage, (DWORD) 0, (LPCSTR)
 | 
						|
                                    instring, inlen,
 | 
						|
                                    (LPWSTR) instr, len);
 | 
						|
 | 
						|
                if (inalloc)
 | 
						|
                {
 | 
						|
                    if (flags & PDC_CONV_TMPALLOC)
 | 
						|
                        pdc_free_tmp(pdc, instring);
 | 
						|
                    else
 | 
						|
                        pdc_free(pdc, instring);
 | 
						|
                }
 | 
						|
 | 
						|
                inalloc = pdc_true;
 | 
						|
                instring = instr;
 | 
						|
                instr = NULL;
 | 
						|
                inlen = len;
 | 
						|
 | 
						|
                inutf = pdc_utf16;
 | 
						|
            }
 | 
						|
            else
 | 
						|
            {
 | 
						|
                inutf = pdc_bytes;
 | 
						|
            }
 | 
						|
#else   /* WIN32 */
 | 
						|
            errcode = PDC_E_CONV_UNSUPP_MBTEXTFORM;
 | 
						|
            goto PDC_CONV_ERROR;
 | 
						|
#endif  /* !WIN32 */
 | 
						|
        }
 | 
						|
        break;
 | 
						|
 | 
						|
        default:
 | 
						|
        break;
 | 
						|
    }
 | 
						|
 | 
						|
    /* analyzing UTF-16 textformat */
 | 
						|
    if (inutf == pdc_utf16)
 | 
						|
    {
 | 
						|
        if (pdc_is_utf16be_unicode(instring))
 | 
						|
            inutf = pdc_utf16be;
 | 
						|
        else if (pdc_is_utf16le_unicode(instring))
 | 
						|
            inutf = pdc_utf16le;
 | 
						|
    }
 | 
						|
 | 
						|
    /* analyzing auto textformat */
 | 
						|
    else if (inutf == pdc_auto)
 | 
						|
    {
 | 
						|
        if (pdc_is_utf8_bytecode(instring))
 | 
						|
            inutf = PDC_UTF8;
 | 
						|
        else if (pdc_is_utf16be_unicode(instring))
 | 
						|
            inutf = pdc_utf16be;
 | 
						|
        else if (pdc_is_utf16le_unicode(instring))
 | 
						|
            inutf = pdc_utf16le;
 | 
						|
        else if (inev && !(flags & PDC_CONV_FORCEUTF16))
 | 
						|
            inutf = pdc_bytes;
 | 
						|
        else
 | 
						|
            inutf = pdc_utf16;
 | 
						|
    }
 | 
						|
 | 
						|
    if (logg)
 | 
						|
        pdc_logg(pdc, "\t\tdetermined textformat: %s\n",
 | 
						|
                 pdc_get_keyword(inutf, pdc_textformat_keylist));
 | 
						|
 | 
						|
    /* only analyzing */
 | 
						|
    if (flags & PDC_CONV_ANALYZE)
 | 
						|
        goto PDC_CONV_EXIT;
 | 
						|
 | 
						|
    /* conversion to UTF-16 by swapping */
 | 
						|
    if ((inutf == pdc_utf16be  || inutf == pdc_utf16le) &&
 | 
						|
        (inutf != oututf || flags & PDC_CONV_TRYBYTES ||
 | 
						|
         flags & PDC_CONV_HTMLCHAR))
 | 
						|
    {
 | 
						|
        if (inlen &&
 | 
						|
            ((inutf == pdc_utf16be && !PDC_ISBIGENDIAN) ||
 | 
						|
             (inutf == pdc_utf16le &&  PDC_ISBIGENDIAN)))
 | 
						|
        {
 | 
						|
            if (inalloc)
 | 
						|
                pdc_swap_bytes2((char *) instring, inlen, NULL);
 | 
						|
            else
 | 
						|
            {
 | 
						|
                instr = (pdc_byte *) ((flags & PDC_CONV_TMPALLOC) ?
 | 
						|
                     pdc_calloc_tmp(pdc, (size_t) (inlen + 2), fn, NULL, NULL) :
 | 
						|
                     pdc_calloc(pdc, (size_t) (inlen + 2), fn));
 | 
						|
                pdc_swap_bytes2((char *) instring, inlen, (char *) instr);
 | 
						|
 | 
						|
                inalloc = pdc_true;
 | 
						|
                instring = instr;
 | 
						|
                instr = NULL;
 | 
						|
            }
 | 
						|
        }
 | 
						|
        inutf = pdc_utf16;
 | 
						|
    }
 | 
						|
 | 
						|
    /* conversion to UTF-32 by swapping */
 | 
						|
    if (inlen && inutf == pdc_utf32)
 | 
						|
    {
 | 
						|
 | 
						|
        if ((pdc_is_utf32be_unicode(instring) && !PDC_ISBIGENDIAN) ||
 | 
						|
            (pdc_is_utf32le_unicode(instring) &&  PDC_ISBIGENDIAN))
 | 
						|
        {
 | 
						|
            if (inalloc)
 | 
						|
                pdc_swap_bytes4((char *) instring, inlen, NULL);
 | 
						|
            else
 | 
						|
            {
 | 
						|
                instr = (pdc_byte *) ((flags & PDC_CONV_TMPALLOC) ?
 | 
						|
                     pdc_calloc_tmp(pdc, (size_t) (inlen + 4), fn, NULL, NULL) :
 | 
						|
                     pdc_calloc(pdc, (size_t) (inlen + 4), fn));
 | 
						|
                pdc_swap_bytes4((char *) instring, inlen, (char *) instr);
 | 
						|
 | 
						|
                inalloc = pdc_true;
 | 
						|
                instring = instr;
 | 
						|
                instr = NULL;
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    /* illegal UTF-16 / UTF-32 */
 | 
						|
    if (inutf >= pdc_utf16 && inlen % 2)
 | 
						|
    {
 | 
						|
        if (inutf == pdc_utf32 && inlen % 4)
 | 
						|
            errcode = PDC_E_CONV_ILLUTF32;
 | 
						|
        else
 | 
						|
            errcode = PDC_E_CONV_ILLUTF16;
 | 
						|
        goto PDC_CONV_ERROR;
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
    /* conversion to UTF-16 by inflation or encoding vector */
 | 
						|
    if (inutf == pdc_bytes &&
 | 
						|
        (oututf != pdc_bytes || flags & PDC_CONV_HTMLCHAR || inev != outev))
 | 
						|
    {
 | 
						|
        if (logg)
 | 
						|
        {
 | 
						|
            if (flags & PDC_CONV_HTMLCHAR)
 | 
						|
                pdc_logg(pdc, "\t\tbyte character entity substitution\n");
 | 
						|
        }
 | 
						|
 | 
						|
        len = 2 * inlen;
 | 
						|
        instr = (pdc_byte *) ((flags & PDC_CONV_TMPALLOC) ?
 | 
						|
             pdc_calloc_tmp(pdc, (size_t) (len + 2), fn, NULL, NULL) :
 | 
						|
             pdc_calloc(pdc, (size_t) (len + 2), fn));
 | 
						|
        usinstr = (pdc_ushort *) instr;
 | 
						|
 | 
						|
        j = 0;
 | 
						|
        for (i = 0; i < inlen; i++)
 | 
						|
        {
 | 
						|
            uv = (pdc_ushort) instring[i];
 | 
						|
            if (inev)
 | 
						|
            {
 | 
						|
                uv = inev->codes[uv];
 | 
						|
                if (!uv && (flags & PDC_CONV_ENCERROR) &&
 | 
						|
                    (!(flags & PDC_CONV_KEEPLBCHAR) ||
 | 
						|
                     !pdc_is_linebreaking_relchar(uv)))
 | 
						|
                {
 | 
						|
                    errcode = PDC_E_ENC_NOTDEF_CODE;
 | 
						|
                    stemp1 = pdc_errprintf(pdc, "x%02X", instring[i]);
 | 
						|
                    stemp2 = inev->apiname;
 | 
						|
                    goto PDC_CONV_ERROR;
 | 
						|
                }
 | 
						|
            }
 | 
						|
 | 
						|
 | 
						|
            usinstr[j] = uv;
 | 
						|
            j++;
 | 
						|
        }
 | 
						|
 | 
						|
        if (inalloc)
 | 
						|
        {
 | 
						|
            if (flags & PDC_CONV_TMPALLOC)
 | 
						|
                pdc_free_tmp(pdc, instring);
 | 
						|
            else
 | 
						|
                pdc_free(pdc, instring);
 | 
						|
        }
 | 
						|
 | 
						|
        inalloc = pdc_true;
 | 
						|
        instring = instr;
 | 
						|
        instr = NULL;
 | 
						|
        inlen = 2 * j;
 | 
						|
        inutf = pdc_utf16;
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
 | 
						|
    /* UTF conversion */
 | 
						|
    oututf_s = oututf;
 | 
						|
    if ((oututf_s == pdc_bytes && inutf == pdc_utf8) ||
 | 
						|
         oututf_s == pdc_utf16be || oututf_s == pdc_utf16le)
 | 
						|
        oututf_s = pdc_utf16;
 | 
						|
    if (inutf != oututf_s && oututf_s != pdc_bytes)
 | 
						|
    {
 | 
						|
        len = 4 * (inlen + 1);
 | 
						|
        instr = (pdc_byte *) ((flags & PDC_CONV_TMPALLOC) ?
 | 
						|
             pdc_calloc_tmp(pdc, (size_t) len, fn, NULL, NULL) :
 | 
						|
             pdc_calloc(pdc, (size_t) len, fn));
 | 
						|
 | 
						|
        if (inlen)
 | 
						|
        {
 | 
						|
            pdc_convers_result result = conversionOK;
 | 
						|
            pdc_byte *instringa, *instra, *instringe, *instre;
 | 
						|
            UTF8 *isa8 = NULL, *ise8 = NULL;
 | 
						|
            UTF16 *isa16, *ise16;
 | 
						|
            UTF32 *isa32, *ise32;
 | 
						|
 | 
						|
            if (logg)
 | 
						|
               pdc_logg(pdc, "\t\tUTF conversion\n");
 | 
						|
 | 
						|
            instringa = instring;
 | 
						|
            instringe = instring + inlen;
 | 
						|
            instra = instr;
 | 
						|
            instre = instr + len;
 | 
						|
 | 
						|
            if (inutf == pdc_utf8)
 | 
						|
            {
 | 
						|
                isa8 = (UTF8 *) instringa;
 | 
						|
                ise8 = (UTF8 *) instringe;
 | 
						|
                if (oututf_s == pdc_utf16)
 | 
						|
                {
 | 
						|
                    isa16 = (UTF16 *) instra;
 | 
						|
                    ise16 = (UTF16 *) instre;
 | 
						|
                    result = pdc_convertUTF8toUTF16(&isa8, ise8,
 | 
						|
                                                    &isa16, ise16,
 | 
						|
                                                    strictConversion);
 | 
						|
                    instra = (pdc_byte *) isa16;
 | 
						|
                    instre = (pdc_byte *) ise16;
 | 
						|
                }
 | 
						|
                else
 | 
						|
                {
 | 
						|
                    isa32 = (UTF32 *) instra;
 | 
						|
                    ise32 = (UTF32 *) instre;
 | 
						|
                    result = pdc_convertUTF8toUTF32(&isa8, ise8,
 | 
						|
                                                    &isa32, ise32,
 | 
						|
                                                    strictConversion);
 | 
						|
                    instra = (pdc_byte *) isa32;
 | 
						|
                    instre = (pdc_byte *) ise32;
 | 
						|
                }
 | 
						|
            }
 | 
						|
            else if (inutf == pdc_utf16)
 | 
						|
            {
 | 
						|
                isa16 = (UTF16 *) instringa;
 | 
						|
                ise16 = (UTF16 *) instringe;
 | 
						|
                if (oututf_s == pdc_utf8)
 | 
						|
                {
 | 
						|
                    isa8 = (UTF8 *) instra;
 | 
						|
                    ise8 = (UTF8 *) instre;
 | 
						|
                    result = pdc_convertUTF16toUTF8(&isa16, ise16, &isa8, ise8,
 | 
						|
                                                    strictConversion);
 | 
						|
                    instra = (pdc_byte *) isa8;
 | 
						|
                    instre = (pdc_byte *) ise8;
 | 
						|
                }
 | 
						|
                else
 | 
						|
                {
 | 
						|
                    isa32 = (UTF32 *) instra;
 | 
						|
                    ise32 = (UTF32 *) instre;
 | 
						|
                    result = pdc_convertUTF16toUTF32(&isa16, ise16,
 | 
						|
                                                     &isa32, ise32,
 | 
						|
                                                     strictConversion);
 | 
						|
                    instra = (pdc_byte *) isa32;
 | 
						|
                    instre = (pdc_byte *) ise32;
 | 
						|
                }
 | 
						|
            }
 | 
						|
            else if (inutf == pdc_utf32)
 | 
						|
            {
 | 
						|
                isa32 = (UTF32 *) instringa;
 | 
						|
                ise32 = (UTF32 *) instringe;
 | 
						|
                if (oututf_s == pdc_utf8)
 | 
						|
                {
 | 
						|
                    isa8 = (UTF8 *) instra;
 | 
						|
                    ise8 = (UTF8 *) instre;
 | 
						|
                    result = pdc_convertUTF32toUTF8(&isa32, ise32,
 | 
						|
                                                    &isa8, ise8,
 | 
						|
                                                    strictConversion);
 | 
						|
                    instra = (pdc_byte *) isa8;
 | 
						|
                    instre = (pdc_byte *) ise8;
 | 
						|
                }
 | 
						|
                else
 | 
						|
                {
 | 
						|
                    isa16 = (UTF16 *) instra;
 | 
						|
                    ise16 = (UTF16 *) instre;
 | 
						|
                    result = pdc_convertUTF32toUTF16(&isa32, ise32,
 | 
						|
                                                     &isa16, ise16,
 | 
						|
                                                     strictConversion);
 | 
						|
                    instra = (pdc_byte *) isa16;
 | 
						|
                    instre = (pdc_byte *) ise16;
 | 
						|
                }
 | 
						|
            }
 | 
						|
 | 
						|
            switch (result)
 | 
						|
            {
 | 
						|
                case targetExhausted:
 | 
						|
                errcode = PDC_E_CONV_MEMOVERFLOW;
 | 
						|
                break;
 | 
						|
 | 
						|
                case sourceExhausted:
 | 
						|
                case sourceIllegal:
 | 
						|
                if (inutf == pdc_utf8)
 | 
						|
                {
 | 
						|
                    UTF8 *bp, *bpe;
 | 
						|
                    char *sb = sbuf;
 | 
						|
 | 
						|
                    bpe = MIN(ise8 - 1, isa8 + 3);
 | 
						|
                    for (bp = isa8; bp <= bpe; bp++)
 | 
						|
                        sb += sprintf(sb, "\\x%02X", *bp);
 | 
						|
                    if (*bp)
 | 
						|
                        sb += sprintf(sb, "...");
 | 
						|
                    sb += sprintf(sb, " (");
 | 
						|
                    for (bp = isa8; bp <= bpe; bp++)
 | 
						|
                        sb += sprintf(sb, "%c", *bp);
 | 
						|
                    if (*bp)
 | 
						|
                        sb += sprintf(sb, "...");
 | 
						|
                    sb += sprintf(sb, ")");
 | 
						|
                    stemp1 = sbuf;
 | 
						|
 | 
						|
                    stemp2 = pdc_errprintf(pdc, "%d", isa8 - (UTF8 *)instringa);
 | 
						|
 | 
						|
                    if (flags & PDC_CONV_INFLATE)
 | 
						|
                    {
 | 
						|
                        pdc_warning(pdc, PDC_E_CONV_ILLUTF8SEQU, stemp1, stemp2,
 | 
						|
                                    0, 0);
 | 
						|
 | 
						|
                        pdc_inflate_ascii((char *) instring, inlen,
 | 
						|
                                          (char *) instr, pdc_utf16);
 | 
						|
                        instra = instr + 2 * inlen;
 | 
						|
                    }
 | 
						|
                    else
 | 
						|
                    {
 | 
						|
                        errcode = PDC_E_CONV_ILLUTF8SEQU;
 | 
						|
                    }
 | 
						|
                }
 | 
						|
                else
 | 
						|
                {
 | 
						|
                    stemp1 = pdc_get_keyword((int)inutf, pdc_utfformat_keylist);
 | 
						|
                    errcode = PDC_E_CONV_ILLUTF;
 | 
						|
                }
 | 
						|
                break;
 | 
						|
 | 
						|
                default:
 | 
						|
                break;
 | 
						|
            }
 | 
						|
 | 
						|
            if (errcode)
 | 
						|
            {
 | 
						|
                if (logg)
 | 
						|
                   pdc_logg(pdc, "\t\tUTF conversion error %d\n", result);
 | 
						|
 | 
						|
                goto PDC_CONV_ERROR;
 | 
						|
            }
 | 
						|
 | 
						|
            inlen = instra - instr;
 | 
						|
        }
 | 
						|
 | 
						|
        if (inalloc)
 | 
						|
        {
 | 
						|
            if (flags & PDC_CONV_TMPALLOC)
 | 
						|
                pdc_free_tmp(pdc, instring);
 | 
						|
            else
 | 
						|
                pdc_free(pdc, instring);
 | 
						|
        }
 | 
						|
 | 
						|
        len = (oututf == pdc_utf32) ? inlen + 4 : inlen + 2;
 | 
						|
        if (inlen + 4 != len)
 | 
						|
            instr = (pdc_byte *) ((flags & PDC_CONV_TMPALLOC) ?
 | 
						|
                 pdc_realloc_tmp(pdc, instr, (size_t) len, fn) :
 | 
						|
                 pdc_realloc(pdc, instr, (size_t) len, fn));
 | 
						|
        instr[inlen] = 0;
 | 
						|
        instr[inlen + 1] = 0;
 | 
						|
        if (oututf == pdc_utf32)
 | 
						|
        {
 | 
						|
            instr[inlen + 2] = 0;
 | 
						|
            instr[inlen + 3] = 0;
 | 
						|
        }
 | 
						|
 | 
						|
        inalloc = pdc_true;
 | 
						|
        instring = instr;
 | 
						|
        instr = NULL;
 | 
						|
        inutf = oututf_s;
 | 
						|
    }
 | 
						|
 | 
						|
    if (inutf == pdc_bytes)
 | 
						|
    {
 | 
						|
        if (!inalloc)
 | 
						|
        {
 | 
						|
            instr = (pdc_byte *) ((flags & PDC_CONV_TMPALLOC) ?
 | 
						|
                 pdc_calloc_tmp(pdc, (size_t) (inlen + 2), fn, NULL, NULL) :
 | 
						|
                 pdc_calloc(pdc, (size_t) (inlen + 2), fn));
 | 
						|
            memcpy(instr, instring, (size_t) inlen);
 | 
						|
 | 
						|
            inalloc = pdc_true;
 | 
						|
            instring = instr;
 | 
						|
            instr = NULL;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    /* trying to reduce UTF-16 string to bytes string */
 | 
						|
    if (inutf == pdc_utf16 &&
 | 
						|
        (oututf == pdc_bytes || flags & PDC_CONV_TRYBYTES))
 | 
						|
    {
 | 
						|
        if (logg)
 | 
						|
           pdc_logg(pdc, "\t\ttry to reduce UTF-16 to bytes\n");
 | 
						|
 | 
						|
        if (pdc_is_utf16be_unicode(instring) ||
 | 
						|
            pdc_is_utf16le_unicode(instring))
 | 
						|
            n = 1;
 | 
						|
        else
 | 
						|
            n = 0;
 | 
						|
 | 
						|
        len = (inlen - n) / 2;
 | 
						|
        instr = (pdc_byte *) ((flags & PDC_CONV_TMPALLOC) ?
 | 
						|
             pdc_calloc_tmp(pdc, (size_t) (len + 2), fn, NULL, NULL) :
 | 
						|
             pdc_calloc(pdc, (size_t) (len + 2), fn));
 | 
						|
        usinstr = (pdc_ushort *) instring;
 | 
						|
 | 
						|
        for (i = 0; i < len; i++)
 | 
						|
        {
 | 
						|
            uv = usinstr[i + n];
 | 
						|
            if (outev && uv)
 | 
						|
            {
 | 
						|
                j = pdc_get_encoding_bytecode(pdc, outev, uv);
 | 
						|
                if (j < 0 && (flags & PDC_CONV_ENCERROR) && oututf == pdc_bytes)
 | 
						|
                {
 | 
						|
                    errcode = PDC_E_ENC_NOTDEF_UNICODE;
 | 
						|
                    stemp1 = pdc_errprintf(pdc, "%04X", uv);
 | 
						|
                    stemp2 = outev->apiname;
 | 
						|
                    goto PDC_CONV_ERROR;
 | 
						|
                }
 | 
						|
                uv = (pdc_ushort) j;
 | 
						|
            }
 | 
						|
            if (uv > PDC_UNICODE_MAXLATIN1)
 | 
						|
                break;
 | 
						|
 | 
						|
            instr[i] = (pdc_byte) uv;
 | 
						|
        }
 | 
						|
 | 
						|
        if (i == len)
 | 
						|
        {
 | 
						|
            if (inalloc)
 | 
						|
            {
 | 
						|
                if (flags & PDC_CONV_TMPALLOC)
 | 
						|
                    pdc_free_tmp(pdc, instring);
 | 
						|
                else
 | 
						|
                    pdc_free(pdc, instring);
 | 
						|
            }
 | 
						|
 | 
						|
            inalloc = pdc_true;
 | 
						|
            instring = instr;
 | 
						|
            instr = NULL;
 | 
						|
            inlen = len;
 | 
						|
            inutf = pdc_bytes;
 | 
						|
        }
 | 
						|
        else
 | 
						|
        {
 | 
						|
            if (flags & PDC_CONV_TMPALLOC)
 | 
						|
                pdc_free_tmp(pdc, instr);
 | 
						|
            else
 | 
						|
                pdc_free(pdc, instr);
 | 
						|
            instr = NULL;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    /* UTF-8 format */
 | 
						|
    if (inutf == pdc_utf8)
 | 
						|
    {
 | 
						|
        hasbom = pdc_is_utf8_unicode(instring);
 | 
						|
 | 
						|
        if (flags & PDC_CONV_TRY7BYTES)
 | 
						|
        {
 | 
						|
            if (logg)
 | 
						|
               pdc_logg(pdc, "\t\ttry to reduce UTF-8 to 7-bit\n");
 | 
						|
 | 
						|
            for (i = hasbom ? 3 : 0; i < inlen; i++)
 | 
						|
                if (instring[i] > PDC_UNICODE_MAXASCII)
 | 
						|
                    break;
 | 
						|
            if (i == inlen)
 | 
						|
            {
 | 
						|
                flags &= ~PDC_CONV_WITHBOM;
 | 
						|
                flags |= PDC_CONV_NOBOM;
 | 
						|
                inutf = pdc_bytes;
 | 
						|
            }
 | 
						|
        }
 | 
						|
        else if (hasbom && (flags & PDC_CONV_AUTOBOM))
 | 
						|
        {
 | 
						|
            flags &= ~PDC_CONV_NOBOM;
 | 
						|
            flags |= PDC_CONV_WITHBOM;
 | 
						|
        }
 | 
						|
        else if ((flags & PDC_CONV_WITHBOM) && (flags & PDC_CONV_NOBOM))
 | 
						|
        {
 | 
						|
            flags &= ~PDC_CONV_NOBOM;
 | 
						|
        }
 | 
						|
 | 
						|
        if (!inalloc || flags & PDC_CONV_WITHBOM || flags & PDC_CONV_NOBOM)
 | 
						|
        {
 | 
						|
            i = (flags & PDC_CONV_WITHBOM && !hasbom) ? 3 : 0;
 | 
						|
            j = (flags & PDC_CONV_NOBOM && hasbom) ? 3 : 0;
 | 
						|
 | 
						|
            len = inlen + i - j;
 | 
						|
            instr = (pdc_byte *) ((flags & PDC_CONV_TMPALLOC) ?
 | 
						|
                 pdc_calloc_tmp(pdc, (size_t) (len + 2), fn, NULL, NULL) :
 | 
						|
                 pdc_calloc(pdc, (size_t) (len + 2), fn));
 | 
						|
            memcpy(&instr[i], &instring[j], (size_t) (inlen - j));
 | 
						|
            instr[len] = 0;
 | 
						|
 | 
						|
            if (inalloc)
 | 
						|
            {
 | 
						|
                if (flags & PDC_CONV_TMPALLOC)
 | 
						|
                    pdc_free_tmp(pdc, instring);
 | 
						|
                else
 | 
						|
                    pdc_free(pdc, instring);
 | 
						|
            }
 | 
						|
 | 
						|
            inalloc = pdc_true;
 | 
						|
            instring = instr;
 | 
						|
            instr = NULL;
 | 
						|
            inlen = len;
 | 
						|
 | 
						|
            hasbom = (flags & PDC_CONV_WITHBOM);
 | 
						|
        }
 | 
						|
 | 
						|
        if (hasbom)
 | 
						|
        {
 | 
						|
            instring[0] = PDF_BOM2;
 | 
						|
            instring[1] = PDF_BOM3;
 | 
						|
            instring[2] = PDF_BOM4;
 | 
						|
        }
 | 
						|
 | 
						|
    }
 | 
						|
 | 
						|
    /* UTF-16 formats */
 | 
						|
    if (inutf == pdc_utf16 || inutf == pdc_utf16be || inutf == pdc_utf16le)
 | 
						|
    {
 | 
						|
        hasbom = pdc_is_utf16be_unicode(instring) ||
 | 
						|
                 pdc_is_utf16le_unicode(instring);
 | 
						|
 | 
						|
        if (hasbom && (flags & PDC_CONV_AUTOBOM))
 | 
						|
        {
 | 
						|
            flags &= ~PDC_CONV_NOBOM;
 | 
						|
            flags |= PDC_CONV_WITHBOM;
 | 
						|
        }
 | 
						|
        else if ((flags & PDC_CONV_WITHBOM) && (flags & PDC_CONV_NOBOM))
 | 
						|
        {
 | 
						|
            flags &= ~PDC_CONV_NOBOM;
 | 
						|
        }
 | 
						|
 | 
						|
        if (!inalloc || oututf == pdc_utf16be || oututf == pdc_utf16le ||
 | 
						|
            flags & PDC_CONV_WITHBOM || flags & PDC_CONV_NOBOM)
 | 
						|
        {
 | 
						|
            i = (flags & PDC_CONV_WITHBOM && !hasbom) ? 2 : 0;
 | 
						|
            j = (flags & PDC_CONV_NOBOM && hasbom) ? 2 : 0;
 | 
						|
 | 
						|
            len = inlen + i - j;
 | 
						|
            instr = (pdc_byte *) ((flags & PDC_CONV_TMPALLOC) ?
 | 
						|
                 pdc_calloc_tmp(pdc, (size_t) (len + 2), fn, NULL, NULL) :
 | 
						|
                 pdc_calloc(pdc, (size_t) (len + 2), fn));
 | 
						|
            memcpy(&instr[i], &instring[j], (size_t) (inlen - j));
 | 
						|
 | 
						|
            if (inalloc)
 | 
						|
            {
 | 
						|
                if (flags & PDC_CONV_TMPALLOC)
 | 
						|
                    pdc_free_tmp(pdc, instring);
 | 
						|
                else
 | 
						|
                    pdc_free(pdc, instring);
 | 
						|
            }
 | 
						|
 | 
						|
            instring = instr;
 | 
						|
            instr = NULL;
 | 
						|
            inlen = len;
 | 
						|
 | 
						|
            hasbom = (flags & PDC_CONV_WITHBOM);
 | 
						|
        }
 | 
						|
 | 
						|
        i = hasbom ? 2 : 0;
 | 
						|
        if (inutf == pdc_utf16)
 | 
						|
        {
 | 
						|
            if (oututf == pdc_utf16be)
 | 
						|
            {
 | 
						|
                inutf = pdc_utf16be;
 | 
						|
                toswap = !PDC_ISBIGENDIAN;
 | 
						|
            }
 | 
						|
            if (oututf == pdc_utf16le)
 | 
						|
            {
 | 
						|
                inutf = pdc_utf16le;
 | 
						|
                toswap = PDC_ISBIGENDIAN;
 | 
						|
            }
 | 
						|
            if (toswap)
 | 
						|
                pdc_swap_bytes2((char *) &instring[i], inlen - i, NULL);
 | 
						|
        }
 | 
						|
 | 
						|
        if (hasbom)
 | 
						|
        {
 | 
						|
            if (inutf == pdc_utf16be ||
 | 
						|
                (inutf == pdc_utf16 && PDC_ISBIGENDIAN))
 | 
						|
            {
 | 
						|
                instring[0] = PDF_BOM0;
 | 
						|
                instring[1] = PDF_BOM1;
 | 
						|
            }
 | 
						|
            if (inutf == pdc_utf16le ||
 | 
						|
                (inutf == pdc_utf16 && !PDC_ISBIGENDIAN))
 | 
						|
            {
 | 
						|
                instring[0] = PDF_BOM1;
 | 
						|
                instring[1] = PDF_BOM0;
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if (logg)
 | 
						|
        pdc_logg(pdc, "\t\ttextformat of converted string: %s\n",
 | 
						|
                 pdc_get_keyword(inutf, pdc_textformat_keylist));
 | 
						|
 | 
						|
    PDC_CONV_EXIT:
 | 
						|
    *oututf_p = inutf;
 | 
						|
    if (outlen)
 | 
						|
        *outlen = inlen;
 | 
						|
    *outstring = instring;
 | 
						|
    return 0;
 | 
						|
 | 
						|
    PDC_CONV_ERROR:
 | 
						|
    if (outlen)
 | 
						|
        *outlen = 0;
 | 
						|
    *outstring = NULL;
 | 
						|
 | 
						|
    if (errcode > 0)
 | 
						|
        pdc_set_errmsg(pdc, errcode, stemp1, stemp2, 0, 0);
 | 
						|
 | 
						|
    if (instr != NULL)
 | 
						|
    {
 | 
						|
        if (flags & PDC_CONV_TMPALLOC)
 | 
						|
            pdc_free_tmp(pdc, instr);
 | 
						|
        else
 | 
						|
            pdc_free(pdc, instr);
 | 
						|
    }
 | 
						|
 | 
						|
    if (inalloc)
 | 
						|
    {
 | 
						|
        if (flags & PDC_CONV_TMPALLOC)
 | 
						|
            pdc_free_tmp(pdc, instring);
 | 
						|
        else
 | 
						|
            pdc_free(pdc, instring);
 | 
						|
    }
 | 
						|
 | 
						|
    if (verbose)
 | 
						|
        PDC_RETHROW(pdc);
 | 
						|
 | 
						|
    return errcode;
 | 
						|
}
 | 
						|
#if defined(_MSC_VER) && defined(_MANAGED)
 | 
						|
#pragma managed
 | 
						|
#endif
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 *  pdc_convert_name_ext converts a string of name data type to UTF-8
 | 
						|
 *
 | 
						|
 *  flags & PDC_CONV_EBCDIC: converts to EBCDIC-UTF-8
 | 
						|
 *
 | 
						|
 *  len == 0: If the string has a [EBCDIC-]UTF-8 BOM or
 | 
						|
 *            flags & PDC_CONV_ISUTF8 is set the string will be duplicated.
 | 
						|
 *            Otherwise the string has encoding enc and codepage
 | 
						|
 *            codepage.
 | 
						|
 *            If enc == pdc_unicode the string is "UTF-16" encoded.
 | 
						|
 *            Otherwise: If enc < pdc_winansi the string is "host" encoded.
 | 
						|
 *
 | 
						|
 *  len  > 0: The string is a UTF-16 string of len bytes.
 | 
						|
 *
 | 
						|
 */
 | 
						|
char *
 | 
						|
pdc_convert_name_ext(pdc_core *pdc, const char *name, int len,
 | 
						|
                     pdc_encoding enc, int codepage, int flags)
 | 
						|
{
 | 
						|
    static const char fn[] = "pdc_convert_name_ext";
 | 
						|
    pdc_encodingvector *ev = NULL;
 | 
						|
    pdc_text_format nameformat = pdc_utf16;
 | 
						|
    pdc_text_format outnameformat = pdc_utf8;
 | 
						|
    pdc_byte *convname;
 | 
						|
    char *outname = NULL;
 | 
						|
    int outlen;
 | 
						|
 | 
						|
    if (name == NULL)
 | 
						|
        return NULL;
 | 
						|
 | 
						|
    if (len == 0)
 | 
						|
    {
 | 
						|
        /* already [EBCDIC-]UTF-8 encoded */
 | 
						|
        if ((flags & PDC_CONV_ISUTF8) || pdc_is_utf8_bytecode(name))
 | 
						|
        {
 | 
						|
            if (!(flags & PDC_CONV_WITHBOM))
 | 
						|
                flags |= PDC_CONV_NOBOM;
 | 
						|
 | 
						|
            if (!(flags & PDC_CONV_EBCDIC))
 | 
						|
                flags |= PDC_CONV_ASCII;
 | 
						|
 | 
						|
            /* On EBCDIC platforms EBCDIC-UTF-8 name strings are expected */
 | 
						|
            outname = pdc_strdup_ext(pdc, name, (flags & ~PDC_CONV_EBCDIC), fn);
 | 
						|
 | 
						|
            if (outname != NULL)
 | 
						|
                return outname;
 | 
						|
        }
 | 
						|
 | 
						|
        /* see bug #1486 */
 | 
						|
        if (enc == pdc_unicode)
 | 
						|
        {
 | 
						|
            /* UTF-16 encoded string */
 | 
						|
            len = (int) pdc_wstrlen(name);
 | 
						|
        }
 | 
						|
        else
 | 
						|
        {
 | 
						|
            /* 8-bit encoded string */
 | 
						|
            nameformat = pdc_bytes;
 | 
						|
            if (enc < pdc_winansi)
 | 
						|
                ev = pdc_get_encoding_vector(pdc,pdc_find_encoding(pdc,"host"));
 | 
						|
            else
 | 
						|
                ev = pdc_get_encoding_vector(pdc, enc);
 | 
						|
 | 
						|
            len = (int) strlen(name);
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if (flags & PDC_CONV_EBCDIC)
 | 
						|
        outnameformat = PDC_UTF8;
 | 
						|
 | 
						|
    flags |= PDC_CONV_TRY7BYTES;
 | 
						|
    if (pdc->charref)
 | 
						|
        flags |= PDC_CONV_HTMLCHAR;
 | 
						|
    if (pdc->escapesequ)
 | 
						|
        flags |= PDC_CONV_BSSEQU;
 | 
						|
 | 
						|
    /* convert to UTF-8 */
 | 
						|
    pdc_convert_string(pdc, nameformat, codepage, ev, (pdc_byte *) name, len,
 | 
						|
                &outnameformat, NULL, &convname, &outlen, flags,
 | 
						|
                pdc_true);
 | 
						|
 | 
						|
    return (char *) convname;
 | 
						|
}
 | 
						|
 | 
						|
char *
 | 
						|
pdc_convert_name(pdc_core *pdc, const char *name, int len, int flags)
 | 
						|
{
 | 
						|
    return pdc_convert_name_ext(pdc, name, len, pdc_invalidenc, 0, flags);
 | 
						|
}
 | 
						|
 | 
						|
/* returned string is temporary allocated
 | 
						|
*/
 | 
						|
char *
 | 
						|
pdc_utf8_to_hostbytes(pdc_core *pdc, pdc_bool honorlang, char *name)
 | 
						|
{
 | 
						|
    static const char fn[] = "pdc_utf8_to_hostbytes";
 | 
						|
    pdc_encoding outenc = pdc_invalidenc;
 | 
						|
    pdc_encodingvector *outev = NULL;
 | 
						|
    pdc_text_format informat = PDC_UTF8;
 | 
						|
    pdc_text_format outformat = pdc_utf16;
 | 
						|
    pdc_byte *outname = NULL;
 | 
						|
    int len = (int) strlen(name);
 | 
						|
 | 
						|
    {
 | 
						|
        (void) fn;
 | 
						|
        (void) honorlang;
 | 
						|
        outenc = pdc_find_encoding(pdc, "host");
 | 
						|
    }
 | 
						|
 | 
						|
    outev = pdc_get_encoding_vector(pdc, outenc);
 | 
						|
 | 
						|
    pdc_convert_string(pdc, informat, 0, NULL, (pdc_byte *) name, len,
 | 
						|
                &outformat, outev, &outname, &len,
 | 
						|
                PDC_CONV_TRYBYTES | PDC_CONV_NOBOM | PDC_CONV_TMPALLOC,
 | 
						|
                pdc_true);
 | 
						|
    if (outformat == pdc_utf16)
 | 
						|
    {
 | 
						|
        pdc_free_tmp(pdc, outname);
 | 
						|
        outname = NULL;
 | 
						|
    }
 | 
						|
 | 
						|
    return (char *) outname;
 | 
						|
}
 | 
						|
 | 
						|
/* returned string is temporary allocated
 | 
						|
*/
 | 
						|
char *
 | 
						|
pdc_hostbytes_to_utf8(pdc_core *pdc, pdc_bool honorlang, char *name)
 | 
						|
{
 | 
						|
    static const char fn[] = "pdc_hostbytes_to_utf8";
 | 
						|
    pdc_encoding inenc = pdc_invalidenc;
 | 
						|
    pdc_encodingvector *inev = NULL;
 | 
						|
    pdc_text_format informat = pdc_bytes;
 | 
						|
    pdc_text_format outformat = PDC_UTF8;
 | 
						|
    pdc_byte *outname = NULL;
 | 
						|
    int len = (int) strlen(name);
 | 
						|
 | 
						|
    {
 | 
						|
        (void) fn;
 | 
						|
        (void) honorlang;
 | 
						|
        inenc = pdc_find_encoding(pdc, "host");
 | 
						|
    }
 | 
						|
 | 
						|
    inev = pdc_get_encoding_vector(pdc, inenc);
 | 
						|
 | 
						|
    pdc_convert_string(pdc, informat, 0, inev, (pdc_byte *) name, len,
 | 
						|
                &outformat, NULL, &outname, &len,
 | 
						|
                PDC_CONV_NOBOM | PDC_CONV_TMPALLOC, pdc_true);
 | 
						|
 | 
						|
    return (char *) outname;
 | 
						|
}
 | 
						|
 | 
						|
/* --------------------- basic UTF conversion functions --------------------- */
 | 
						|
 | 
						|
char *
 | 
						|
pdc_utf16_to_utf8(pdc_core *pdc, const char *utf16string, int len, int flags,
 | 
						|
                  int *size)
 | 
						|
{
 | 
						|
    pdc_text_format outtextformat = pdc_utf8;
 | 
						|
    pdc_byte *utf8string = NULL;
 | 
						|
    int outlen;
 | 
						|
 | 
						|
    if (!utf16string)
 | 
						|
        pdc_error(pdc, PDC_E_ILLARG_EMPTY, "utf16string", 0, 0, 0);
 | 
						|
 | 
						|
    if (flags & PDC_CONV_EBCDIC)
 | 
						|
        outtextformat = PDC_UTF8;
 | 
						|
 | 
						|
    flags |= PDC_CONV_AUTOBOM;
 | 
						|
    pdc_convert_string(pdc, pdc_utf16, 0, NULL,
 | 
						|
                       (pdc_byte *) utf16string, len,
 | 
						|
                       &outtextformat, NULL, &utf8string, &outlen,
 | 
						|
                       flags, pdc_true);
 | 
						|
 | 
						|
    if (size) *size = outlen;
 | 
						|
 | 
						|
    return (char *) utf8string;
 | 
						|
}
 | 
						|
 | 
						|
char *
 | 
						|
pdc_utf8_to_utf16(pdc_core *pdc, const char *utf8string, const char *format,
 | 
						|
                  int flags, int *size)
 | 
						|
{
 | 
						|
    pdc_text_format textformat = pdc_utf8;
 | 
						|
    pdc_text_format outtextformat = pdc_utf16;
 | 
						|
    pdc_byte *utf16string = NULL;
 | 
						|
    int len;
 | 
						|
 | 
						|
    if (!utf8string)
 | 
						|
        pdc_error(pdc, PDC_E_ILLARG_EMPTY, "utf8string", 0, 0, 0);
 | 
						|
    len = (int) strlen(utf8string);
 | 
						|
 | 
						|
    if (format && *format)
 | 
						|
    {
 | 
						|
        int k = pdc_get_keycode_ci(format, pdc_textformat_keylist);
 | 
						|
 | 
						|
        /* see bug #2175 */
 | 
						|
        if (k == PDC_KEY_NOTFOUND)
 | 
						|
        {
 | 
						|
            char **sfl;
 | 
						|
            const char *sf;
 | 
						|
            int ns, i;
 | 
						|
 | 
						|
            sf = NULL;
 | 
						|
            ns = pdc_split_stringlist(pdc, format, NULL, 0, &sfl);
 | 
						|
            for (i = 0; i < ns; i++)
 | 
						|
            {
 | 
						|
                if (!strcmp(sfl[i], "inflate"))
 | 
						|
                    flags |= PDC_CONV_INFLATE;
 | 
						|
                else
 | 
						|
                    sf = sfl[i];
 | 
						|
            }
 | 
						|
            if (sf != NULL)
 | 
						|
                k = pdc_get_keycode_ci(sf, pdc_textformat_keylist);
 | 
						|
            else
 | 
						|
                k = pdc_utf16;
 | 
						|
 | 
						|
            pdc_cleanup_stringlist(pdc, sfl);
 | 
						|
        }
 | 
						|
 | 
						|
        if (k == PDC_KEY_NOTFOUND ||
 | 
						|
            ((pdc_text_format) k != pdc_utf16 &&
 | 
						|
             (pdc_text_format) k != pdc_utf16be &&
 | 
						|
             (pdc_text_format) k != pdc_utf16le))
 | 
						|
            pdc_error(pdc, PDC_E_ILLARG_STRING, "format", format, 0, 0);
 | 
						|
 | 
						|
        outtextformat = (pdc_text_format) k;
 | 
						|
    }
 | 
						|
 | 
						|
    if (flags & PDC_CONV_EBCDIC)
 | 
						|
        textformat = PDC_UTF8;
 | 
						|
 | 
						|
    if (outtextformat == pdc_utf16)
 | 
						|
        flags |= PDC_CONV_AUTOBOM;
 | 
						|
    else
 | 
						|
        flags |= PDC_CONV_WITHBOM;
 | 
						|
    pdc_convert_string(pdc, textformat, 0, NULL,
 | 
						|
                      (pdc_byte *) utf8string, len,
 | 
						|
                      &outtextformat, NULL, &utf16string, size,
 | 
						|
                      flags, pdc_true);
 | 
						|
 | 
						|
    return (char *) utf16string;
 | 
						|
}
 | 
						|
 | 
						|
char *
 | 
						|
pdc_utf16_to_utf32(pdc_core *pdc, const char *utf16string, int len, int *size)
 | 
						|
{
 | 
						|
    pdc_text_format outtextformat = pdc_utf32;
 | 
						|
    pdc_byte *utf32string = NULL;
 | 
						|
 | 
						|
    if (!utf16string)
 | 
						|
        pdc_error(pdc, PDC_E_ILLARG_EMPTY, "utf16string", 0, 0, 0);
 | 
						|
 | 
						|
    pdc_convert_string(pdc, pdc_utf16, 0, NULL,
 | 
						|
                       (pdc_byte *) utf16string, len,
 | 
						|
                       &outtextformat, NULL, &utf32string, size,
 | 
						|
                       0, pdc_true);
 | 
						|
 | 
						|
    return (char *) utf32string;
 | 
						|
}
 | 
						|
 | 
						|
char *
 | 
						|
pdc_utf32_to_utf8(pdc_core *pdc, const char *utf32string, int len, int flags,
 | 
						|
                  int *size)
 | 
						|
{
 | 
						|
    pdc_text_format outtextformat = pdc_utf8;
 | 
						|
    pdc_byte *utf8string = NULL;
 | 
						|
    int outlen;
 | 
						|
 | 
						|
    if (!utf32string)
 | 
						|
        pdc_error(pdc, PDC_E_ILLARG_EMPTY, "utf32string", 0, 0, 0);
 | 
						|
 | 
						|
    if (flags & PDC_CONV_EBCDIC)
 | 
						|
        outtextformat = PDC_UTF8;
 | 
						|
 | 
						|
    flags |= PDC_CONV_AUTOBOM;
 | 
						|
    pdc_convert_string(pdc, pdc_utf32, 0, NULL,
 | 
						|
                       (pdc_byte *) utf32string, len,
 | 
						|
                       &outtextformat, NULL, &utf8string, &outlen,
 | 
						|
                       flags, pdc_true);
 | 
						|
 | 
						|
    if (size) *size = outlen;
 | 
						|
 | 
						|
    return (char *) utf8string;
 | 
						|
}
 | 
						|
 | 
						|
char *
 | 
						|
pdc_utf32_to_utf16(pdc_core *pdc, const char *utf32string, int len,
 | 
						|
                   const char *format, int flags, int *size)
 | 
						|
{
 | 
						|
    pdc_text_format textformat = pdc_utf32;
 | 
						|
    pdc_text_format outtextformat = pdc_utf16;
 | 
						|
    pdc_byte *utf16string = NULL;
 | 
						|
 | 
						|
    if (!utf32string)
 | 
						|
        pdc_error(pdc, PDC_E_ILLARG_EMPTY, "utf32string", 0, 0, 0);
 | 
						|
 | 
						|
    if (format && *format)
 | 
						|
    {
 | 
						|
        int k = pdc_get_keycode_ci(format, pdc_textformat_keylist);
 | 
						|
        if (k == PDC_KEY_NOTFOUND ||
 | 
						|
            ((pdc_text_format) k != pdc_utf16 &&
 | 
						|
             (pdc_text_format) k != pdc_utf16be &&
 | 
						|
             (pdc_text_format) k != pdc_utf16le))
 | 
						|
            pdc_error(pdc, PDC_E_ILLARG_STRING, "format", format, 0, 0);
 | 
						|
        outtextformat = (pdc_text_format) k;
 | 
						|
    }
 | 
						|
 | 
						|
    if (outtextformat == pdc_utf16)
 | 
						|
        flags |= PDC_CONV_AUTOBOM;
 | 
						|
    else
 | 
						|
        flags |= PDC_CONV_WITHBOM;
 | 
						|
    pdc_convert_string(pdc, textformat, 0, NULL,
 | 
						|
                      (pdc_byte *) utf32string, len,
 | 
						|
                      &outtextformat, NULL, &utf16string, size,
 | 
						|
                      flags, pdc_true);
 | 
						|
 | 
						|
    return (char *) utf16string;
 | 
						|
}
 | 
						|
 | 
						|
int
 | 
						|
pdc_char16_to_char32(pdc_core *pdc, const pdc_ushort *ustext, int *ic, int len,
 | 
						|
                     pdc_bool verbose)
 | 
						|
{
 | 
						|
    pdc_ushort uvh = ustext[*ic];
 | 
						|
 | 
						|
    if (uvh < PDC_UNICODE_MINHIGHSUR || uvh > PDC_UNICODE_MAXLOWSUR)
 | 
						|
    {
 | 
						|
        return (int) uvh;
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        UTF16 *isa16 = (UTF16 *) &ustext[*ic];
 | 
						|
        pdc_ushort uvl = 0;
 | 
						|
        int icn = *ic + 1;
 | 
						|
 | 
						|
        if (icn < len)
 | 
						|
        {
 | 
						|
            uvl = ustext[icn];
 | 
						|
            if (uvh <= PDC_UNICODE_MAXHIGHSUR)
 | 
						|
            {
 | 
						|
                if (uvl >= PDC_UNICODE_MINLOWSUR &&
 | 
						|
                    uvl <= PDC_UNICODE_MAXLOWSUR)
 | 
						|
                {
 | 
						|
                    int usv;
 | 
						|
                    UTF16 *ise16 = isa16 + 2;
 | 
						|
                    UTF32 *isa32 = (UTF32 *) &usv;
 | 
						|
                    UTF32 *ise32 = isa32 + 1;
 | 
						|
 | 
						|
                    pdc_convers_result result = pdc_convertUTF16toUTF32(
 | 
						|
                                &isa16, ise16, &isa32, ise32, strictConversion);
 | 
						|
                    if (result == conversionOK)
 | 
						|
                    {
 | 
						|
                        *ic = icn;
 | 
						|
                        return usv;
 | 
						|
                    }
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        pdc_set_errmsg(pdc, PDC_E_CONV_ILLUTF16SUR,
 | 
						|
                       pdc_errprintf(pdc, "%04X", uvh),
 | 
						|
                       pdc_errprintf(pdc, "%04X", uvl), 0, 0);
 | 
						|
 | 
						|
        if (verbose)
 | 
						|
            pdc_error(pdc, -1, 0, 0, 0, 0);
 | 
						|
    }
 | 
						|
 | 
						|
    return -1;
 | 
						|
}
 | 
						|
 | 
						|
int
 | 
						|
pdc_char32_to_char16(pdc_core *pdc, int usv, pdc_ushort *uvlist,
 | 
						|
                     pdc_bool verbose)
 | 
						|
{
 | 
						|
    if (usv < PDC_NUM_BMPVAL)
 | 
						|
    {
 | 
						|
        uvlist[0] = (pdc_ushort) usv;
 | 
						|
        return 1;
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        UTF32 *isa32 = (UTF32 *) &usv;
 | 
						|
        UTF32 *ise32 = isa32 + 1;
 | 
						|
        UTF16 *isa16 = (UTF16 *) uvlist;
 | 
						|
        UTF16 *ise16 = isa16 + 2;
 | 
						|
 | 
						|
        pdc_convers_result result = pdc_convertUTF32toUTF16(
 | 
						|
                    &isa32, ise32, &isa16, ise16, strictConversion);
 | 
						|
        if (result == conversionOK)
 | 
						|
        {
 | 
						|
            return 2;
 | 
						|
        }
 | 
						|
 | 
						|
        pdc_set_errmsg(pdc, PDC_E_CONV_ILLUTF32CHAR,
 | 
						|
                       pdc_errprintf(pdc, "%05X", usv), 0, 0, 0);
 | 
						|
 | 
						|
        if (verbose)
 | 
						|
            pdc_error(pdc, -1, 0, 0, 0, 0);
 | 
						|
    }
 | 
						|
 | 
						|
    return 0;
 | 
						|
}
 |