blob: e371d7996c19a9d13062d5d96031e5deac9dddab [file] [log] [blame]
/*=============================================================================#
# Copyright (c) 2009, 2018 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.Externalizable;
import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectOutput;
import org.eclipse.statet.rj.data.RJIO;
/**
* Based on byte array.
*
* This implementation is limited to length of 2<sup>31</sup>-1.
*/
public class RLogicalByte32Store extends AbstractLogicalStore
implements RDataResizeExtension<Boolean>, ExternalizableRStore, Externalizable {
public static final byte TRUE= TRUE_BYTE;
public static final byte FALSE= FALSE_BYTE;
private int length;
private byte[] boolValues;
public RLogicalByte32Store() {
this.length= 0;
this.boolValues= EMPTY_BYTE_ARRAY;
}
public RLogicalByte32Store(final int length) {
this.length= length;
this.boolValues= new byte[length];
}
public RLogicalByte32Store(final boolean[] values) {
this.length= values.length;
this.boolValues= new byte[values.length];
for (int i= values.length-1; i >= 0; i--) {
this.boolValues[i]= (values[i]) ? TRUE_BYTE : FALSE_BYTE;
}
}
public RLogicalByte32Store(final boolean[] values, final int[] naIdxs) {
this.length= values.length;
this.boolValues= new byte[values.length];
for (int i= values.length-1; i >= 0; i--) {
this.boolValues[i]= (values[i]) ? TRUE_BYTE : FALSE_BYTE;
}
if (naIdxs != null) {
for (int i= 0; i < naIdxs.length; i++) {
this.boolValues[naIdxs[i]]= NA_logical_BYTE;
}
}
}
public RLogicalByte32Store(final byte[] values) {
this.length= values.length;
this.boolValues= values;
}
public RLogicalByte32Store(final byte[] values, final byte trueCode, final byte naCode) {
this.length= values.length;
if (trueCode != TRUE_BYTE || naCode != NA_logical_BYTE) {
for (int i= values.length-1; i >= 0; i--) {
final int value= values[i];
if (value == trueCode) {
values[i]= TRUE_BYTE;
}
else if (value == naCode) {
values[i]= NA_logical_BYTE;
}
else {
values[i]= FALSE_BYTE;
}
}
}
this.boolValues= values;
}
public RLogicalByte32Store(final RJIO io, final int length) throws IOException {
this.length= length;
this.boolValues= io.readByteData(new byte[length], length);
}
@Override
public void writeExternal(final RJIO io) throws IOException {
io.writeByteData(this.boolValues, this.length);
}
@Override
public void readExternal(final ObjectInput in) throws IOException {
this.length= in.readInt();
this.boolValues= new byte[this.length];
in.readFully(this.boolValues, 0, this.length);
}
@Override
public void writeExternal(final ObjectOutput out) throws IOException {
out.writeInt(this.length);
out.write(this.boolValues, 0, this.length);
}
@Override
protected final boolean isStructOnly() {
return false;
}
protected final int length() {
return this.length;
}
@Override
public final long getLength() {
return this.length;
}
@Override
public boolean isNA(final int idx) {
return (this.boolValues[idx] == NA_logical_BYTE);
}
@Override
public boolean isNA(final long idx) {
if (idx < 0 || idx >= length()) {
throw new IndexOutOfBoundsException(Long.toString(idx));
}
return (this.boolValues[(int) idx] == NA_logical_BYTE);
}
@Override
public boolean isMissing(final int idx) {
return (this.boolValues[idx] == NA_logical_BYTE);
}
@Override
public boolean isMissing(final long idx) {
if (idx < 0 || idx >= length()) {
throw new IndexOutOfBoundsException(Long.toString(idx));
}
return (this.boolValues[(int) idx] == NA_logical_BYTE);
}
@Override
public void setNA(final int idx) {
this.boolValues[idx]= NA_logical_BYTE;
}
@Override
public void setNA(final long idx) {
if (idx < 0 || idx >= length()) {
throw new IndexOutOfBoundsException(Long.toString(idx));
}
this.boolValues[(int) idx]= NA_logical_BYTE;
}
@Override
public boolean getLogi(final int idx) {
return (this.boolValues[idx] == TRUE_BYTE);
}
@Override
public boolean getLogi(final long idx) {
if (idx < 0 || idx >= length()) {
throw new IndexOutOfBoundsException(Long.toString(idx));
}
return (this.boolValues[(int) idx] == TRUE_BYTE);
}
@Override
public void setLogi(final int idx, final boolean value) {
this.boolValues[idx]= (value) ? TRUE_BYTE : FALSE_BYTE;
}
@Override
public void setLogi(final long idx, final boolean value) {
if (idx < 0 || idx >= length()) {
throw new IndexOutOfBoundsException(Long.toString(idx));
}
this.boolValues[(int) idx]= (value) ? TRUE_BYTE : FALSE_BYTE;
}
private void prepareInsert(final int[] idxs) {
this.boolValues= prepareInsert(this.boolValues, this.length, idxs);
this.length+= idxs.length;
}
public void insertLogi(final int idx, final boolean value) {
prepareInsert(new int[] { idx });
this.boolValues[idx]= value ? TRUE_BYTE : FALSE_BYTE;
}
@Override
public void insertNA(final int idx) {
prepareInsert(new int[] { idx });
this.boolValues[idx]= NA_logical_BYTE;
}
@Override
public void insertNA(final int[] idxs) {
if (idxs.length == 0) {
return;
}
prepareInsert(idxs);
for (int idx= 0; idx < idxs.length; idx++) {
this.boolValues[idxs[idx]+idx]= NA_logical_BYTE;
}
}
@Override
public void remove(final int idx) {
this.boolValues= remove(this.boolValues, this.length, new int[] { idx });
this.length--;
}
@Override
public void remove(final int[] idxs) {
this.boolValues= remove(this.boolValues, this.length, idxs);
this.length-= idxs.length;
}
@Override
public Boolean get(final int idx) {
if (idx < 0 || idx >= length()) {
throw new IndexOutOfBoundsException(Long.toString(idx));
}
switch(this.boolValues[idx]) {
case TRUE_BYTE:
return Boolean.TRUE;
case FALSE_BYTE:
return Boolean.FALSE;
default:
return null;
}
}
@Override
public Boolean get(final long idx) {
if (idx < 0 || idx >= length()) {
throw new IndexOutOfBoundsException(Long.toString(idx));
}
switch(this.boolValues[(int) idx]) {
case TRUE_BYTE:
return Boolean.TRUE;
case FALSE_BYTE:
return Boolean.FALSE;
default:
return null;
}
}
@Override
public Boolean[] toArray() {
final Boolean[] array= new Boolean[length()];
final byte[] bools= this.boolValues;
for (int i= 0; i < array.length; i++) {
switch(bools[i]) {
case TRUE_BYTE:
array[i]= Boolean.TRUE;
continue;
case FALSE_BYTE:
array[i]= Boolean.FALSE;
continue;
default:
continue;
}
}
return array;
}
@Override
public long indexOfNA(long fromIdx) {
if (fromIdx >= Integer.MAX_VALUE) {
return -1;
}
if (fromIdx < 0) {
fromIdx= 0;
}
final long l= getLength();
final byte[] bools= this.boolValues;
for (int i= (int) fromIdx; i < l; i++) {
if (bools[i] == NA_logical_BYTE) {
return i;
}
}
return -1;
}
@Override
public long indexOf(final int integer, long fromIdx) {
if (fromIdx >= Integer.MAX_VALUE
|| integer == NA_integer_INT ) {
return -1;
}
if (fromIdx < 0) {
fromIdx= 0;
}
final int l= length();
final byte[] bools= this.boolValues;
if (integer != 0) {
for (int i= (int) fromIdx; i < l; i++) {
if (bools[i] == TRUE_BYTE) {
return i;
}
}
}
else {
for (int i= (int) fromIdx; i < l; i++) {
if (bools[i] == FALSE_BYTE) {
return i;
}
}
}
return -1;
}
}