public class

CharToByteEUC_TW

extends CharToByteConverter
/*
 * Copyright (c) 1996, 2003, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Oracle designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */

package sun.io;

import sun.nio.cs.ext.EUC_TW;

/*
 * @author Limin Shi
 */

public class CharToByteEUC_TW extends CharToByteConverter
{
    private final byte MSB = (byte)0x80;
    private final byte SS2 = (byte) 0x8E;
    private final byte P2 = (byte) 0xA2;
    private final byte P3 = (byte) 0xA3;

    private final static EUC_TW nioCoder = new EUC_TW();

    private static String uniTab1 = nioCoder.getUniTab1();
    private static String uniTab2 = nioCoder.getUniTab2();
    private static String uniTab3 = nioCoder.getUniTab3();
    private static String cnsTab1 = nioCoder.getCNSTab1();
    private static String cnsTab2 = nioCoder.getCNSTab2();
    private static String cnsTab3 = nioCoder.getCNSTab3();

    public int flush(byte[] output, int outStart, int outEnd)
        throws MalformedInputException
    {
        reset();
        return 0;
    }

    public void reset() {
        byteOff = charOff = 0;
    }

    public boolean canConvert(char ch){
        if (((0xFF00 & ch) != 0) && (getNative(ch) != -1)){
            return true;
        }
        return false;
    }

    /**
     * Character conversion
     */
    public int convert(char[] input, int inOff, int inEnd,
                       byte[] output, int outOff, int outEnd)
        throws UnknownCharacterException, MalformedInputException,
               ConversionBufferFullException
    {
        int outputSize;
        byte [] tmpbuf = new byte[4];
        byte [] outputByte;

        byteOff = outOff;

        //Fixed 4122961 by bringing the charOff++ out to this
        // loop where it belongs, changing the loop from
        // while(){} to for(){}.
        for (charOff = inOff; charOff < inEnd; charOff++) {
            outputByte = tmpbuf;
            if ( input[charOff] < 0x80) {       // ASCII
                outputSize = 1;
                outputByte[0] = (byte)(input[charOff] & 0x7f);
            } else {
                outputSize = unicodeToEUC(input[charOff], outputByte);
            }

            if (outputSize == -1) {
                if (subMode) {
                    outputByte = subBytes;
                    outputSize = subBytes.length;
                } else {
                    badInputLength = 1;
                    throw new UnknownCharacterException();
                }
            }

            if (outEnd - byteOff < outputSize)
                throw new ConversionBufferFullException();

            for (int i = 0; i < outputSize; i++)
                output[byteOff++] = outputByte[i];
        }

        return byteOff - outOff;

    }


    /**
     * returns the maximum number of bytes needed to convert a char
     */
    public int getMaxBytesPerChar() {
        return 4;
    }


    /**
     * Return the character set ID
     */
    public String getCharacterEncoding() {
        return "EUC_TW";
    }


    protected int getNative(char unicode) {
        int  i,
             cns;       // 2 chars in CNS table make 1 CNS code

        if (unicode < UniTab2[0]) {
            if ((i = searchTab(unicode, UniTab1)) == -1)
                return -1;
            cns = (CNSTab1[2*i] << 16) + CNSTab1[2*i+1];
            return cns;
        } else  if (unicode < UniTab3[0]) {
            if ((i = searchTab(unicode, UniTab2)) == -1)
                return -1;
            cns = (CNSTab2[2*i] << 16) + CNSTab2[2*i+1];
            return cns;
        } else {
            if ((i = searchTab(unicode, UniTab3)) == -1)
                return -1;
            cns = (CNSTab3[2*i] << 16) + CNSTab3[2*i+1];
            return cns;
        }
    }

    protected int searchTab(char code, char [] table) {
        int     i = 0, l, h;

        for (l = 0, h = table.length - 1; l < h; ) {
                if (table[l] == code) {
                        i = l;
                        break;
                }
                if (table[h] == code) {
                        i = h;
                        break;
                }
                i = (l + h) / 2;
                if (table[i] == code)
                        break;
                if (table[i] < code)
                        l = i + 1;
                else    h = i - 1;
        }
        if (code == table[i]) {
            return i;
        } else {
            return -1;
        }
    }


    private int unicodeToEUC(char unicode, byte ebyte[]) {
        int cns = getNative(unicode);

        if ((cns >> 16) == 0x01) {      // Plane 1
            ebyte[0] = (byte) (((cns & 0xff00) >> 8) | MSB);
            ebyte[1] = (byte) ((cns & 0xff) | MSB);
            return 2;
        }

        byte cnsPlane = (byte)(cns >> 16);
        if (cnsPlane >= (byte)0x02) {   // Plane 2
            ebyte[0] = SS2;
            ebyte[1] = (byte) (cnsPlane | (byte)0xA0);
            ebyte[2] = (byte) (((cns & 0xff00) >> 8) | MSB);
            ebyte[3] = (byte) ((cns & 0xff) | MSB);
            return 4;
        }

        return -1;
    }

    protected int unicodeToEUC(char unicode) {
        if (unicode <= 0x7F) { // ASCII falls into EUC_TW CS0
            return unicode;
        }

        int cns = getNative(unicode);
        int plane = cns >> 16;
        int euc = (cns & 0x0000FFFF) | 0x00008080;

        if (plane == 1) {
            return euc;
        } else if (plane == 2) {
            return ((SS2 << 24) & 0xFF000000) | ((P2 << 16) & 0x00FF0000) |
                euc;
        } else if (plane == 3) {
            return ((SS2 << 24) & 0xFF000000) | ((P3 << 16) & 0x00FF0000) |
                euc;
        }

        return -1;
    }

    private char [] UniTab1 = uniTab1.toCharArray();
    private char [] UniTab2 = uniTab2.toCharArray();
    private char [] UniTab3 = uniTab3.toCharArray();
    private char [] CNSTab1 = cnsTab1.toCharArray();
    private char [] CNSTab2 = cnsTab2.toCharArray();
    private char [] CNSTab3 = cnsTab3.toCharArray();
}