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