blob: bc53351df9b587be269edc221686bf1755ec14f9 [file] [log] [blame]
/*=============================================================================#
# Copyright (c) 2009, 2020 Stephan Wahlbrink and others.
#
# This program and the accompanying materials are made available under the
# terms of the Eclipse Public License 2.0 which is available at
# https://www.eclipse.org/legal/epl-2.0, or the Apache License, Version 2.0
# which is available at https://www.apache.org/licenses/LICENSE-2.0.
#
# SPDX-License-Identifier: EPL-2.0 OR Apache-2.0
#
# Contributors:
# Stephan Wahlbrink <sw@wahlbrink.eu> - initial API and implementation
#=============================================================================*/
package org.eclipse.statet.rj.data.impl;
import java.io.IOException;
import org.eclipse.statet.jcommons.lang.NonNull;
import org.eclipse.statet.jcommons.lang.NonNullByDefault;
import org.eclipse.statet.jcommons.lang.Nullable;
import org.eclipse.statet.rj.data.RJIO;
@NonNullByDefault
public class RNumericBFix64Store extends AbstractNumericStore
implements ExternalizableRStore {
public static final int SEGMENT_LENGTH= DEFAULT_LONG_DATA_SEGMENT_LENGTH;
private final long length;
protected final double [] @NonNull[] realValues;
public RNumericBFix64Store(final long length) {
this.length= length;
this.realValues= new2dDoubleArray(length, SEGMENT_LENGTH);
}
public RNumericBFix64Store(final double [] @NonNull[] values) {
this.length= check2dArrayLength(values, SEGMENT_LENGTH);
this.realValues= values;
}
public RNumericBFix64Store(final double [] @NonNull[] values,
final boolean @Nullable[] @NonNull[] isNAs) {
this.length= check2dArrayLength(values, SEGMENT_LENGTH);
this.realValues= values;
if (isNAs != null) {
if (check2dArrayLength(isNAs, SEGMENT_LENGTH) != this.length) {
throw new IllegalArgumentException();
}
for (int i= 0; i < isNAs.length; i++) {
final boolean[] isNAi= isNAs[i];
for (int j= 0; j < isNAi.length; j++) {
if (isNAi[j]) {
this.realValues[i][j]= NA_numeric_DOUBLE;
}
}
}
}
}
public RNumericBFix64Store(final RJIO io, final long length) throws IOException {
this.length= length;
this.realValues= new2dDoubleArray(length, SEGMENT_LENGTH);
for (int i= 0; i < this.realValues.length; i++) {
io.readDoubleData(this.realValues[i], this.realValues[i].length);
}
}
@Override
public void writeExternal(final RJIO io) throws IOException {
for (int i= 0; i < this.realValues.length; i++) {
io.writeDoubleData(this.realValues[i], this.realValues[i].length);
}
}
@Override
protected final boolean isStructOnly() {
return false;
}
@Override
public final long getLength() {
return this.length;
}
@Override
public boolean isNA(final int idx) {
final double v= this.realValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH];
return (Double.isNaN(v)
&& (int)Double.doubleToRawLongBits(v) == NA_numeric_INT_MATCH);
}
@Override
public boolean isNA(final long idx) {
final double v= this.realValues[(int)(idx / SEGMENT_LENGTH)][(int)(idx % SEGMENT_LENGTH)];
return (Double.isNaN(v)
&& (int)Double.doubleToRawLongBits(v) == NA_numeric_INT_MATCH);
}
@Override
public void setNA(final int idx) {
this.realValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH] =
NA_numeric_DOUBLE;
}
@Override
public void setNA(final long idx) {
this.realValues[(int)(idx / SEGMENT_LENGTH)][(int)(idx % SEGMENT_LENGTH)] =
NA_numeric_DOUBLE;
}
@Override
public boolean isNaN(final int idx) {
final double v= this.realValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH];
return (Double.isNaN(v)
&& (int)Double.doubleToRawLongBits(v) != NA_numeric_INT_MATCH);
}
@Override
public boolean isNaN(final long idx) {
final double v= this.realValues[(int)(idx / SEGMENT_LENGTH)][(int)(idx % SEGMENT_LENGTH)];
return (Double.isNaN(v)
&& (int)Double.doubleToRawLongBits(v) != NA_numeric_INT_MATCH);
}
@Override
public boolean isMissing(final int idx) {
return (Double.isNaN(
this.realValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH] ));
}
@Override
public boolean isMissing(final long idx) {
return (Double.isNaN(
this.realValues[(int)(idx / SEGMENT_LENGTH)][(int)(idx % SEGMENT_LENGTH)] ));
}
@Override
public double getNum(final int idx) {
return this.realValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH];
}
@Override
public double getNum(final long idx) {
return this.realValues[(int)(idx / SEGMENT_LENGTH)][(int)(idx % SEGMENT_LENGTH)];
}
@Override
public void setNum(final int idx, final double value) {
this.realValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH] =
(Double.isNaN(value)) ? NaN_numeric_DOUBLE : value;
}
@Override
public void setNum(final long idx, final double value) {
this.realValues[(int)(idx / SEGMENT_LENGTH)][(int)(idx % SEGMENT_LENGTH)] =
(Double.isNaN(value)) ? NaN_numeric_DOUBLE : value;
}
@Override
public @Nullable Double get(final int idx) {
if (idx < 0 || idx >= this.length) {
throw new IndexOutOfBoundsException(Long.toString(idx));
}
final double v= this.realValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH];
return (!Double.isNaN(v)
|| (int)Double.doubleToRawLongBits(v) != NA_numeric_INT_MATCH) ?
Double.valueOf(v) :
null;
}
@Override
public @Nullable Double get(final long idx) {
if (idx < 0 || idx >= this.length) {
throw new IndexOutOfBoundsException(Long.toString(idx));
}
final double v= this.realValues[(int)(idx / SEGMENT_LENGTH)][(int)(idx % SEGMENT_LENGTH)];
return (!Double.isNaN(v)
|| (int)Double.doubleToRawLongBits(v) != NA_numeric_INT_MATCH) ?
Double.valueOf(v) :
null;
}
@Override
public @Nullable Double [] toArray() {
final int l= checkToArrayLength();
final var array= new @Nullable Double [l];
int k= 0;
for (int i= 0; i < this.realValues.length; i++, k++) {
final double[] reals= this.realValues[i];
for (int j= 0; j < reals.length; j++) {
final double v= reals[j];
if (!Double.isNaN(v)
|| (int)Double.doubleToRawLongBits(v) != NA_numeric_INT_MATCH) {
array[k]= Double.valueOf(v);
}
}
}
return array;
}
}