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

src.com.ibm.as400.access.GenerateConverterTable Maven / Gradle / Ivy

///////////////////////////////////////////////////////////////////////////////
//
// JTOpen (IBM Toolbox for Java - OSS version)
//
// Filename:  AS400.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) 1997-2016 International Business Machines Corporation and
// others.  All rights reserved.
//
///////////////////////////////////////////////////////////////////////////////

package com.ibm.as400.access;

import java.io.*;
import java.lang.reflect.Field;
import java.sql.Blob;
import java.sql.Clob;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.util.*;

public class GenerateConverterTable {
  private static final String copyright = "Copyright (C) 1997-2016 International Business Machines Corporation and others.";
  private static final int MAX_SURROGATE_LENGTH = 2000;
  private static final int MAX_TO_EBCDIC_LENGTH = 20000;
  static AS400 sys = null;
  static Connection connection_ = null ;
  static boolean compress_ = true; // Compress the conversion table
                                   // Note: turn this off for debugging purposes

  static boolean codePointPerLine_ = false; // Should only 1 code point be
                                            // printed per line
  static boolean ascii_ = false; // Indicates if listed ccsids are ascii tables
                                 // or not

  static boolean bidi_ = false; // Indicates if listed ccsids are bidi tables or
                                // not
  // Note: bidi_ and ascii_ cannot both be true

  static boolean showOffsets_ = false; // Indicates of the offsets should be
                                       // printed in the tables
  
  static boolean useJdbc_ = false;   // Use JDBC to retrieve the table information 

  public static void main(String[] args) {
    if (args.length < 4) {
      System.out
          .println("Usage: java com.ibm.as400.access.GenerateConverterTable system uid pwd [-nocompress] [-ascii] [-bidi] [-showOffsets] [-codePointPerLine] [-useJdbc] ccsid [ccsid2] [ccsid3] [ccsid4] ...");
      System.exit(0);
    }

    try {
      sys = new AS400(args[0], args[1], args[2]);
      sys.connectService(AS400.CENTRAL);
    } catch (Exception e) {
      e.printStackTrace();
      System.exit(0);
    }

    int start = 3;
    if (args[start].equals("-nocompress")) {
      compress_ = false;
      ++start;
    }
    if (args[start].equals("-ascii")) {
      ascii_ = true;
      ++start;
    }
    if (args[start].equals("-bidi")) {
      bidi_ = true;
      ++start;
    }

    if (args[start].equals("-showOffsets")) {
      showOffsets_ = true;
      ++start;
    }

    if (args[start].equals("-codePointPerLine")) {
      codePointPerLine_ = true;
      ++start;
    }

    if (args[start].equals("-useJdbc")) {
      useJdbc_ = true;
      
      try {
        Class.forName("com.ibm.as400.access.AS400JDBCDriver"); 
        connection_ = DriverManager.getConnection("jdbc:as400:"+args[0], args[1], args[2]);
        
      } catch (Exception e) {
        e.printStackTrace();
        System.exit(0);
      }

      
      ++start;
    }

    for (int i = start; i < args.length; ++i) {
      go((new Integer(args[i])).intValue());
    }
  }

  static String formattedChar(char x) {
    int num = 0xFFFF & (int) x;

    String s = "\\u";
    if (num < 16)
      s += "0";
    if (num < 256)
      s += "0";
    if (num < 4096)
      s += "0";
    s += Integer.toHexString(num).toUpperCase();
    return s;
  }

  static void go(int ccsid) {
    char[] tableToUnicode = new char[0];
    char[] tableToEbcdic = new char[0];
    char[][] surrogateTable = null;

    // int numTables1 = 1;
    // int numTables2 = 1;

    boolean ebcdicIsDBCS = false;
    int doubleByteFormat = NLSTableDownload.DOUBLE_BYTE_FROM_CCSID; 
    int originalCcsid = ccsid; 
    
    try {
      if (useJdbc_) {
        if (ConvTable.isMixedCCSID(originalCcsid)) {
          // 
          // If we have a mixed CCSID then we will download in two pieces
          // We use the convention that a CCSID that starts with 10xxxxx is SINGLE BYTE part of mixed
          //                        and a CCSID that starts with 20xxxxx is DOUBLE BYTE part of mixed
          go(1000000+ccsid); 
          go(2000000+ccsid); 
          
          return; 
        } else {
          Class.forName("com.ibm.as400.access.AS400JDBCDriver"); 
          try { 
          if (ccsid > 2000000 ) {
             ebcdicIsDBCS = true; 
          } else if (ccsid > 1000000) {
            ebcdicIsDBCS = false; 
          } else { 
            ebcdicIsDBCS = jdbcIsDBCS(connection_, ccsid) ; 
          }

          if (ebcdicIsDBCS) {
            tableToUnicode =  jdbcToUnicodeDBCS(connection_, ccsid); 
            tableToEbcdic =   jdbcToEbcdicDBCS(connection_, ccsid); 
          } else {
            tableToUnicode =   jdbcToUnicode(connection_, ccsid); 
            tableToEbcdic =    jdbcToEbcdic(connection_, ccsid);  
          }
          } catch (Exception e) { 
            System.out.println("Error downloading table using JDBC "); 
            e.printStackTrace(System.out) ;
            System.exit(1); 
          }
          
          
        }
      } else {
        AS400ImplRemote impl = (AS400ImplRemote) sys.getImpl();

        NLSTableDownload down = new NLSTableDownload(impl);
        down.connect();

        if (ccsid == 1089) // There are currently no tables for
                           // 1089->13488->1089;
                           // use 61952 instead, since it would be the same
                           // anyway.
        {
          System.out.println("Special case for ccsid 1089.");
          System.out.println("Retrieving " + ccsid + "->61952 table...");

          tableToUnicode = down.download(ccsid, 61952,
              NLSTableDownload.SINGLE_BYTE_FROM_CCSID);
        } else if (ccsid == 61175) {
          System.out.println("Special case for ccsid 61175.");
          System.out.println("Retrieving 1026->13488 table and adjusting...");
          tableToUnicode = down.download(1026, 13488,
              NLSTableDownload.SINGLE_BYTE_FROM_CCSID);
          tableToUnicode[0xFC] = tableToUnicode[0x7F];
          tableToUnicode[0x7F] = '"';
        } else if (ccsid == 1376) {
          // This is double byte so fall into bottom path
          tableToUnicode = null;
        } else if (ccsid == 1371) {
          tableToUnicode = null;
          doubleByteFormat = NLSTableDownload.MIXED_BYTE_FROM_CCSID;
        } else {
          System.out.println("Retrieving " + ccsid + "->13488 table...");
          tableToUnicode = down.download(ccsid, 13488,
              NLSTableDownload.SINGLE_BYTE_FROM_CCSID);
        }
        if (tableToUnicode == null || tableToUnicode.length == 0) {
          String reason = "";
          if (tableToUnicode == null) {
            reason = "tableToUnicode is null";
          } else {
            reason = "tableToUnicode.length is 0";
          }
          if (ccsid == 1175) {
            System.out.println("Aborting since CCSD 1175 failed to download");
            throw new Exception("Aborting since CCSD 1175 failed to download");
          }
          System.out.println(ccsid
              + " must be double-byte because download failed (" + reason
              + "). Performing secondary retrieve of " + ccsid
              + "->1200 table...");
          ebcdicIsDBCS = true;
          down.disconnect();
          down.connect();
          tableToUnicode = down.download(ccsid, 1200, doubleByteFormat);
        }
        down.disconnect();
        down.connect();
        
        if (ccsid == 1089) {
          System.out.println("Special case for ccsid 1089.");
          System.out.println("Retrieving 61952->" + ccsid + " table...");
          tableToEbcdic = down.download(61952, ccsid,
              NLSTableDownload.DOUBLE_BYTE_FROM_CCSID);
        } else {
          /* Use 1200 instead of 13488 */
          System.out.println("Retrieving 1200->" + ccsid + " table...");
          tableToEbcdic = down.download(1200, ccsid,
              NLSTableDownload.DOUBLE_BYTE_FROM_CCSID);
        }

      }
      System.out.println("  Size: " + tableToUnicode.length);
      if (tableToUnicode.length > 65536) {
        System.out.println("Size is > 65536.  Fixing table");
        int next = 0;
        int from = 0;
        char[] newTable = new char[65536];
        int lastFrom = 0; 
        int lastTo = 0; 
        while (from < tableToUnicode.length && next < 65536) {

          int c = 0xFFFF & (int) tableToUnicode[from];
          
          // If we didn't process a variation selector, ignore it. 
          while (c >= 0xFE00 && c <= 0xFE0F) {
            from++;
            c = 0xFFFF & (int) tableToUnicode[from];
          }
          
          if (next > 0xECAA && next <= 0xECD0) {
            System.out.println("Next=0x" + Integer.toHexString(next) + " to="
                + Integer.toHexString(c));
          }

          int nextchar = 0;
          if (from + 1 < tableToUnicode.length) {
            nextchar = 0xFFFF & (int) tableToUnicode[from + 1];
          }

          
          if (
          // in surrogate range
          ((c >= 0xD800) && (c <= 0xDFFF))
              ||
              // Uses Variation selector
              ((nextchar >= 0xFE00) && (nextchar <= 0xFE0F))
              
              ||
              // Uses combining character
              ((nextchar == 0x309A) && (c != 0x3099)) ||  /* In 835 there are two combining characters next to each other */ 
                                                        /* In that case, we do not combine */ 
              (c != 0xFFfd && nextchar == 0x300)
              || (c != 0xffd && c != 0x300 && nextchar == 0x301)
              ||
              // Weird cases..
              (c == 0x2e5 && nextchar == 0x2e9)
              || (c == 0x2e9 && nextchar == 0x2e5)) {
            // Mark as surrogate
            
            newTable[next] = (char) 0xD800;
            lastFrom =next; 
            lastTo = 0xD800; 
            
            // add to surrogate table
            if (surrogateTable == null) {
              surrogateTable = new char[65536][];
            }
            char[] pair = new char[2];
            surrogateTable[next] = pair;
            pair[0] = (char) (0xFFFF & (int) tableToUnicode[from]);
            pair[1] = (char) (0xFFFF & (int) tableToUnicode[from + 1]);
            /*
             * System.out.println("Warning: Sub at offset "+Integer.toHexString(next
             * )+" for "+Integer.toHexString(0xFFFF & (int)
             * table1[from])+" "+Integer.toHexString(0xFFFF & (int)
             * table1[from+1]));
             */
            from += 2;
          } else {
            newTable[next] = (char) c;
            if (c != 0xFFFD) {
              lastFrom = next; 
              lastTo = c; 
            }
            from++;
          }
          next++;
        }
        tableToUnicode = newTable;

      }
      System.out.println("  Size: " + tableToEbcdic.length);

      // If a mixed CCSID, then we need to fix the tableToEbcdic
      // Convert the table to a byte array
      // Go through table and process SI/I)
      if (doubleByteFormat == NLSTableDownload.MIXED_BYTE_FROM_CCSID) {
        char[] newTableToEbcdic = new char[65536];
        byte[] oldTableToEbcdic = new byte[tableToEbcdic.length * 2]; 
        for (int i = 0; i < tableToEbcdic.length; i++) {
          oldTableToEbcdic[2*i] = (byte) ( 0xFF & (tableToEbcdic[i] >> 8)); 
          oldTableToEbcdic[2*i+1] = (byte)( tableToEbcdic[i] & 0xFF); 
        }
        boolean singleByte = true; 
        int newTableOffset = 0; 
        for (int i = 0; i < oldTableToEbcdic.length; i++) {
           int b = 0xFF & oldTableToEbcdic[i]; 
           while ((i > 0x0F) && (i < oldTableToEbcdic.length) && ((b == 0x0E) || (b == 0x0F)) ) {
             if (b == 0x0E) {
                singleByte = false;
                i++; 
             } else {
               singleByte = true;
               i++; 
             }
             if (i < oldTableToEbcdic.length) { 
                b = 0xFF & oldTableToEbcdic[i];
             }
           } 
          if (i < oldTableToEbcdic.length) {
            if (singleByte) {
              if (newTableOffset < newTableToEbcdic.length) { 
                newTableToEbcdic[newTableOffset] = (char) b;
                newTableOffset++;
              }
            } else {
              i++;
              if (i < oldTableToEbcdic.length) { 
                 int c =  0xFF & oldTableToEbcdic[i];
                 if (newTableOffset < newTableToEbcdic.length) {
                   if (newTableOffset == 0x431) {
                     newTableToEbcdic[newTableOffset] = (char) ((b << 8) + c);
                     newTableOffset++;
                     
                   } else { 
                     newTableToEbcdic[newTableOffset] = (char) ((b << 8) + c);
                     newTableOffset++;
                   }
                 }
              }

            }
          }
        } /* for i */ 
        while (newTableOffset < newTableToEbcdic.length) {
            newTableToEbcdic[newTableOffset]=(char) 0xFEFE; 
            newTableOffset++; 
        }
        tableToEbcdic=newTableToEbcdic;
      } /* If Mixed byte */ 
      sys.disconnectAllServices();
    } catch (Exception e) {
      e.printStackTrace(System.out);
    }

    // Do any necessary fixup
    if (ccsid == 290) {
      
      tableToUnicode[0xE1] = '\u20ac'; 
      char toEbcdic = tableToEbcdic[0x20ac / 2]; 
      toEbcdic = (char)((0xE1 << 8) | (toEbcdic & 0xFF)); 
      tableToEbcdic[0x20ac / 2] = toEbcdic; 
    }
    // Verify the mapping
    verifyRoundTrip(tableToUnicode, tableToEbcdic, ebcdicIsDBCS, ccsid);

    System.out.println("****************************************");
    System.out.println("Verify round 2 ");
    System.out.println("****************************************");
    verifyRoundTrip(tableToUnicode, tableToEbcdic, ebcdicIsDBCS, ccsid);

    // Compress the ccsid table
    if (ebcdicIsDBCS) {
      if (compress_) {
        System.out.println("Compressing " + ccsid
            + "->13488 conversion table...");
        char[] arr = compress(tableToUnicode);
        System.out.println("Old compression length: " + arr.length
            + " characters.");
        char[] temparr = compressBetter(tableToUnicode);
        System.out.println("New compression length: " + temparr.length
            + " characters.");
        if (temparr.length > arr.length) {
          System.out
              .println("WARNING: New algorithm WORSE than old algorithm!");
        }
        System.out.println("Verifying compressed table...");
        arr = decompressBetter(temparr);
        if (arr.length != tableToUnicode.length) {
          System.out.println("Verification failed, lengths not equal: "
              + arr.length + " != " + tableToUnicode.length);
          int c = 0;
          while (c < arr.length && arr[c] == tableToUnicode[c])
            ++c;
          System.out.println("First mismatch at index " + c + ": "
              + (int) arr[c] + " != " + (int) tableToUnicode[c]);
        } else {
          boolean bad = false;
          for (int c = 0; c < arr.length; ++c) {
            if (arr[c] != tableToUnicode[c]) {
              bad = true;
              System.out.println(c + ": " + Integer.toHexString((int) arr[c])
                  + " != " + Integer.toHexString((int) tableToUnicode[c]));
            }
          }
          if (bad) {
            System.out.println("Mismatches found in table.");
          } else {
            tableToUnicode = temparr;
            System.out.println("Table verified.");
          }
        }
      }
    }

    int fileCcsid = ccsid; 
    if ( doubleByteFormat  == NLSTableDownload.MIXED_BYTE_FROM_CCSID) {
        fileCcsid = ccsid + 1100000; 
        System.out.println("Create file using "+fileCcsid+" since MIXED CCSID "); 
    }
    // Write out the ccsid table
    StringBuffer surrogateInitStringBuffer = new StringBuffer(); 

    try {
      String fName = "ConvTable" + fileCcsid + ".java";
      FileWriter f = new FileWriter(fName);
      writeHeader(f, fileCcsid, sys.getSystemName());
      if (ascii_) {
        f.write("class ConvTable" + fileCcsid + " extends ConvTableAsciiMap\n{\n");
      } else if (bidi_) {
        f.write("class ConvTable" + fileCcsid + " extends ConvTableBidiMap\n{\n");
      } else if (ebcdicIsDBCS) {
        f.write("class ConvTable" + fileCcsid + " extends ConvTableDoubleMap\n{\n");
      } else {
        f.write("class ConvTable" + fileCcsid + " extends ConvTableSingleMap\n{\n");
      }

      f.write("  private static char[] toUnicodeArray_;  \n");
      f.write("  private static final String copyright = \"Copyright (C) 1997-2016 International Business Machines Corporation and others.\";\n");
      f.write("  // toUnicode_ length is "+tableToUnicode.length+"\n"); 
      f.write("  private static final String toUnicode_ = \n");
      System.out.print("Writing table for conversion from " + ccsid
          + " to 13488... to " + fName + "\n");
      writeTable(f, tableToUnicode, 0, tableToUnicode.length );
      f.write("\n");
      f.write("\n");

      // Write out the surrogateTable if it exists
      int surrogateLength = 0; 
      if (surrogateTable != null) {

        f.write("\n");

        for (int i = 0; i < surrogateTable.length; i++) {
          char[] pair = surrogateTable[i];
          if (pair != null) {
            surrogateLength++; 
          }
        } /* for i */
        
        int surrogateCount = 0; 
        char[][] compressedSurrogateTable = new char[surrogateLength][]; 
        for (int i = 0; i < surrogateTable.length; i++) {
          char[] pair = surrogateTable[i];
          if (pair != null) {
            char[] triplet = new char[3]; 
            triplet[0] = (char) i; 
            triplet[1] = pair[0];
            triplet[2] = pair[1];
            compressedSurrogateTable[surrogateCount] = triplet; 
            surrogateCount++; 
          }
        } /* for i */
        
        
        f.write("  // Number of surrogateMappings is "+surrogateLength+"\n"); 
        if (surrogateLength < MAX_SURROGATE_LENGTH) { 
          f.write("  private static final char[][] toUnicodeSurrogateMappings = { \n");
          System.out.print("Writing surrogate table for conversion from " + ccsid
              + " to 13488... to " + fName + "\n");
          for (int i = 0; i < compressedSurrogateTable.length; i++) {
            char[] triplet = surrogateTable[i];
            if (triplet != null) {
              f.write("{'" + formattedChar((char) triplet[0]) + "','"
                  + formattedChar(triplet[1]) + "','" + formattedChar(triplet[2])
                  + "'},\n");
            }
          } /* for i */
          f.write("};\n");
          f.write("\n");
          f.write("\n");
        } else {
          // We must break into pieces
          f.write("  private static char[][] toUnicodeSurrogateMappings = new char["+surrogateLength+"][];\n");  
          int startIndex = 0; 
          while (startIndex < surrogateLength) {
            f.write("  private static void initToUnicodeSurrogateMappings"+startIndex+"() { \n"); 
            f.write("  char[][] toUnicodeSurrogateMappingsPiece = {\n");
            for (int i = 0; i < MAX_SURROGATE_LENGTH && (i+startIndex < surrogateLength); i++) {
              char[] triplet = compressedSurrogateTable[startIndex+i];
              if (triplet != null) {
                f.write("    {'" + formattedChar((char) (triplet[0])) + "','"
                    + formattedChar(triplet[1]) + "','" + formattedChar(triplet[2])
                    + "'},\n");
              }
            } /* for i */
            f.write("  };\n");

            f.write("    for (int j = 0; j < toUnicodeSurrogateMappingsPiece.length ; j++) {\n"); 
            f.write("      toUnicodeSurrogateMappings["+startIndex+"+j]= new char[3];\n");  
            f.write("      toUnicodeSurrogateMappings["+startIndex+"+j][0] = toUnicodeSurrogateMappingsPiece[j][0];\n");
            f.write("      toUnicodeSurrogateMappings["+startIndex+"+j][1] = toUnicodeSurrogateMappingsPiece[j][1];\n");
            f.write("      toUnicodeSurrogateMappings["+startIndex+"+j][2] = toUnicodeSurrogateMappingsPiece[j][2];\n");
            f.write("    }\n"); 
            f.write("  }\n"); 
            f.write("\n");
            surrogateInitStringBuffer.append("   initToUnicodeSurrogateMappings"+startIndex+"();\n"); 
            
            startIndex += MAX_SURROGATE_LENGTH; 
          }
        }
      } /* if surrogate table */

      f.close();
    } catch (Exception e) {
      e.printStackTrace();
    }

    // Compress the Unicode table
    if (compress_) {
      System.out
          .println("Compressing 13488->" + ccsid + " conversion table...");
      char[] arr = compress(tableToEbcdic);
      System.out.println("Old compression length: " + arr.length
          + " characters.");
      char[] temparr = compressBetter(tableToEbcdic);
      System.out.println("New compression length: " + temparr.length
          + " characters.");
      if (temparr.length > arr.length) {
        System.out.println("WARNING: New algorithm WORSE than old algorithm!");
      }
      System.out.println("Verifying compressed table...");
      arr = decompressBetter(temparr);
      if (arr.length != tableToEbcdic.length) {
        System.out.println("Verification failed, lengths not equal: "
            + arr.length + " != " + tableToEbcdic.length);
        int c = 0;
        while (c < arr.length && arr[c] == tableToEbcdic[c])
          ++c;
        System.out.println("First mismatch at index " + c + ": " + (int) arr[c]
            + " != " + (int) tableToEbcdic[c]);
        tableToEbcdic = temparr;
      } else {
        boolean bad = false;
        for (int c = 0; c < arr.length; ++c) {
          if (arr[c] != tableToEbcdic[c]) {
            bad = true;
            System.out.println(c + ": " + Integer.toHexString((int) arr[c])
                + " != " + Integer.toHexString((int) tableToEbcdic[c]));
          }
        }
        if (bad) {
          System.out.println("Mismatches found in table.");
        } else {
          tableToEbcdic = temparr;
          System.out.println("Table verified.");
        }
      }
    }

    // Write out the Unicode table
    try {
      String fName = "ConvTable" + fileCcsid + ".java";
      FileWriter f = new FileWriter(fName, true);

      System.out.print("Writing table for conversion from 13488 to " + ccsid
          + "... to " + fName + "\n");
      f.write("  private static char[] fromUnicodeArray_; \n");
      f.write("  // fromUnicode length = "+tableToEbcdic.length+"\n");
      if (tableToEbcdic.length < MAX_TO_EBCDIC_LENGTH ) {
        f.write("  private static final String fromUnicode_ = \n");
        writeTable(f, tableToEbcdic, 0, tableToEbcdic.length);
        f.write("\n");
      } else {
        int oneFourth = tableToEbcdic.length/4;
        f.write("  private static final String fromUnicode0_ = \n");
        writeTable(f, tableToEbcdic, 0, oneFourth );
        f.write("\n");
        f.write("  private static final String fromUnicode1_ = \n");
        writeTable(f, tableToEbcdic,oneFourth, oneFourth);
        f.write("\n");
        f.write("  private static final String fromUnicode2_ = \n");
        writeTable(f, tableToEbcdic,2 * oneFourth, oneFourth);
        f.write("\n");

        f.write("  private static final String fromUnicode3_ = \n");
        writeTable(f, tableToEbcdic,3 * oneFourth, tableToEbcdic.length - 3 * oneFourth);
        f.write("\n");

        
      }

      f.write("  static {\n");
      f.write("    toUnicodeArray_ = toUnicode_.toCharArray();\n");
      if (tableToEbcdic.length < MAX_TO_EBCDIC_LENGTH ) {
        f.write("    fromUnicodeArray_ = fromUnicode_.toCharArray();\n");
      } else {
        /* Note:  recent compilers try to optimized and add fromUnicode0_+fromUnicode1_ to the constant pool */
        /* Using sb.append disables this optimization */ 
        f.write("    StringBuffer sb = new StringBuffer(); \n");
        f.write("    sb.append(fromUnicode0_); \n");
        f.write("    sb.append(fromUnicode1_); \n");
        f.write("    sb.append(fromUnicode2_); \n");
        f.write("    sb.append(fromUnicode3_); \n");
        f.write("    fromUnicodeArray_ = sb.toString().toCharArray();\n");
      }
      f.write(surrogateInitStringBuffer.toString());
      f.write("  }\n");

      f.write("\n  ConvTable" + fileCcsid + "()\n  {\n");
      f.write("    super(" + fileCcsid + ", ");
      f.write("toUnicodeArray_, ");
      if (surrogateTable != null) {
        f.write("fromUnicodeArray_,");
        f.write("toUnicodeSurrogateMappings);\n");
      } else {
        f.write("fromUnicodeArray_);\n");
      }
      f.write("  }\n\n");

      f.write("\n  ConvTable" + fileCcsid + "(int ccsid)\n  {\n");
      f.write("    super(ccsid, ");
      f.write("toUnicodeArray_, ");
      if (surrogateTable != null) {
        f.write("fromUnicodeArray_,");
        f.write("toUnicodeSurrogateMappings);\n");
      } else {
        f.write("fromUnicodeArray_);\n");
      }

      f.write("  }\n}\n");

      f.close();
    } catch (Exception e) {
      e.printStackTrace();
    }
    System.out.print("Done.\n");
  }

  private static void writeTable(FileWriter f, char[] table, int start, int length )
      throws IOException {
    for (int i = start; i < start+length; i = i + 16) {
      if (showOffsets_) {
        f.write("/* " + Integer.toHexString(i) + " */ \"");
      } else {
        f.write("    \"");
      }
      for (int j = 0; j < 16 && (i + j) < start+length; ++j) {
        int num = (int) table[i + j]; // these each contain 2 single
                                              // byte chars, but we write it
                                              // like this to save space
        if (num == 0x0008)
          f.write("\\b");
        else if (num == 0x0009)
          f.write("\\t");
        // else if(num == 0x000A) f.write("\\r");
        else if (num == 0x000A)
          f.write("\\n");
        else if (num == 0x000C)
          f.write("\\f");
        // else if(num == 0x000D) f.write("\\n");
        else if (num == 0x000D)
          f.write("\\r");
        else if (num == 0x0022)
          f.write("\\\"");
        else if (num == 0x0027)
          f.write("\\'");
        else if (num == 0x005C)
          f.write("\\\\");
        else {
          String s = "\\u";
          if (num < 16)
            s += "0";
          if (num < 256)
            s += "0";
          if (num < 4096)
            s += "0";
          s += Integer.toHexString(num).toUpperCase();
          f.write(s);
        }
        if (codePointPerLine_) {
          if (j < 15) {
            if (showOffsets_) {
              f.write("\" +\n/* " + Integer.toHexString(i + j + 1) + " */ \"");
            } else {
              f.write("\" +\n    \"");
            }
          }
        }

      }
      if (i + 16 < start + length)
        f.write("\" +\n");
      else
        f.write("\";\n");
    }
  }

 

 
  private static boolean verifyRoundTrip(char[] tableToUnicode,
      char[] tableToEbcdic, boolean ebcdicIsDBCS, int ccsid) {

    String ebcdicPrefix = "X";
    if (ebcdicIsDBCS) {
      ebcdicPrefix = "GX";
    }

    // Make sure substitution is correct for Unicode to ebcdic
    if (!ebcdicIsDBCS) {
      int piece = 0xFFFF & tableToEbcdic[0x1a / 2];
      if ((piece >> 8) != 0x3f) {
        System.out.println("Fixing sub char in tableToEbcdic == sub was 0x"
            + Integer.toHexString(piece >> 8));
        piece = (0x3f << 8) | (0xFF & piece);
        tableToEbcdic[0x1a / 2] = (char) piece;
      }
    }

    System.out.println("Checking round trip for CCSID "+ccsid);
    boolean passed = true;
    StringBuffer sb1 = new StringBuffer();
    StringBuffer sb2 = new StringBuffer();
    StringBuffer sb3 = new StringBuffer();

    for (int i = 0; i < tableToUnicode.length; i++) {
      int unicodeChar = 0xFFFF & tableToUnicode[i];
      if (unicodeChar != 0xfffd && unicodeChar != 0xD800) {
        int ebcdicChar;
        if (ebcdicIsDBCS) {
          ebcdicChar = 0xFFFF & tableToEbcdic[unicodeChar];
        } else {
          int piece = 0xFFFF & tableToEbcdic[unicodeChar / 2];
          if (unicodeChar % 2 == 0) {
            ebcdicChar = piece >> 8;
          } else {
            ebcdicChar = piece & 0xFF;
          }
        }
        if (i != ebcdicChar) {
          if ((unicodeChar != 0x1a)
              && ((ebcdicChar == 0xFEFE) || (ebcdicChar == 0x3F))) {
            sb1.append("Fixing up EBCDIC RoundTrip Failure " + ebcdicPrefix
                + "'" + Integer.toHexString(i) + "'" + " -> UX'"
                + Integer.toHexString(unicodeChar) + "'" + " -> "
                + ebcdicPrefix + "'" + Integer.toHexString(ebcdicChar) + "'\n");
            if (ebcdicIsDBCS) {
              tableToEbcdic[unicodeChar] = (char) i;
            } else {
              int piece = 0xFFFF & tableToEbcdic[unicodeChar / 2];
              if (unicodeChar % 2 == 0) {

                piece = (i << 8) | (piece & 0x00FF);
              } else {
                piece = (piece & 0xFF00) | i;
              }
              tableToEbcdic[unicodeChar / 2] = (char) piece;
            }
            passed = false;
          } else {
            int unicodeChar2 = 0;
            try {
              unicodeChar2 = 0xFFFF & tableToUnicode[ebcdicChar];
            } catch (ArrayIndexOutOfBoundsException e) {
              System.out.println("ERROR.. ArrayIndexOutOfBounds");
              System.out.println("ebcdicChar=0x"
                  + Integer.toHexString((int) ebcdicChar));
              System.out.println("i=" + i);
              System.out.println("unicodeChar=0x"
                  + Integer.toHexString((int) unicodeChar));
              throw e;

            }
            if (unicodeChar2 == unicodeChar) {
              sb2.append("Secondary EBCDIC mapping " + ebcdicPrefix + "'"
                  + Integer.toHexString(i) + "'" + " -> UX'"
                  + Integer.toHexString(unicodeChar) + "'" + " -> "
                  + ebcdicPrefix + "'" + Integer.toHexString(ebcdicChar) + "'"
                  + " -> UX'" + Integer.toHexString(unicodeChar2) + "'\n");

            } else {
              sb3.append("EBCDIC RoundTrip Failure2 " + ebcdicPrefix + "'"
                  + Integer.toHexString(i) + "'" + " -> UX'"
                  + Integer.toHexString(unicodeChar) + "'" + " -> "
                  + ebcdicPrefix + "'" + Integer.toHexString(ebcdicChar) + "'"
                  + " -> UX'" + Integer.toHexString(unicodeChar2) + "'\n");
              passed = false;

            }
          }
        }
      }
    }
    System.out.println(sb2);
    System.out.println(sb1);
    System.out.println(sb3);

    sb1.setLength(0);
    sb2.setLength(0);
    sb3.setLength(0);

    for (int i = 0; i < tableToEbcdic.length; i++) {

      int ebcdicChar;
      if (ebcdicIsDBCS) {
        ebcdicChar = 0xFFFF & tableToEbcdic[i];
      } else {
        int piece = 0xFFFF & tableToEbcdic[i / 2];
        if (i % 2 == 0) {
          ebcdicChar = piece >> 8;
        } else {
          ebcdicChar = piece & 0xFF;
        }
      }

      if ((ebcdicChar != 0xfefe) && (ebcdicChar != 0x3f)) {
        int unicodeChar = 0xFFFF & tableToUnicode[ebcdicChar];
        if (i != unicodeChar) {
          if (unicodeChar == 0xFFFD) {
            sb1.append("Unicode RoundTrip Failure UX'" + Integer.toHexString(i)
                + "'" + " -> " + ebcdicPrefix + "'"
                + Integer.toHexString(ebcdicChar) + "'" + " -> UX'"
                + Integer.toHexString(unicodeChar) + "'\n");
            passed = false;
          } else {
            int ebcdicChar2;
            if (ebcdicIsDBCS) {
              ebcdicChar2 = 0xFFFF & tableToEbcdic[unicodeChar];
            } else {
              int piece = 0xFFFF & tableToEbcdic[unicodeChar / 2];
              if (unicodeChar % 2 == 0) {
                ebcdicChar2 = piece >> 8;
              } else {
                ebcdicChar2 = piece & 0xFF;
              }
            }

            if (ebcdicChar2 == ebcdicChar) {
              sb2.append("Secondary Unicode mapping UX'"
                  + Integer.toHexString(i) + "'" + " -> " + ebcdicPrefix + "'"
                  + Integer.toHexString(ebcdicChar) + "'" + " -> UX'"
                  + Integer.toHexString(unicodeChar) + "'" + " -> "
                  + ebcdicPrefix + "'" + Integer.toHexString(ebcdicChar2)
                  + "'\n");

            } else {
              sb3.append("Unicode RoundTrip Failure2 UX'"
                  + Integer.toHexString(i) + "'" + " -> " + ebcdicPrefix + "'"
                  + Integer.toHexString(ebcdicChar) + "'" + " -> UX'"
                  + Integer.toHexString(unicodeChar) + "'" + " -> "
                  + ebcdicPrefix + "'" + Integer.toHexString(ebcdicChar2)
                  + "'\n");
              passed = false;

            }

          }
        }
      }
    }
    System.out.println(sb2);
    System.out.println(sb1);
    System.out.println(sb3);

    return passed;

  }

  private static final char repSig = '\uFFFF'; // compression indication
                                               // character
  private static final char cic_ = repSig;

  private static final char rampSig = '\uFFFE'; // ramp indication character
  private static final char ric_ = rampSig;

  private static final char hbSig = '\u0000'; // high-byte compression
                                              // indication character
  private static final char pad = '\u0000'; // pad character

  static int repeatCheck(char[] arr, int startingIndex) {
    int index = startingIndex + 1;
    while (index < arr.length && arr[index] == arr[index - 1]) {
      ++index;
    }
    return (index - startingIndex);
  }

  static final int rampCheck(char[] arr, int startingIndex) {
    int index = startingIndex + 1;
    while (index < arr.length && arr[index] == arr[index - 1] + 1) {
      ++index;
    }
    return (index - startingIndex);
  }

  static int hbCheck(char[] arr, int startingIndex) {
    int index = startingIndex + 1;
    while (index < arr.length) {
      // check for repeat
      // for 6 repeated chars, we'd need either 3 hb-compressed chars or 3
      // repeatsig chars, so it's a toss up
      if (repeatCheck(arr, index) > 6)
        return (index - startingIndex); // at this point though, it's better to
                                        // stop and do the repeat

      // check for ramp, same reason
      if (rampCheck(arr, index) > 6)
        return (index - startingIndex);

      // OK, finally check for hb
      if ((arr[index] & 0xFF00) != (arr[index - 1] & 0xFF00))
        return (index - startingIndex);

      ++index;
    }
    return (index - startingIndex);
  }

  static int numRepeats;
  static int numRamps;
  static int hbRepeats;
  static int charRepeats;

  // This is the new way - 05/04/2000.
  static char[] compressBetter(char[] arr) {
    numRepeats = 0;
    numRamps = 0;
    hbRepeats = 0;
    charRepeats = 0;

    // This uses the "correct" compression scheme from my invention disclosure
    // It also employs high-byte compression, something that I did not include
    // in my disclosure.
    StringBuffer buf = new StringBuffer();

    for (int i = 0; i < arr.length; ++i) {
      int repNum = repeatCheck(arr, i);
      if (repNum > 3) // had enough repeats
      {
        numRepeats++;
        buf.append(repSig);
        buf.append((char) repNum);
        buf.append(arr[i]);
        i += repNum - 1;
      } else {
        int rampNum = rampCheck(arr, i);
        if (rampNum > 3) // had enough in the ramp
        {
          numRamps++;
          buf.append(rampSig);
          buf.append((char) rampNum);
          buf.append(arr[i]);
          i += rampNum - 1;
        } else {
          int hbNum = hbCheck(arr, i);
          --hbNum; // don't include the first char, since we always append it.
          if (hbNum >= 6) {
            // System.out.print("HBNUM is "+Integer.toHexString((int)hbNum)+"; ");
            hbRepeats++;
            // pattern is this: ss ss nn nn hh tt xx xx xx xx ...
            // where ss ss is hbSig
            // nn nn is hbNum
            // hh tt is the first char (hh is the repeated high byte)
            // xx is the lower byte of the next char in the sequence
            // xx repeats hbNum/2 times so that
            // hbNum is the total number of repeated db chars in the ciphertext,
            // not including the first char.
            // Note that there may be, in actuality, hbNum*2 +1 chars in the
            // cleartext that fit into the
            // conversion, but since we'd have to fill out the last char with an
            // empty byte, there's no point
            // in doing it anyway. Besides, it might be able to be compressed
            // via another scheme with itself as
            // the starting character.
            // int start = i;
            buf.append(hbSig);
            if (hbNum % 2 == 1) // odd number
            {
              --hbNum; // no point in doing the last char
            }
            // System.out.println("Appending "+Integer.toHexString((int)((char)(hbNum/2))));
            buf.append((char) (hbNum / 2)); // hbNum is always even, so this
                                            // comes out.
            // System.out.print("hb comp: "+Integer.toHexString(hbNum)+": ");
            // for (int b=0; b>> 8));
            char c2 = (char) (highByteMask + (0x00FF & both));
            buf.append(c1);
            buf.append(c2);
            // System.out.print(Integer.toHexString((int)c1)+" "+Integer.toHexString((int)c2)+" ");
          }
          // System.out.println(Integer.toHexString((int)arr[i+hbNum]));
          i = i + hbNum - 1;
        }
      } else {
        buf.append(arr[i]);
        charRepeats++;
      }
    }
    System.out.println("Decompression stats: " + numRepeats + " repeats, "
        + numRamps + " ramps, " + hbRepeats + " highbytes, " + charRepeats
        + " regular.");
    numRepeats = 0;
    numRamps = 0;
    hbRepeats = 0;
    charRepeats = 0;
    return buf.toString().toCharArray();
  }

  // This is the old way
  static char[] compress(char[] arr) {

    if (arr.length < 3)
      return arr;
    StringBuffer buf = new StringBuffer();
    char oldold = arr[0];
    char old = arr[1];
    int count = 0;
    boolean inCompression = false; // this flags if we are repeating the same
                                   // character
    boolean inRamp = false; // this flags if each subsequent characters is the
                            // previous character + 1

    for (int i = 2; i < arr.length; ++i) {
      if (!inCompression && !inRamp) {
        if (arr[i] == old && arr[i] == oldold) // Check for repeating
        {
          inCompression = true;
          buf.append(cic_);
          buf.append(oldold);
          count = 3;
        } else if (arr[i] == old + 1 && arr[i] == oldold + 2) // Check for ramp
        {
          inRamp = true;
          buf.append(ric_);
          buf.append(oldold);
        } else if (oldold == cic_) // Check for duplicate cic
        {
          buf.append(cic_);
        } else if (oldold == ric_) // Check for duplicate ric
        {
          buf.append(ric_);
        } else // Just copy it normal
        {
          buf.append(oldold);
        }
        oldold = old;
        old = arr[i];
      } else if (inCompression) {
        if (arr[i] == old && arr[i] == oldold) // Still repeating?
        {
          ++count;
          oldold = old;
          old = arr[i];
        } else // Not repeating anymore
        {
          inCompression = false;
          char c = (char) count;
          if (count == 0x0008)
            c = '\b';
          else if (count == 0x0009)
            c = '\t';
          // else if(count == 0x000A) c = '\r'; // Had trouble with 0x0A and
          // 0x0D getting messed up.
          else if (count == 0x000A)
            c = '\n'; // Had trouble with 0x0A and 0x0D getting messed up.
          else if (count == 0x000C)
            c = '\f';
          // else if(count == 0x000D) c = '\n';
          else if (count == 0x000D)
            c = '\r';
          else if (count == 0x0022)
            c = '\"';
          else if (count == 0x0027)
            c = '\'';
          else if (count == 0x005C)
            c = '\\';
          buf.append(c);
          oldold = arr[i++]; // yes this is right... think about it.
          old = arr[i];
        }
      } else { // must be in ramp
        if (arr[i] == old + 1 && arr[i] == oldold + 2) {
          oldold = old;
          old = arr[i];
        } else {
          inRamp = false;
          buf.append(old);
          oldold = arr[i++];
          old = arr[i];
        }
      }
    }
    if (inCompression) {
      char c = (char) count;
      if (count == 0x0008)
        c = '\b';
      else if (count == 0x0009)
        c = '\t';
      else if (count == 0x000A)
        c = '\n';
      else if (count == 0x000C)
        c = '\f';
      // else if(count == 0x000D) c = '\n';
      else if (count == 0x000D)
        c = '\r';
      else if (count == 0x0022)
        c = '\"';
      else if (count == 0x0027)
        c = '\'';
      else if (count == 0x005C)
        c = '\\';
      buf.append(c);
    }
    if (inRamp) {
      buf.append(old);
    }

    return buf.toString().toCharArray();
  }

  static void writeHeader(FileWriter f, int ccsid, String system)
      throws Exception {
    int dotIndex = system.indexOf('.');
    if (dotIndex > 0) {
      system = system.substring(0, dotIndex);
    }
    Date currentDate = new Date();
    // Look up the version dynamically
    Class copyrightClass = Copyright.class;
    Field field = copyrightClass.getField("version");
    String jtopenVersion = (String) field.get(null);

    f.write("///////////////////////////////////////////////////////////////////////////////\n");
    f.write("//\n");
    f.write("// JTOpen (IBM Toolbox for Java - OSS version)\n");
    f.write("//\n");
    f.write("// Filename:  ConvTable" + ccsid + ".java\n");
    f.write("//\n");
    f.write("// The source code contained herein is licensed under the IBM Public License\n");
    f.write("// Version 1.0, which has been approved by the Open Source Initiative.\n");
    f.write("// Copyright (C) 1997-2016 International Business Machines Corporation and\n");
    f.write("// others.  All rights reserved.\n");
    f.write("//\n");
    f.write("// Generated " + currentDate + " from " + system + "\n");
  
    StringBuffer sb = new StringBuffer(); 
    if (compress_ == false) sb.append(" -nocompress");
    if (ascii_ == true) sb.append(" -ascii");
    if (bidi_ == true) sb.append(" -bidi");
    if (showOffsets_ == true) sb.append(" -showOffsets");
    if (codePointPerLine_ == true) sb.append(" -codePointPerLine");
    if (useJdbc_ == true)  sb.append(" -useJdbc"); 
    if (sb.length() > 0 ) {
    f.write("// Generation Options:"+sb.toString()+"\n");  
    }
    f.write("// Using " + jtopenVersion + "\n");
    f.write("///////////////////////////////////////////////////////////////////////////////\n\n");
    f.write("package com.ibm.as400.access;\n\n");
  }

  
  
  
  
  //
  // Methods for using JDBC to handle the translation tables
  // 
  
 static boolean jdbcIsDBCS(Connection connection, int ccsid) throws SQLException {
    boolean isDBCS;
    
    Statement stmt = connection.createStatement(); 
    try { 
      stmt.executeUpdate("CREATE TABLE QTEMP.GENERATE"+ccsid+"(C1 VARCHAR(80) CCSID "+ccsid+")"); 
      isDBCS = false; 
    } catch (SQLException sqlex) { 
      int sqlcode = sqlex.getErrorCode(); 
      if (sqlcode == -189) {   // CCSID not valid 
        stmt.executeUpdate("CREATE TABLE QTEMP.GENERATE"+ccsid+"(C1 VARGRAPHIC(80) CCSID "+ccsid+")"); 
        isDBCS = true; 
      } else {
        throw sqlex; 
      }
    }
    
    
    stmt.close(); 
    return isDBCS; 
    
 }
 
 
 private static char[] jdbcToEbcdic(Connection connection, int ccsid) throws Exception  {
   if (ccsid > 1000000) {
     ccsid = ccsid - 1000000; 
   }
   PreparedStatement ps = connection.prepareStatement("select cast(CAST(CAST(? AS DBCLOB(1M) CCSID 1200) AS CLOB(1M) CCSID "+ccsid+") as BLOB(1M)) from sysibm.sysdummy1");
   
   char[] allChar65536 = new char[65536]; 
   for (int i = 0; i < 0xD800; i++) {
     allChar65536[i] = (char) i; 
   }
   // Fill in the surrogate range with substitution characters
   for (int i = 0xD800; i < 0xF900; i++) {
     allChar65536[i] = '\u001a'; 
   }
   for (int i = 0xF900; i < 0x10000; i++) {
     allChar65536[i] = (char) i; 
   }
   
   Clob clob = ((AS400JDBCConnection)connection).createClob(); 
   clob.setString(1, new String(allChar65536)); 
   ps.setClob(1, clob); 
   ResultSet rs = ps.executeQuery(); 
   rs.next(); 
   byte[] byteAnswer = rs.getBytes(1); 
   if (byteAnswer.length != 65536) {
     // We must have shift.out shift in combination that we don't use for single byte conversions
     // Remove them. 
     byteAnswer = removeDoubleByteEbcdic(byteAnswer); 
   }
   rs.close(); 
   ps.close(); 
   
   char[] answer = new char[byteAnswer.length / 2]; 
   
   for (int i = 0; i < byteAnswer.length; i+= 2) {
     answer[i/2] = (char) ((byteAnswer[i] << 8) | (0xFF & byteAnswer[i+1])); 
   }
   
   return answer; 
   
   
   
 }

 private static byte[] removeDoubleByteEbcdic(byte[] inBytes) throws Exception {
   byte[] outBytes = new byte[65536]; 
   
   // Just copy the control characters
   for (int i = 0; i < 0x40; i++)  {
       outBytes[i] = inBytes[i]; 
   }
   int toIndex = 0x40; 
   boolean singleByte = true; 
   for (int i = 0x40; i < inBytes.length; i++) { 
     byte b = inBytes[i]; 
     if (singleByte) {
       if (b == 0x0E) {
         singleByte = false; 
       } else if (b == 0x0F) {
         throw new Exception("Illegal 0x0f found in singleByte mode"); 
       } else {
         outBytes[toIndex]=b; 
         toIndex++; 
       }
     } else {
       if (b == 0x0F) {
         singleByte = true; 
       } else if (b == 0x0e) {
         throw new Exception("Illegal 0x0e found in doubleByte mode"); 
       } else {
         outBytes[toIndex]=0x3f;
         toIndex++; 
         i++;   // Skip extra DB charater
       }
     }
     
   }
   if (toIndex != 65536) {
     throw new Exception("To index is "+toIndex+" should be 65536"); 
   }
   
   return outBytes; 
   
}

 
 private static char[] removeSingleByteEbcdic(byte[] inBytes, boolean mixedCcsid) throws Exception {
   char[] outChars = new char[65536]; 

   boolean singleByte = false; 
   if (mixedCcsid) singleByte = true; 
   int toIndex = 0x0; 
   for (int i = 0x0; i < inBytes.length; i++) { 
     byte b = inBytes[i]; 
     if (singleByte) {
       if (b == 0x0E) {
         singleByte = false; 
       } else if (b == 0x0F) {
         throw new Exception("Illegal 0x0f found in singleByte mode"); 
       } else {
         outChars[toIndex]='\uFEFE'; 
         toIndex++; 
       }
     } else {
       if (b == 0x0F && mixedCcsid ) {
         singleByte = true; 
       } else if (b == 0x0e  && mixedCcsid ) {
         throw new Exception("Illegal 0x0e found in doubleByte mode"); 
       } else {
         outChars[toIndex]= (char)((b << 8) | (0xFF & inBytes[i+1])); 
         toIndex++; 
         i++; 
       }
     }
     
   }
   if (toIndex != 65536) {
     throw new Exception("To index is "+toIndex+" should be 65536"); 
   }
   
   return outChars; 
   
}
 
 
 
private static char[] jdbcToUnicode(Connection connection, int ccsid) throws SQLException  {
   
   if (ccsid > 1000000) {
     ccsid = ccsid - 1000000; 
   }
   PreparedStatement ps = connection.prepareStatement("select cast(CAST(CAST(? AS VARCHAR(256) FOR BIT DATA) AS VARCHAR(256) CCSID "+ccsid+") as VARGRAPHIC(256) CCSID 1200) from sysibm.sysdummy1");
   byte[] all256 = new byte[256]; 
   for (int i = 0; i < 256; i++) {
     // For 0x0E and 0x0F, they do not translate correctly for mixed CCSID
     // We will fix them up below. 
     if (i == 0x0E) {
       all256[i] = (byte) 0x3F; 
     } else if (i == 0x0F) { 
       all256[i] = (byte) 0x3F; 
     } else {
     all256[i] = (byte) i; 
     }  
   }
   ps.setBytes(1,all256); 
   ResultSet rs = ps.executeQuery(); 
   rs.next();
   String answer = rs.getString(1); 
   rs.close();
   ps.close(); 
   char[] charAnswer = answer.toCharArray(); 
   // Fix 0x0e/ 0x0F
   charAnswer[0x0e]='\u000e';
   charAnswer[0x0f]='\u000f';
   return charAnswer; 
   
 }

