All Downloads are FREE. Search and download functionalities are using the official Maven repository.

jtopenlite.com.ibm.jtopenlite.Conv Maven / Gradle / Ivy

///////////////////////////////////////////////////////////////////////////////
//
// JTOpenLite
//
// Filename:  Conv.java
//
// The source code contained herein is licensed under the IBM Public License
// Version 1.0, which has been approved by the Open Source Initiative.
// Copyright (C) 2011-2012 International Business Machines Corporation and
// others.  All rights reserved.
//
///////////////////////////////////////////////////////////////////////////////

package com.ibm.jtopenlite;

import java.io.*;
import java.util.*;
import java.math.*;

import com.ibm.jtopenlite.ccsidConversion.CcsidConversion;

/**
 * Utility class for converting data from one format to another.
**/
public final class Conv
{
	
  private static Hashtable localeNlvMap_;

  private static final char[] NUM = new char[] { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F'};
  private static final byte[] CHAR_HIGH = new byte[10];
  private static final byte[] CHAR_LOW = new byte[10];
  static
  {
    for (int i=0; i<=9; ++i)
    {
      int val = i;
      CHAR_HIGH[i] = (byte)(val << 4);
      CHAR_LOW[i] = (byte)val;
    }
  }

  // The array offset is the Unicode character value, the array value is the EBCDIC 37 value.
  // e.g. CONV_TO_37['0'] == 0xF0  and  CONV_TO_37[' '] == 0x40
  private static final byte[] CONV_TO_37 = new byte[65536];

  private static final byte[] INIT_TO_37 = new byte[]
  {
    0x00, 0x01, 0x02, 0x03, 0x37, 0x2D, 0x2E, 0x2F, 0x16, 0x05, 0x25, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F,
    0x10, 0x11, 0x12, 0x13, 0x3C, 0x3D, 0x32, 0x26, 0x18, 0x19, 0x3F, 0x27, 0x1C, 0x1D, 0x1E, 0x1F,
    0x40, 0x5A, 0x7F, 0x7B, 0x5B, 0x6C, 0x50, 0x7D, 0x4D, 0x5D, 0x5C, 0x4E, 0x6B, 0x60, 0x4B, 0x61,
    (byte)0xF0, (byte)0xF1, (byte)0xF2, (byte)0xF3, (byte)0xF4, (byte)0xF5, (byte)0xF6, (byte)0xF7, (byte)0xF8, (byte)0xF9, 0x7A, 0x5E, 0x4C, 0x7E, 0x6E, 0x6F,
    0x7C, (byte)0xC1, (byte)0xC2, (byte)0xC3, (byte)0xC4, (byte)0xC5, (byte)0xC6, (byte)0xC7, (byte)0xC8, (byte)0xC9, (byte)0xD1, (byte)0xD2, (byte)0xD3, (byte)0xD4, (byte)0xD5, (byte)0xD6,
    (byte)0xD7, (byte)0xD8, (byte)0xD9, (byte)0xE2, (byte)0xE3, (byte)0xE4, (byte)0xE5, (byte)0xE6, (byte)0xE7, (byte)0xE8, (byte)0xE9, (byte)0xBA, (byte)0xE0, (byte)0xBB, (byte)0xB0, 0x6D,
    0x79, (byte)0x81, (byte)0x82, (byte)0x83, (byte)0x84, (byte)0x85, (byte)0x86, (byte)0x87, (byte)0x88, (byte)0x89, (byte)0x91, (byte)0x92, (byte)0x93, (byte)0x94, (byte)0x95, (byte)0x96,
    (byte)0x97, (byte)0x98, (byte)0x99, (byte)0xA2, (byte)0xA3, (byte)0xA4, (byte)0xA5, (byte)0xA6, (byte)0xA7, (byte)0xA8, (byte)0xA9, (byte)0xC0, 0x4F, (byte)0xD0, (byte)0xA1, 0x07,
    0x20, 0x21, 0x22, 0x23, 0x24, 0x15, 0x06, 0x17, 0x28, 0x29, 0x2A, 0x2B, 0x2C, 0x09, 0x0A, 0x1B,
    0x30, 0x31, 0x1A, 0x33, 0x34, 0x35, 0x36, 0x08, 0x38, 0x39, 0x3A, 0x3B, 0x04, 0x14, 0x3E, (byte)0xFF,
    0x41, (byte)0xAA, 0x4A, (byte)0xB1, (byte)0x9F, (byte)0xB2, 0x6A, (byte)0xB5, (byte)0xBD, (byte)0xB4, (byte)0x9A, (byte)0x8A, 0x5F, (byte)0xCA, (byte)0xAF, (byte)0xBC,
    (byte)0x90, (byte)0x8F, (byte)0xEA, (byte)0xFA, (byte)0xBE, (byte)0xA0, (byte)0xB6, (byte)0xB3, (byte)0x9D, (byte)0xDA, (byte)0x9B, (byte)0x8B, (byte)0xB7, (byte)0xB8, (byte)0xB9, (byte)0xAB,
    0x64, 0x65, 0x62, 0x66, 0x63, 0x67, (byte)0x9E, 0x68, 0x74, 0x71, 0x72, 0x73, 0x78, 0x75, 0x76, 0x77,
    (byte)0xAC, 0x69, (byte)0xED, (byte)0xEE, (byte)0xEB, (byte)0xEF, (byte)0xEC, (byte)0xBF, (byte)0x80, (byte)0xFD, (byte)0xFE, (byte)0xFB, (byte)0xFC, (byte)0xAD, (byte)0xAE, 0x59,
    0x44, 0x45, 0x42, 0x46, 0x43, 0x47, (byte)0x9C, 0x48, 0x54, 0x51, 0x52, 0x53, 0x58, 0x55, 0x56, 0x57,
    (byte)0x8C, 0x49, (byte)0xCD, (byte)0xCE, (byte)0xCB, (byte)0xCF, (byte)0xCC, (byte)0xE1, 0x70, (byte)0xDD, (byte)0xDE, (byte)0xDB, (byte)0xDC, (byte)0x8D, (byte)0x8E, (byte)0xDF
  };

  // The array offset is the EBCDIC 37 character value, the array value is the Unicode value.
  // e.g. CONV_FROM_37[0xF0] == '0'  and  CONV_FROM_37[0x40] == ' '
  private static final char[] CONV_FROM_37 = new char[]
  {
    0x0000, 0x0001, 0x0002, 0x0003, 0x009C, 0x0009, 0x0086, 0x007F, 0x0097, 0x008D, 0x008E, 0x000B, 0x000C, 0x000D, 0x000E, 0x000F,
    0x0010, 0x0011, 0x0012, 0x0013, 0x009D, 0x0085, 0x0008, 0x0087, 0x0018, 0x0019, 0x0092, 0x008F, 0x001C, 0x001D, 0x001E, 0x001F,
    0x0080, 0x0081, 0x0082, 0x0083, 0x0084, 0x000A, 0x0017, 0x001B, 0x0088, 0x0089, 0x008A, 0x008B, 0x008C, 0x0005, 0x0006, 0x0007,
    0x0090, 0x0091, 0x0016, 0x0093, 0x0094, 0x0095, 0x0096, 0x0004, 0x0098, 0x0099, 0x009A, 0x009B, 0x0014, 0x0015, 0x009E, 0x001A,
    0x0020, 0x00A0, 0x00E2, 0x00E4, 0x00E0, 0x00E1, 0x00E3, 0x00E5, 0x00E7, 0x00F1, 0x00A2, 0x002E, 0x003C, 0x0028, 0x002B, 0x007C,
    0x0026, 0x00E9, 0x00EA, 0x00EB, 0x00E8, 0x00ED, 0x00EE, 0x00EF, 0x00EC, 0x00DF, 0x0021, 0x0024, 0x002A, 0x0029, 0x003B, 0x00AC,
    0x002D, 0x002F, 0x00C2, 0x00C4, 0x00C0, 0x00C1, 0x00C3, 0x00C5, 0x00C7, 0x00D1, 0x00A6, 0x002C, 0x0025, 0x005F, 0x003E, 0x003F,
    0x00F8, 0x00C9, 0x00CA, 0x00CB, 0x00C8, 0x00CD, 0x00CE, 0x00CF, 0x00CC, 0x0060, 0x003A, 0x0023, 0x0040, 0x0027, 0x003D, 0x0022,
    0x00D8, 0x0061, 0x0062, 0x0063, 0x0064, 0x0065, 0x0066, 0x0067, 0x0068, 0x0069, 0x00AB, 0x00BB, 0x00F0, 0x00FD, 0x00FE, 0x00B1,
    0x00B0, 0x006A, 0x006B, 0x006C, 0x006D, 0x006E, 0x006F, 0x0070, 0x0071, 0x0072, 0x00AA, 0x00BA, 0x00E6, 0x00B8, 0x00C6, 0x00A4,
    0x00B5, 0x007E, 0x0073, 0x0074, 0x0075, 0x0076, 0x0077, 0x0078, 0x0079, 0x007A, 0x00A1, 0x00BF, 0x00D0, 0x00DD, 0x00DE, 0x00AE,
    0x005E, 0x00A3, 0x00A5, 0x00B7, 0x00A9, 0x00A7, 0x00B6, 0x00BC, 0x00BD, 0x00BE, 0x005B, 0x005D, 0x00AF, 0x00A8, 0x00B4, 0x00D7,
    0x007B, 0x0041, 0x0042, 0x0043, 0x0044, 0x0045, 0x0046, 0x0047, 0x0048, 0x0049, 0x00AD, 0x00F4, 0x00F6, 0x00F2, 0x00F3, 0x00F5,
    0x007D, 0x004A, 0x004B, 0x004C, 0x004D, 0x004E, 0x004F, 0x0050, 0x0051, 0x0052, 0x00B9, 0x00FB, 0x00FC, 0x00F9, 0x00FA, 0x00FF,
    0x005C, 0x00F7, 0x0053, 0x0054, 0x0055, 0x0056, 0x0057, 0x0058, 0x0059, 0x005A, 0x00B2, 0x00D4, 0x00D6, 0x00D2, 0x00D3, 0x00D5,
    0x0030, 0x0031, 0x0032, 0x0033, 0x0034, 0x0035, 0x0036, 0x0037, 0x0038, 0x0039, 0x00B3, 0x00DB, 0x00DC, 0x00D9, 0x00DA, 0x009F
  };

  private static final String[] CACHE_FROM_37 = new String[256];
  private static final boolean cacheFrom37Init_;

  static
  {
    System.arraycopy(INIT_TO_37, 0, CONV_TO_37, 0, INIT_TO_37.length);
    for (int i=INIT_TO_37.length; i=offset; --i)
    {
      int low = data[i] & 0x000F;
      int high = (data[i] >> 4) & 0x000F;
      buffer[--count] = NUM[low];
      buffer[--count] = NUM[high];
    }
    return new String(buffer, 0, numChars);
  }

  /**
   * Converts the specified hexadecimal String into its constituent byte values.
  **/
  public static final byte[] hexStringToBytes(final String value)
  {
    int len = value.length();
    /* this comparison works with negative numbers */
    if (len % 2 != 0) ++len;
    final byte[] data = new byte[len>>1];
    hexStringToBytes(value, data, 0);
    return data;
  }

  /**
   * Converts the specified hexadecimal String into its constituent byte values.
  **/
  public static final int hexStringToBytes(final String value, final byte[] data, final int offset)
  {
    final int len = value.length();
    final int odd = len % 2;
    if (odd == 1)
    {
      data[offset] = 0;
    }
    for (int i=0; i= '0' && c <= '9')
      {
        val = c - '0';
      }
      else if (c >= 'A' && c <= 'F')
      {
        val = c - 'A' + 10;
      }
      else if (c >= 'a' && c <= 'f')
      {
        val = c - 'a' + 10;
      }
      final int arrOff = offset + ((i+odd)>>1);
      data[arrOff] = i % 2 == odd ? (byte)(val << 4) : (byte)(data[arrOff] | val);
    }
    final int num = len >> 1;
    return odd == 0 ? num : num+1;
  }


  /**
   * Converts the specified String into CCSID 37 bytes.
  **/
  public static final byte[] stringToEBCDICByteArray37(final String s)
  {
    final byte[] b = new byte[s.length()];
    stringToEBCDICByteArray37(s, b, 0);
    return b;
  }

  /**
   * Converts the specified String into CCSID 37 bytes.
  **/
  public static final int stringToEBCDICByteArray37(final String s, final byte[] data, final int offset)
  {
      return stringToEBCDICByteArray37(s, s.length(), data, offset);
  }


  public static final int stringToEBCDICByteArray37(final String s, int length,  final byte[] data, final int offset)
  {
	int sLength = s.length();
	if (sLength < length) {
		length = sLength;
	}
    final int stop = offset+length;
    for (int i=offset; i sLength) length = sLength;

    final int ccsidToUse = ccsid & 0x00FFFF; // So we don't overflow our encodings_ table.
    if (ccsidToUse == 37) return stringToEBCDICByteArray37(s, length, data, offset);
    String encoding = encodings_[ccsidToUse];
    if (encoding != null)
    {
    	try {
    	      // BOOOO!

      byte[] b = s.substring(0,length).getBytes(encoding);
      System.arraycopy(b, 0, data, offset, b.length);
      return b.length;
    	} catch (UnsupportedEncodingException ex) {
    		encodings_[ccsidToUse] = null;
    	}
    }
    return CcsidConversion.stringToEBCDICByteArray(s, length,  data, offset, ccsidToUse);

  }


  /**
   * Converts the specified String into Unicode bytes.
   * returns the number of bytes.
  **/
  public static final byte[] stringToUnicodeByteArray(final String s)
  {
    final byte[] b = new byte[s.length()*2];
    stringToUnicodeByteArray(s, b, 0);
    return b;
  }

  /**
   * Converts the specified String into Unicode bytes.
  **/
  public static final int stringToUnicodeByteArray(final String s, final byte[] data, final int offset)
  {
      return stringToUnicodeByteArray(s, s.length(), data, offset);
  }


  public static final int stringToUnicodeByteArray(final String s, int length, final byte[] data, final int offset)
  {
    for (int i=0; i> 8);
      byte low = (byte)c;
      data[offset+(i*2)] = high;
      data[offset+(i*2)+1] = low;
    }
    return length*2;
  }

  /* Version that pads with spaces */
  public static void stringToUnicodeByteArray(String s, byte[] data, int offset, int byteLength) {
    int sLength = s.length();

    for (int i=0; i> 8);
      byte low = (byte)c;
      data[offset+(i*2)] = high;
      data[offset+(i*2)+1] = low;
    }

  }

  public static final int stringToUtf8ByteArray(final String s, int length, final byte[] data, final int offset)
  {
	  int sLength = s.length();
	  if (length > sLength) length = sLength;

    	      // BOOOO!

      byte[] b ;
      try {
      b = s.substring(0,length).getBytes("UTF-8");
      System.arraycopy(b, 0, data, offset, b.length);
      return b.length;
      } catch (UnsupportedEncodingException uee) {
    	  // should never happen
    	  return 0;
      }


  }



  /**
   * Converts the specified String into Unicode bytes, padding the byte array with Unicode
   * spaces (0x0020) up to length bytes.
  **/
  public static final void stringToBlankPadUnicodeByteArray(final String s,
		  final byte[] data, final int offset, final int length)
  {
    int counter = 0;
    if (s != null)
    {
      for (int i=0; i> 8);
        byte low = (byte)s.charAt(i);
        data[offset+counter] = high;
        ++counter;
        data[offset+counter] = low;
        ++counter;
      }
    }
    while ((counter+2) <= length)
    {
      data[offset+counter] = 0x00;
      ++counter;
      data[offset+counter] = 0x20;
      ++counter;
    }
  }

  /**
   * Converts the specified Unicode bytes into a String.
   * The length is in bytes, and should be twice the length of the returned String.
  **/
  public static final String unicodeByteArrayToString(final byte[] data, final int offset, final int length)
  {
    char[] buf = new char[length];
    return unicodeByteArrayToString(data, offset, length, buf);
  }

  /**
   * Converts the specified Unicode bytes into a String.
   * The length is in bytes, and should be twice the length of the returned String.
  **/
  public static final String unicodeByteArrayToString(final byte[] data, final int offset, final int length, final char[] buffer)
  {
    final int numChars = length/2;
    int count = numChars;
    for (int i=offset+length-1; i>=offset; i-=2)
    {
      int low = data[i] & 0x00FF;
      int high = data[i-1] & 0x00FF;
      char c = (char)((high << 8) | low);
      buffer[--count] = c;
    }
    return new String(buffer, 0, numChars);
  }

  /**
   * Converts the specified String into CCSID 37 bytes, padding the byte array with EBCDIC spaces (0x40) up to length bytes.
  **/
  public static final void stringToBlankPadEBCDICByteArray(final String s, final byte[] data, final int offset, final int length)
  {
    for (int i=0; ilength bytes.
   * @exception UnsupportedEncodingException Thrown if conversion to or from the specified CCSID is not supported.
  **/
  public static final void stringToBlankPadEBCDICByteArray(final String s, final byte[] data, final int offset, final int length, final int ccsid) throws UnsupportedEncodingException
  {
    final int ccsidToUse = ccsid & 0x00FFFF; // So we don't overflow our encodings_ table.
    if (ccsidToUse == 37)
    {
      stringToBlankPadEBCDICByteArray(s, data, offset, length);
    }
    else
    {
      String encoding = encodings_[ccsidToUse];
      if (encoding != null)
      {
        // BOOOO!
        byte[] b = s.getBytes(encoding);
        int len = b.length;
        int total = len < length ? len : length;
        System.arraycopy(b, 0, data, offset, total);
        int rem = length-len;
        if (rem > 0)
        {
          final byte[] blank = " ".getBytes(encoding);
          while (rem > 0)
          {
            System.arraycopy(blank, 0, data, offset+total, blank.length);
            total += blank.length;
            rem = rem - blank.length;
          }
        }
      }
      else
      {
        throw new UnsupportedEncodingException("CCSID "+ccsidToUse);
      }
    }
  }

  /**
   * Converts the specified CCSID 37 byte into a Unicode char without creating any intermediate objects.
  **/
  public static final char ebcdicByteToChar(final byte b)
  {
    return CONV_FROM_37[b & 0x00FF];
  }

  /**
   * Converts the specified CCSID 37 bytes into a String.
   * Note: You might as well just use new String(data,"Cp037") to avoid the extra char array this method needs to create.
   * Note: You cannot use new String(data,"Cp037" because this is not supported on all JVMS
  **/
  public static final String ebcdicByteArrayToString(final byte[] data, final int offset, final int length)
  {
    if (length == 1 && cacheFrom37Init_) return CACHE_FROM_37[data[offset] & 0x00FF];
    return ebcdicByteArrayToString(data, offset, length, new char[length]);
  }

  public static final String ebcdicByteArrayToString(final byte[] data,  final char[] buffer) {
    int offset = 0;
    int length = data.length;
    return ebcdicByteArrayToString(data, offset, length, buffer);
  }

  // @csmith: Perf testing:
  // This uses the same amount of memory as new String(data, "Cp037")
  // but is about 2x faster on IBM 1.5 Windows 32-bit and about 3x faster on Sun 1.4 Windows 32-bit.
  /**
   * Converts the specified CCSID 37 bytes into a String.
  **/
  public static final String ebcdicByteArrayToString(final byte[] data, final int offset, final int length, final char[] buffer)
  {
    if (length == 1 && cacheFrom37Init_) return CACHE_FROM_37[data[offset] & 0x00FF];
    int counter = length;
    for (int i=offset+length-1; i>=offset; --i)
    {
      buffer[--counter] = CONV_FROM_37[data[i] & 0x00FF];
    }
    return new String(buffer, 0, length);
  }

  // Conversion maps copied from Toolbox/JTOpen.
  private static final HashMap encodingCcsid_ = new HashMap();
  private static final HashMap ccsidEncoding_ = new HashMap();

  private static final String[] encodings_ = new String[65536];

  static
  {
    // 137+ possible Java encodings. 13 have unknown CCSIDs.
    // We have 128 known in this table.
    encodingCcsid_.put("ASCII",         "367");  // ANSI X.34 ASCI.
    encodingCcsid_.put("Cp1252",        "1252");
    encodingCcsid_.put("ISO8859_1",     "819");
    encodingCcsid_.put("Unicode",       "13488");
    encodingCcsid_.put("UnicodeBig",    "13488");  // BOM is 0xFEFF.
    // encodingCcsid_.put("UnicodeBigUnmarked", 13488);
    encodingCcsid_.put("UnicodeLittle", "1202"); // BOM is 0xFFFE.
    // encodingCcsid_.put("UnicodeLittleUnmarked", 13488);
    encodingCcsid_.put("UTF8",          "1208");
    encodingCcsid_.put("UTF-8",         "1208");
    encodingCcsid_.put("UTF-16BE",      "1200");

    encodingCcsid_.put("Big5",      "950");
    // encodingCcsid_.put("Big5 HKSCS", ???); // Big5 with Hong Kong extensions.
    encodingCcsid_.put("CNS11643",  "964");
    encodingCcsid_.put("Cp037",     "37");
    encodingCcsid_.put("Cp256",     "256");
    encodingCcsid_.put("Cp273",     "273");
    encodingCcsid_.put("Cp277",     "277");
    encodingCcsid_.put("Cp278",     "278");
    encodingCcsid_.put("Cp280",     "280");
    encodingCcsid_.put("Cp284",     "284");
    encodingCcsid_.put("Cp285",     "285");
    encodingCcsid_.put("Cp290",     "290");
    encodingCcsid_.put("Cp297",     "297");
    encodingCcsid_.put("Cp420",     "420");
    encodingCcsid_.put("Cp423",     "423");
    encodingCcsid_.put("Cp424",     "424");
    encodingCcsid_.put("Cp437",     "437");
    encodingCcsid_.put("Cp500",     "500");
    encodingCcsid_.put("Cp737",     "737");
    encodingCcsid_.put("Cp775",     "775");
    encodingCcsid_.put("Cp833",     "833");
    encodingCcsid_.put("Cp838",     "838");
    encodingCcsid_.put("Cp850",     "850");
    encodingCcsid_.put("Cp852",     "852");
    encodingCcsid_.put("Cp855",     "855");
    encodingCcsid_.put("Cp856",     "856");
    encodingCcsid_.put("Cp857",     "857");
    encodingCcsid_.put("Cp858",     "858");
    encodingCcsid_.put("Cp860",     "860");
    encodingCcsid_.put("Cp861",     "861");
    encodingCcsid_.put("Cp862",     "862");
    encodingCcsid_.put("Cp863",     "863");
    encodingCcsid_.put("Cp864",     "864");
    encodingCcsid_.put("Cp865",     "865");
    encodingCcsid_.put("Cp866",     "866");
    encodingCcsid_.put("Cp868",     "868");
    encodingCcsid_.put("Cp869",     "869");
    encodingCcsid_.put("Cp870",     "870");
    encodingCcsid_.put("Cp871",     "871");
    encodingCcsid_.put("Cp874",     "874");
    encodingCcsid_.put("Cp875",     "875");
    encodingCcsid_.put("Cp880",     "880");
    encodingCcsid_.put("Cp905",     "905");
    encodingCcsid_.put("Cp918",     "918");
    encodingCcsid_.put("Cp921",     "921");
    encodingCcsid_.put("Cp922",     "922");
    encodingCcsid_.put("Cp923",     "923");  // IBM Latin-9.
    encodingCcsid_.put("Cp924",     "924");
    encodingCcsid_.put("Cp930",     "930");
    encodingCcsid_.put("Cp933",     "933");
    encodingCcsid_.put("Cp935",     "935");
    encodingCcsid_.put("Cp937",     "937");
    encodingCcsid_.put("Cp939",     "939");
    encodingCcsid_.put("Cp942",     "942");
    // encodingCcsid_.put("Cp942C",    ???);  // Don't know the CCSID - unclear what the 'C' means.
    encodingCcsid_.put("Cp943",     "943");
    // encodingCcsid_.put("Cp943C",    ???); // Don't know the CCSID - unclear what the 'C' means.
    encodingCcsid_.put("Cp948",     "948");
    encodingCcsid_.put("Cp949",     "949");
    // encodingCcsid_.put("Cp949C",    ???); // Don't know the CCSID - unclear what the 'C' means.
    encodingCcsid_.put("Cp950",     "950");
    encodingCcsid_.put("Cp964",     "964");
    encodingCcsid_.put("Cp970",     "970");
    encodingCcsid_.put("Cp1006",   "1006");
    encodingCcsid_.put("Cp1025",   "1025");
    encodingCcsid_.put("Cp1026",   "1026");
    encodingCcsid_.put("Cp1027",   "1027");
    encodingCcsid_.put("Cp1046",   "1046");
    encodingCcsid_.put("Cp1097",   "1097");
    encodingCcsid_.put("Cp1098",   "1098");
    encodingCcsid_.put("Cp1112",   "1112");
    encodingCcsid_.put("Cp1122",   "1122");
    encodingCcsid_.put("Cp1123",   "1123");
    encodingCcsid_.put("Cp1124",   "1124");
    encodingCcsid_.put("Cp1130",   "1130");
    encodingCcsid_.put("Cp1132",   "1132");
    encodingCcsid_.put("Cp1137",   "1137");
    encodingCcsid_.put("Cp1140",   "1140");
    encodingCcsid_.put("Cp1141",   "1141");
    encodingCcsid_.put("Cp1142",   "1142");
    encodingCcsid_.put("Cp1143",   "1143");
    encodingCcsid_.put("Cp1144",   "1144");
    encodingCcsid_.put("Cp1145",   "1145");
    encodingCcsid_.put("Cp1146",   "1146");
    encodingCcsid_.put("Cp1147",   "1147");
    encodingCcsid_.put("Cp1148",   "1148");
    encodingCcsid_.put("Cp1149",   "1149");
    encodingCcsid_.put("Cp1153",   "1153");
    encodingCcsid_.put("Cp1154",   "1154");
    encodingCcsid_.put("Cp1155",   "1155");
    encodingCcsid_.put("Cp1156",   "1156");
    encodingCcsid_.put("Cp1157",   "1157");
    encodingCcsid_.put("Cp1158",   "1158");
    encodingCcsid_.put("Cp1160",   "1160");
    encodingCcsid_.put("Cp1164",   "1164");
    encodingCcsid_.put("Cp1250",   "1250");
    encodingCcsid_.put("Cp1251",   "1251");
    encodingCcsid_.put("Cp1253",   "1253");
    encodingCcsid_.put("Cp1254",   "1254");
    encodingCcsid_.put("Cp1255",   "1255");
    encodingCcsid_.put("Cp1256",   "1256");
    encodingCcsid_.put("Cp1257",   "1257");
    encodingCcsid_.put("Cp1258",   "1258");
    encodingCcsid_.put("Cp1364",   "1364");
    encodingCcsid_.put("Cp1381",   "1381");
    encodingCcsid_.put("Cp1383",   "1383");
    encodingCcsid_.put("Cp1388",   "1388");
    encodingCcsid_.put("Cp1399",   "1399");
    encodingCcsid_.put("Cp4971",   "4971");
    encodingCcsid_.put("Cp5123",   "5123");
    encodingCcsid_.put("Cp9030",   "9030");
    encodingCcsid_.put("Cp13121", "13121");
    encodingCcsid_.put("Cp13124", "13124");
    encodingCcsid_.put("Cp28709", "28709");
    encodingCcsid_.put("Cp33722", "33722");

    // The Toolbox does not directly support EUC at this time, Java will do the conversion.
    encodingCcsid_.put("EUC_CN", "1383");  // Superset of 5479.
    encodingCcsid_.put("EUC_JP", "33722");
    encodingCcsid_.put("EUC_KR", "970");  // Superset of 5066.
    encodingCcsid_.put("EUC_TW", "964");  // Superset of 5060.

    encodingCcsid_.put("GB2312", "1381");
    encodingCcsid_.put("GB18030", "1392"); //1392 is mixed 4-byte; the individual component CCSIDs are not supported.
    encodingCcsid_.put("GBK",    "1386");

    // encodingCcsid_.put("ISCII91", ???); // Indic scripts.

    // The Toolbox does not directly support ISO2022.
    // encodingCcsid_.put("ISO2022CN",     ???);  // Not sure of the CCSID, possibly 9575?
    // encodingCcsid_.put("ISO2022CN_CNS", "965");  // Java doesn't support this one?
    // encodingCcsid_.put("ISO2022CN_GB",  "9575");  // Java doesn't support this one?

    encodingCcsid_.put("ISO2022JP", "5054"); // Could be 956 also, but the IBM i JVM uses 5054.
    encodingCcsid_.put("ISO2022KR", "25546"); // Could be 17354 also, but the IBM i JVM uses 25546.

    encodingCcsid_.put("ISO8859_2", "912");
    encodingCcsid_.put("ISO8859_3", "913");
    encodingCcsid_.put("ISO8859_4", "914");
    encodingCcsid_.put("ISO8859_5", "915");
    encodingCcsid_.put("ISO8859_6", "1089");
    encodingCcsid_.put("ISO8859_7", "813");
    encodingCcsid_.put("ISO8859_8", "916");
    encodingCcsid_.put("ISO8859_9", "920");
    // encodingCcsid_.put("ISO8859_13", ???);  // Latin alphabet No. 7.
    // encodingCcsid_.put("ISO8859_15_FDIS", ???); // Don't know the CCSID; FYI, this codepage is ISO 28605.

    // The Toolbox does not directly support JIS.
    encodingCcsid_.put("JIS0201",       "897"); // Could be 895, but the IBM i JVM uses 897.
    encodingCcsid_.put("JIS0208",       "952");
    encodingCcsid_.put("JIS0212",       "953");
    // encodingCcsid_.put("JISAutoDetect", ???); // Can't do this one. Would need to look at the bytes to determine the CCSID.

    encodingCcsid_.put("Johab",  "1363");
    encodingCcsid_.put("KOI8_R", "878");
    encodingCcsid_.put("KSC5601", "949");

    encodingCcsid_.put("MS874", "874");
    encodingCcsid_.put("MS932", "943");
    encodingCcsid_.put("MS936", "1386");
    encodingCcsid_.put("MS949", "949");
    encodingCcsid_.put("MS950", "950");

    // encodingCcsid_.put("MacArabic", ???); // Don't know.
    encodingCcsid_.put("MacCentralEurope", "1282");
    encodingCcsid_.put("MacCroatian", "1284");
    encodingCcsid_.put("MacCyrillic", "1283");
    // encodingCcsid_.put("MacDingbat", ???); // Don't know.
    encodingCcsid_.put("MacGreek", "1280");
    // encodingCcsid_.put("MacHebrew", ???); // Don't know.
    encodingCcsid_.put("MacIceland", "1286");
    encodingCcsid_.put("MacRoman", "1275");
    encodingCcsid_.put("MacRomania", "1285");
    // encodingCcsid_.put("MacSymbol", ???); // Don't know.
    // encodingCcsid_.put("MacThai", ???); // Don't know.
    encodingCcsid_.put("MacTurkish", "1281");
    // encodingCcsid_.put("MacUkraine", ???); // Don't know.

    encodingCcsid_.put("SJIS", "932"); // Could be 943, but the IBM i JVM uses 932.
    encodingCcsid_.put("TIS620", "874"); // IBM i JVM uses 874.
  }

  static
  {
    // Build the CCSID to encoding map.
    Iterator it = encodingCcsid_.keySet().iterator();
    while (it.hasNext())
    {
      Object key = it.next();
      ccsidEncoding_.put(encodingCcsid_.get(key), key);
    }

    ccsidEncoding_.put("13488", "UTF-16BE");
    ccsidEncoding_.put("61952", "UTF-16BE");
    ccsidEncoding_.put("17584", "UTF-16BE"); // IBM i doesn't support this, but other people use it.

    it = ccsidEncoding_.keySet().iterator();
    while (it.hasNext())
    {
      String ccsid = (String)it.next();
      String encoding = (String)ccsidEncoding_.get(ccsid);
      int i = new Integer(ccsid).intValue();
      encodings_[i] = encoding;
    }
  }

  /**
   * Converts the specific CCSID bytes into a String.
   * @exception UnsupportedEncodingException Thrown if conversion to or from the specified CCSID is not supported.
  **/
  public static final String ebcdicByteArrayToString(final byte[] data, final int offset, final int length, final int ccsid) throws UnsupportedEncodingException
  {
    final int ccsidToUse = ccsid & 0x00FFFF; // So we don't overflow our encodings_ table.
    if (ccsidToUse == 37) return ebcdicByteArrayToString(data, offset, length);
    String encoding = encodings_[ccsidToUse];
    if (encoding != null)
    {
      return new String(data, offset, length, encoding);
    }
    throw new UnsupportedEncodingException("CCSID "+ccsidToUse);
  }

  /**
   * Converts the specific CCSID bytes into a String.
   * @exception UnsupportedEncodingException Thrown if conversion to or from the specified CCSID is not supported.
  **/
  public static final String ebcdicByteArrayToString(final byte[] data, final int offset, final int length, final char[] buffer, final int ccsid) throws UnsupportedEncodingException
  {
    final int ccsidToUse = ccsid & 0x00FFFF; // So we don't overflow our encodings_ table.
    if (ccsidToUse == 37) return ebcdicByteArrayToString(data, offset, length, buffer);
    String encoding = encodings_[ccsidToUse];
    if (encoding != null)
    {
      try {
         return new String(data, offset, length, encoding);
      } catch (UnsupportedEncodingException ex) {
    	  // Mark as unsupported
    	  encodings_[ccsidToUse] = null;
    	  // Fall through and convert
    	  encoding = null;

      }
    }
   return CcsidConversion.createString(data, offset, length, ccsidToUse);
  }

  /**
   * Returns true if the conversion to or from the specific CCSID is supported by the methods on this class.
  **/
  public static boolean isSupported(final int ccsid)
  {
    if (ccsid < 0 || ccsid > 65535) return false;
    return ccsid == 37 || encodings_[ccsid] != null;
  }

  /**
   * Converts the specified bytes into a long value.
  **/
  public static final long byteArrayToLong(final byte[] data, final int offset)
  {
    int p0 = (0x00FF & data[offset]) << 24;
    int p1 = (0x00FF & data[offset+1]) << 16;
    int p2 = (0x00FF & data[offset+2]) << 8;
    int p3 = 0x00FF & data[offset+3];
    int p4 = (0x00FF & data[offset+4]) << 24;
    int p5 = (0x00FF & data[offset+5]) << 16;
    int p6 = (0x00FF & data[offset+6]) << 8;
    int p7 = (0x00FF & data[offset+7]);
    long l1 = (long)(p0 | p1 | p2 | p3);
    long l2 = (long)(p4 | p5 | p6 | p7);
    return(l1 << 32) | (l2 & 0x00FFFFFFFFL);
  }

  /**
   * Converts the specified bytes into an int value.
  **/
  public static final int byteArrayToInt(final byte[] data, final int offset)
  {
    int p0 = (0x00FF & data[offset]) << 24;
    int p1 = (0x00FF & data[offset+1]) << 16;
    int p2 = (0x00FF & data[offset+2]) << 8;
    int p3 = 0x00FF & data[offset+3];
    return p0 | p1 | p2 | p3;
  }

  /**
   * Converts the specified bytes into a short value.
  **/
  public static final short byteArrayToShort(final byte[] data, final int offset)
  {
    short p0 = (short) (( 0x00FF & data[offset]) << 8);
    short p1 = (short) (0x00FF & data[offset+1]);
    return (short) (p0 | p1);
  }

  /**
   * Converts the specified short value into 2 bytes.
  **/
  public static final void shortToByteArray(final int value, final byte[] data, final int offset)
  {
    data[offset] = (byte)(value >> 8);
    data[offset+1] = (byte)value;
  }

  /**
   * Converts the specified int value into 4 bytes.
  **/
  public static final byte[] intToByteArray(final int value)
  {
    final byte[] val = new byte[4];
    intToByteArray(value, val, 0);
    return val;
  }

  /**
   * Converts the specified int value into 4 bytes.
  **/
  public static final void intToByteArray(final int value, final byte[] data, final int offset)
  {
    data[offset] = (byte)(value >> 24);
    data[offset+1] = (byte)(value >> 16);
    data[offset+2] = (byte)(value >>  8);
    data[offset+3] = (byte)value;
  }

  /**
   * Converts the specified long value into 8 bytes.
  **/
  public static final byte[] longToByteArray(final long longValue)
  {
    final byte[] val = new byte[8];
    longToByteArray(longValue, val, 0);
    return val;
  }

  // @csmith: Perf testing:
  // Confirmed breaking the long into two int's is much faster.
  // Using sign extension >> rather than unsigned shift >>> is faster on IBM JRE but not on Sun (Windows 32-bit).
  /**
   * Converts the specified long value into 8 bytes.
  **/
  public static final void longToByteArray(final long longValue, final byte[] data, final int offset)
  {
    // Do in two parts to avoid long temps.
    final int high = (int)(longValue >> 32);
    final int low = (int)longValue;

    data[offset] = (byte)(high >> 24);
    data[offset+1] = (byte)(high >> 16);
    data[offset+2] = (byte)(high >>  8);
    data[offset+3] = (byte)high;

    data[offset+4] = (byte)(low >> 24);
    data[offset+5] = (byte)(low >> 16);
    data[offset+6] = (byte)(low >>  8);
    data[offset+7] = (byte)low;
  }

  static final void writeStringToUnicodeBytes(final String s, final HostServerConnection.HostOutputStream out) throws IOException
  {
    for (int i=0; i> 58;

    //compute sign here so we can get -+Infinity values
    int sign = ((decFloat16Bits & DEC_FLOAT_16_SIGN_MASK) == DEC_FLOAT_16_SIGN_MASK) ? -1 : 1;

    // deal with special numbers. (not a number and infinity)
    if ((combination == 0x1fL) && (sign == 1))
    {
      long nanSignal = (decFloat16Bits & DEC_FLOAT_16_SIGNAL_MASK) >> 57; //shift first 7 bits to get signal bit out  //@snan
      return nanSignal == 1 ? "SNaN" : "NaN";
    }
    else if ((combination == 0x1fL) && (sign == -1))
    {
      long nanSignal = (decFloat16Bits & DEC_FLOAT_16_SIGNAL_MASK) >> 57; //shift first 7 bits to get signal bit out  //@snan
      return nanSignal == 1 ? "-SNaN" : "-NaN";
    }
    else if ((combination == 0x1eL) && (sign == 1))
    {
      return "Infinity";
    }
    else if ((combination == 0x1eL) && (sign == -1))
    {
      return "-Infinity";
    }

    // compute the exponent MSD and the coefficient MSD.
    int exponentMSD;
    long coefficientMSD;
    if ((combination & 0x18L) == 0x18L)
    {
      // format of 11xxx:
      exponentMSD = (int) ((combination & 0x06L) >> 1);
      coefficientMSD = 8 + (combination & 0x01L);
    }
    else
    {
      // format of xxxxx:
      exponentMSD = (int) ((combination & 0x18L) >> 3);
      coefficientMSD = (combination & 0x07L);
    }

    // compute the exponent.
    int exponent = (int) ((decFloat16Bits & DEC_FLOAT_16_EXPONENT_CONTINUATION_MASK) >> 50);
    exponent |= (exponentMSD << 8);
    exponent -= DEC_FLOAT_16_BIAS;

    // compute the coefficient.
    long coefficientContinuation = decFloat16Bits & DEC_FLOAT_16_COEFFICIENT_CONTINUATION_MASK;
    int coefficientLo = decFloatBitsToDigits((int) (coefficientContinuation & 0x3fffffff)); // low 30 bits (9 digits)
    int coefficientHi = decFloatBitsToDigits((int) ((coefficientContinuation >> 30) & 0xfffff)); // high 20 bits (6 digits)
    coefficientHi += coefficientMSD * 1000000L;

    // compute the int array of coefficient.
    int[] value = computeMagnitude(new int[] { coefficientHi, coefficientLo});

    // convert value to a byte array of coefficient.
    byte[] magnitude = new byte[8];
    magnitude[0] = (byte) (value[0] >>> 24);
    magnitude[1] = (byte) (value[0] >>> 16);
    magnitude[2] = (byte) (value[0] >>> 8);
    magnitude[3] = (byte) (value[0]);
    magnitude[4] = (byte) (value[1] >>> 24);
    magnitude[5] = (byte) (value[1] >>> 16);
    magnitude[6] = (byte) (value[1] >>> 8);
    magnitude[7] = (byte) (value[1]);

    BigInteger bigInt = new BigInteger(sign, magnitude);
    BigDecimal bigDec = new BigDecimal(bigInt, -exponent);
    return bigDec.toString();
  }

  // Copied from JTOpen. TODO - Needs optimization.
  /**
   * Converts the specified 16 bytes in decfloat34 format into a String.
  **/
  public static final String decfloat34ByteArrayToString(final byte[] data, final int offset)
  {
    long decFloat34BitsHi = byteArrayToLong(data, offset);
    long decFloat34BitsLo = byteArrayToLong(data, offset + 8);
    long combination = (decFloat34BitsHi & DEC_FLOAT_34_COMBINATION_MASK) >> 58;

    //compute sign.
    int sign = ((decFloat34BitsHi & DEC_FLOAT_34_SIGN_MASK) == DEC_FLOAT_34_SIGN_MASK) ? -1 : 1;

    // deal with special numbers.
    if ((combination == 0x1fL) && (sign == 1))
    {
      long nanSignal = (decFloat34BitsHi & DEC_FLOAT_34_SIGNAL_MASK) >> 57; //shift first 7 bits to get signal bit out  //@snan
      return nanSignal == 1 ? "SNaN" : "NaN";
    }
    else if ((combination == 0x1fL) && (sign == -1))
    {
      long nanSignal = (decFloat34BitsHi & DEC_FLOAT_34_SIGNAL_MASK) >> 57; //shift first 7 bits to get signal bit out  //@snan
      return nanSignal == 1 ? "-SNaN" : "-NaN";
    }
    else if ((combination == 0x1eL) && (sign == 1))
    {
      return "Infinity";
    }
    else if ((combination == 0x1eL) && (sign == -1))
    {
      return "-Infinity";
    }

    // compute the exponent MSD and the coefficient MSD.
    int exponentMSD;
    long coefficientMSD;
    if ((combination & 0x18L) == 0x18L)
    {
      // format of 11xxx:
      exponentMSD = (int) ((combination & 0x06L) >> 1);
      coefficientMSD = 8 + (combination & 0x01L);
    }
    else
    {
      // format of xxxxx:
      exponentMSD = (int) ((combination & 0x18L) >> 3);
      coefficientMSD = (combination & 0x07L);
    }

    // compute the exponent.
    int exponent = (int) ((decFloat34BitsHi & DEC_FLOAT_34_EXPONENT_CONTINUATION_MASK) >> 46);
    exponent |= (exponentMSD << 12);
    exponent -= DEC_FLOAT_34_BIAS;

    // compute the coefficient.
    int coefficientLo = decFloatBitsToDigits((int) (decFloat34BitsLo & 0x3fffffff)); // last 30 bits (9 digits)
    // another 30 bits (9 digits)
    int coefficientMeLo = decFloatBitsToDigits((int) ((decFloat34BitsLo >> 30) & 0x3fffffff));
    // another 30 bits (9 digits). 26 bits from hi and 4 bits from lo.
    int coefficientMeHi = decFloatBitsToDigits((int) (((decFloat34BitsHi & 0x3ffffff) << 4) | ((decFloat34BitsLo >> 60) & 0xf)));
    int coefficientHi = decFloatBitsToDigits((int) ((decFloat34BitsHi >> 26) & 0xfffff)); // high 20 bits (6 digits)
    coefficientHi += coefficientMSD * 1000000L;

    // compute the int array of coefficient.
    int[] value = computeMagnitude(new int[] { coefficientHi, coefficientMeHi, coefficientMeLo, coefficientLo});

    // convert value to a byte array of coefficient.
    byte[] magnitude = new byte[16];
    magnitude[0] = (byte) (value[0] >>> 24);
    magnitude[1] = (byte) (value[0] >>> 16);
    magnitude[2] = (byte) (value[0] >>> 8);
    magnitude[3] = (byte) (value[0]);
    magnitude[4] = (byte) (value[1] >>> 24);
    magnitude[5] = (byte) (value[1] >>> 16);
    magnitude[6] = (byte) (value[1] >>> 8);
    magnitude[7] = (byte) (value[1]);
    magnitude[8] = (byte) (value[2] >>> 24);
    magnitude[9] = (byte) (value[2] >>> 16);
    magnitude[10] = (byte) (value[2] >>> 8);
    magnitude[11] = (byte) (value[2]);
    magnitude[12] = (byte) (value[3] >>> 24);
    magnitude[13] = (byte) (value[3] >>> 16);
    magnitude[14] = (byte) (value[3] >>> 8);
    magnitude[15] = (byte) (value[3]);

    BigInteger bigInt = new BigInteger(sign, magnitude);
    BigDecimal bigDec = new BigDecimal(bigInt, -exponent);
    return bigDec.toString();
  }

  // Copied from JTOpen. TODO - Needs optimization.
  // Compute the int array of magnitude from input value segments.
  private static final int[] computeMagnitude(final int[] input)
  {
    final int length = input.length;
    final int[] mag = new int[length];

    final int stop = length-1;
    mag[stop] = input[stop];
    for (int i=0; i= 0; --j, --k)
      {
        long product = (input[length-2-i] & 0xFFFFFFFFL) * (TEN_RADIX_MAGNITUDE[i][j] & 0xFFFFFFFFL)
                       + (mag[k] & 0xFFFFFFFFL) // add previous value
                       + (carry & 0xFFFFFFFFL); // add carry
        carry = (int) (product >>> 32);
        mag[k] = (int) (product & 0xFFFFFFFFL);
      }
      mag[k] = (int) carry;
    }
    return mag;
  }

  // Copied from JTOpen. TODO - Needs optimization.
  // Convert 30 binary bits coefficient to 9 decimal digits.
  private static final int decFloatBitsToDigits (int bits)
  {
    int decimal = 0;
    for (int i=2; i>=0; --i)
    {
      decimal *= 1000;
      decimal += unpackDenselyPackedDecimal((int)((bits >> (i * 10)) & 0x03ffL));
    }
    return decimal;
  }

  // Copied from JTOpen. TODO - Needs optimization.
  // Internal declet decoding helper method.
  private static int unpackDenselyPackedDecimal (int bits)
  {
    //Declet is the three bit encoding of one decimal digit.  The Decfloat is made up of declets to represent
    //the decfloat 16 or 34 digits
    int combination;
    if ((bits & 14) == 14)
    {
      combination = ((bits & 96) >> 5) | 4;
    }
    else
    {
      combination = ((bits & 8) == 8) ? (((~bits) & 6) >> 1) : 0;
    }
    int decoded = 0;
    switch (combination)
    {
      case 0:  // bit 6 is 0
        decoded = ((bits & 896) << 1) | (bits & 119);
        break;
      case 1:  // bits 6,7,8 are 1-1-0
        decoded = ((bits & 128) << 1) | (bits & 113) | ((bits & 768) >> 7) | 2048;
        break;
      case 2:  // bits 6,7,8 are 1-0-1
        decoded = ((bits & 896) << 1) | (bits & 17) | ((bits & 96) >> 4) | 128;
        break;
      case 3:  // bits 6,7,8 are 1-0-0
        decoded = ((bits & 896) << 1) | (bits & 113) | 8;
        break;
      case 4:  // bits 6,7,8 are 1-1-1, bits 3,4 are 0-0
        decoded = ((bits & 128) << 1) | (bits & 17) | ((bits & 768) >> 7) | 2176;
        break;
      case 5:  // bits 6,7,8 are 1-1-1, bits 3,4 are 0-1
        decoded = ((bits & 128) << 1) | (bits & 17) | ((bits & 768) >> 3) | 2056;
        break;
      case 6:  // bits 6,7,8 are 1-1-1, bits 3,4 are 1-0
        decoded = ((bits & 896) << 1) | (bits & 17) | 136;
        break;
      case 7:  // bits 6,7,8 are 1-1-1, bits 3,4 are 1-1
        // NB: we ignore values of bits 0,1 in this case
        decoded = ((bits & 128) << 1) | (bits & 17) | 2184;
        break;
    }
    return((decoded & 3840) >> 8) * 100 + ((decoded & 240) >> 4) *10 + (decoded & 15);
  }

  /**
   * Converts the specified packed decimal bytes into a String.
  **/
  public static final String packedDecimalToString(final byte[] data, final int offset, final int numDigits, final int scale)
  {
    final int len = numDigits/2+1;
    int sign = data[offset+len-1] & 0x0F;
    boolean isNegative = sign == 0x0B || sign == 0x0D;
    char[] buf = new char[numDigits+(scale > 0 ? 1 : 0)+(isNegative ? 1 : 0)];
    return packedDecimalToString(data, offset, numDigits, scale, buf);
  }

  /**
   * Converts the specified packed decimal bytes into a String.
   * The number of bytes used from data is equal to numDigits/2+1.
  **/
  public static final String packedDecimalToString(final byte[] data, int offset, int numDigits, final int scale, final char[] buffer)
  {

    // even number of digits will have a leading zero
    if (numDigits%2 == 0) ++numDigits;

    final int len = numDigits/2+1;

    int sign = data[offset+len-1] & 0x0F;
    boolean isNegative = sign == 0x0B || sign == 0x0D;
    int count = 0;
    if (isNegative)
    {
      buffer[count++] = '-';
    }
    // boolean doHigh = numDigits % 2 == 1;
    // FindBugs says:  The code uses x % 2 == 1 to check to see if a value is odd, but this won't work for negative numbers (e.g., (-5) % 2 == -1).
    // If this code is intending to check for oddness, consider using x & 1 == 1, or x % 2 != 0.
    boolean doHigh = numDigits % 2 != 0;

    int digitsBeforeDecimal = numDigits-scale;
    boolean foundNonZero = false;
    for (int i=0; i> 4) : data[offset]) & 0x0F;
      if (foundNonZero || nibble != 0)
      {
        buffer[count++] = NUM[nibble];
        foundNonZero = true;
      }
      if (!doHigh)
      {
        doHigh = true;
        ++offset;
      }
      else
      {
        doHigh = false;
      }
    }
    if (count == 0 || (isNegative && count == 1))
    {
      buffer[count++] = '0';
    }
    if (scale > 0)
    {
      buffer[count++] = '.';
    }
    for (int i=digitsBeforeDecimal; i> 4) : data[offset]) & 0x0F;
      buffer[count++] = NUM[nibble];
      if (!doHigh)
      {
        doHigh = true;
        ++offset;
      }
      else
      {
        doHigh = false;
      }
    }
    return new String(buffer, 0, count);
  }

  // Copied from JTOpen AS400PackedDecimal.
  public static final double packedDecimalToDouble(final byte[] data, final int offset, final int numDigits, final int scale)
  {
    // Compute the value.
    double doubleValue = 0;
    double multiplier = Math.pow(10, -scale);
    int rightMostOffset = offset + numDigits/2;
    boolean nibble = true; // true for left nibble, false for right nibble.
    for (int i = rightMostOffset; i >= offset;)
    {
      if (nibble)
      {
        doubleValue += (byte)((data[i] & 0x00F0) >> 4) * multiplier;
        --i;
      }
      else
      {
        doubleValue += ((byte)(data[i] & 0x000F)) * multiplier;
      }

      multiplier *= 10;
      nibble = ! nibble;
    }

    // Determine the sign.
    switch (data[rightMostOffset] & 0x000F)
    {
      case 0x000B:
      case 0x000D:
        // Negative.
        doubleValue *= -1;
        break;
      case 0x000A:
      case 0x000C:
      case 0x000E:
      case 0x000F:
        // Positive.
        break;
      default:
        throw new NumberFormatException("Byte sequence not valid for packed decimal ("+rightMostOffset+": "+(data[rightMostOffset] & 0x000F)+").");
    }

    return doubleValue;
  }

  public static final byte[] doubleToPackedDecimal(final double d, final int numDigits, final int scale)
  {
    byte[] data = new byte[numDigits/2+1];
    doubleToPackedDecimal(d, data, 0, numDigits, scale);
    return data;
  }

  // Copied from JTOpen AS400PackedDecimal.
  public static final void doubleToPackedDecimal(final double d, final byte[] data, final int offset, final int numDigits, final int scale)
  {
    // GOAL:  For performance reasons, we need to do this conversion
    //        without creating any Java objects (e.g., BigDecimals,
    //        Strings).

    // If the number is too big, we can't do anything with it.
    double absValue = Math.abs(d);
    if (absValue > Long.MAX_VALUE)
    {
      throw new NumberFormatException("Double value is too big: "+d);
    }

    // Extract the normalized value.  This is the value represented by
    // two longs (one for each side of the decimal point).  Using longs
    // here improves the quality of the algorithm as well as the
    // performance of arithmetic operations.  We may need to use an
    // "effective" scale due to the lack of precision representable
    // by a long.
    long leftSide = (long)absValue;
    int effectiveScale = (scale > 15) ? 15 : scale;
    long rightSide = (long)Math.round((absValue - (double)leftSide) * Math.pow(10, effectiveScale));

    // Ok, now we are done with any double arithmetic!
    int length = numDigits/2;
    int b = offset + length;
    boolean nibble = true; // true for left nibble, false for right nibble.

    // If the effective scale is different than the actual scale,
    // then pad with zeros.
    int scaleDifference = scale - effectiveScale;
    for (int i = 1; i <= scaleDifference; ++i)
    {
      if (nibble)
      {
        data[b] &= (byte)(0x000F);
        --b;
      }
      else
      {
        data[b] &= (byte)(0x00F0);
      }
      nibble = !nibble;
    }

    // Compute the bytes for the right side of the decimal point.
    int nextDigit;
    for (int i = 1; i <= effectiveScale; ++i)
    {
      nextDigit = (int)(rightSide % 10);
      if (nibble)
      {
        data[b] &= (byte)(0x000F);
        data[b] |= ((byte)nextDigit << 4);
        --b;
      }
      else
      {
        data[b] &= (byte)(0x00F0);
        data[b] |= (byte)nextDigit;
      }
      nibble = !nibble;
      rightSide /= 10;
    }

    // Compute the bytes for the left side of the decimal point.
    int leftSideDigits = numDigits - scale;
    for (int i = 1; i <= leftSideDigits; ++i)
    {
      nextDigit = (int)(leftSide % 10);
      if (nibble)
      {
        data[b] &= (byte)(0x000F);
        data[b] |= ((byte)nextDigit << 4);
        --b;
      }
      else
      {
        data[b] &= (byte)(0x00F0);
        data[b] |= (byte)nextDigit;
      }
      nibble = !nibble;
      leftSide /= 10;
    }

    // Zero out the left part of the value, if needed.
    while (b >= offset)
    {
      if (nibble)
      {
        data[b] &= (byte)(0x000F);
        --b;
      }
      else
      {
        data[b] &= (byte)(0x00F0);
      }
      nibble = !nibble;
    }

    // Fix the sign.
    b = offset + length;
    data[b] &= (byte)(0x00F0);
    data[b] |= (byte)((d >= 0) ? 0x000F : 0x000D);

    // If left side still has digits, then the value was too big
    // to fit.
    if (leftSide > 0)
    {
      throw new NumberFormatException("Double value "+d+" too big for output array.");
    }
  }

  /**
   * Converts the specified String (number) into packed decimal bytes.
  **/
  public static final byte[] stringToPackedDecimal(final String s, final int numDigits)
  {
    byte[] b = new byte[numDigits/2+1];
    stringToPackedDecimal(s, numDigits, b, 0);
    return b;
  }

  /**
   * Converts the specified String (number) into packed decimal bytes.
   * The string must have the correct number of decimal digits for
   * the conversion to be correct.
  **/
  public static final void stringToPackedDecimal(final String s, final int numDigits, final byte[] buffer, final int offset)
  {
    final int len = numDigits/2+1;
    final boolean isNegative = s != null && s.length() > 0 && s.charAt(0) == '-';
    int counter = offset+len-1;
    buffer[counter] = isNegative ? (byte)0x0D : (byte)0x0F;
    final int stop = isNegative ? 1 : 0;
    boolean doHigh = true;
    if ( s != null) {

    for (int i = s.length()-1; i>=stop; --i)
    {
      char c = s.charAt(i);
      if (c != '.')
      {
    	  int index = (int)c-'0';
    	  if (index < 0 || index > 9) {
    		  throw new NumberFormatException("Invalid character "+c);
    	  }
        if (doHigh)
        {
          buffer[counter--] |= CHAR_HIGH[index];
          doHigh = false;
        }
        else
        {
          buffer[counter] = CHAR_LOW[index];
          doHigh = true;
        }
      }
    }
    }
  }

  /**
   * Converts the specified zoned decimal bytes into a String.
  **/
  public static final String zonedDecimalToString(final byte[] data, final int offset, final int numDigits, final int scale)
  {
    int sign = (data[offset+numDigits-1] >> 4) & 0x0F;
    boolean isNegative = sign == 0x0B || sign == 0x0D;
    char[] buf = new char[numDigits+(scale > 0 ? 1 : 0)+(isNegative ? 1 : 0)];
    return zonedDecimalToString(data, offset, numDigits, scale, buf);
  }

  /**
   * Converts the specified zoned decimal bytes into a String.
  **/
  public static final String zonedDecimalToString(final byte[] data, int offset, final int numDigits, final int scale, final char[] buffer)
  {
    int sign = (data[offset+numDigits-1] >> 4) & 0x0F;
    boolean isNegative = sign == 0x0B || sign == 0x0D;
    int count = 0;
    boolean foundNonZero = false;
    if (isNegative)
    {
      buffer[count++] = '-';
    }
    int digitsBeforeDecimal = numDigits-scale;
    for (int i=0; i 0)
    {
      buffer[count++] = '.';
    }
    for (int i=digitsBeforeDecimal; i Long.MAX_VALUE)
    {
      throw new NumberFormatException("Double value is too big: "+d);
    }

    // Extract the normalized value.  This is the value represented by
    // two longs (one for each side of the decimal point).  Using longs
    // here improves the quality of the algorithm as well as the
    // performance of arithmetic operations.  We may need to use an
    // "effective" scale due to the lack of precision representable
    // by a long.
    long leftSide = (long)absValue;
    int effectiveScale = (scale > 15) ? 15 : scale;
    long rightSide = (long)Math.round((absValue - (double)leftSide) * Math.pow(10, effectiveScale));

    // Ok, now we are done with any double arithmetic!

    // If the effective scale is different than the actual scale,
    // then pad with zeros.
    int rightmostOffset = offset + numDigits - 1;
    int padOffset = rightmostOffset - (scale - effectiveScale);
    for (int i = rightmostOffset; i > padOffset; --i)
      data[i] = (byte)0x00F0;

    // Compute the bytes for the right side of the decimal point.
    int decimalOffset = rightmostOffset - scale;
    int nextDigit;
    for (int i = padOffset; i > decimalOffset; --i)
    {
      nextDigit = (int)(rightSide % 10);
      data[i] = (byte)(0x00F0 | nextDigit);
      rightSide /= 10;
    }

    // Compute the bytes for the left side of the decimal point.
    for (int i = decimalOffset; i >= offset; --i)
    {
      nextDigit = (int)(leftSide % 10);
      data[i] = (byte)(0x00F0 | nextDigit);
      leftSide /= 10;
    }

    // Fix the sign, if negative.
    if (d < 0)
      data[rightmostOffset] = (byte)(data[rightmostOffset] & 0x00DF);

    // If left side still has digits, then the value was too big
    // to fit.
    if (leftSide > 0)
    {
      throw new NumberFormatException("Double value "+d+" too big for output array.");
    }
  }

  /**
   * The scale is 0, and 0 < numDigits <= 20.
  **/
  public static final void longToZonedDecimal(long l, final byte[] data, final int offset, final int numDigits)
  {
    int rightmostOffset = offset + numDigits - 1;
    boolean isNegative = (l < 0);
    // Compute the bytes for the left side of the decimal point.
    for (int i = rightmostOffset; i >= offset; --i)
    {
      int nextDigit = (int)(l % 10);
      data[i] = (byte)(0x00F0 | nextDigit);
      l /= 10;
    }

    if (isNegative) data[rightmostOffset] = (byte)(data[rightmostOffset] & 0x00DF);

    if (l != 0)
    {
      throw new NumberFormatException("Long value too big for ZONED("+numDigits+",0).");
    }
  }

  /**
   * Converts the specified String (number) into zoned decimal bytes.
  **/
  public static final byte[] stringToZonedDecimal(final String s, final int numDigits)
  {
    byte[] b = new byte[numDigits];
    stringToZonedDecimal(s, numDigits, b, 0);
    return b;
  }

  /**
   * Converts the specified String (number) into zoned decimal bytes.
  **/
  public static final void stringToZonedDecimal(final String s, final int numDigits, final byte[] buffer, final int offset)
  {
    int counter = offset+numDigits-1;
    final boolean isNegative = s != null && s.length() > 0 && s.charAt(0) == '-';
    final int stop = isNegative ? 1 : 0;
    if (s != null) {
    for (int i = s.length()-1; i>=stop; --i)
    {
      char c = s.charAt(i);
      if (c != '.')
      {
    	int index = c - '0';
    	if (index < 0 || index > 9 ) {
    		throw new NumberFormatException("Invalid character "+c);
    	}
        buffer[counter--] = (byte) ( CHAR_LOW[index] | 0xF0) ;
      }
    }
    }
    if (isNegative)
    {
      buffer[offset+numDigits-1] = (byte) ((buffer[offset+numDigits-1] & ((byte) 0x0F)) | (byte) 0xD0);
    }
  }


  /**
   * Return the default NLV for the client corresponding to the Java locale.
   * @return the default client NLV
   */
  public static String getDefaultNLV() {
	  if (localeNlvMap_ == null) { 
		  localeNlvMap_ = new Hashtable(100); // 74 actual entries.
	        // 74 entries.
	        localeNlvMap_.put("ar", "2954");
	        localeNlvMap_.put("ar_SA", "2954");
	        localeNlvMap_.put("be", "2979");
	        localeNlvMap_.put("bg", "2974");
	        localeNlvMap_.put("ca", "2931");
	        localeNlvMap_.put("cs", "2975");
	        localeNlvMap_.put("da", "2926");
	        localeNlvMap_.put("de", "2929");
	        localeNlvMap_.put("de_CH", "2939");
	        localeNlvMap_.put("de_DE", "2929");
	        localeNlvMap_.put("el", "2957");
	        localeNlvMap_.put("en", "2924");
	        localeNlvMap_.put("en_BE", "2909");
	        localeNlvMap_.put("en_CN", "2984");
	        localeNlvMap_.put("en_JP", "2938");
	        localeNlvMap_.put("en_KR", "2984");
	        localeNlvMap_.put("en_SG", "2984");
	        localeNlvMap_.put("en_TW", "2984");
	        localeNlvMap_.put("es", "2931");
	        localeNlvMap_.put("es_ES", "2931");
	        localeNlvMap_.put("et", "2902");
	        localeNlvMap_.put("fa", "2998");
	        localeNlvMap_.put("fi", "2925");
	        localeNlvMap_.put("fr", "2928");
	        localeNlvMap_.put("fr_BE", "2966");
	        localeNlvMap_.put("fr_CA", "2981");
	        localeNlvMap_.put("fr_CH", "2940");
	        localeNlvMap_.put("fr_FR", "2928");
	        localeNlvMap_.put("hr", "2912");
	        localeNlvMap_.put("hu", "2976");
	        localeNlvMap_.put("is", "2958");
	        localeNlvMap_.put("it", "2932");
	        localeNlvMap_.put("it_CH", "2942");
	        localeNlvMap_.put("iw", "2961");
	        localeNlvMap_.put("ja", "2962");
	        localeNlvMap_.put("ji", "2961");
	        localeNlvMap_.put("ka", "2979");
	        localeNlvMap_.put("kk", "2979");
	        localeNlvMap_.put("ko", "2986");
	        localeNlvMap_.put("ko_KR", "2986");
	        localeNlvMap_.put("lo", "2906");
	        localeNlvMap_.put("lt", "2903");
	        localeNlvMap_.put("lv", "2904");
	        localeNlvMap_.put("mk", "2913");
	        localeNlvMap_.put("nl", "2923");
	        localeNlvMap_.put("nl_BE", "2963");
	        localeNlvMap_.put("nl_NL", "2923");
	        localeNlvMap_.put("no", "2933");
	        localeNlvMap_.put("pl", "2978");
	        localeNlvMap_.put("pt", "2996");
	        localeNlvMap_.put("pt_BR", "2980");
	        localeNlvMap_.put("pt_PT", "2922");
	        localeNlvMap_.put("ro", "2992");
	        localeNlvMap_.put("ru", "2979");
	        localeNlvMap_.put("sh", "2912");
	        localeNlvMap_.put("sk", "2994");
	        localeNlvMap_.put("sl", "2911");
	        localeNlvMap_.put("sq", "2995");
	        localeNlvMap_.put("sr", "2914");
	        localeNlvMap_.put("sv", "2937");
	        localeNlvMap_.put("sv_SE", "2937");
	        localeNlvMap_.put("th", "2972");
	        localeNlvMap_.put("th_TH", "2972");
	        localeNlvMap_.put("tr", "2956");
	        localeNlvMap_.put("uk", "2979");
	        localeNlvMap_.put("uz", "2979");
	        localeNlvMap_.put("vi", "2905");
	        localeNlvMap_.put("zh", "2989");
	        localeNlvMap_.put("zh_CN", "2989");
	        localeNlvMap_.put("zh_HK", "2987");
	        localeNlvMap_.put("zh_SG", "2989");
	        localeNlvMap_.put("zh_TW", "2987");
	        localeNlvMap_.put("cht", "2987");  // Chinese/Taiwan
	        localeNlvMap_.put("cht_CN", "2987");  // Chinese/Taiwan
	  }
	  String defaultNLV = null; 
	  try { 
		  Locale locale = Locale.getDefault(); 
		  if (locale != null) { 
		    String localeString = locale.toString();
		    defaultNLV = (String) localeNlvMap_.get(localeString);
		    if (defaultNLV == null) { 
		    	// Check for underscore index in locale
		    	int underscoreIndex = localeString.indexOf('_'); 
		    	if (underscoreIndex > 0) { 
		    		localeString = localeString.substring(0,underscoreIndex); 
				    defaultNLV = (String) localeNlvMap_.get(localeString);
		    	}
		    }
		  }
	  } catch (Exception e) { 
		  // Ignore any errors 
	  }
	  if (defaultNLV == null) defaultNLV="2924"; 
	  return defaultNLV; 

  }



}





© 2015 - 2025 Weber Informatics LLC | Privacy Policy