pl.edu.icm.jlargearrays.FloatLargeArray Maven / Gradle / Ivy
Show all versions of JLargeArrays Show documentation
/* ***** BEGIN LICENSE BLOCK *****
* JLargeArrays
* Copyright (C) 2013 onward University of Warsaw, ICM
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* 1. Redistributions of source code must retain the above copyright notice, this
* list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
* ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
* ***** END LICENSE BLOCK ***** */
package pl.edu.icm.jlargearrays;
import sun.misc.Cleaner;
import org.apache.commons.math3.util.FastMath;
/**
*
* An array of floats that can store up to 263 elements.
*
* @author Piotr Wendykier ([email protected])
*/
public class FloatLargeArray extends LargeArray
{
private static final long serialVersionUID = -8342458159338079576L;
private float[] data;
/**
* Creates new instance of this class.
*
* @param length number of elements
*/
public FloatLargeArray(long length)
{
this(length, true);
}
/**
* Creates new instance of this class.
*
* @param length number of elements
* @param zeroNativeMemory if true, then the native memory is zeroed.
*/
public FloatLargeArray(long length, boolean zeroNativeMemory)
{
this.type = LargeArrayType.FLOAT;
this.sizeof = 4;
if (length <= 0) {
throw new IllegalArgumentException(length + " is not a positive long value");
}
this.length = length;
if (length > getMaxSizeOf32bitArray()) {
this.ptr = LargeArrayUtils.UNSAFE.allocateMemory(this.length * this.sizeof);
if (zeroNativeMemory) {
zeroNativeMemory(length);
}
Cleaner.create(this, new Deallocator(this.ptr, this.length, this.sizeof));
MemoryCounter.increaseCounter(this.length * this.sizeof);
} else {
data = new float[(int) length];
}
}
/**
* Creates a constant array.
*
* @param length number of elements
* @param constantValue value
*/
public FloatLargeArray(long length, float constantValue)
{
this.type = LargeArrayType.FLOAT;
this.sizeof = 4;
if (length <= 0) {
throw new IllegalArgumentException(length + " is not a positive long value");
}
this.length = length;
this.isConstant = true;
this.data = new float[]{constantValue};
}
/**
* Creates new instance of this class.
*
* @param data data array, this reference is used internally.
*/
public FloatLargeArray(float[] data)
{
this.type = LargeArrayType.FLOAT;
this.sizeof = 4;
this.length = data.length;
this.data = data;
}
/**
* Returns a deep copy of this instance. (The elements themselves are copied.)
*
* @return a clone of this instance
*/
@Override
public FloatLargeArray clone()
{
if (isConstant) {
return new FloatLargeArray(length, getFloat(0));
} else {
FloatLargeArray v = new FloatLargeArray(length, false);
LargeArrayUtils.arraycopy(this, 0, v, 0, length);
return v;
}
}
@Override
public boolean equals(Object o)
{
if (super.equals(o)) {
FloatLargeArray la = (FloatLargeArray) o;
return this.data == la.data;
}
return false;
}
@Override
public int hashCode()
{
return 29 * super.hashCode() + (this.data != null ? this.data.hashCode() : 0);
}
@Override
public final Float get(long i)
{
return getFloat(i);
}
@Override
public final Float getFromNative(long i)
{
return LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i);
}
@Override
public final boolean getBoolean(long i)
{
if (ptr != 0) {
return (LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i)) != 0;
} else if (isConstant) {
return data[0] != 0;
} else {
return data[(int) i] != 0;
}
}
@Override
public final byte getByte(long i)
{
if (ptr != 0) {
return (byte) (LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i));
} else if (isConstant) {
return (byte) data[0];
} else {
return (byte) data[(int) i];
}
}
@Override
public final short getUnsignedByte(long i)
{
if (ptr != 0) {
return (short) (0xFF & (int) LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i));
} else if (isConstant) {
return (short) (0xFF & (int) data[0]);
} else {
return (short) (0xFF & (int) data[(int) i]);
}
}
@Override
public final short getShort(long i)
{
if (ptr != 0) {
return (short) (LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i));
} else if (isConstant) {
return (short) data[0];
} else {
return (short) data[(int) i];
}
}
@Override
public final int getInt(long i)
{
if (ptr != 0) {
return (int) (LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i));
} else if (isConstant) {
return (int) data[0];
} else {
return (int) data[(int) i];
}
}
@Override
public final long getLong(long i)
{
if (ptr != 0) {
return (long) (LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i));
} else if (isConstant) {
return (long) data[0];
} else {
return (long) data[(int) i];
}
}
@Override
public final float getFloat(long i)
{
if (ptr != 0) {
return (LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i));
} else if (isConstant) {
return data[0];
} else {
return data[(int) i];
}
}
@Override
public final double getDouble(long i)
{
if (ptr != 0) {
return (double) LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i);
} else if (isConstant) {
return (double) data[0];
} else {
return (double) data[(int) i];
}
}
@Override
public final float[] getData()
{
return data;
}
@Override
public final boolean[] getBooleanData()
{
if (length > LargeArray.LARGEST_SUBARRAY) return null;
boolean[] out = new boolean[(int) length];
if (ptr != 0) {
for (int i = 0; i < length; i++) {
float v = LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i);
out[i] = v != 0;
}
} else if (isConstant) {
boolean elem = data[0] != 0;
for (int i = 0; i < length; i++) {
out[i] = elem;
}
} else {
for (int i = 0; i < length; i++) {
out[i] = data[i] != 0;
}
}
return out;
}
@Override
public final boolean[] getBooleanData(boolean[] a, long startPos, long endPos, long step)
{
if (startPos < 0 || startPos >= length) {
throw new ArrayIndexOutOfBoundsException("startPos < 0 || startPos >= length");
}
if (endPos < 0 || endPos > length || endPos < startPos) {
throw new ArrayIndexOutOfBoundsException("endPos < 0 || endPos > length || endPos < startPos");
}
if (step < 1) {
throw new IllegalArgumentException("step < 1");
}
long len = (long) FastMath.ceil((endPos - startPos) / (double) step);
if (len > LargeArray.LARGEST_SUBARRAY) {
return null;
} else {
boolean[] out;
if (a != null && a.length >= len) {
out = a;
} else {
out = new boolean[(int) len];
}
int idx = 0;
if (ptr != 0) {
for (long i = startPos; i < endPos; i += step) {
float v = LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i);
out[idx++] = v != 0;
}
} else if (isConstant) {
for (long i = startPos; i < endPos; i += step) {
out[idx++] = data[0] != 0;
}
} else {
for (long i = startPos; i < endPos; i += step) {
float v = data[(int) i];
out[idx++] = v != 0;
}
}
return out;
}
}
@Override
public final byte[] getByteData()
{
if (length > LargeArray.LARGEST_SUBARRAY) return null;
byte[] out = new byte[(int) length];
if (ptr != 0) {
for (int i = 0; i < length; i++) {
out[i] = (byte) LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i);
}
} else if (isConstant) {
float elem = data[0];
for (int i = 0; i < length; i++) {
out[i] = (byte) elem;
}
} else {
for (int i = 0; i < length; i++) {
out[i] = (byte) data[i];
}
}
return out;
}
@Override
public final byte[] getByteData(byte[] a, long startPos, long endPos, long step)
{
if (startPos < 0 || startPos >= length) {
throw new ArrayIndexOutOfBoundsException("startPos < 0 || startPos >= length");
}
if (endPos < 0 || endPos > length || endPos < startPos) {
throw new ArrayIndexOutOfBoundsException("endPos < 0 || endPos > length || endPos < startPos");
}
if (step < 1) {
throw new IllegalArgumentException("step < 1");
}
long len = (long) FastMath.ceil((endPos - startPos) / (double) step);
if (len > LargeArray.LARGEST_SUBARRAY) {
return null;
} else {
byte[] out;
if (a != null && a.length >= len) {
out = a;
} else {
out = new byte[(int) len];
}
int idx = 0;
if (ptr != 0) {
for (long i = startPos; i < endPos; i += step) {
out[idx++] = (byte) LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i);
}
} else if (isConstant) {
for (long i = startPos; i < endPos; i += step) {
out[idx++] = (byte) data[0];
}
} else {
for (long i = startPos; i < endPos; i += step) {
out[idx++] = (byte) data[(int) i];
}
}
return out;
}
}
@Override
public final short[] getShortData()
{
if (length > LargeArray.LARGEST_SUBARRAY) return null;
short[] out = new short[(int) length];
if (ptr != 0) {
for (int i = 0; i < length; i++) {
out[i] = (short) LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i);
}
} else if (isConstant) {
float elem = data[0];
for (int i = 0; i < length; i++) {
out[i] = (short) elem;
}
} else {
for (int i = 0; i < length; i++) {
out[i] = (short) data[i];
}
}
return out;
}
@Override
public final short[] getShortData(short[] a, long startPos, long endPos, long step)
{
if (startPos < 0 || startPos >= length) {
throw new ArrayIndexOutOfBoundsException("startPos < 0 || startPos >= length");
}
if (endPos < 0 || endPos > length || endPos < startPos) {
throw new ArrayIndexOutOfBoundsException("endPos < 0 || endPos > length || endPos < startPos");
}
if (step < 1) {
throw new IllegalArgumentException("step < 1");
}
long len = (long) FastMath.ceil((endPos - startPos) / (double) step);
if (len > LargeArray.LARGEST_SUBARRAY) {
return null;
} else {
short[] out;
if (a != null && a.length >= len) {
out = a;
} else {
out = new short[(int) len];
}
int idx = 0;
if (ptr != 0) {
for (long i = startPos; i < endPos; i += step) {
out[idx++] = (short) LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i);
}
} else if (isConstant) {
for (long i = startPos; i < endPos; i += step) {
out[idx++] = (short) data[0];
}
} else {
for (long i = startPos; i < endPos; i += step) {
out[idx++] = (short) data[(int) i];
}
}
return out;
}
}
@Override
public final int[] getIntData()
{
if (length > LargeArray.LARGEST_SUBARRAY) return null;
int[] out = new int[(int) length];
if (ptr != 0) {
for (int i = 0; i < length; i++) {
out[i] = (int) LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i);
}
} else if (isConstant) {
float elem = data[0];
for (int i = 0; i < length; i++) {
out[i] = (int) elem;
}
} else {
for (int i = 0; i < length; i++) {
out[i] = (int) data[i];
}
}
return out;
}
@Override
public final int[] getIntData(int[] a, long startPos, long endPos, long step)
{
if (startPos < 0 || startPos >= length) {
throw new ArrayIndexOutOfBoundsException("startPos < 0 || startPos >= length");
}
if (endPos < 0 || endPos > length || endPos < startPos) {
throw new ArrayIndexOutOfBoundsException("endPos < 0 || endPos > length || endPos < startPos");
}
if (step < 1) {
throw new IllegalArgumentException("step < 1");
}
long len = (long) FastMath.ceil((endPos - startPos) / (double) step);
if (len > LargeArray.LARGEST_SUBARRAY) {
return null;
} else {
int[] out;
if (a != null && a.length >= len) {
out = a;
} else {
out = new int[(int) len];
}
int idx = 0;
if (ptr != 0) {
for (long i = startPos; i < endPos; i += step) {
out[idx++] = (int) LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i);
}
} else if (isConstant) {
for (long i = startPos; i < endPos; i += step) {
out[idx++] = (int) data[0];
}
} else {
for (long i = startPos; i < endPos; i += step) {
out[idx++] = (int) data[(int) i];
}
}
return out;
}
}
@Override
public final long[] getLongData()
{
if (length > LargeArray.LARGEST_SUBARRAY) return null;
long[] out = new long[(int) length];
if (ptr != 0) {
for (int i = 0; i < length; i++) {
out[i] = (long) LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i);
}
} else if (isConstant) {
float elem = data[0];
for (int i = 0; i < length; i++) {
out[i] = (long) elem;
}
} else {
for (int i = 0; i < length; i++) {
out[i] = (long) data[i];
}
}
return out;
}
@Override
public final long[] getLongData(long[] a, long startPos, long endPos, long step)
{
if (startPos < 0 || startPos >= length) {
throw new ArrayIndexOutOfBoundsException("startPos < 0 || startPos >= length");
}
if (endPos < 0 || endPos > length || endPos < startPos) {
throw new ArrayIndexOutOfBoundsException("endPos < 0 || endPos > length || endPos < startPos");
}
if (step < 1) {
throw new IllegalArgumentException("step < 1");
}
long len = (long) FastMath.ceil((endPos - startPos) / (double) step);
if (len > LargeArray.LARGEST_SUBARRAY) {
return null;
} else {
long[] out;
if (a != null && a.length >= len) {
out = a;
} else {
out = new long[(int) len];
}
int idx = 0;
if (ptr != 0) {
for (long i = startPos; i < endPos; i += step) {
out[idx++] = (long) LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i);
}
} else if (isConstant) {
for (long i = startPos; i < endPos; i += step) {
out[idx++] = (long) data[0];
}
} else {
for (long i = startPos; i < endPos; i += step) {
out[idx++] = (long) data[(int) i];
}
}
return out;
}
}
@Override
public final float[] getFloatData()
{
if (length > LargeArray.LARGEST_SUBARRAY) return null;
float[] out = new float[(int) length];
if (ptr != 0) {
for (int i = 0; i < length; i++) {
out[i] = LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i);
}
} else if (isConstant) {
float elem = data[0];
for (int i = 0; i < length; i++) {
out[i] = elem;
}
} else {
System.arraycopy(data, 0, out, 0, (int) length);
}
return out;
}
@Override
public final float[] getFloatData(float[] a, long startPos, long endPos, long step)
{
if (startPos < 0 || startPos >= length) {
throw new ArrayIndexOutOfBoundsException("startPos < 0 || startPos >= length");
}
if (endPos < 0 || endPos > length || endPos < startPos) {
throw new ArrayIndexOutOfBoundsException("endPos < 0 || endPos > length || endPos < startPos");
}
if (step < 1) {
throw new IllegalArgumentException("step < 1");
}
long len = (long) FastMath.ceil((endPos - startPos) / (double) step);
if (len > LargeArray.LARGEST_SUBARRAY) {
return null;
} else {
float[] out;
if (a != null && a.length >= len) {
out = a;
} else {
out = new float[(int) len];
}
int idx = 0;
if (ptr != 0) {
for (long i = startPos; i < endPos; i += step) {
out[idx++] = LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i);
}
} else if (isConstant) {
for (long i = startPos; i < endPos; i += step) {
out[idx++] = data[0];
}
} else {
for (long i = startPos; i < endPos; i += step) {
out[idx++] = data[(int) i];
}
}
return out;
}
}
@Override
public final double[] getDoubleData()
{
if (length > LargeArray.LARGEST_SUBARRAY) return null;
double[] out = new double[(int) length];
if (ptr != 0) {
for (int i = 0; i < length; i++) {
out[i] = LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i);
}
} else if (isConstant) {
double elem = data[0];
for (int i = 0; i < length; i++) {
out[i] = elem;
}
} else {
for (int i = 0; i < length; i++) {
out[i] = data[i];
}
}
return out;
}
@Override
public final double[] getDoubleData(double[] a, long startPos, long endPos, long step)
{
if (startPos < 0 || startPos >= length) {
throw new ArrayIndexOutOfBoundsException("startPos < 0 || startPos >= length");
}
if (endPos < 0 || endPos > length || endPos < startPos) {
throw new ArrayIndexOutOfBoundsException("endPos < 0 || endPos > length || endPos < startPos");
}
if (step < 1) {
throw new IllegalArgumentException("step < 1");
}
long len = (long) FastMath.ceil((endPos - startPos) / (double) step);
if (len > LargeArray.LARGEST_SUBARRAY) {
return null;
} else {
double[] out;
if (a != null && a.length >= len) {
out = a;
} else {
out = new double[(int) len];
}
int idx = 0;
if (ptr != 0) {
for (long i = startPos; i < endPos; i += step) {
out[idx++] = (double) LargeArrayUtils.UNSAFE.getFloat(ptr + sizeof * i);
}
} else if (isConstant) {
for (long i = startPos; i < endPos; i += step) {
out[idx++] = (double) data[0];
}
} else {
for (long i = startPos; i < endPos; i += step) {
out[idx++] = (double) data[(int) i];
}
}
return out;
}
}
@Override
public final void setToNative(long i, Object value)
{
LargeArrayUtils.UNSAFE.putFloat(ptr + sizeof * i, (Float) value);
}
@Override
public final void setBoolean(long i, boolean value)
{
if (ptr != 0) {
LargeArrayUtils.UNSAFE.putFloat(ptr + sizeof * i, value == true ? 1.0f : 0.0f);
} else {
if (isConstant) {
throw new IllegalAccessError("Constant arrays cannot be modified.");
}
data[(int) i] = value == true ? 1.0f : 0.0f;
}
}
@Override
public final void setByte(long i, byte value)
{
if (ptr != 0) {
LargeArrayUtils.UNSAFE.putFloat(ptr + sizeof * i, (float) value);
} else {
if (isConstant) {
throw new IllegalAccessError("Constant arrays cannot be modified.");
}
data[(int) i] = (float) value;
}
}
@Override
public final void setUnsignedByte(long i, short value)
{
setShort(i, value);
}
@Override
public final void setShort(long i, short value)
{
if (ptr != 0) {
LargeArrayUtils.UNSAFE.putFloat(ptr + sizeof * i, (float) value);
} else {
if (isConstant) {
throw new IllegalAccessError("Constant arrays cannot be modified.");
}
data[(int) i] = (float) value;
}
}
@Override
public final void setInt(long i, int value)
{
if (ptr != 0) {
LargeArrayUtils.UNSAFE.putFloat(ptr + sizeof * i, (float) value);
} else {
if (isConstant) {
throw new IllegalAccessError("Constant arrays cannot be modified.");
}
data[(int) i] = (float) value;
}
}
@Override
public final void setLong(long i, long value)
{
if (ptr != 0) {
LargeArrayUtils.UNSAFE.putFloat(ptr + sizeof * i, (float) value);
} else {
if (isConstant) {
throw new IllegalAccessError("Constant arrays cannot be modified.");
}
data[(int) i] = (float) value;
}
}
@Override
public final void setFloat(long i, float value)
{
if (ptr != 0) {
LargeArrayUtils.UNSAFE.putFloat(ptr + sizeof * i, value);
} else {
if (isConstant) {
throw new IllegalAccessError("Constant arrays cannot be modified.");
}
data[(int) i] = value;
}
}
@Override
public final void setDouble(long i, double value)
{
if (ptr != 0) {
LargeArrayUtils.UNSAFE.putFloat(ptr + sizeof * i, (float) value);
} else {
if (isConstant) {
throw new IllegalAccessError("Constant arrays cannot be modified.");
}
data[(int) i] = (float) value;
}
}
}