 private static char[] jdbcToEbcdicDBCS(Connection connection, int ccsid) throws Exception  {
  
   boolean mixedCcsid = false; 
   if (ccsid > 2000000) {
     ccsid = ccsid - 2000000; 
     mixedCcsid = true; 
   }
   PreparedStatement ps = connection.prepareStatement("select cast(CAST(CAST(? AS DBCLOB(1M) CCSID 1200) AS CLOB(1M) CCSID "+ccsid+") as BLOB(1M)) from sysibm.sysdummy1");
   
   char[] allChar65536 = new char[65536]; 
   for (int i = 0; i < 0xD800; i++) {
     if (i <= 0x80) { 
       allChar65536[i] = (char) '\u001a';
     } else { 
       allChar65536[i] = (char) i;
     }
   }
   // Fill in the surrogate range with double width substitution characters
   for (int i = 0xD800; i < 0xE000; i++) {
     allChar65536[i] = '\ufffd'; 
   }
   for (int i = 0xE000; i < 0x10000; i++) {
     // Don't translate Variation selectors 
     if (i >= 0xFE00 && i <= 0xFF0F) {
       allChar65536[i] = '\ufffd'; 
     } else { 
        allChar65536[i] = (char) i;
     }
   }
   
   Clob clob = ((AS400JDBCConnection)connection).createClob(); 
   clob.setString(1, new String(allChar65536)); 
   ps.setClob(1, clob); 
   ResultSet rs = ps.executeQuery(); 
   rs.next(); 
   byte[] byteAnswer = rs.getBytes(1); 
  
   rs.close(); 
   ps.close(); 
   
   char[] answer;  
   
   
   if (byteAnswer.length != 2 * 65536) {
     // We must have shift.out shift.in combinations. 
     // We need to remove the single bytes.
     
     answer = removeSingleByteEbcdic(byteAnswer, mixedCcsid); 
   } else { 
     answer = new char[byteAnswer.length / 2];
   for (int i = 0; i < byteAnswer.length; i+= 2) {
     answer[i/2] = (char) ((byteAnswer[i] << 8) | (0xFF & byteAnswer[i+1])); 
   }
   }
   return answer; 
   
   
   
   
 }

 private static char[] jdbcToUnicodeDBCS(Connection connection, int ccsid) throws SQLException  {

   // The database doesn't allow CLOB CCSID 65535 or BLOB to be converted to CLOB / DBCLOB.  We'll need to process this in pieces. 
   // The database doesn't allow VARCHAR CCSID 65535 to be convert to GRAPHIC.   Need to figure out how to handle that also. 
   
   // Create a huge SQL statement with the necessary literals.  
   // For this to work, the job must be changed to the specified CCSID if mixed
   // or the associated CCSID if no double byte
   
   boolean mixed = false;
   // NOTE: This doesn't work... 
   String sql = "select cast(CAST(CAST(? AS CLOB(1M) FOR BIT DATA ) AS DBCLOB(1M) CCSID "+ccsid+") as DBCLOB(1M) CCSID 1200) from sysibm.sysdummy1";
   if (ccsid > 2000000) {
     ccsid = ccsid - 2000000; 
     mixed = true; 
     sql  = "select cast(CAST(CAST(? AS VARCHAR(16390) FOR BIT DATA) AS VARCHAR(16390) CCSID "+ccsid+") as VARGRAPHIC(8200) CCSID 1200) from sysibm.sysdummy1";
   }
   PreparedStatement ps = connection.prepareStatement(sql);
   
   // Start at 0x00 and handle 8192 double byte characters at at time
   // int BLOCKSIZE = 8192;
   int BLOCKSIZE = 32;
   int OUTERLOOP = 65536 / BLOCKSIZE; 
   byte[] piece8192; 
   if (mixed) {
     piece8192 = new byte[BLOCKSIZE * 2 + 2];
   } else {
     piece8192 = new byte[BLOCKSIZE * 2];
   }
   int offset = 0; 
   if (mixed) {
     piece8192[0]=0x0E;
     piece8192[BLOCKSIZE * 2 + 1]=0x0F; 
     offset = 1; 
   }
   
   StringBuffer sb = new StringBuffer(); 
   int cp; 
   for (int i = 0; i < OUTERLOOP; i++) {
     for (int j = 0; j < BLOCKSIZE; j++) {
    cp = i * BLOCKSIZE  + j ; 
      // Filter out all the 0E/0F for mixed CCSIDS
      // Filter out the lower 0x100 for mixed CCSIDS 
    
    
      if (mixed) {
        if (cp <0x100  || (cp / 256) == 0x0E || (cp / 256) == 0x0F ||  (cp % 256) == 0x0E || (cp % 256) == 0x0F) {
          cp = 0xFEFE;
        }
      }
      piece8192[offset+2*j] = (byte) (cp / 256);
      piece8192[offset+2*j+1] = (byte) cp ;
    } /* for j */ 
     if (offset == 1) { 
       piece8192[BLOCKSIZE * 2 + 1]=0x0F;
     }
    
     ps.setBytes(1,piece8192); 
     ResultSet rs = ps.executeQuery(); 
     rs.next();
     String answer = rs.getString(1);
     if (answer == null) { 
       System.out.println("ERROR: got null processing block "+i+" of size "+BLOCKSIZE);
       System.out.println("INPUT BYTES: = "+dumpBytes(" ", piece8192)); 
     } else if (answer.length() != BLOCKSIZE ) {
       // This is OK since there may be surrogate pairs which will be handled later
       // 
       // System.out.println("ERROR: got size = "+answer.length()+" processing block "+i+" of size "+BLOCKSIZE);
       // System.out.println("INPUT BYTES: = "+dumpBytes(" ",piece8192)); 
       // System.out.println("OUTPUT STRING: =  "+dumpUnicodeString("  ", answer)); 

     }
     sb.append(answer); 
     rs.close();

     
     
   } /* for i*/ 
   ps.close(); 
   
   return sb.toString().toCharArray(); 
 } /* jdbcToUnicodeDBCS */

private static String dumpUnicodeString(String pad , String data) {
  StringBuffer sb = new StringBuffer(); 
  char[] charArray = data.toCharArray(); 
  for (int i = 0; i < charArray.length; i++) { 
    sb.append(pad); 
    int value = 0xffff & charArray[i]; 
    if (value < 0x10) sb.append("0"); 
    if (value < 0x100) sb.append("0"); 
    if (value < 0x1000) sb.append("0"); 
    sb.append(Integer.toHexString(value)); 
  }
  
  
  return sb.toString(); 
  
}

private static String dumpBytes(String pad, byte[] block) {
   StringBuffer sb = new StringBuffer(); 
   for (int i = 0; i < block.length; i++) { 
     sb.append(pad); 
     int value = 0xff & block[i]; 
     if (value < 0x10) sb.append("0"); 
     sb.append(Integer.toHexString(value)); 
   }
   

   return sb.toString(); 
}

  
  
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy