org.geolatte.geom.ByteBuffer Maven / Gradle / Ivy
/*
* This file is part of the GeoLatte project.
*
* GeoLatte is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* GeoLatte is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with GeoLatte. If not, see .
*
* Copyright (C) 2010 - 2011 and Ownership of code is shared by:
* Qmino bvba - Romeinsestraat 18 - 3001 Heverlee (http://www.qmino.com)
* Geovise bvba - Generaal Eisenhowerlei 9 - 2140 Antwerpen (http://www.geovise.com)
*/
package org.geolatte.geom;
import java.nio.BufferOverflowException;
import java.nio.BufferUnderflowException;
import java.util.List;
import java.util.Objects;
/**
* A byte buffer class.
*
* This class is modeled on the java.nio.Buffer
interface. Specifically, the properties capacity and limit are
* defined as for Buffer
.
*
*
* - A buffer's capacity is the number of elements it contains. The capacity of a buffer is never negative and never changes.
* - A buffer's limit is the index of the first element that should not be read or written. A buffer's limit is never negative and is never greater than its capacity.
*
*
* @author Karel Maesen, Geovise BVBA
* creation-date: Oct 29, 2010
*/
public class ByteBuffer {
/**
* byte size for unsigned int
*/
public static final int UINT_SIZE = 4;
/**
* byte size for doubles
*/
public static final int DOUBLE_SIZE = 8;
/**
* Max. permissible value for an unsigned int.
*/
public static final long UINT_MAX_VALUE = 4294967295L;
private final java.nio.ByteBuffer buffer;
private ByteBuffer(java.nio.ByteBuffer buffer) {
this.buffer = buffer;
}
/**
* Creates a ByteBuffer
from a hexadecimal string.
*
* Every two chars in the string are interpreted as the hexadecimal representation of a byte.
* If the string length is odd, the last character will be ignored.
*
* @param hexString the bytes represented in hexadecimal form
* @return A ByteBuffer based on the hexadecimal string
*/
public static ByteBuffer from(String hexString) {
if (hexString == null) throw new IllegalArgumentException("Cannot create ByteBuffer from null input String.");
int size = hexString.length() / 2; // this will drop the last char, if hexString is not even.
java.nio.ByteBuffer buffer = java.nio.ByteBuffer.allocate(size);
for (int i = 0; i < size * 2; i += 2) {
final char firstLetterOrNumber = hexString.charAt(i);
final char secondLetterOrNumber = hexString.charAt(i + 1);
final byte firstDigit;
final byte secondDigit;
if (firstLetterOrNumber == '+') {
firstDigit = 0;
secondDigit = charToHex(secondLetterOrNumber);
}
else if (firstLetterOrNumber == '-'){
firstDigit = 0;
secondDigit = (byte)-charToHex(secondLetterOrNumber);
}else {
firstDigit = charToHex(firstLetterOrNumber);
secondDigit = charToHex(secondLetterOrNumber);
}
final byte b = (byte)((firstDigit << 4) | secondDigit);
buffer.put(b);
}
buffer.rewind();
return new ByteBuffer(buffer);
}
/**
* Returns this instance as a hexadecimal string.
*
* @return A string representation of this ByteBuffer in hexadecimal form
*/
public String toString() {
StringBuilder builder = new StringBuilder();
int savedPosition = buffer.position();
rewind();
for (int i = 0; i < limit(); i++) {
int hexValue = readByte();
appendHexByteRepresentation(builder, hexValue);
}
buffer.position(savedPosition);
return builder.toString();
}
private void appendHexByteRepresentation(StringBuilder builder, int hexValue) {
String byteStr = Integer.toHexString(hexValue).toUpperCase();
if (byteStr.length() == 1) {
builder.append("0").append(byteStr);
} else {
builder.append(byteStr);
}
}
private int readByte() {
int hexValue = get();
hexValue = (hexValue << 24) >>> 24;
return hexValue;
}
/**
* Wraps a byte array into a ByteBuffer
.
*
* The new buffer will be backed by the given byte array; that is, modifications to the buffer will cause the array
* to be modified and vice versa. The new buffer's capacity and limit will be bytes.length.
*
* @param bytes The array that will back this buffer
* @return The new byte buffer.
*/
public static ByteBuffer from(byte[] bytes) {
java.nio.ByteBuffer buffer = java.nio.ByteBuffer.wrap(bytes);
return new ByteBuffer(buffer);
}
/**
* Transforms a List of ByteBuffers to a single ByteBuffer
*/
public static ByteBuffer collect(List buffers) {
if(buffers == null || buffers.isEmpty()) return allocate(0);
int totalSize = 0;
for( ByteBuffer b: buffers){
totalSize += b.capacity();
}
byte[] bytes = new byte[totalSize];
int p = 0;
for( ByteBuffer b: buffers){
java.lang.System.arraycopy(b.toByteArray(), 0, bytes, p, b.capacity());
p += b.capacity();
}
return from(bytes);
}
/**
* Allocates a new ByteBuffer
of the specified capacity.
*
* The new buffer's position will be zero, its limit will be its capacity and each of its elements will be
* initialized to zero.
*
* @param capacity The new buffer's capacity, in bytes
* @return a new ByteBuffer
instance of the specified capacity
*/
public static ByteBuffer allocate(int capacity) {
return new ByteBuffer(java.nio.ByteBuffer.allocate(capacity));
}
/**
* Relative get method. Reads the byte at this buffer's current position, and then increments the position.
*
* @return The byte at the buffer's current position
* @throws BufferAccessException If the buffer's current position is not smaller than its limit.
*/
public byte get() {
try {
return buffer.get();
} catch (BufferOverflowException e) {
throw new BufferAccessException(e.getMessage());
}
}
/**
* Writes the given byte into this buffer at the current position, and then increments the position.
*
* @param value The byte to be written
* @throws BufferAccessException If this buffer's current position is not smaller than its limit.
*/
public void put(byte value) {
try {
buffer.put(value);
} catch (BufferOverflowException e) {
throw new BufferAccessException(e.getMessage());
}
}
/**
* Returns this buffer's capacity.
*
* @return The capacity of this buffer
*/
public int capacity() {
return buffer.capacity();
}
/**
* Returns this buffer's limit.
*
* @return The limit of this buffer
*/
public int limit() {
return buffer.limit();
}
/**
* Rewinds the buffer.
*
* After rewind, the next get() or put() will take place on the first element of this instance.
*/
public void rewind() {
buffer.rewind();
}
/**
* Reports if this buffer is empty (holds no bytes).
*
* @return True if limit is 0, otherwise false
*/
public boolean isEmpty() {
return buffer.limit() == 0;
}
/**
* Sets the byte order for this instance.
*
* @param wbo The new byte order, either {@link ByteOrder#XDR XDR} or {@link ByteOrder#NDR NDR}
*/
public void setByteOrder(ByteOrder wbo) {
buffer.order(wbo.getByteOrder());
}
/**
* Reads the next 4 bytes as an int from this instance at the current position, taking into account the byte-order,
* and then increments the position by four.
*
* @return The int value at the buffer's current position
* @throws BufferAccessException If there are fewer than four bytes remaining in this buffer.
*/
public int getInt() {
try {
return buffer.getInt();
} catch (BufferUnderflowException e) {
throw new BufferAccessException(e.getMessage());
}
}
/**
* Writes the specified int-value as 4 bytes to this instance at the current position, respecting the byte-order,
* and then increments the position by four.
*
* @param value The int value to be written
* @throws BufferAccessException If there are fewer than four bytes remaining in this buffer.
*/
public void putInt(int value) {
try {
buffer.putInt(value);
} catch (BufferOverflowException e) {
throw new BufferAccessException(e.getMessage());
}
}
/**
* Reads the next 8 bytes as a long from this instance at the current position, taking into account the byte-order,
* and then increments the position by eight.
*
* @return The long value at the buffer's current position
* @throws BufferAccessException If there are fewer than eight bytes remaining in this buffer.
*/
public long getLong() {
try {
return buffer.getLong();
} catch (BufferUnderflowException e) {
throw new BufferAccessException(e.getMessage());
}
}
/**
* Writes the specified long value as 8 bytes to this instance at the current position, respecting the byte-order,
* and then increments the position by eight.
*
* @param value The long value to be written
* @throws BufferAccessException If there are fewer than eight bytes remaining in this buffer.
*/
public void putLong(long value) {
try {
buffer.putLong(value);
} catch (BufferOverflowException e) {
throw new BufferAccessException(e.getMessage());
}
}
/**
* Reads the next 4 bytes as a float from this instance at the current position, taking into account the byte-order,
* and then increments the position by four.
*
* @return The float value at the buffer's current position
* @throws BufferAccessException If there are fewer than four bytes remaining in this buffer.
*/
public float getFloat() {
try {
return buffer.getFloat();
} catch (BufferUnderflowException e) {
throw new BufferAccessException(e.getMessage());
}
}
/**
* Writes the specified float-value as 4 bytes to this instance at the current position, respecting the byte-order,
* and then increments the position by four.
*
* @param value The float value to be written
* @throws BufferAccessException If there are fewer than four bytes remaining in this buffer.
*/
public void putFloat(float value) {
try {
buffer.putFloat(value);
} catch (BufferOverflowException e) {
throw new BufferAccessException(e.getMessage());
}
}
/**
* Reads the next 8 bytes as a double from this instance at the current position, taking into account the byte-order,
* and then increments the position by eight.
*
* @return The double value at the buffer's current position
* @throws BufferAccessException If there are fewer than eight bytes remaining in this buffer.
*/
public double getDouble() {
try {
return buffer.getDouble();
} catch (BufferUnderflowException e) {
throw new BufferAccessException(e.getMessage());
}
}
/**
* Writes the specified double-value as 8 bytes to this instance at the current position, respecting the byte-order,
* and then increments the position by eight.
*
* @param value The double value to be written
* @throws BufferAccessException If there are fewer than eight bytes remaining in this buffer.
*/
public void putDouble(Double value) {
try {
buffer.putDouble(value);
} catch (BufferOverflowException e) {
throw new BufferAccessException(e.getMessage());
}
}
/**
* Reads the next 4 bytes as an unsigned integer from this instance at the current position,
* taking into account the byte-order, and then increments the position by four.
*
* @return The value of the 4-byte unsigned integer at the current position as a long
* @throws BufferAccessException If there are fewer than four bytes remaining in this buffer
*/
public long getUInt() {
try {
int signedInt = buffer.getInt();
return signedInt & 0xffffffffL; //cast the signed int to an unsigned value in the bottom 32 bits of a long
} catch (BufferUnderflowException e) {
throw new BufferAccessException(e.getMessage());
}
}
/**
* Interprets the specified long-value as and unsigned integer, and appends it as 4 bytes
* to this instance at the current position, respecting the byte-order.
*
* @param value The unsigned integer value to be written
* @throws RuntimeException If the specified value is larger than the largest unsigned integer (4294967295L)
* @throws BufferAccessException If there are fewer than eight bytes remaining in this buffer.
*/
public void putUInt(long value) {
if (value > UINT_MAX_VALUE) throw new RuntimeException("Value received doesn't fit in unsigned integer");
try {
buffer.putInt((int) value);
} catch (BufferOverflowException e) {
throw new BufferAccessException(e.getMessage());
}
}
/**
* Gets the byte order of this instance.
*
* @return This buffer's byte order
*/
public ByteOrder getByteOrder() {
java.nio.ByteOrder order = buffer.order();
return ByteOrder.valueOf(order);
}
/**
* Returns the byte array that backs this buffer.
*
* @return The array that backs this buffer
*/
public byte[] toByteArray(){
return buffer.array();
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
ByteBuffer that = (ByteBuffer) o;
if (buffer != null ? !buffer.equals(that.buffer) : that.buffer != null) return false;
return true;
}
@Override
public int hashCode() {
return buffer != null ? buffer.hashCode() : 0;
}
/**
* Used for testing purposes.
*
* @param other another ByteBuffer
* @return true if both buffers contain the same bytes, false otherwise
*/
public boolean hasSameContent(ByteBuffer other) {
if (other == null) return false;
if (this.limit() != other.limit()) return false;
int thisSavedPosition = this.buffer.position();
int otherSavedPosition = other.buffer.position();
this.rewind();
other.rewind();
for (int i = 0; i < this.limit(); i++) {
if (this.get() != other.get()) return false;
}
this.buffer.position(thisSavedPosition);
other.buffer.position(otherSavedPosition);
return true;
}
private static byte charToHex(int letterOrNumber) {
final int number;
if (letterOrNumber <= '9'){
number = letterOrNumber - '0';
if (number < 0) throw numberFormatException(letterOrNumber);
return (byte) number;
} else {
// The letters 'A' - 'F' and 'a' - 'f' differ only at bit position 6.
// If bit 6 is zero, then it is uppercase
int letterCaseInsensitive = letterOrNumber & ~(1 << 5);
if (letterCaseInsensitive <= 'F') {
number = letterCaseInsensitive - 'A' + 10;
if (number < 10) throw numberFormatException(letterOrNumber);
return (byte) number;
}
}
throw numberFormatException(letterOrNumber);
}
private static NumberFormatException numberFormatException(int letterOrNumber) {
return new NumberFormatException((char)letterOrNumber + " is not a hex digit");
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy