org.deepsymmetry.beatlink.dbserver.NumberField Maven / Gradle / Ivy
package org.deepsymmetry.beatlink.dbserver;
import org.deepsymmetry.beatlink.Util;
import java.io.DataInputStream;
import java.io.IOException;
import java.nio.ByteBuffer;
/**
* A number field represents an integer, and can take up 1, 2, or 4 bytes, depending on the tag which
* introduces it.
*
* @author James Elliott
*/
public class NumberField extends Field {
/**
* The byte which identifies the specific type of number field that is coming next in a network stream.
*/
private final byte typeTag;
/**
* The number of bytes making up the network representation of the value of this field, excluding the type tag.
*/
private final int size;
/**
* Holds the value represented by this field. We use a long so we don't have to worry about sign issues
* when the full four byte version is used.
*/
private final long value;
/**
* Holds the actual bytes used to transmit this field, including the type tag.
*/
private final ByteBuffer buffer;
/**
* A four-byte field representing the number zero, used in many requests, so they can save the time and garbage
* of constructing throwaway versions.
*/
public final static NumberField WORD_0 = new NumberField(0);
/**
* A four-byte field representing the number one, used in some requests, so they can save the time and garbage
* of constructing throwaway versions.
*/
public final static NumberField WORD_1 = new NumberField(1);
/**
* Constructor for reading from the network.
*
* @param typeTag the tag which identified this field as a NumberField, and which allows us to determine the
* proper size.
* @param is the stream on which the field value is to be read.
*
* @throws IllegalArgumentException if tag is not a valid number field tag.
* @throws IOException if there is a problem reading the value.
*/
public NumberField(final byte typeTag, final DataInputStream is) throws IOException {
this.typeTag = typeTag;
switch (typeTag) {
case 0x0f:
size = 1;
break;
case 0x10:
size = 2;
break;
case 0x11:
size = 4;
break;
default:
throw new IllegalArgumentException("NumberField cannot have tag " + typeTag);
}
byte[] bufBytes = new byte[size + 1];
bufBytes[0] = typeTag;
is.readFully(bufBytes, 1, size);
buffer = ByteBuffer.wrap(bufBytes).asReadOnlyBuffer();
value = Util.bytesToNumber(bufBytes, 1, size);
}
/**
* Constructor from code.
*
* @param value the desired value to be represented by this field.
* @param size the number of bytes to be used to hold the value: 1, 2, or 4.
*
* @throws IllegalArgumentException if the specified size is not a supported number field size.
*/
public NumberField(final long value, final int size) {
this.value = value & 0xffffffffL;
this.size = size;
byte[] bufBytes = new byte[size + 1];
switch (size) {
case 1:
typeTag = (byte)0x0f;
bufBytes[1] = (byte)(value & 0xff);
break;
case 2:
typeTag = (byte)0x10;
bufBytes[1] = (byte)((value & 0xff00) >> 8);
bufBytes[2] = (byte)(value & 0xff);
break;
case 4:
typeTag = (byte)0x11;
bufBytes[1] = (byte)((value & 0xff000000) >> 24);
bufBytes[2] = (byte)((value & 0xff0000) >> 16);
bufBytes[3] = (byte)((value & 0xff00) >> 8);
bufBytes[4] = (byte)(value & 0xff);
break;
default:
throw new IllegalArgumentException("NumberField cannot have size " + size);
}
bufBytes[0] = typeTag;
buffer = ByteBuffer.wrap(bufBytes).asReadOnlyBuffer();
}
/**
* Convenience constructor from code, for the most common case of wanting a 4-byte number field.
*
* @param value the desired value to be represented by this field.
*/
public NumberField(final long value) {
this(value, 4);
}
/**
* Get the numeric value this field represents.
*
* @return the number which this field contains.
*/
public long getValue() {
return value;
}
@Override
public byte getTypeTag() {
return typeTag;
}
@Override
public byte getArgumentTag() {
return (byte) 0x06;
}
@Override
public long getSize() {
return size;
}
@Override
public ByteBuffer getBytes() {
buffer.rewind();
return buffer.slice();
}
@Override
public String toString() {
return "NumberField[ size: " + size + ", value: " + value + ", bytes: " + getHexString() + "]";
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy