src.com.ibm.as400.access.BinaryConverter Maven / Gradle / Ivy
///////////////////////////////////////////////////////////////////////////////
//
// JTOpen (IBM Toolbox for Java - OSS version)
//
// Filename: BinaryConverter.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-2003 International Business Machines Corporation and
// others. All rights reserved.
//
///////////////////////////////////////////////////////////////////////////////
package com.ibm.as400.access;
/**
A binary types converter between Java byte arrays and Java simple types.
Note: Some methods of this class accept an array-valued argument in addition to arguments that specify an array offset and possibly also a length.
All such methods will throw an {@link ArrayIndexOutOfBoundsException ArrayIndexOutOfBoundsException} if the offset and/or length are not valid for the array.
**/
public class BinaryConverter
{
// Don't allow instances of this class.
private BinaryConverter()
{
}
/**
Convert the specified short into IBM i format in the specified byte array.
The converted value will occupy 2 bytes.
@param shortValue The value to be converted to IBM i format.
@param serverValue The array to receive the data type in IBM i format.
@param offset The offset into the byte array for the start of the IBM i value.
**/
public static void shortToByteArray(short shortValue, byte[] serverValue, int offset)
{
if (serverValue == null) throw new NullPointerException("serverValue");
serverValue[offset] = (byte)(shortValue >>> 8);
serverValue[offset+1] = (byte) shortValue;
}
/**
Convert the specified short into IBM i format in a byte array.
@param shortValue The value to be converted to IBM i format.
@return An array (of length 2) with the data type in IBM i format.
**/
public static byte[] shortToByteArray(short shortValue)
{
byte[] serverValue = new byte[2];
shortToByteArray(shortValue, serverValue, 0);
return serverValue;
}
/**
Convert the specified IBM i data type to a short.
@param serverValue The array containing the data type in IBM i format.
@param offset The offset into the byte array for the start of the IBM i value.
@return a short corresponding to the data type.
**/
public static short byteArrayToShort(byte[] serverValue, int offset)
{
checkArgs(serverValue, offset);
return(short)(((serverValue[offset] & 0xFF) << 8) +
(serverValue[offset+1] & 0xFF));
}
/**
Convert the specified int into IBM i format in the specified byte array.
The converted value will occupy 4 bytes.
@param intValue The value to be converted to IBM i format.
@param serverValue The array to receive the data type in IBM i format.
@param offset The offset into the byte array for the start of the IBM i value.
**/
public static void intToByteArray(int intValue, byte[] serverValue, int offset)
{
checkArgs(serverValue, offset);
serverValue[offset] = (byte)(intValue >>> 24);
serverValue[offset+1] = (byte)(intValue >>> 16);
serverValue[offset+2] = (byte)(intValue >>> 8);
serverValue[offset+3] = (byte) intValue;
}
/**
Convert the specified int into IBM i format in a byte array.
@param intValue The value to be converted to IBM i format.
@return An array (of length 4) with the data type in IBM i format.
**/
public static byte[] intToByteArray(int intValue)
{
byte[] serverValue = new byte[4];
intToByteArray(intValue, serverValue, 0);
return serverValue;
}
/**
Convert the specified IBM i data type to an int.
@param serverValue The array containing the data type in IBM i format.
@param offset The offset into the byte array for the start of the IBM i value.
@return an int corresponding to the data type.
**/
public static int byteArrayToInt(byte[] serverValue, int offset)
{
checkArgs(serverValue, offset);
return((serverValue[offset] & 0xFF) << 24) +
((serverValue[offset+1] & 0xFF) << 16) +
((serverValue[offset+2] & 0xFF) << 8) +
(serverValue[offset+3] & 0xFF);
}
/**
Convert the specified float into IBM i format in the specified byte array.
The converted value will occupy 4 bytes.
@param floatValue The value to be converted to IBM i format.
@param serverValue The array to receive the data type in IBM i format.
@param offset The offset into the byte array for the start of the IBM i value.
**/
public static void floatToByteArray(float floatValue, byte[] serverValue, int offset)
{
checkArgs(serverValue, offset);
int bits = Float.floatToIntBits(floatValue);
serverValue[offset] = (byte)(bits >>> 24);
serverValue[offset+1] = (byte)(bits >>> 16);
serverValue[offset+2] = (byte)(bits >>> 8);
serverValue[offset+3] = (byte) bits;
}
/**
Convert the specified float into IBM i format in a byte array.
@param floatValue The value to be converted to IBM i format.
@return An array (of length 4) with the data type in IBM i format.
@deprecated Use floatToByteArray(float) instead.
**/
public static byte[] floatToByteArray(int floatValue)
{
byte[] serverValue = new byte[4];
floatToByteArray(floatValue, serverValue, 0);
return serverValue;
}
/**
Convert the specified float into IBM i format in a byte array.
@param floatValue The value to be converted to IBM i format.
@return An array (of length 4) with the data type in IBM i format.
**/
public static byte[] floatToByteArray(float floatValue)
{
byte[] serverValue = new byte[4];
floatToByteArray(floatValue, serverValue, 0);
return serverValue;
}
/**
Convert the specified IBM i data type to a float.
@param serverValue The array containing the data type in IBM i format.
@param offset The offset into the byte array for the start of the IBM i value.
@return a float corresponding to the data type.
**/
public static float byteArrayToFloat(byte[] serverValue, int offset)
{
checkArgs(serverValue, offset);
int bits = ((serverValue[offset] & 0xFF) << 24) +
((serverValue[offset+1] & 0xFF) << 16) +
((serverValue[offset+2] & 0xFF) << 8) +
(serverValue[offset+3] & 0xFF);
return Float.intBitsToFloat(bits);
}
/**
Convert the specified double into IBM i format in the specified byte array.
The converted value will occupy 8 bytes.
@param doubleValue The value to be converted to IBM i format.
@param serverValue The array to receive the data type in IBM i format.
@param offset The offset into the byte array for the start of the IBM i value.
**/
public static void doubleToByteArray(double doubleValue, byte[] serverValue, int offset)
{
checkArgs(serverValue, offset);
long bits = Double.doubleToLongBits(doubleValue);
// Do in two parts to avoid long temps.
int high = (int)(bits >>> 32);
int low = (int)bits;
serverValue[offset] = (byte)(high >>> 24);
serverValue[offset+1] = (byte)(high >>> 16);
serverValue[offset+2] = (byte)(high >>> 8);
serverValue[offset+3] = (byte) high;
serverValue[offset+4] = (byte)(low >>> 24);
serverValue[offset+5] = (byte)(low >>> 16);
serverValue[offset+6] = (byte)(low >>> 8);
serverValue[offset+7] = (byte) low;
}
/**
Convert the specified double into IBM i format in a byte array.
@param doubleValue The value to be converted to IBM i format.
@return An array (of length 8) with the data type in IBM i format.
**/
public static byte[] doubleToByteArray(double doubleValue)
{
byte[] serverValue = new byte[8];
doubleToByteArray(doubleValue, serverValue, 0);
return serverValue;
}
/**
Convert the specified IBM i data type to a double.
@param serverValue The array containing the data type in IBM i format.
@param offset The offset into the byte array for the start of the IBM i value.
@return a double corresponding to the data type.
**/
public static double byteArrayToDouble(byte[] serverValue, int offset)
{
checkArgs(serverValue, offset);
// Do in two parts to avoid long temps.
int firstPart = ((serverValue[offset] & 0xFF) << 24) +
((serverValue[offset+1] & 0xFF) << 16) +
((serverValue[offset+2] & 0xFF) << 8) +
(serverValue[offset+3] & 0xFF);
int secondPart = ((serverValue[offset+4] & 0xFF) << 24) +
((serverValue[offset+5] & 0xFF) << 16) +
((serverValue[offset+6] & 0xFF) << 8) +
(serverValue[offset+7] & 0xFF);
long bits = ((long)firstPart << 32) + (secondPart & 0xFFFFFFFFL);
return Double.longBitsToDouble(bits);
}
/**
Convert the specified int into IBM i format in the specified byte array.
The converted value will occupy 2 bytes.
@param intValue The value to be converted to IBM i format. The integer should be greater than or equal to zero and representable in two bytes.
@param serverValue The array to receive the data type in IBM i format.
@param offset The offset into the byte array for the start of the IBM i value.
**/
public static void unsignedShortToByteArray(int intValue, byte[] serverValue, int offset)
{
checkArgs(serverValue, offset);
serverValue[offset] = (byte)(intValue >>> 8);
serverValue[offset+1] = (byte) intValue;
}
/**
Convert the specified int into IBM i format in a byte array.
@param intValue The value to be converted to IBM i format. The integer should be greater than or equal to zero and representable in two bytes.
@return An array (of length 2) with the data type in IBM i format.
**/
public static byte[] unsignedShortToByteArray(int intValue)
{
byte[] serverValue = new byte[2];
unsignedShortToByteArray(intValue, serverValue, 0);
return serverValue;
}
/**
Convert the specified IBM i data type to an int.
@param serverValue The array containing the data type in IBM i format.
@param offset The offset into the byte array for the start of the IBM i value.
@return an int corresponding to the data type, derived from the 2 bytes of data starting at the specified offset.
**/
public static int byteArrayToUnsignedShort(byte[] serverValue, int offset)
{
checkArgs(serverValue, offset);
return((serverValue[offset] & 0xFF) << 8) +
(serverValue[offset+1] & 0xFF);
}
/**
Convert the specified long into IBM i format in the specified byte array.
The converted value will occupy 4 bytes.
@param longValue The value to be converted to IBM i format. The long should be greater than or equal to zero and representable in four bytes.
@param serverValue The array to receive the data type in IBM i format.
@param offset The offset into the byte array for the start of the IBM i value.
**/
public static void unsignedIntToByteArray(long longValue, byte[] serverValue, int offset)
{
checkArgs(serverValue, offset);
// Grab bits into int to avoid long temps.
int bits = (int)longValue;
serverValue[offset] = (byte)(bits >>> 24);
serverValue[offset+1] = (byte)(bits >>> 16);
serverValue[offset+2] = (byte)(bits >>> 8);
serverValue[offset+3] = (byte) bits;
}
/**
Convert the specified long into IBM i format in a byte array.
@param longValue The value to be converted to IBM i format. The long should be greater than or equal to zero and representable in four bytes.
@return An array (of length 4) with the data type in IBM i format.
**/
public static byte[] unsignedIntToByteArray(long longValue)
{
byte[] serverValue = new byte[4];
unsignedIntToByteArray(longValue, serverValue, 0);
return serverValue;
}
/**
Convert the specified IBM i data type to a long.
@param serverValue The array containing the data type in IBM i format.
@param offset The offset into the byte array for the start of the IBM i value.
@return a long corresponding to the data type.
**/
public static long byteArrayToUnsignedInt(byte[] serverValue, int offset)
{
checkArgs(serverValue, offset);
return(((serverValue[offset] & 0xFF) << 24) +
((serverValue[offset+1] & 0xFF) << 16) +
((serverValue[offset+2] & 0xFF) << 8) +
(serverValue[offset+3] & 0xFF)) & 0xFFFFFFFFL;
}
/**
Convert the specified long into IBM i format in the specified byte array.
The converted value will occupy 8 bytes.
@param longValue The value to be converted to IBM i format.
@param serverValue The array to receive the data type in IBM i format.
@param offset The offset into the byte array for the start of the IBM i value.
**/
public static void longToByteArray(long longValue, byte[] serverValue, int offset)
{
checkArgs(serverValue, offset);
// Do in two parts to avoid long temps.
int high = (int)(longValue >>> 32);
int low = (int)longValue;
serverValue[offset] = (byte)(high >>> 24);
serverValue[offset+1] = (byte)(high >>> 16);
serverValue[offset+2] = (byte)(high >>> 8);
serverValue[offset+3] = (byte) high;
serverValue[offset+4] = (byte)(low >>> 24);
serverValue[offset+5] = (byte)(low >>> 16);
serverValue[offset+6] = (byte)(low >>> 8);
serverValue[offset+7] = (byte) low;
}
/**
Convert the specified long into IBM i format in a byte array.
@param longValue The value to be converted to IBM i format.
@return An array (of length 8) with the data type in IBM i format.
**/
public static byte[] longToByteArray(long longValue)
{
byte[] serverValue = new byte[8];
longToByteArray(longValue, serverValue, 0);
return serverValue;
}
/**
Convert the specified IBM i data type to a long.
@param serverValue The array containing the data type in IBM i format.
@param offset The offset into the byte array for the start of the IBM i value.
@return a long corresponding to the data type.
**/
public static long byteArrayToLong(byte[] serverValue, int offset)
{
checkArgs(serverValue, offset);
// Do in two parts to avoid long temps.
int firstPart = ((serverValue[offset] & 0xFF) << 24) +
((serverValue[offset+1] & 0xFF) << 16) +
((serverValue[offset+2] & 0xFF) << 8) +
(serverValue[offset+3] & 0xFF);
int secondPart = ((serverValue[offset+4] & 0xFF) << 24) +
((serverValue[offset+5] & 0xFF) << 16) +
((serverValue[offset+6] & 0xFF) << 8) +
(serverValue[offset+7] & 0xFF);
return((long)firstPart << 32) + (secondPart & 0xFFFFFFFFL);
}
static byte[] charArrayToByteArray(char[] charValue)
{
if (charValue == null) return null;
byte[] byteValue = new byte[charValue.length * 2];
int inPos = 0;
int outPos = 0;
while (inPos < charValue.length)
{
byteValue[outPos++] = (byte)(charValue[inPos] >> 8);
byteValue[outPos++] = (byte)charValue[inPos++];
}
return byteValue;
}
static char[] byteArrayToCharArray(byte[] byteValue)
{
if (byteValue == null) return null;
char[] charValue = new char[byteValue.length / 2];
int inPos = 0;
int outPos = 0;
while (inPos < byteValue.length)
{
charValue[outPos++] = (char)(((byteValue[inPos++] & 0xFF) << 8) + (byteValue[inPos++] & 0xFF));
}
return charValue;
}
// Constant used in bytesToHexString()
private static final char[] c_ = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F'};
static final char hiNibbleToChar(byte b)
{
return c_[(b >>> 4) & 0x0F];
}
static final char loNibbleToChar(byte b)
{
return c_[b & 0x0F];
}
/**
Convert the specified byte array to its hexadecimal String representation.
Example generated string: "010203047E7F".
@param b The array containing the data.
@return A String containing the hex characters that represent the byte data.
@deprecated Replaced by bytesToHexString(). (Same implementation, better name.)
**/
public static final String bytesToString(final byte[] b)
{
if (b == null) throw new NullPointerException("b");
return bytesToHexString(b, 0, b.length);
}
/**
Convert the specified byte array to its hexadecimal String representation.
Example generated string: "010203047E7F".
@param b The array containing the data.
@param offset The offset into the array at which to begin reading bytes.
@param length The number of bytes to read out of the array.
@return A String containing the hex characters that represent the byte data.
@deprecated Replaced by bytesToHexString(). (Same implementation, better name.)
**/
public static final String bytesToString(final byte[] b, int offset, int length)
{
return bytesToHexString(b, offset, length);
}
/**
Convert the specified byte array to its hexadecimal String representation.
Example generated string: "010203047E7F".
@param b The array containing the data.
@return A String containing the hex characters that represent the byte data.
**/
public static final String bytesToHexString(final byte[] b)
{
if (b == null) throw new NullPointerException("b");
return bytesToHexString(b, 0, b.length);
}
/**
Convert the specified byte array to its hexadecimal String representation.
Example generated string: "010203047E7F".
@param b The array containing the data.
@param offset The offset into the array at which to begin reading bytes.
@param length The number of bytes to read out of the array.
@return A String containing the hex characters that represent the byte data.
**/
public static final String bytesToHexString(final byte[] b, int offset, int length)
{
if (b == null) throw new NullPointerException("b");
char[] c = new char[length*2];
int num = bytesToHexString(b, offset, length, c, 0);
return new String(c, 0, num);
}
// Helper method to convert a byte array into its hex string representation.
// This is faster than calling Integer.toHexString(...)
static final int bytesToHexString(final byte[] b, int offset, int length, final char[] c, int coffset)
{
for (int i=0; i>>4) & 0x0F);
final byte lo = (byte)((b[i+offset] & 0x0F));
c[j+coffset] = c_[hi];
c[j+coffset+1] = c_[lo];
}
return length*2;
}
// Constant used in stringToBytes()
// Note that 0x11 is "undefined".
private static final byte[] b_ =
{
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
0x11, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
0x11, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11
};
static final byte charsToByte(char hi, char lo) throws NumberFormatException
{
int c1 = 0x00FFFF & hi;
int c2 = 0x00FFFF & lo;
if (c1 > 255 || c2 > 255) throw new NumberFormatException();
byte b1 = b_[c1];
byte b2 = b_[c2];
if (b1 == 0x11 || b2 == 0x11) return 0;
return(byte)(((byte)(b1 << 4)) + b2);
}
static final void checkArgs(byte[] serverValue, int offset)
throws NullPointerException
{
if (serverValue == null) throw new NullPointerException("serverValue");
if (offset < 0 || (offset > serverValue.length-1))
{
if (Trace.traceOn_) Trace.log(Trace.ERROR, "Offset " + offset + " was specified for array of length " + serverValue.length + ".");
ArrayIndexOutOfBoundsException ex =new ArrayIndexOutOfBoundsException(String.valueOf(offset));
if (Trace.traceOn_) Trace.log(Trace.ERROR, ex);
throw ex;
}
}
/**
Convert the specified hexadecimal String into a byte array containing the byte values for the
hexadecimal characters in the String. If the String contains characters other than those allowed
for a hexadecimal String (0-9 and A-F), an exception will be thrown.
@param s The String containing the hexadecimal representation of the data.
@return A byte array containing the byte values of the hex characters.
**/
public static final byte[] stringToBytes(String s) throws NumberFormatException
{
if (s == null) throw new NullPointerException("s");
int stringLength = s.length(); //@KBA check for empty string
if(stringLength > 2) //@KBA //@pdc allow for input single digit input like "X" so NumberFormatException is thrown instead of array bounds error
{ //@KBA
if(s.charAt(0) == '0' && (s.charAt(1) == 'x' || s.charAt(1) == 'X')) //@KBA
s = s.substring(2); //@KBA
else if((s.charAt(0) == 'x' || s.charAt(0) == 'X') && s.charAt(1) == '\'') //@KBA
{ //@KBA
if(s.charAt(stringLength - 1) == '\'') //@KBA
s = s.substring(2, stringLength-1); //@KBA
else //@KBA
throw new NumberFormatException(); //@KBA
} //@KBA
} //@KBA
//check if an odd number of characters //@KBA
if(s.length()%2 != 0) //@KBA
s = "0" + s; //@KBA
char[] c = s.toCharArray();
return stringToBytes(c, 0, c.length);
}
static final byte[] stringToBytes(char[] hex, int offset, int length)
{
if (hex.length == 0) return new byte[0];
byte[] buf = new byte[(length+1)/2];
int num = stringToBytes(hex, offset, length, buf, 0);
if (num < buf.length)
{
byte[] temp = buf;
buf = new byte[num];
System.arraycopy(temp, 0, buf, 0, num);
}
return buf;
}
// Helper method to convert a String in hex into its corresponding byte array.
// Called only by HexReaderInputStream
// length is the length of the string
static final int stringToBytes(char[] hex, int offset, int length, final byte[] b, int boff) throws NumberFormatException
{
if (hex.length == 0) return 0;
// account for char[] of odd length
//@KBD if (hex.length % 2 == 1) //@KBD put checks in stringToBytes(String s)
//@KBD {
//@KBD char[] temp = hex;
//@KBD hex = new char[temp.length+1];
//@KBD System.arraycopy(temp, 0, hex, 0, temp.length-1);
//@KBD hex[hex.length-2] = '0';
//@KBD hex[hex.length-1] = temp[temp.length-1];
//@KBD }
//@KBD if (hex[offset] == '0' && (hex.length > offset+1 && (hex[offset+1] == 'X' || hex[offset+1] == 'x')))
//@KBD {
//@KBD offset += 2;
//@KBD length -= 2;
//@KBD }
// Fixed loop condition to be length / 2 not b.length
for (int i=0; i< length / 2; ++i)
{
final int j = i*2;
final int c1 = 0x00FFFF & hex[j+offset];
final int c2 = 0x00FFFF & hex[j+offset+1];
if (c1 > 255 || c2 > 255) // out of range
{
//b[i+boff] = 0x00;
throw new NumberFormatException();
}
else
{
final byte b1 = b_[c1];
final byte b2 = b_[c2];
if (b1 == 0x11 || b2 == 0x11) // out of range
{
//b[i+boff] = 0x00;
throw new NumberFormatException();
}
else
{
final byte hi = (byte)(b1<<4);
b[i+boff] = (byte)(hi + b2);
}
}
}
return length / 2;
}
}