diff --git a/core/trino-main/src/main/java/io/trino/operator/aggregation/listagg/GroupListaggAggregationState.java b/core/trino-main/src/main/java/io/trino/operator/aggregation/listagg/GroupListaggAggregationState.java index 863c3987c51a..0da363839766 100644 --- a/core/trino-main/src/main/java/io/trino/operator/aggregation/listagg/GroupListaggAggregationState.java +++ b/core/trino-main/src/main/java/io/trino/operator/aggregation/listagg/GroupListaggAggregationState.java @@ -14,6 +14,7 @@ package io.trino.operator.aggregation.listagg; import com.google.common.primitives.Ints; +import io.airlift.slice.DynamicSliceOutput; import io.airlift.slice.SliceOutput; import io.trino.spi.block.ValueBlock; import io.trino.spi.block.VariableWidthBlockBuilder; @@ -40,6 +41,7 @@ public class GroupListaggAggregationState private static final VarHandle LONG_HANDLE = MethodHandles.byteArrayViewVarHandle(long[].class, LITTLE_ENDIAN); private final int recordNextIndexOffset; + private final DynamicSliceOutput out = new DynamicSliceOutput(0); private long[] groupHeadPositions = new long[0]; private long[] groupTailPositions = new long[0]; @@ -130,7 +132,9 @@ public void write(VariableWidthBlockBuilder blockBuilder) blockBuilder.appendNull(); return; } - blockBuilder.buildEntry(this::write); + out.reset(); + write(out); + blockBuilder.writeEntry(out.slice()); } private void write(SliceOutput out) diff --git a/core/trino-main/src/main/java/io/trino/operator/aggregation/listagg/SingleListaggAggregationState.java b/core/trino-main/src/main/java/io/trino/operator/aggregation/listagg/SingleListaggAggregationState.java index 2fcd62699907..892508fc51d1 100644 --- a/core/trino-main/src/main/java/io/trino/operator/aggregation/listagg/SingleListaggAggregationState.java +++ b/core/trino-main/src/main/java/io/trino/operator/aggregation/listagg/SingleListaggAggregationState.java @@ -13,6 +13,7 @@ */ package io.trino.operator.aggregation.listagg; +import io.airlift.slice.DynamicSliceOutput; import io.airlift.slice.SliceOutput; import io.trino.spi.block.SqlRow; import io.trino.spi.block.VariableWidthBlockBuilder; @@ -25,6 +26,8 @@ public class SingleListaggAggregationState extends AbstractListaggAggregationState { + private final DynamicSliceOutput out = new DynamicSliceOutput(0); + private SqlRow tempSerializedState; public SingleListaggAggregationState() @@ -46,7 +49,9 @@ public void write(VariableWidthBlockBuilder blockBuilder) blockBuilder.appendNull(); return; } - blockBuilder.buildEntry(this::writeNotGrouped); + out.reset(); + writeNotGrouped(out); + blockBuilder.writeEntry(out.slice()); } private void writeNotGrouped(SliceOutput out) diff --git a/core/trino-main/src/main/java/io/trino/operator/output/BytePositionsAppender.java b/core/trino-main/src/main/java/io/trino/operator/output/BytePositionsAppender.java deleted file mode 100644 index 8599de929e4b..000000000000 --- a/core/trino-main/src/main/java/io/trino/operator/output/BytePositionsAppender.java +++ /dev/null @@ -1,212 +0,0 @@ -/* - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -package io.trino.operator.output; - -import io.trino.spi.block.Block; -import io.trino.spi.block.ByteArrayBlock; -import io.trino.spi.block.RunLengthEncodedBlock; -import io.trino.spi.block.ValueBlock; -import it.unimi.dsi.fastutil.ints.IntArrayList; - -import java.util.Arrays; -import java.util.Optional; - -import static com.google.common.base.Preconditions.checkArgument; -import static io.airlift.slice.SizeOf.instanceSize; -import static io.airlift.slice.SizeOf.sizeOf; -import static io.trino.operator.output.PositionsAppenderUtil.calculateBlockResetSize; -import static io.trino.operator.output.PositionsAppenderUtil.calculateNewArraySize; -import static java.lang.Math.max; - -public class BytePositionsAppender - implements PositionsAppender -{ - private static final int INSTANCE_SIZE = instanceSize(BytePositionsAppender.class); - private static final Block NULL_VALUE_BLOCK = new ByteArrayBlock(1, Optional.of(new boolean[] {true}), new byte[1]); - - private boolean initialized; - private int initialEntryCount; - - private int positionCount; - private boolean hasNullValue; - private boolean hasNonNullValue; - - // it is assumed that these arrays are the same length - private boolean[] valueIsNull = new boolean[0]; - private byte[] values = new byte[0]; - - private long retainedSizeInBytes; - private long sizeInBytes; - - public BytePositionsAppender(int expectedEntries) - { - this.initialEntryCount = max(expectedEntries, 1); - - updateRetainedSize(); - } - - @Override - public void append(IntArrayList positions, ValueBlock block) - { - checkArgument(block instanceof ByteArrayBlock, "Block must be instance of %s", ByteArrayBlock.class); - - if (positions.isEmpty()) { - return; - } - int[] positionArray = positions.elements(); - int positionsSize = positions.size(); - ensureCapacity(positionCount + positionsSize); - - if (block.mayHaveNull()) { - for (int i = 0; i < positionsSize; i++) { - int position = positionArray[i]; - boolean isNull = block.isNull(position); - int positionIndex = positionCount + i; - if (isNull) { - valueIsNull[positionIndex] = true; - hasNullValue = true; - } - else { - values[positionIndex] = block.getByte(position, 0); - hasNonNullValue = true; - } - } - positionCount += positionsSize; - } - else { - for (int i = 0; i < positionsSize; i++) { - int position = positionArray[i]; - values[positionCount + i] = block.getByte(position, 0); - } - positionCount += positionsSize; - hasNonNullValue = true; - } - - updateSize(positionsSize); - } - - @Override - public void appendRle(ValueBlock block, int rlePositionCount) - { - checkArgument(block instanceof ByteArrayBlock, "Block must be instance of %s", ByteArrayBlock.class); - - if (rlePositionCount == 0) { - return; - } - int sourcePosition = 0; - ensureCapacity(positionCount + rlePositionCount); - if (block.isNull(sourcePosition)) { - Arrays.fill(valueIsNull, positionCount, positionCount + rlePositionCount, true); - hasNullValue = true; - } - else { - byte value = block.getByte(sourcePosition, 0); - Arrays.fill(values, positionCount, positionCount + rlePositionCount, value); - hasNonNullValue = true; - } - positionCount += rlePositionCount; - - updateSize(rlePositionCount); - } - - @Override - public void append(int sourcePosition, ValueBlock source) - { - checkArgument(source instanceof ByteArrayBlock, "Block must be instance of %s", ByteArrayBlock.class); - - ensureCapacity(positionCount + 1); - if (source.isNull(sourcePosition)) { - valueIsNull[positionCount] = true; - hasNullValue = true; - } - else { - values[positionCount] = source.getByte(sourcePosition, 0); - hasNonNullValue = true; - } - positionCount++; - - updateSize(1); - } - - @Override - public Block build() - { - Block result; - if (hasNonNullValue) { - result = new ByteArrayBlock(positionCount, hasNullValue ? Optional.of(valueIsNull) : Optional.empty(), values); - } - else { - result = RunLengthEncodedBlock.create(NULL_VALUE_BLOCK, positionCount); - } - reset(); - return result; - } - - @Override - public long getRetainedSizeInBytes() - { - return retainedSizeInBytes; - } - - @Override - public long getSizeInBytes() - { - return sizeInBytes; - } - - @Override - public void reset() - { - initialEntryCount = calculateBlockResetSize(positionCount); - initialized = false; - valueIsNull = new boolean[0]; - values = new byte[0]; - positionCount = 0; - sizeInBytes = 0; - hasNonNullValue = false; - hasNullValue = false; - updateRetainedSize(); - } - - private void ensureCapacity(int capacity) - { - if (values.length >= capacity) { - return; - } - - int newSize; - if (initialized) { - newSize = calculateNewArraySize(values.length); - } - else { - newSize = initialEntryCount; - initialized = true; - } - newSize = max(newSize, capacity); - - valueIsNull = Arrays.copyOf(valueIsNull, newSize); - values = Arrays.copyOf(values, newSize); - updateRetainedSize(); - } - - private void updateSize(long positionsSize) - { - sizeInBytes += ByteArrayBlock.SIZE_IN_BYTES_PER_POSITION * positionsSize; - } - - private void updateRetainedSize() - { - retainedSizeInBytes = INSTANCE_SIZE + sizeOf(valueIsNull) + sizeOf(values); - } -} diff --git a/core/trino-main/src/main/java/io/trino/operator/output/Fixed12PositionsAppender.java b/core/trino-main/src/main/java/io/trino/operator/output/Fixed12PositionsAppender.java deleted file mode 100644 index 685db0271293..000000000000 --- a/core/trino-main/src/main/java/io/trino/operator/output/Fixed12PositionsAppender.java +++ /dev/null @@ -1,229 +0,0 @@ -/* - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -package io.trino.operator.output; - -import io.trino.spi.block.Block; -import io.trino.spi.block.Fixed12Block; -import io.trino.spi.block.RunLengthEncodedBlock; -import io.trino.spi.block.ValueBlock; -import it.unimi.dsi.fastutil.ints.IntArrayList; - -import java.util.Arrays; -import java.util.Optional; - -import static com.google.common.base.Preconditions.checkArgument; -import static io.airlift.slice.SizeOf.SIZE_OF_INT; -import static io.airlift.slice.SizeOf.instanceSize; -import static io.airlift.slice.SizeOf.sizeOf; -import static io.trino.operator.output.PositionsAppenderUtil.calculateBlockResetSize; -import static io.trino.operator.output.PositionsAppenderUtil.calculateNewArraySize; -import static java.lang.Math.max; - -public class Fixed12PositionsAppender - implements PositionsAppender -{ - private static final int INSTANCE_SIZE = instanceSize(Fixed12PositionsAppender.class); - private static final Block NULL_VALUE_BLOCK = new Fixed12Block(1, Optional.of(new boolean[] {true}), new int[3]); - - private boolean initialized; - private int initialEntryCount; - - private int positionCount; - private boolean hasNullValue; - private boolean hasNonNullValue; - - // it is assumed that these arrays are the same length - private boolean[] valueIsNull = new boolean[0]; - private int[] values = new int[0]; - - private long retainedSizeInBytes; - private long sizeInBytes; - - public Fixed12PositionsAppender(int expectedEntries) - { - this.initialEntryCount = max(expectedEntries, 1); - - updateRetainedSize(); - } - - @Override - public void append(IntArrayList positions, ValueBlock block) - { - checkArgument(block instanceof Fixed12Block, "Block must be instance of %s", Fixed12Block.class); - - if (positions.isEmpty()) { - return; - } - int[] positionArray = positions.elements(); - int positionsSize = positions.size(); - ensureCapacity(positionCount + positionsSize); - - if (block.mayHaveNull()) { - for (int i = 0; i < positionsSize; i++) { - int position = positionArray[i]; - boolean isNull = block.isNull(position); - if (isNull) { - valueIsNull[positionCount + i] = true; - hasNullValue = true; - } - else { - int valuesIndex = (positionCount + i) * 3; - values[valuesIndex] = block.getInt(position, 0); - values[valuesIndex + 1] = block.getInt(position, SIZE_OF_INT); - values[valuesIndex + 2] = block.getInt(position, SIZE_OF_INT + SIZE_OF_INT); - hasNonNullValue = true; - } - } - positionCount += positionsSize; - } - else { - for (int i = 0; i < positionsSize; i++) { - int position = positionArray[i]; - int valuesIndex = (positionCount + i) * 3; - values[valuesIndex] = block.getInt(position, 0); - values[valuesIndex + 1] = block.getInt(position, SIZE_OF_INT); - values[valuesIndex + 2] = block.getInt(position, SIZE_OF_INT + SIZE_OF_INT); - } - positionCount += positionsSize; - hasNonNullValue = true; - } - - updateSize(positionsSize); - } - - @Override - public void appendRle(ValueBlock block, int rlePositionCount) - { - checkArgument(block instanceof Fixed12Block, "Block must be instance of %s", Fixed12Block.class); - - if (rlePositionCount == 0) { - return; - } - int sourcePosition = 0; - ensureCapacity(positionCount + rlePositionCount); - if (block.isNull(sourcePosition)) { - Arrays.fill(valueIsNull, positionCount, positionCount + rlePositionCount, true); - hasNullValue = true; - } - else { - int valueHigh = block.getInt(sourcePosition, 0); - int valueMid = block.getInt(sourcePosition, SIZE_OF_INT); - int valueLow = block.getInt(sourcePosition, SIZE_OF_INT + SIZE_OF_INT); - int positionIndex = positionCount * 3; - for (int i = 0; i < rlePositionCount; i++) { - values[positionIndex] = valueHigh; - values[positionIndex + 1] = valueMid; - values[positionIndex + 2] = valueLow; - positionIndex += 3; - } - hasNonNullValue = true; - } - positionCount += rlePositionCount; - - updateSize(rlePositionCount); - } - - @Override - public void append(int sourcePosition, ValueBlock source) - { - checkArgument(source instanceof Fixed12Block, "Block must be instance of %s", Fixed12Block.class); - - ensureCapacity(positionCount + 1); - if (source.isNull(sourcePosition)) { - valueIsNull[positionCount] = true; - hasNullValue = true; - } - else { - int positionIndex = positionCount * 3; - values[positionIndex] = source.getInt(sourcePosition, 0); - values[positionIndex + 1] = source.getInt(sourcePosition, SIZE_OF_INT); - values[positionIndex + 2] = source.getInt(sourcePosition, SIZE_OF_INT + SIZE_OF_INT); - hasNonNullValue = true; - } - positionCount++; - - updateSize(1); - } - - @Override - public Block build() - { - Block result; - if (hasNonNullValue) { - result = new Fixed12Block(positionCount, hasNullValue ? Optional.of(valueIsNull) : Optional.empty(), values); - } - else { - result = RunLengthEncodedBlock.create(NULL_VALUE_BLOCK, positionCount); - } - reset(); - return result; - } - - @Override - public long getRetainedSizeInBytes() - { - return retainedSizeInBytes; - } - - @Override - public long getSizeInBytes() - { - return sizeInBytes; - } - - @Override - public void reset() - { - initialEntryCount = calculateBlockResetSize(positionCount); - initialized = false; - valueIsNull = new boolean[0]; - values = new int[0]; - positionCount = 0; - sizeInBytes = 0; - hasNonNullValue = false; - hasNullValue = false; - updateRetainedSize(); - } - - private void ensureCapacity(int capacity) - { - if (valueIsNull.length >= capacity) { - return; - } - - int newSize; - if (initialized) { - newSize = calculateNewArraySize(valueIsNull.length); - } - else { - newSize = initialEntryCount; - initialized = true; - } - newSize = max(newSize, capacity); - - valueIsNull = Arrays.copyOf(valueIsNull, newSize); - values = Arrays.copyOf(values, newSize * 3); - updateRetainedSize(); - } - - private void updateSize(long positionsSize) - { - sizeInBytes += Fixed12Block.SIZE_IN_BYTES_PER_POSITION * positionsSize; - } - - private void updateRetainedSize() - { - retainedSizeInBytes = INSTANCE_SIZE + sizeOf(valueIsNull) + sizeOf(values); - } -} diff --git a/core/trino-main/src/main/java/io/trino/operator/output/Int128PositionsAppender.java b/core/trino-main/src/main/java/io/trino/operator/output/Int128PositionsAppender.java deleted file mode 100644 index 251a7f25eff4..000000000000 --- a/core/trino-main/src/main/java/io/trino/operator/output/Int128PositionsAppender.java +++ /dev/null @@ -1,227 +0,0 @@ -/* - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -package io.trino.operator.output; - -import io.trino.spi.block.Block; -import io.trino.spi.block.Int128ArrayBlock; -import io.trino.spi.block.RunLengthEncodedBlock; -import io.trino.spi.block.ValueBlock; -import it.unimi.dsi.fastutil.ints.IntArrayList; - -import java.util.Arrays; -import java.util.Optional; - -import static com.google.common.base.Preconditions.checkArgument; -import static io.airlift.slice.SizeOf.SIZE_OF_LONG; -import static io.airlift.slice.SizeOf.instanceSize; -import static io.airlift.slice.SizeOf.sizeOf; -import static io.trino.operator.output.PositionsAppenderUtil.calculateBlockResetSize; -import static io.trino.operator.output.PositionsAppenderUtil.calculateNewArraySize; -import static java.lang.Math.max; - -public class Int128PositionsAppender - implements PositionsAppender -{ - private static final int INSTANCE_SIZE = instanceSize(Int128PositionsAppender.class); - private static final Block NULL_VALUE_BLOCK = new Int128ArrayBlock(1, Optional.of(new boolean[] {true}), new long[2]); - - private boolean initialized; - private int initialEntryCount; - - private int positionCount; - private boolean hasNullValue; - private boolean hasNonNullValue; - - // it is assumed that these arrays are the same length - private boolean[] valueIsNull = new boolean[0]; - private long[] values = new long[0]; - - private long retainedSizeInBytes; - private long sizeInBytes; - - public Int128PositionsAppender(int expectedEntries) - { - this.initialEntryCount = max(expectedEntries, 1); - - updateRetainedSize(); - } - - @Override - public void append(IntArrayList positions, ValueBlock block) - { - checkArgument(block instanceof Int128ArrayBlock, "Block must be instance of %s", Int128ArrayBlock.class); - - if (positions.isEmpty()) { - return; - } - int[] positionArray = positions.elements(); - int positionsSize = positions.size(); - ensureCapacity(positionCount + positionsSize); - - if (block.mayHaveNull()) { - int positionIndex = positionCount * 2; - for (int i = 0; i < positionsSize; i++) { - int position = positionArray[i]; - boolean isNull = block.isNull(position); - - if (isNull) { - valueIsNull[positionCount + i] = true; - hasNullValue = true; - } - else { - values[positionIndex] = block.getLong(position, 0); - values[positionIndex + 1] = block.getLong(position, SIZE_OF_LONG); - hasNonNullValue = true; - } - positionIndex += 2; - } - positionCount += positionsSize; - } - else { - int positionIndex = positionCount * 2; - for (int i = 0; i < positionsSize; i++) { - int position = positionArray[i]; - values[positionIndex] = block.getLong(position, 0); - values[positionIndex + 1] = block.getLong(position, SIZE_OF_LONG); - positionIndex += 2; - } - positionCount += positionsSize; - hasNonNullValue = true; - } - - updateSize(positionsSize); - } - - @Override - public void appendRle(ValueBlock block, int rlePositionCount) - { - checkArgument(block instanceof Int128ArrayBlock, "Block must be instance of %s", Int128ArrayBlock.class); - - if (rlePositionCount == 0) { - return; - } - int sourcePosition = 0; - ensureCapacity(positionCount + rlePositionCount); - if (block.isNull(sourcePosition)) { - Arrays.fill(valueIsNull, positionCount, positionCount + rlePositionCount, true); - hasNullValue = true; - } - else { - long valueHigh = block.getLong(sourcePosition, 0); - long valueLow = block.getLong(sourcePosition, SIZE_OF_LONG); - int positionIndex = positionCount * 2; - for (int i = 0; i < rlePositionCount; i++) { - values[positionIndex] = valueHigh; - values[positionIndex + 1] = valueLow; - positionIndex += 2; - } - hasNonNullValue = true; - } - positionCount += rlePositionCount; - - updateSize(rlePositionCount); - } - - @Override - public void append(int sourcePosition, ValueBlock source) - { - checkArgument(source instanceof Int128ArrayBlock, "Block must be instance of %s", Int128ArrayBlock.class); - - ensureCapacity(positionCount + 1); - if (source.isNull(sourcePosition)) { - valueIsNull[positionCount] = true; - hasNullValue = true; - } - else { - int positionIndex = positionCount * 2; - values[positionIndex] = source.getLong(sourcePosition, 0); - values[positionIndex + 1] = source.getLong(sourcePosition, SIZE_OF_LONG); - hasNonNullValue = true; - } - positionCount++; - - updateSize(1); - } - - @Override - public Block build() - { - Block result; - if (hasNonNullValue) { - result = new Int128ArrayBlock(positionCount, hasNullValue ? Optional.of(valueIsNull) : Optional.empty(), values); - } - else { - result = RunLengthEncodedBlock.create(NULL_VALUE_BLOCK, positionCount); - } - reset(); - return result; - } - - @Override - public long getRetainedSizeInBytes() - { - return retainedSizeInBytes; - } - - @Override - public long getSizeInBytes() - { - return sizeInBytes; - } - - @Override - public void reset() - { - initialEntryCount = calculateBlockResetSize(positionCount); - initialized = false; - valueIsNull = new boolean[0]; - values = new long[0]; - positionCount = 0; - sizeInBytes = 0; - hasNonNullValue = false; - hasNullValue = false; - updateRetainedSize(); - } - - private void ensureCapacity(int capacity) - { - if (valueIsNull.length >= capacity) { - return; - } - - int newSize; - if (initialized) { - newSize = calculateNewArraySize(valueIsNull.length); - } - else { - newSize = initialEntryCount; - initialized = true; - } - newSize = max(newSize, capacity); - - valueIsNull = Arrays.copyOf(valueIsNull, newSize); - values = Arrays.copyOf(values, newSize * 2); - updateRetainedSize(); - } - - private void updateSize(long positionsSize) - { - sizeInBytes += Int128ArrayBlock.SIZE_IN_BYTES_PER_POSITION * positionsSize; - } - - private void updateRetainedSize() - { - retainedSizeInBytes = INSTANCE_SIZE + sizeOf(valueIsNull) + sizeOf(values); - } -} diff --git a/core/trino-main/src/main/java/io/trino/operator/output/IntPositionsAppender.java b/core/trino-main/src/main/java/io/trino/operator/output/IntPositionsAppender.java deleted file mode 100644 index bcb7d73b046d..000000000000 --- a/core/trino-main/src/main/java/io/trino/operator/output/IntPositionsAppender.java +++ /dev/null @@ -1,212 +0,0 @@ -/* - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -package io.trino.operator.output; - -import io.trino.spi.block.Block; -import io.trino.spi.block.IntArrayBlock; -import io.trino.spi.block.RunLengthEncodedBlock; -import io.trino.spi.block.ValueBlock; -import it.unimi.dsi.fastutil.ints.IntArrayList; - -import java.util.Arrays; -import java.util.Optional; - -import static com.google.common.base.Preconditions.checkArgument; -import static io.airlift.slice.SizeOf.instanceSize; -import static io.airlift.slice.SizeOf.sizeOf; -import static io.trino.operator.output.PositionsAppenderUtil.calculateBlockResetSize; -import static io.trino.operator.output.PositionsAppenderUtil.calculateNewArraySize; -import static java.lang.Math.max; - -public class IntPositionsAppender - implements PositionsAppender -{ - private static final int INSTANCE_SIZE = instanceSize(IntPositionsAppender.class); - private static final Block NULL_VALUE_BLOCK = new IntArrayBlock(1, Optional.of(new boolean[] {true}), new int[1]); - - private boolean initialized; - private int initialEntryCount; - - private int positionCount; - private boolean hasNullValue; - private boolean hasNonNullValue; - - // it is assumed that these arrays are the same length - private boolean[] valueIsNull = new boolean[0]; - private int[] values = new int[0]; - - private long retainedSizeInBytes; - private long sizeInBytes; - - public IntPositionsAppender(int expectedEntries) - { - this.initialEntryCount = max(expectedEntries, 1); - - updateRetainedSize(); - } - - @Override - public void append(IntArrayList positions, ValueBlock block) - { - checkArgument(block instanceof IntArrayBlock, "Block must be instance of %s", IntArrayBlock.class); - - if (positions.isEmpty()) { - return; - } - int[] positionArray = positions.elements(); - int positionsSize = positions.size(); - ensureCapacity(positionCount + positionsSize); - - if (block.mayHaveNull()) { - for (int i = 0; i < positionsSize; i++) { - int position = positionArray[i]; - boolean isNull = block.isNull(position); - int positionIndex = positionCount + i; - if (isNull) { - valueIsNull[positionIndex] = true; - hasNullValue = true; - } - else { - values[positionIndex] = block.getInt(position, 0); - hasNonNullValue = true; - } - } - positionCount += positionsSize; - } - else { - for (int i = 0; i < positionsSize; i++) { - int position = positionArray[i]; - values[positionCount + i] = block.getInt(position, 0); - } - positionCount += positionsSize; - hasNonNullValue = true; - } - - updateSize(positionsSize); - } - - @Override - public void appendRle(ValueBlock block, int rlePositionCount) - { - checkArgument(block instanceof IntArrayBlock, "Block must be instance of %s", IntArrayBlock.class); - - if (rlePositionCount == 0) { - return; - } - int sourcePosition = 0; - ensureCapacity(positionCount + rlePositionCount); - if (block.isNull(sourcePosition)) { - Arrays.fill(valueIsNull, positionCount, positionCount + rlePositionCount, true); - hasNullValue = true; - } - else { - int value = block.getInt(sourcePosition, 0); - Arrays.fill(values, positionCount, positionCount + rlePositionCount, value); - hasNonNullValue = true; - } - positionCount += rlePositionCount; - - updateSize(rlePositionCount); - } - - @Override - public void append(int sourcePosition, ValueBlock source) - { - checkArgument(source instanceof IntArrayBlock, "Block must be instance of %s", IntArrayBlock.class); - - ensureCapacity(positionCount + 1); - if (source.isNull(sourcePosition)) { - valueIsNull[positionCount] = true; - hasNullValue = true; - } - else { - values[positionCount] = source.getInt(sourcePosition, 0); - hasNonNullValue = true; - } - positionCount++; - - updateSize(1); - } - - @Override - public Block build() - { - Block result; - if (hasNonNullValue) { - result = new IntArrayBlock(positionCount, hasNullValue ? Optional.of(valueIsNull) : Optional.empty(), values); - } - else { - result = RunLengthEncodedBlock.create(NULL_VALUE_BLOCK, positionCount); - } - reset(); - return result; - } - - @Override - public long getRetainedSizeInBytes() - { - return retainedSizeInBytes; - } - - @Override - public long getSizeInBytes() - { - return sizeInBytes; - } - - @Override - public void reset() - { - initialEntryCount = calculateBlockResetSize(positionCount); - initialized = false; - valueIsNull = new boolean[0]; - values = new int[0]; - positionCount = 0; - sizeInBytes = 0; - hasNonNullValue = false; - hasNullValue = false; - updateRetainedSize(); - } - - private void ensureCapacity(int capacity) - { - if (values.length >= capacity) { - return; - } - - int newSize; - if (initialized) { - newSize = calculateNewArraySize(values.length); - } - else { - newSize = initialEntryCount; - initialized = true; - } - newSize = max(newSize, capacity); - - valueIsNull = Arrays.copyOf(valueIsNull, newSize); - values = Arrays.copyOf(values, newSize); - updateRetainedSize(); - } - - private void updateSize(long positionsSize) - { - sizeInBytes += IntArrayBlock.SIZE_IN_BYTES_PER_POSITION * positionsSize; - } - - private void updateRetainedSize() - { - retainedSizeInBytes = INSTANCE_SIZE + sizeOf(valueIsNull) + sizeOf(values); - } -} diff --git a/core/trino-main/src/main/java/io/trino/operator/output/LongPositionsAppender.java b/core/trino-main/src/main/java/io/trino/operator/output/LongPositionsAppender.java deleted file mode 100644 index 6fc555f02a01..000000000000 --- a/core/trino-main/src/main/java/io/trino/operator/output/LongPositionsAppender.java +++ /dev/null @@ -1,212 +0,0 @@ -/* - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -package io.trino.operator.output; - -import io.trino.spi.block.Block; -import io.trino.spi.block.LongArrayBlock; -import io.trino.spi.block.RunLengthEncodedBlock; -import io.trino.spi.block.ValueBlock; -import it.unimi.dsi.fastutil.ints.IntArrayList; - -import java.util.Arrays; -import java.util.Optional; - -import static com.google.common.base.Preconditions.checkArgument; -import static io.airlift.slice.SizeOf.instanceSize; -import static io.airlift.slice.SizeOf.sizeOf; -import static io.trino.operator.output.PositionsAppenderUtil.calculateBlockResetSize; -import static io.trino.operator.output.PositionsAppenderUtil.calculateNewArraySize; -import static java.lang.Math.max; - -public class LongPositionsAppender - implements PositionsAppender -{ - private static final int INSTANCE_SIZE = instanceSize(LongPositionsAppender.class); - private static final Block NULL_VALUE_BLOCK = new LongArrayBlock(1, Optional.of(new boolean[] {true}), new long[1]); - - private boolean initialized; - private int initialEntryCount; - - private int positionCount; - private boolean hasNullValue; - private boolean hasNonNullValue; - - // it is assumed that these arrays are the same length - private boolean[] valueIsNull = new boolean[0]; - private long[] values = new long[0]; - - private long retainedSizeInBytes; - private long sizeInBytes; - - public LongPositionsAppender(int expectedEntries) - { - this.initialEntryCount = max(expectedEntries, 1); - - updateRetainedSize(); - } - - @Override - public void append(IntArrayList positions, ValueBlock block) - { - checkArgument(block instanceof LongArrayBlock, "Block must be instance of %s", LongArrayBlock.class); - - if (positions.isEmpty()) { - return; - } - int[] positionArray = positions.elements(); - int positionsSize = positions.size(); - ensureCapacity(positionCount + positionsSize); - - if (block.mayHaveNull()) { - for (int i = 0; i < positionsSize; i++) { - int position = positionArray[i]; - int positionIndex = positionCount + i; - boolean isNull = block.isNull(position); - if (isNull) { - valueIsNull[positionIndex] = true; - hasNullValue = true; - } - else { - values[positionIndex] = block.getLong(position, 0); - hasNonNullValue = true; - } - } - positionCount += positionsSize; - } - else { - for (int i = 0; i < positionsSize; i++) { - int position = positionArray[i]; - values[positionCount + i] = block.getLong(position, 0); - } - positionCount += positionsSize; - hasNonNullValue = true; - } - - updateSize(positionsSize); - } - - @Override - public void appendRle(ValueBlock block, int rlePositionCount) - { - checkArgument(block instanceof LongArrayBlock, "Block must be instance of %s", LongArrayBlock.class); - - if (rlePositionCount == 0) { - return; - } - int sourcePosition = 0; - ensureCapacity(positionCount + rlePositionCount); - if (block.isNull(sourcePosition)) { - Arrays.fill(valueIsNull, positionCount, positionCount + rlePositionCount, true); - hasNullValue = true; - } - else { - long value = block.getLong(sourcePosition, 0); - Arrays.fill(values, positionCount, positionCount + rlePositionCount, value); - hasNonNullValue = true; - } - positionCount += rlePositionCount; - - updateSize(rlePositionCount); - } - - @Override - public void append(int sourcePosition, ValueBlock source) - { - checkArgument(source instanceof LongArrayBlock, "Block must be instance of %s", LongArrayBlock.class); - - ensureCapacity(positionCount + 1); - if (source.isNull(sourcePosition)) { - valueIsNull[positionCount] = true; - hasNullValue = true; - } - else { - values[positionCount] = source.getLong(sourcePosition, 0); - hasNonNullValue = true; - } - positionCount++; - - updateSize(1); - } - - @Override - public Block build() - { - Block result; - if (hasNonNullValue) { - result = new LongArrayBlock(positionCount, hasNullValue ? Optional.of(valueIsNull) : Optional.empty(), values); - } - else { - result = RunLengthEncodedBlock.create(NULL_VALUE_BLOCK, positionCount); - } - reset(); - return result; - } - - @Override - public long getRetainedSizeInBytes() - { - return retainedSizeInBytes; - } - - @Override - public long getSizeInBytes() - { - return sizeInBytes; - } - - @Override - public void reset() - { - initialEntryCount = calculateBlockResetSize(positionCount); - initialized = false; - valueIsNull = new boolean[0]; - values = new long[0]; - positionCount = 0; - sizeInBytes = 0; - hasNonNullValue = false; - hasNullValue = false; - updateRetainedSize(); - } - - private void ensureCapacity(int capacity) - { - if (values.length >= capacity) { - return; - } - - int newSize; - if (initialized) { - newSize = calculateNewArraySize(values.length); - } - else { - newSize = initialEntryCount; - initialized = true; - } - newSize = max(newSize, capacity); - - valueIsNull = Arrays.copyOf(valueIsNull, newSize); - values = Arrays.copyOf(values, newSize); - updateRetainedSize(); - } - - private void updateSize(long positionsSize) - { - sizeInBytes += LongArrayBlock.SIZE_IN_BYTES_PER_POSITION * positionsSize; - } - - private void updateRetainedSize() - { - retainedSizeInBytes = INSTANCE_SIZE + sizeOf(valueIsNull) + sizeOf(values); - } -} diff --git a/core/trino-main/src/main/java/io/trino/operator/output/PositionsAppenderFactory.java b/core/trino-main/src/main/java/io/trino/operator/output/PositionsAppenderFactory.java index 34eab30e020e..e972baad6d22 100644 --- a/core/trino-main/src/main/java/io/trino/operator/output/PositionsAppenderFactory.java +++ b/core/trino-main/src/main/java/io/trino/operator/output/PositionsAppenderFactory.java @@ -13,14 +13,9 @@ */ package io.trino.operator.output; -import io.trino.spi.block.ByteArrayBlock; -import io.trino.spi.block.Fixed12Block; -import io.trino.spi.block.Int128ArrayBlock; -import io.trino.spi.block.IntArrayBlock; -import io.trino.spi.block.LongArrayBlock; import io.trino.spi.block.RowBlock; -import io.trino.spi.block.ShortArrayBlock; import io.trino.spi.block.VariableWidthBlock; +import io.trino.spi.block.VariableWidthBlockBuilder; import io.trino.spi.type.RowType; import io.trino.spi.type.Type; import io.trino.type.BlockTypeOperators; @@ -28,11 +23,14 @@ import java.util.Optional; +import static io.trino.operator.output.PositionsAppenderUtil.MAX_ARRAY_SIZE; +import static java.lang.Math.min; import static java.util.Objects.requireNonNull; public class PositionsAppenderFactory { private final BlockTypeOperators blockTypeOperators; + private static final int EXPECTED_VARIABLE_WIDTH_BYTES_PER_ENTRY = 32; public PositionsAppenderFactory(BlockTypeOperators blockTypeOperators) { @@ -50,30 +48,15 @@ public UnnestingPositionsAppender create(Type type, int expectedPositions, long private PositionsAppender createPrimitiveAppender(Type type, int expectedPositions, long maxPageSizeInBytes) { - if (type.getValueBlockType() == ByteArrayBlock.class) { - return new BytePositionsAppender(expectedPositions); - } - if (type.getValueBlockType() == ShortArrayBlock.class) { - return new ShortPositionsAppender(expectedPositions); - } - if (type.getValueBlockType() == IntArrayBlock.class) { - return new IntPositionsAppender(expectedPositions); - } - if (type.getValueBlockType() == LongArrayBlock.class) { - return new LongPositionsAppender(expectedPositions); - } - if (type.getValueBlockType() == Fixed12Block.class) { - return new Fixed12PositionsAppender(expectedPositions); - } - if (type.getValueBlockType() == Int128ArrayBlock.class) { - return new Int128PositionsAppender(expectedPositions); - } - if (type.getValueBlockType() == VariableWidthBlock.class) { - return new SlicePositionsAppender(expectedPositions, maxPageSizeInBytes); - } if (type.getValueBlockType() == RowBlock.class) { return RowPositionsAppender.createRowAppender(this, (RowType) type, expectedPositions, maxPageSizeInBytes); } + if (type.getValueBlockType() == VariableWidthBlock.class) { + // it is guaranteed Math.min will not overflow; safe to cast + int expectedBytes = (int) min((long) expectedPositions * EXPECTED_VARIABLE_WIDTH_BYTES_PER_ENTRY, maxPageSizeInBytes); + expectedBytes = min(expectedBytes, MAX_ARRAY_SIZE); + return new TypedPositionsAppender(new VariableWidthBlockBuilder(null, expectedPositions, expectedBytes)); + } return new TypedPositionsAppender(type, expectedPositions); } } diff --git a/core/trino-main/src/main/java/io/trino/operator/output/PositionsAppenderUtil.java b/core/trino-main/src/main/java/io/trino/operator/output/PositionsAppenderUtil.java index 0d1d6b642096..b6efce94d2f1 100644 --- a/core/trino-main/src/main/java/io/trino/operator/output/PositionsAppenderUtil.java +++ b/core/trino-main/src/main/java/io/trino/operator/output/PositionsAppenderUtil.java @@ -24,9 +24,7 @@ final class PositionsAppenderUtil // See java.util.ArrayList for an explanation static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8; - private PositionsAppenderUtil() - { - } + private PositionsAppenderUtil() {} // Copied from io.trino.spi.block.BlockUtil#calculateNewArraySize static int calculateNewArraySize(int currentSize) @@ -61,14 +59,4 @@ else if (newSize > MAX_ARRAY_SIZE) { } return (int) newSize; } - - // Copied from io.trino.spi.block.BlockUtil#calculateBlockResetBytes - static int calculateBlockResetBytes(int currentBytes) - { - long newBytes = (long) ceil(currentBytes * BLOCK_RESET_SKEW); - if (newBytes > MAX_ARRAY_SIZE) { - return MAX_ARRAY_SIZE; - } - return (int) newBytes; - } } diff --git a/core/trino-main/src/main/java/io/trino/operator/output/ShortPositionsAppender.java b/core/trino-main/src/main/java/io/trino/operator/output/ShortPositionsAppender.java deleted file mode 100644 index 16739ae1ea04..000000000000 --- a/core/trino-main/src/main/java/io/trino/operator/output/ShortPositionsAppender.java +++ /dev/null @@ -1,212 +0,0 @@ -/* - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -package io.trino.operator.output; - -import io.trino.spi.block.Block; -import io.trino.spi.block.RunLengthEncodedBlock; -import io.trino.spi.block.ShortArrayBlock; -import io.trino.spi.block.ValueBlock; -import it.unimi.dsi.fastutil.ints.IntArrayList; - -import java.util.Arrays; -import java.util.Optional; - -import static com.google.common.base.Preconditions.checkArgument; -import static io.airlift.slice.SizeOf.instanceSize; -import static io.airlift.slice.SizeOf.sizeOf; -import static io.trino.operator.output.PositionsAppenderUtil.calculateBlockResetSize; -import static io.trino.operator.output.PositionsAppenderUtil.calculateNewArraySize; -import static java.lang.Math.max; - -public class ShortPositionsAppender - implements PositionsAppender -{ - private static final int INSTANCE_SIZE = instanceSize(ShortPositionsAppender.class); - private static final Block NULL_VALUE_BLOCK = new ShortArrayBlock(1, Optional.of(new boolean[] {true}), new short[1]); - - private boolean initialized; - private int initialEntryCount; - - private int positionCount; - private boolean hasNullValue; - private boolean hasNonNullValue; - - // it is assumed that these arrays are the same length - private boolean[] valueIsNull = new boolean[0]; - private short[] values = new short[0]; - - private long retainedSizeInBytes; - private long sizeInBytes; - - public ShortPositionsAppender(int expectedEntries) - { - this.initialEntryCount = max(expectedEntries, 1); - - updateRetainedSize(); - } - - @Override - public void append(IntArrayList positions, ValueBlock block) - { - checkArgument(block instanceof ShortArrayBlock, "Block must be instance of %s", ShortArrayBlock.class); - - if (positions.isEmpty()) { - return; - } - int[] positionArray = positions.elements(); - int positionsSize = positions.size(); - ensureCapacity(positionCount + positionsSize); - - if (block.mayHaveNull()) { - for (int i = 0; i < positionsSize; i++) { - int position = positionArray[i]; - boolean isNull = block.isNull(position); - int positionIndex = positionCount + i; - if (isNull) { - valueIsNull[positionIndex] = true; - hasNullValue = true; - } - else { - values[positionIndex] = block.getShort(position, 0); - hasNonNullValue = true; - } - } - positionCount += positionsSize; - } - else { - for (int i = 0; i < positionsSize; i++) { - int position = positionArray[i]; - values[positionCount + i] = block.getShort(position, 0); - } - positionCount += positionsSize; - hasNonNullValue = true; - } - - updateSize(positionsSize); - } - - @Override - public void appendRle(ValueBlock block, int rlePositionCount) - { - checkArgument(block instanceof ShortArrayBlock, "Block must be instance of %s", ShortArrayBlock.class); - - if (rlePositionCount == 0) { - return; - } - int sourcePosition = 0; - ensureCapacity(positionCount + rlePositionCount); - if (block.isNull(sourcePosition)) { - Arrays.fill(valueIsNull, positionCount, positionCount + rlePositionCount, true); - hasNullValue = true; - } - else { - short value = block.getShort(sourcePosition, 0); - Arrays.fill(values, positionCount, positionCount + rlePositionCount, value); - hasNonNullValue = true; - } - positionCount += rlePositionCount; - - updateSize(rlePositionCount); - } - - @Override - public void append(int sourcePosition, ValueBlock source) - { - checkArgument(source instanceof ShortArrayBlock, "Block must be instance of %s", ShortArrayBlock.class); - - ensureCapacity(positionCount + 1); - if (source.isNull(sourcePosition)) { - valueIsNull[positionCount] = true; - hasNullValue = true; - } - else { - values[positionCount] = source.getShort(sourcePosition, 0); - hasNonNullValue = true; - } - positionCount++; - - updateSize(1); - } - - @Override - public Block build() - { - Block result; - if (hasNonNullValue) { - result = new ShortArrayBlock(positionCount, hasNullValue ? Optional.of(valueIsNull) : Optional.empty(), values); - } - else { - result = RunLengthEncodedBlock.create(NULL_VALUE_BLOCK, positionCount); - } - reset(); - return result; - } - - @Override - public long getRetainedSizeInBytes() - { - return retainedSizeInBytes; - } - - @Override - public long getSizeInBytes() - { - return sizeInBytes; - } - - @Override - public void reset() - { - initialEntryCount = calculateBlockResetSize(positionCount); - initialized = false; - valueIsNull = new boolean[0]; - values = new short[0]; - positionCount = 0; - sizeInBytes = 0; - hasNonNullValue = false; - hasNullValue = false; - updateRetainedSize(); - } - - private void ensureCapacity(int capacity) - { - if (values.length >= capacity) { - return; - } - - int newSize; - if (initialized) { - newSize = calculateNewArraySize(values.length); - } - else { - newSize = initialEntryCount; - initialized = true; - } - newSize = max(newSize, capacity); - - valueIsNull = Arrays.copyOf(valueIsNull, newSize); - values = Arrays.copyOf(values, newSize); - updateRetainedSize(); - } - - private void updateSize(long positionsSize) - { - sizeInBytes += ShortArrayBlock.SIZE_IN_BYTES_PER_POSITION * positionsSize; - } - - private void updateRetainedSize() - { - retainedSizeInBytes = INSTANCE_SIZE + sizeOf(valueIsNull) + sizeOf(values); - } -} diff --git a/core/trino-main/src/main/java/io/trino/operator/output/SlicePositionsAppender.java b/core/trino-main/src/main/java/io/trino/operator/output/SlicePositionsAppender.java deleted file mode 100644 index 204cf679520c..000000000000 --- a/core/trino-main/src/main/java/io/trino/operator/output/SlicePositionsAppender.java +++ /dev/null @@ -1,333 +0,0 @@ -/* - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -package io.trino.operator.output; - -import com.google.common.annotations.VisibleForTesting; -import io.airlift.slice.Slice; -import io.airlift.slice.Slices; -import io.trino.spi.block.Block; -import io.trino.spi.block.RunLengthEncodedBlock; -import io.trino.spi.block.ValueBlock; -import io.trino.spi.block.VariableWidthBlock; -import it.unimi.dsi.fastutil.ints.IntArrayList; - -import java.util.Arrays; -import java.util.Optional; - -import static com.google.common.base.Preconditions.checkArgument; -import static io.airlift.slice.SizeOf.SIZE_OF_BYTE; -import static io.airlift.slice.SizeOf.SIZE_OF_INT; -import static io.airlift.slice.SizeOf.instanceSize; -import static io.airlift.slice.SizeOf.sizeOf; -import static io.airlift.slice.Slices.EMPTY_SLICE; -import static io.trino.operator.output.PositionsAppenderUtil.MAX_ARRAY_SIZE; -import static io.trino.operator.output.PositionsAppenderUtil.calculateBlockResetBytes; -import static io.trino.operator.output.PositionsAppenderUtil.calculateBlockResetSize; -import static io.trino.operator.output.PositionsAppenderUtil.calculateNewArraySize; -import static java.lang.Math.min; -import static java.lang.Math.toIntExact; - -public class SlicePositionsAppender - implements PositionsAppender -{ - private static final int EXPECTED_BYTES_PER_ENTRY = 32; - private static final int INSTANCE_SIZE = instanceSize(SlicePositionsAppender.class); - private static final Block NULL_VALUE_BLOCK = new VariableWidthBlock(1, EMPTY_SLICE, new int[] {0, 0}, Optional.of(new boolean[] {true})); - - private boolean initialized; - private int initialEntryCount; - private int initialBytesSize; - - private byte[] bytes = new byte[0]; - - private boolean hasNullValue; - private boolean hasNonNullValue; - // it is assumed that the offset array is one position longer than the valueIsNull array - private boolean[] valueIsNull = new boolean[0]; - private int[] offsets = new int[1]; - - private int positionCount; - - private long retainedSizeInBytes; - private long sizeInBytes; - - public SlicePositionsAppender(int expectedEntries, long maxPageSizeInBytes) - { - this(expectedEntries, getExpectedBytes(maxPageSizeInBytes, expectedEntries)); - } - - public SlicePositionsAppender(int expectedEntries, int expectedBytes) - { - initialEntryCount = expectedEntries; - initialBytesSize = min(expectedBytes, MAX_ARRAY_SIZE); - - updateRetainedSize(); - } - - @Override - public void append(IntArrayList positions, ValueBlock block) - { - checkArgument(block instanceof VariableWidthBlock, "Block must be instance of %s", VariableWidthBlock.class); - - if (positions.isEmpty()) { - return; - } - ensurePositionCapacity(positionCount + positions.size()); - VariableWidthBlock variableWidthBlock = (VariableWidthBlock) block; - int newByteCount = 0; - int[] lengths = new int[positions.size()]; - int[] sourceOffsets = new int[positions.size()]; - int[] positionArray = positions.elements(); - - if (block.mayHaveNull()) { - for (int i = 0; i < positions.size(); i++) { - int position = positionArray[i]; - int length = variableWidthBlock.getSliceLength(position); - lengths[i] = length; - sourceOffsets[i] = variableWidthBlock.getRawSliceOffset(position); - newByteCount += length; - boolean isNull = block.isNull(position); - valueIsNull[positionCount + i] = isNull; - offsets[positionCount + i + 1] = offsets[positionCount + i] + length; - hasNullValue |= isNull; - hasNonNullValue |= !isNull; - } - } - else { - for (int i = 0; i < positions.size(); i++) { - int position = positionArray[i]; - int length = variableWidthBlock.getSliceLength(position); - lengths[i] = length; - sourceOffsets[i] = variableWidthBlock.getRawSliceOffset(position); - newByteCount += length; - offsets[positionCount + i + 1] = offsets[positionCount + i] + length; - } - hasNonNullValue = true; - } - copyBytes(variableWidthBlock.getRawSlice(), lengths, sourceOffsets, positions.size(), newByteCount); - } - - @Override - public void appendRle(ValueBlock block, int rlePositionCount) - { - checkArgument(block instanceof VariableWidthBlock, "Block must be instance of %s", VariableWidthBlock.class); - - if (rlePositionCount == 0) { - return; - } - ensurePositionCapacity(positionCount + rlePositionCount); - if (block.isNull(0)) { - Arrays.fill(valueIsNull, positionCount, positionCount + rlePositionCount, true); - Arrays.fill(offsets, positionCount + 1, positionCount + rlePositionCount + 1, getCurrentOffset()); - positionCount += rlePositionCount; - - hasNullValue = true; - updateSize(rlePositionCount, 0); - } - else { - hasNonNullValue = true; - duplicateBytes(block.getSlice(0, 0, block.getSliceLength(0)), rlePositionCount); - } - } - - @Override - public void append(int position, ValueBlock source) - { - checkArgument(source instanceof VariableWidthBlock, "Block must be instance of %s but is %s".formatted(VariableWidthBlock.class, source.getClass())); - - ensurePositionCapacity(positionCount + 1); - if (source.isNull(position)) { - valueIsNull[positionCount] = true; - offsets[positionCount + 1] = getCurrentOffset(); - positionCount++; - - hasNullValue = true; - updateSize(1, 0); - } - else { - hasNonNullValue = true; - int currentOffset = getCurrentOffset(); - int sliceLength = source.getSliceLength(position); - Slice slice = source.getSlice(position, 0, sliceLength); - - ensureExtraBytesCapacity(sliceLength); - - slice.getBytes(0, bytes, currentOffset, sliceLength); - - offsets[positionCount + 1] = currentOffset + sliceLength; - - positionCount++; - updateSize(1, sliceLength); - } - } - - @Override - public Block build() - { - Block result; - if (hasNonNullValue) { - result = new VariableWidthBlock( - positionCount, - Slices.wrappedBuffer(bytes, 0, getCurrentOffset()), - offsets, - hasNullValue ? Optional.of(valueIsNull) : Optional.empty()); - } - else { - result = RunLengthEncodedBlock.create(NULL_VALUE_BLOCK, positionCount); - } - reset(); - return result; - } - - @Override - public long getRetainedSizeInBytes() - { - return retainedSizeInBytes; - } - - @Override - public long getSizeInBytes() - { - return sizeInBytes; - } - - private void copyBytes(Slice rawSlice, int[] lengths, int[] sourceOffsets, int count, int newByteCount) - { - ensureExtraBytesCapacity(newByteCount); - - byte[] base = rawSlice.byteArray(); - int byteArrayOffset = rawSlice.byteArrayOffset(); - for (int i = 0; i < count; i++) { - System.arraycopy(base, byteArrayOffset + sourceOffsets[i], bytes, offsets[positionCount + i], lengths[i]); - } - - positionCount += count; - updateSize(count, newByteCount); - } - - /** - * Copy all bytes from {@code slice} to {@code count} consecutive positions in the {@link #bytes} array. - */ - private void duplicateBytes(Slice slice, int count) - { - int length = slice.length(); - int newByteCount = toIntExact((long) count * length); - int startOffset = getCurrentOffset(); - ensureExtraBytesCapacity(newByteCount); - - duplicateBytes(slice, bytes, startOffset, count); - - int currentStartOffset = startOffset + length; - for (int i = 0; i < count; i++) { - offsets[positionCount + i + 1] = currentStartOffset; - currentStartOffset += length; - } - - positionCount += count; - updateSize(count, newByteCount); - } - - /** - * Copy {@code length} bytes from {@code slice}, starting at offset {@code sourceOffset} to {@code count} consecutive positions in the {@link #bytes} array. - */ - @VisibleForTesting - static void duplicateBytes(Slice slice, byte[] bytes, int startOffset, int count) - { - int length = slice.length(); - if (length == 0) { - // nothing to copy - return; - } - // copy slice to the first position - slice.getBytes(0, bytes, startOffset, length); - int totalDuplicatedBytes = count * length; - int duplicatedBytes = length; - // copy every byte copied so far, doubling the number of bytes copied on evey iteration - while (duplicatedBytes * 2 <= totalDuplicatedBytes) { - System.arraycopy(bytes, startOffset, bytes, startOffset + duplicatedBytes, duplicatedBytes); - duplicatedBytes = duplicatedBytes * 2; - } - // copy the leftover - System.arraycopy(bytes, startOffset, bytes, startOffset + duplicatedBytes, totalDuplicatedBytes - duplicatedBytes); - } - - @Override - public void reset() - { - initialEntryCount = calculateBlockResetSize(positionCount); - initialBytesSize = calculateBlockResetBytes(getCurrentOffset()); - initialized = false; - valueIsNull = new boolean[0]; - offsets = new int[1]; - bytes = new byte[0]; - positionCount = 0; - sizeInBytes = 0; - hasNonNullValue = false; - hasNullValue = false; - updateRetainedSize(); - } - - private int getCurrentOffset() - { - return offsets[positionCount]; - } - - private void updateSize(long positionsSize, int bytesWritten) - { - sizeInBytes += (SIZE_OF_BYTE + SIZE_OF_INT) * positionsSize + bytesWritten; - } - - private void ensureExtraBytesCapacity(int extraBytesCapacity) - { - int totalBytesCapacity = getCurrentOffset() + extraBytesCapacity; - if (bytes.length < totalBytesCapacity) { - int newBytesLength = Math.max(bytes.length, initialBytesSize); - if (totalBytesCapacity > newBytesLength) { - newBytesLength = Math.max(totalBytesCapacity, calculateNewArraySize(newBytesLength)); - } - bytes = Arrays.copyOf(bytes, newBytesLength); - updateRetainedSize(); - } - } - - private void ensurePositionCapacity(int capacity) - { - if (valueIsNull.length < capacity) { - int newSize; - if (initialized) { - newSize = calculateNewArraySize(valueIsNull.length); - } - else { - newSize = initialEntryCount; - initialized = true; - } - newSize = Math.max(newSize, capacity); - - valueIsNull = Arrays.copyOf(valueIsNull, newSize); - offsets = Arrays.copyOf(offsets, newSize + 1); - updateRetainedSize(); - } - } - - private void updateRetainedSize() - { - retainedSizeInBytes = INSTANCE_SIZE + sizeOf(valueIsNull) + sizeOf(offsets) + sizeOf(bytes); - } - - private static int getExpectedBytes(long maxPageSizeInBytes, int expectedPositions) - { - // it is guaranteed Math.min will not overflow; safe to cast - return (int) min((long) expectedPositions * EXPECTED_BYTES_PER_ENTRY, maxPageSizeInBytes); - } -} diff --git a/core/trino-main/src/main/java/io/trino/operator/output/TypedPositionsAppender.java b/core/trino-main/src/main/java/io/trino/operator/output/TypedPositionsAppender.java index 2fcc0fda3b52..141c26811ca6 100644 --- a/core/trino-main/src/main/java/io/trino/operator/output/TypedPositionsAppender.java +++ b/core/trino-main/src/main/java/io/trino/operator/output/TypedPositionsAppender.java @@ -20,43 +20,40 @@ import it.unimi.dsi.fastutil.ints.IntArrayList; import static io.airlift.slice.SizeOf.instanceSize; -import static java.util.Objects.requireNonNull; class TypedPositionsAppender implements PositionsAppender { private static final int INSTANCE_SIZE = instanceSize(TypedPositionsAppender.class); - private final Type type; private BlockBuilder blockBuilder; TypedPositionsAppender(Type type, int expectedPositions) { - this.type = requireNonNull(type, "type is null"); this.blockBuilder = type.createBlockBuilder(null, expectedPositions); } + public TypedPositionsAppender(BlockBuilder blockBuilder) + { + this.blockBuilder = blockBuilder; + } + @Override - public void append(IntArrayList positions, ValueBlock source) + public void append(IntArrayList positions, ValueBlock block) { - int[] positionArray = positions.elements(); - for (int i = 0; i < positions.size(); i++) { - type.appendTo(source, positionArray[i], blockBuilder); - } + blockBuilder.appendPositions(block, positions.elements(), 0, positions.size()); } @Override - public void appendRle(ValueBlock block, int rlePositionCount) + public void appendRle(ValueBlock block, int count) { - for (int i = 0; i < rlePositionCount; i++) { - type.appendTo(block, 0, blockBuilder); - } + blockBuilder.appendRepeated(block, 0, count); } @Override - public void append(int position, ValueBlock source) + public void append(int position, ValueBlock block) { - type.appendTo(source, position, blockBuilder); + blockBuilder.append(block, position); } @Override diff --git a/core/trino-main/src/main/java/io/trino/type/LikePatternType.java b/core/trino-main/src/main/java/io/trino/type/LikePatternType.java index 180f9a33fa2a..c4e442b69919 100644 --- a/core/trino-main/src/main/java/io/trino/type/LikePatternType.java +++ b/core/trino-main/src/main/java/io/trino/type/LikePatternType.java @@ -14,6 +14,7 @@ package io.trino.type; import io.airlift.slice.Slice; +import io.airlift.slice.Slices; import io.trino.spi.block.Block; import io.trino.spi.block.BlockBuilder; import io.trino.spi.block.VariableWidthBlock; @@ -55,7 +56,7 @@ public Object getObject(Block block, int position) int valuePosition = block.getUnderlyingValuePosition(position); Slice slice = valueBlock.getSlice(valuePosition); - // layout is: ? + // layout is: ? int length = slice.getInt(0); String pattern = slice.toString(4, length, UTF_8); @@ -73,18 +74,25 @@ public Object getObject(Block block, int position) public void writeObject(BlockBuilder blockBuilder, Object value) { LikePattern likePattern = (LikePattern) value; - ((VariableWidthBlockBuilder) blockBuilder).buildEntry(valueWriter -> { - Slice pattern = utf8Slice(likePattern.getPattern()); - int length = pattern.length(); - valueWriter.writeInt(length); - valueWriter.writeBytes(pattern, 0, length); - if (likePattern.getEscape().isEmpty()) { - valueWriter.writeByte(0); - } - else { - valueWriter.writeByte(1); - valueWriter.writeInt(likePattern.getEscape().get()); - } - }); + Slice pattern = utf8Slice(likePattern.getPattern()); + + Slice slice = Slices.allocate( + Integer.BYTES + + pattern.length() + + Byte.BYTES + + (likePattern.getEscape().isPresent() ? Integer.BYTES : 0)); + + // layout is: ? + slice.setInt(0, pattern.length()); + slice.setBytes(4, pattern); + if (likePattern.getEscape().isEmpty()) { + slice.setByte(4 + pattern.length(), (byte) 0); + } + else { + slice.setByte(4 + pattern.length(), (byte) 1); + slice.setInt(4 + pattern.length() + 1, likePattern.getEscape().get()); + } + + ((VariableWidthBlockBuilder) blockBuilder).writeEntry(slice); } } diff --git a/core/trino-main/src/test/java/io/trino/execution/buffer/TestPagesSerde.java b/core/trino-main/src/test/java/io/trino/execution/buffer/TestPagesSerde.java index 5340489dc313..676c46f4b1b2 100644 --- a/core/trino-main/src/test/java/io/trino/execution/buffer/TestPagesSerde.java +++ b/core/trino-main/src/test/java/io/trino/execution/buffer/TestPagesSerde.java @@ -18,6 +18,7 @@ import io.airlift.slice.Slice; import io.airlift.slice.SliceInput; import io.airlift.slice.SliceOutput; +import io.airlift.slice.Slices; import io.trino.metadata.BlockEncodingManager; import io.trino.metadata.InternalBlockEncodingSerde; import io.trino.spi.Page; @@ -26,7 +27,6 @@ import io.trino.spi.block.BlockBuilder; import io.trino.spi.block.BlockEncodingSerde; import io.trino.spi.block.VariableWidthBlock; -import io.trino.spi.block.VariableWidthBlockBuilder; import io.trino.spi.type.Type; import io.trino.tpch.LineItem; import io.trino.tpch.LineItemGenerator; @@ -288,14 +288,13 @@ private void testDeserializationWithRollover(boolean encryptionEnabled, boolean private static Page createTestPage(int numberOfEntries) { - VariableWidthBlockBuilder blockBuilder = new VariableWidthBlockBuilder(null, 1, 1000); - blockBuilder.buildEntry(value -> { - value.writeInt(numberOfEntries); - for (int i = 0; i < numberOfEntries; i++) { - value.writeLong(i); - } - }); - return new Page(blockBuilder.build()); + Slice slice = Slices.allocate(Integer.BYTES + numberOfEntries * Long.BYTES); + SliceOutput out = slice.getOutput(); + out.writeInt(numberOfEntries); + for (int i = 0; i < numberOfEntries; i++) { + out.writeLong(i); + } + return new Page(new VariableWidthBlock(1, slice, new int[] {0, slice.length()}, Optional.empty())); } private static class RolloverBlockSerde @@ -305,15 +304,15 @@ private static class RolloverBlockSerde public Block readBlock(SliceInput input) { int numberOfEntries = input.readInt(); - VariableWidthBlockBuilder blockBuilder = new VariableWidthBlockBuilder(null, 1, 1000); - blockBuilder.buildEntry(value -> { - value.writeInt(numberOfEntries); - for (int i = 0; i < numberOfEntries; ++i) { - // read 8 bytes at a time - value.writeLong(input.readLong()); - } - }); - return blockBuilder.build(); + + Slice slice = Slices.allocate(Integer.BYTES + numberOfEntries * Long.BYTES); + SliceOutput out = slice.getOutput(); + out.writeInt(numberOfEntries); + for (int i = 0; i < numberOfEntries; ++i) { + // read 8 bytes at a time + out.writeLong(input.readLong()); + } + return new VariableWidthBlock(1, slice, new int[] {0, slice.length()}, Optional.empty()); } @Override diff --git a/core/trino-main/src/test/java/io/trino/operator/output/TestSlicePositionsAppender.java b/core/trino-main/src/test/java/io/trino/operator/output/TestSlicePositionsAppender.java deleted file mode 100644 index c90ffe5234b0..000000000000 --- a/core/trino-main/src/test/java/io/trino/operator/output/TestSlicePositionsAppender.java +++ /dev/null @@ -1,122 +0,0 @@ -/* - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -package io.trino.operator.output; - -import io.airlift.slice.Slice; -import io.airlift.slice.Slices; -import io.trino.spi.block.Block; -import io.trino.spi.block.RunLengthEncodedBlock; -import io.trino.spi.block.ValueBlock; -import org.junit.jupiter.api.Test; - -import java.util.Arrays; - -import static io.trino.block.BlockAssertions.assertBlockEquals; -import static io.trino.block.BlockAssertions.createStringsBlock; -import static io.trino.operator.output.SlicePositionsAppender.duplicateBytes; -import static io.trino.spi.type.VarcharType.VARCHAR; -import static org.testng.internal.junit.ArrayAsserts.assertArrayEquals; - -public class TestSlicePositionsAppender -{ - @Test - public void testAppendEmptySliceRle() - { - // test SlicePositionAppender.appendRle with empty value (Slice with length 0) - PositionsAppender positionsAppender = new SlicePositionsAppender(1, 100); - ValueBlock value = createStringsBlock(""); - positionsAppender.appendRle(value, 10); - - Block actualBlock = positionsAppender.build(); - - assertBlockEquals(VARCHAR, actualBlock, RunLengthEncodedBlock.create(value, 10)); - } - - @Test - public void testDuplicateZeroLength() - { - Slice slice = Slices.wrappedBuffer(); - byte[] target = new byte[] {-1}; - duplicateBytes(slice, target, 0, 100); - assertArrayEquals(new byte[] {-1}, target); - } - - @Test - public void testDuplicate1Byte() - { - Slice slice = Slices.wrappedBuffer(new byte[] {2}); - byte[] target = new byte[5]; - Arrays.fill(target, (byte) -1); - duplicateBytes(slice, target, 3, 2); - assertArrayEquals(new byte[] {-1, -1, -1, 2, 2}, target); - } - - @Test - public void testDuplicate2Bytes() - { - Slice slice = Slices.wrappedBuffer(new byte[] {1, 2}); - byte[] target = new byte[8]; - Arrays.fill(target, (byte) -1); - duplicateBytes(slice, target, 1, 3); - assertArrayEquals(new byte[] {-1, 1, 2, 1, 2, 1, 2, -1}, target); - } - - @Test - public void testDuplicate1Time() - { - Slice slice = Slices.wrappedBuffer(new byte[] {1, 2}); - byte[] target = new byte[8]; - Arrays.fill(target, (byte) -1); - - duplicateBytes(slice, target, 1, 1); - - assertArrayEquals(new byte[] {-1, 1, 2, -1, -1, -1, -1, -1}, target); - } - - @Test - public void testDuplicateMultipleBytesOffNumberOfTimes() - { - Slice slice = Slices.wrappedBuffer(new byte[] {5, 3, 1}); - byte[] target = new byte[17]; - Arrays.fill(target, (byte) -1); - - duplicateBytes(slice, target, 1, 5); - - assertArrayEquals(new byte[] {-1, 5, 3, 1, 5, 3, 1, 5, 3, 1, 5, 3, 1, 5, 3, 1, -1}, target); - } - - @Test - public void testDuplicateMultipleBytesEvenNumberOfTimes() - { - Slice slice = Slices.wrappedBuffer(new byte[] {5, 3, 1}); - byte[] target = new byte[20]; - Arrays.fill(target, (byte) -1); - - duplicateBytes(slice, target, 1, 6); - - assertArrayEquals(new byte[] {-1, 5, 3, 1, 5, 3, 1, 5, 3, 1, 5, 3, 1, 5, 3, 1, 5, 3, 1, -1}, target); - } - - @Test - public void testDuplicateMultipleBytesPowerOfTwoNumberOfTimes() - { - Slice slice = Slices.wrappedBuffer(new byte[] {5, 3, 1}); - byte[] target = new byte[14]; - Arrays.fill(target, (byte) -1); - - duplicateBytes(slice, target, 1, 4); - - assertArrayEquals(new byte[] {-1, 5, 3, 1, 5, 3, 1, 5, 3, 1, 5, 3, 1, -1}, target); - } -} diff --git a/core/trino-spi/pom.xml b/core/trino-spi/pom.xml index 1c0bb8249690..eb497d8c8ffc 100644 --- a/core/trino-spi/pom.xml +++ b/core/trino-spi/pom.xml @@ -212,6 +212,41 @@ + + java.class.removed + interface io.trino.spi.block.VariableWidthBlockBuilder.VariableWidthEntryBuilder<E extends java.lang.Throwable> + + + java.method.removed + method <E extends java.lang.Throwable> void io.trino.spi.block.VariableWidthBlockBuilder::buildEntry(io.trino.spi.block.VariableWidthBlockBuilder.VariableWidthEntryBuilder<E>) throws E + + + true + java.method.addedToInterface + method void io.trino.spi.block.BlockBuilder::append(io.trino.spi.block.ValueBlock, int) + + + java.method.addedToInterface + method void io.trino.spi.block.BlockBuilder::appendPositions(io.trino.spi.block.ValueBlock, int[], int, int) + + + java.method.addedToInterface + method void io.trino.spi.block.BlockBuilder::appendRange(io.trino.spi.block.ValueBlock, int, int) + + + java.method.addedToInterface + method void io.trino.spi.block.BlockBuilder::appendRepeated(io.trino.spi.block.ValueBlock, int, int) + + + java.method.returnTypeChangedCovariantly + method io.trino.spi.block.BlockBuilder io.trino.spi.block.ShortArrayBlockBuilder::appendNull() + method io.trino.spi.block.ShortArrayBlockBuilder io.trino.spi.block.ShortArrayBlockBuilder::appendNull() + + + java.method.returnTypeChangedCovariantly + method io.trino.spi.block.BlockBuilder io.trino.spi.block.ShortArrayBlockBuilder::writeShort(short) + method io.trino.spi.block.ShortArrayBlockBuilder io.trino.spi.block.ShortArrayBlockBuilder::writeShort(short) + diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/ArrayBlock.java b/core/trino-spi/src/main/java/io/trino/spi/block/ArrayBlock.java index 9aad1d9da67e..ee7ab4f5bc85 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/ArrayBlock.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/ArrayBlock.java @@ -177,6 +177,11 @@ int[] getOffsets() return offsets; } + boolean[] getRawValueIsNull() + { + return valueIsNull; + } + int getOffsetBase() { return arrayOffset; diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/ArrayBlockBuilder.java b/core/trino-spi/src/main/java/io/trino/spi/block/ArrayBlockBuilder.java index df28648003e7..4e5baa0c5bbd 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/ArrayBlockBuilder.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/ArrayBlockBuilder.java @@ -21,6 +21,8 @@ import static io.airlift.slice.SizeOf.instanceSize; import static io.airlift.slice.SizeOf.sizeOf; import static io.trino.spi.block.ArrayBlock.createArrayBlockInternal; +import static io.trino.spi.block.BlockUtil.appendRawBlockRange; +import static io.trino.spi.block.BlockUtil.calculateNewArraySize; import static java.lang.Math.max; import static java.util.Objects.requireNonNull; @@ -28,6 +30,7 @@ public class ArrayBlockBuilder implements BlockBuilder { private static final int INSTANCE_SIZE = instanceSize(ArrayBlockBuilder.class); + private static final int SIZE_IN_BYTES_PER_POSITION = Integer.BYTES + Byte.BYTES; private int positionCount; @@ -39,7 +42,7 @@ public class ArrayBlockBuilder private int[] offsets = new int[1]; private boolean[] valueIsNull = new boolean[0]; private boolean hasNullValue; - private boolean hasNonNullRow; + private boolean hasNonNullValue; private final BlockBuilder values; private boolean currentEntryOpened; @@ -82,7 +85,7 @@ private ArrayBlockBuilder(@Nullable BlockBuilderStatus blockBuilderStatus, Block this.values = requireNonNull(values, "values is null"); this.initialEntryCount = max(expectedEntries, 1); - updateDataSize(); + updateRetainedSize(); } @Override @@ -94,7 +97,7 @@ public int getPositionCount() @Override public long getSizeInBytes() { - return values.getSizeInBytes() + ((Integer.BYTES + Byte.BYTES) * (long) positionCount); + return values.getSizeInBytes() + (SIZE_IN_BYTES_PER_POSITION * (long) positionCount); } @Override @@ -116,6 +119,97 @@ public void buildEntry(ArrayValueBuilder builder) currentEntryOpened = false; } + @Override + public void append(ValueBlock block, int position) + { + if (currentEntryOpened) { + throw new IllegalStateException("Current entry must be closed before a null can be written"); + } + + ArrayBlock arrayBlock = (ArrayBlock) block; + if (block.isNull(position)) { + entryAdded(true); + return; + } + + int offsetBase = arrayBlock.getOffsetBase(); + int[] offsets = arrayBlock.getOffsets(); + int startOffset = offsets[offsetBase + position]; + int length = offsets[offsetBase + position + 1] - startOffset; + + appendRawBlockRange(arrayBlock.getRawElementBlock(), startOffset, length, values); + entryAdded(false); + } + + @Override + public void appendRepeated(ValueBlock block, int position, int count) + { + if (currentEntryOpened) { + throw new IllegalStateException("Current entry must be closed before a null can be written"); + } + for (int i = 0; i < count; i++) { + append(block, position); + } + } + + @Override + public void appendRange(ValueBlock block, int offset, int length) + { + if (currentEntryOpened) { + throw new IllegalStateException("Current entry must be closed before a null can be written"); + } + + ensureCapacity(positionCount + length); + + ArrayBlock arrayBlock = (ArrayBlock) block; + + int rawOffsetBase = arrayBlock.getOffsetBase(); + int[] rawOffsets = arrayBlock.getOffsets(); + int startOffset = rawOffsets[rawOffsetBase + offset]; + int endOffset = rawOffsets[rawOffsetBase + offset + length]; + + appendRawBlockRange(arrayBlock.getRawElementBlock(), startOffset, endOffset - startOffset, values); + + // update offsets for copied data + for (int i = 0; i < length; i++) { + int entrySize = rawOffsets[rawOffsetBase + offset + i + 1] - rawOffsets[rawOffsetBase + offset + i]; + offsets[positionCount + i + 1] = offsets[positionCount + i] + entrySize; + } + + boolean[] rawValueIsNull = arrayBlock.getRawValueIsNull(); + if (rawValueIsNull != null) { + for (int i = 0; i < length; i++) { + if (rawValueIsNull[rawOffsetBase + offset + i]) { + valueIsNull[positionCount + i] = true; + hasNullValue = true; + } + else { + hasNonNullValue = true; + } + } + } + else { + hasNonNullValue = true; + } + + positionCount += length; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(length * SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public void appendPositions(ValueBlock block, int[] positions, int offset, int length) + { + if (currentEntryOpened) { + throw new IllegalStateException("Current entry must be closed before a null can be written"); + } + for (int i = 0; i < length; i++) { + append(block, positions[offset + i]); + } + } + @Override public BlockBuilder appendNull() { @@ -129,37 +223,41 @@ public BlockBuilder appendNull() private void entryAdded(boolean isNull) { - if (valueIsNull.length <= positionCount) { - growCapacity(); - } + ensureCapacity(positionCount + 1); + offsets[positionCount + 1] = values.getPositionCount(); valueIsNull[positionCount] = isNull; hasNullValue |= isNull; - hasNonNullRow |= !isNull; + hasNonNullValue |= !isNull; positionCount++; if (blockBuilderStatus != null) { - blockBuilderStatus.addBytes(Integer.BYTES + Byte.BYTES); + blockBuilderStatus.addBytes(SIZE_IN_BYTES_PER_POSITION); } } - private void growCapacity() + private void ensureCapacity(int capacity) { + if (valueIsNull.length >= capacity) { + return; + } + int newSize; if (initialized) { - newSize = BlockUtil.calculateNewArraySize(valueIsNull.length); + newSize = calculateNewArraySize(capacity); } else { newSize = initialEntryCount; initialized = true; } + newSize = max(newSize, capacity); valueIsNull = Arrays.copyOf(valueIsNull, newSize); offsets = Arrays.copyOf(offsets, newSize + 1); - updateDataSize(); + updateRetainedSize(); } - private void updateDataSize() + private void updateRetainedSize() { retainedSizeInBytes = INSTANCE_SIZE + sizeOf(valueIsNull) + sizeOf(offsets); if (blockBuilderStatus != null) { @@ -173,7 +271,7 @@ public Block build() if (currentEntryOpened) { throw new IllegalStateException("Current entry must be closed before the block can be built"); } - if (!hasNonNullRow) { + if (!hasNonNullValue) { return nullRle(positionCount); } return buildValueBlock(); @@ -197,10 +295,9 @@ public BlockBuilder newBlockBuilderLike(int expectedEntries, BlockBuilderStatus @Override public String toString() { - StringBuilder sb = new StringBuilder("ArrayBlockBuilder{"); - sb.append("positionCount=").append(getPositionCount()); - sb.append('}'); - return sb.toString(); + return "ArrayBlockBuilder{" + + "positionCount=" + getPositionCount() + + '}'; } private Block nullRle(int positionCount) diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/BlockBuilder.java b/core/trino-spi/src/main/java/io/trino/spi/block/BlockBuilder.java index 7d458991497e..97df9b75b235 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/BlockBuilder.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/BlockBuilder.java @@ -36,6 +36,26 @@ public interface BlockBuilder */ long getRetainedSizeInBytes(); + /** + * Append the specified value. + */ + void append(ValueBlock block, int position); + + /** + * Append the specified value multiple times. + */ + void appendRepeated(ValueBlock block, int position, int count); + + /** + * Append the values in the specified range. + */ + void appendRange(ValueBlock block, int offset, int length); + + /** + * Append the values at the specified positions. + */ + void appendPositions(ValueBlock block, int[] positions, int offset, int length); + /** * Appends a null value to the block. */ diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/BlockUtil.java b/core/trino-spi/src/main/java/io/trino/spi/block/BlockUtil.java index e17b4ab28ef4..06af44be190e 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/BlockUtil.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/BlockUtil.java @@ -28,7 +28,8 @@ final class BlockUtil private static final int DEFAULT_CAPACITY = 64; // See java.util.ArrayList for an explanation - static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8; + // Two additional positions are reserved for a spare null position and offset position + static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8 - 2; private BlockUtil() { @@ -78,19 +79,25 @@ static void checkReadablePosition(Block block, int position) static int calculateNewArraySize(int currentSize) { - // grow array by 50% - long newSize = (long) currentSize + (currentSize >> 1); + return calculateNewArraySize(currentSize, DEFAULT_CAPACITY); + } - // verify new size is within reasonable bounds - if (newSize < DEFAULT_CAPACITY) { - newSize = DEFAULT_CAPACITY; + static int calculateNewArraySize(int currentSize, int minimumSize) + { + if (currentSize < 0 || currentSize > MAX_ARRAY_SIZE || minimumSize < 0 || minimumSize > MAX_ARRAY_SIZE) { + throw new IllegalArgumentException("Invalid currentSize or minimumSize"); } - else if (newSize > MAX_ARRAY_SIZE) { - newSize = MAX_ARRAY_SIZE; - if (newSize == currentSize) { - throw new IllegalArgumentException(format("Cannot grow array beyond '%s'", MAX_ARRAY_SIZE)); - } + if (currentSize == MAX_ARRAY_SIZE) { + throw new IllegalArgumentException("Cannot grow array beyond size " + MAX_ARRAY_SIZE); } + + minimumSize = Math.max(minimumSize, DEFAULT_CAPACITY); + + // grow the array by 50% if possible + long newSize = (long) currentSize + (currentSize >> 1); + + // ensure new size is within bounds + newSize = Math.min(Math.max(newSize, minimumSize), MAX_ARRAY_SIZE); return (int) newSize; } @@ -350,4 +357,21 @@ else if (buffer.length < capacity) { return buffer; } + + static void appendRawBlockRange(Block rawBlock, int offset, int length, BlockBuilder blockBuilder) + { + rawBlock = rawBlock.getLoadedBlock(); + if (rawBlock instanceof RunLengthEncodedBlock rleBlock) { + blockBuilder.appendRepeated(rleBlock.getValue(), 0, length); + } + else if (rawBlock instanceof DictionaryBlock dictionaryBlock) { + blockBuilder.appendPositions(dictionaryBlock.getDictionary(), dictionaryBlock.getRawIds(), offset, length); + } + else if (rawBlock instanceof ValueBlock valueBlock) { + blockBuilder.appendRange(valueBlock, offset, length); + } + else { + throw new IllegalArgumentException("Unsupported block type " + rawBlock.getClass().getSimpleName()); + } + } } diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/ByteArrayBlock.java b/core/trino-spi/src/main/java/io/trino/spi/block/ByteArrayBlock.java index 744c6753445c..e5fa09d77611 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/ByteArrayBlock.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/ByteArrayBlock.java @@ -242,4 +242,19 @@ Slice getValuesSlice() { return Slices.wrappedBuffer(values, arrayOffset, positionCount); } + + int getRawValuesOffset() + { + return arrayOffset; + } + + boolean[] getRawValueIsNull() + { + return valueIsNull; + } + + byte[] getRawValues() + { + return values; + } } diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/ByteArrayBlockBuilder.java b/core/trino-spi/src/main/java/io/trino/spi/block/ByteArrayBlockBuilder.java index 559ead304ead..ef9971ca5253 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/ByteArrayBlockBuilder.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/ByteArrayBlockBuilder.java @@ -19,6 +19,7 @@ import static io.airlift.slice.SizeOf.instanceSize; import static io.airlift.slice.SizeOf.sizeOf; +import static io.trino.spi.block.BlockUtil.calculateNewArraySize; import static java.lang.Math.max; public class ByteArrayBlockBuilder @@ -47,14 +48,12 @@ public ByteArrayBlockBuilder(@Nullable BlockBuilderStatus blockBuilderStatus, in this.blockBuilderStatus = blockBuilderStatus; this.initialEntryCount = max(expectedEntries, 1); - updateDataSize(); + updateRetainedSize(); } public BlockBuilder writeByte(byte value) { - if (values.length <= positionCount) { - growCapacity(); - } + ensureCapacity(positionCount + 1); values[positionCount] = value; @@ -67,12 +66,147 @@ public BlockBuilder writeByte(byte value) } @Override - public BlockBuilder appendNull() + public void append(ValueBlock block, int position) + { + ensureCapacity(positionCount + 1); + + ByteArrayBlock byteArrayBlock = (ByteArrayBlock) block; + if (byteArrayBlock.isNull(position)) { + valueIsNull[positionCount] = true; + hasNullValue = true; + } + else { + values[positionCount] = byteArrayBlock.getByte(position); + hasNonNullValue = true; + } + positionCount++; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(ByteArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public void appendRepeated(ValueBlock block, int position, int count) + { + if (count == 0) { + return; + } + if (count == 1) { + append(block, position); + return; + } + + ensureCapacity(positionCount + count); + + ByteArrayBlock byteArrayBlock = (ByteArrayBlock) block; + + if (byteArrayBlock.isNull(position)) { + Arrays.fill(valueIsNull, positionCount, positionCount + count, true); + hasNullValue = true; + } + else { + byte value = byteArrayBlock.getByte(position); + Arrays.fill(values, positionCount, positionCount + count, value); + hasNonNullValue = true; + } + positionCount += count; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(count * ByteArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public void appendRange(ValueBlock block, int offset, int length) + { + if (length == 0) { + return; + } + if (length == 1) { + append(block, offset); + return; + } + + ensureCapacity(positionCount + length); + + ByteArrayBlock byteArrayBlock = (ByteArrayBlock) block; + int rawOffset = byteArrayBlock.getRawValuesOffset(); + + byte[] rawValues = byteArrayBlock.getRawValues(); + System.arraycopy(rawValues, rawOffset + offset, values, positionCount, length); + + boolean[] rawValueIsNull = byteArrayBlock.getRawValueIsNull(); + if (rawValueIsNull != null) { + for (int i = 0; i < length; i++) { + if (rawValueIsNull[rawOffset + offset + i]) { + valueIsNull[positionCount + i] = true; + hasNullValue = true; + } + else { + hasNonNullValue = true; + } + } + } + else { + hasNonNullValue = true; + } + positionCount += length; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(length * ByteArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public void appendPositions(ValueBlock block, int[] positions, int offset, int length) { - if (values.length <= positionCount) { - growCapacity(); + if (length == 0) { + return; + } + if (length == 1) { + append(block, positions[offset]); + return; } + ensureCapacity(positionCount + length); + + ByteArrayBlock byteArrayBlock = (ByteArrayBlock) block; + int rawOffset = byteArrayBlock.getRawValuesOffset(); + byte[] rawValues = byteArrayBlock.getRawValues(); + boolean[] rawValueIsNull = byteArrayBlock.getRawValueIsNull(); + if (rawValueIsNull != null) { + for (int i = 0; i < length; i++) { + int rawPosition = positions[offset + i] + rawOffset; + if (rawValueIsNull[rawPosition]) { + valueIsNull[positionCount + i] = true; + hasNullValue = true; + } + else { + values[positionCount + i] = rawValues[rawPosition]; + hasNonNullValue = true; + } + } + } + else { + for (int i = 0; i < length; i++) { + int rawPosition = positions[offset + i] + rawOffset; + values[positionCount + i] = rawValues[rawPosition]; + } + hasNonNullValue = true; + } + positionCount += length; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(length * ByteArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public BlockBuilder appendNull() + { + ensureCapacity(positionCount + 1); + valueIsNull[positionCount] = true; hasNullValue = true; @@ -104,23 +238,28 @@ public BlockBuilder newBlockBuilderLike(int expectedEntries, BlockBuilderStatus return new ByteArrayBlockBuilder(blockBuilderStatus, expectedEntries); } - private void growCapacity() + private void ensureCapacity(int capacity) { + if (values.length >= capacity) { + return; + } + int newSize; if (initialized) { - newSize = BlockUtil.calculateNewArraySize(values.length); + newSize = calculateNewArraySize(capacity); } else { newSize = initialEntryCount; initialized = true; } + newSize = max(newSize, capacity); valueIsNull = Arrays.copyOf(valueIsNull, newSize); values = Arrays.copyOf(values, newSize); - updateDataSize(); + updateRetainedSize(); } - private void updateDataSize() + private void updateRetainedSize() { retainedSizeInBytes = INSTANCE_SIZE + sizeOf(valueIsNull) + sizeOf(values); if (blockBuilderStatus != null) { diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/Fixed12Block.java b/core/trino-spi/src/main/java/io/trino/spi/block/Fixed12Block.java index f38e059ad2f6..a8694f321806 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/Fixed12Block.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/Fixed12Block.java @@ -294,11 +294,17 @@ public static int decodeFixed12Second(int[] values, int position) return values[offset + 2]; } - int getPositionOffset() + int getRawOffset() { return positionOffset; } + @Nullable + boolean[] getRawValueIsNull() + { + return valueIsNull; + } + int[] getRawValues() { return values; diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/Fixed12BlockBuilder.java b/core/trino-spi/src/main/java/io/trino/spi/block/Fixed12BlockBuilder.java index f0d9e278510f..3a30f6fca05c 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/Fixed12BlockBuilder.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/Fixed12BlockBuilder.java @@ -19,6 +19,7 @@ import static io.airlift.slice.SizeOf.instanceSize; import static io.airlift.slice.SizeOf.sizeOf; +import static io.trino.spi.block.BlockUtil.calculateNewArraySize; import static io.trino.spi.block.Fixed12Block.FIXED12_BYTES; import static io.trino.spi.block.Fixed12Block.encodeFixed12; import static java.lang.Math.max; @@ -49,14 +50,12 @@ public Fixed12BlockBuilder(@Nullable BlockBuilderStatus blockBuilderStatus, int this.blockBuilderStatus = blockBuilderStatus; this.initialEntryCount = max(expectedEntries, 1); - updateDataSize(); + updateRetainedSize(); } public void writeFixed12(long first, int second) { - if (valueIsNull.length <= positionCount) { - growCapacity(); - } + ensureCapacity(positionCount + 1); encodeFixed12(first, second, values, positionCount); @@ -68,12 +67,174 @@ public void writeFixed12(long first, int second) } @Override - public BlockBuilder appendNull() + public void append(ValueBlock block, int position) + { + ensureCapacity(positionCount + 1); + + Fixed12Block fixed12Block = (Fixed12Block) block; + if (fixed12Block.isNull(position)) { + valueIsNull[positionCount] = true; + hasNullValue = true; + } + else { + int[] rawValues = fixed12Block.getRawValues(); + int rawValuePosition = (fixed12Block.getRawOffset() + position) * 3; + + int positionIndex = positionCount * 3; + values[positionIndex] = rawValues[rawValuePosition]; + values[positionIndex + 1] = rawValues[rawValuePosition + 1]; + values[positionIndex + 2] = rawValues[rawValuePosition + 2]; + hasNonNullValue = true; + } + positionCount++; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(Fixed12Block.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public void appendRepeated(ValueBlock block, int position, int count) + { + if (count == 0) { + return; + } + if (count == 1) { + append(block, position); + return; + } + + ensureCapacity(positionCount + count); + + Fixed12Block fixed12Block = (Fixed12Block) block; + if (fixed12Block.isNull(position)) { + Arrays.fill(valueIsNull, positionCount, positionCount + count, true); + hasNullValue = true; + } + else { + int[] rawValues = fixed12Block.getRawValues(); + int rawValuePosition = (fixed12Block.getRawOffset() + position) * 3; + int valueFirst = rawValues[rawValuePosition]; + int valueSecond = rawValues[rawValuePosition + 1]; + int valueThird = rawValues[rawValuePosition + 2]; + + int positionIndex = positionCount * 3; + for (int i = 0; i < count; i++) { + values[positionIndex] = valueFirst; + values[positionIndex + 1] = valueSecond; + values[positionIndex + 2] = valueThird; + positionIndex += 3; + } + + hasNonNullValue = true; + } + positionCount += count; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(count * Fixed12Block.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public void appendRange(ValueBlock block, int offset, int length) + { + if (length == 0) { + return; + } + if (length == 1) { + append(block, offset); + return; + } + + ensureCapacity(positionCount + length); + + Fixed12Block fixed12Block = (Fixed12Block) block; + int rawOffset = fixed12Block.getRawOffset(); + + int[] rawValues = fixed12Block.getRawValues(); + System.arraycopy(rawValues, (rawOffset + offset) * 3, values, positionCount * 3, length * 3); + + boolean[] rawValueIsNull = fixed12Block.getRawValueIsNull(); + if (rawValueIsNull != null) { + for (int i = 0; i < length; i++) { + if (rawValueIsNull[rawOffset + offset + i]) { + valueIsNull[positionCount + i] = true; + hasNullValue = true; + } + else { + hasNonNullValue = true; + } + } + } + else { + hasNonNullValue = true; + } + positionCount += length; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(length * LongArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public void appendPositions(ValueBlock block, int[] positions, int offset, int length) { - if (valueIsNull.length <= positionCount) { - growCapacity(); + if (length == 0) { + return; + } + if (length == 1) { + append(block, positions[offset]); + return; } + ensureCapacity(positionCount + length); + + Fixed12Block fixed12Block = (Fixed12Block) block; + int rawOffset = fixed12Block.getRawOffset(); + int[] rawValues = fixed12Block.getRawValues(); + boolean[] rawValueIsNull = fixed12Block.getRawValueIsNull(); + + int positionIndex = positionCount * 3; + if (rawValueIsNull != null) { + for (int i = 0; i < length; i++) { + int rawPosition = positions[offset + i] + rawOffset; + if (rawValueIsNull[rawPosition]) { + valueIsNull[positionCount + i] = true; + hasNullValue = true; + } + else { + int rawValuePosition = rawPosition * 3; + values[positionIndex] = rawValues[rawValuePosition]; + values[positionIndex + 1] = rawValues[rawValuePosition + 1]; + values[positionIndex + 2] = rawValues[rawValuePosition + 2]; + hasNonNullValue = true; + } + positionIndex += 3; + } + } + else { + for (int i = 0; i < length; i++) { + int rawPosition = positions[offset + i] + rawOffset; + int rawValuePosition = rawPosition * 3; + values[positionIndex] = rawValues[rawValuePosition]; + values[positionIndex + 1] = rawValues[rawValuePosition + 1]; + values[positionIndex + 2] = rawValues[rawValuePosition + 2]; + positionIndex += 3; + } + hasNonNullValue = true; + } + positionCount += length; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(length * Fixed12Block.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public BlockBuilder appendNull() + { + ensureCapacity(positionCount + 1); + valueIsNull[positionCount] = true; hasNullValue = true; @@ -105,23 +266,28 @@ public BlockBuilder newBlockBuilderLike(int expectedEntries, BlockBuilderStatus return new Fixed12BlockBuilder(blockBuilderStatus, expectedEntries); } - private void growCapacity() + private void ensureCapacity(int capacity) { + if (valueIsNull.length >= capacity) { + return; + } + int newSize; if (initialized) { - newSize = BlockUtil.calculateNewArraySize(valueIsNull.length); + newSize = calculateNewArraySize(capacity); } else { newSize = initialEntryCount; initialized = true; } + newSize = max(newSize, capacity); valueIsNull = Arrays.copyOf(valueIsNull, newSize); values = Arrays.copyOf(values, newSize * 3); - updateDataSize(); + updateRetainedSize(); } - private void updateDataSize() + private void updateRetainedSize() { retainedSizeInBytes = INSTANCE_SIZE + sizeOf(valueIsNull) + sizeOf(values); if (blockBuilderStatus != null) { diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/Fixed12BlockEncoding.java b/core/trino-spi/src/main/java/io/trino/spi/block/Fixed12BlockEncoding.java index 131837f74c86..5d1799f83c4a 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/Fixed12BlockEncoding.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/Fixed12BlockEncoding.java @@ -40,7 +40,7 @@ public void writeBlock(BlockEncodingSerde blockEncodingSerde, SliceOutput sliceO encodeNullsAsBits(sliceOutput, fixed12Block); if (!fixed12Block.mayHaveNull()) { - sliceOutput.writeInts(fixed12Block.getRawValues(), fixed12Block.getPositionOffset() * 3, fixed12Block.getPositionCount() * 3); + sliceOutput.writeInts(fixed12Block.getRawValues(), fixed12Block.getRawOffset() * 3, fixed12Block.getPositionCount() * 3); } else { int[] valuesWithoutNull = new int[positionCount * 3]; diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/Int128ArrayBlock.java b/core/trino-spi/src/main/java/io/trino/spi/block/Int128ArrayBlock.java index 57b8e4ac9bd4..0cdf686a3d21 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/Int128ArrayBlock.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/Int128ArrayBlock.java @@ -256,13 +256,19 @@ public String toString() return sb.toString(); } - long[] getRawValues() + int getRawOffset() { - return values; + return positionOffset; } - int getPositionOffset() + @Nullable + boolean[] getRawValueIsNull() { - return positionOffset; + return valueIsNull; + } + + long[] getRawValues() + { + return values; } } diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/Int128ArrayBlockBuilder.java b/core/trino-spi/src/main/java/io/trino/spi/block/Int128ArrayBlockBuilder.java index f22ae8951fea..5d512b718453 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/Int128ArrayBlockBuilder.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/Int128ArrayBlockBuilder.java @@ -19,6 +19,7 @@ import static io.airlift.slice.SizeOf.instanceSize; import static io.airlift.slice.SizeOf.sizeOf; +import static io.trino.spi.block.BlockUtil.calculateNewArraySize; import static java.lang.Math.max; public class Int128ArrayBlockBuilder @@ -47,14 +48,12 @@ public Int128ArrayBlockBuilder(@Nullable BlockBuilderStatus blockBuilderStatus, this.blockBuilderStatus = blockBuilderStatus; this.initialEntryCount = max(expectedEntries, 1); - updateDataSize(); + updateRetainedSize(); } public void writeInt128(long high, long low) { - if (valueIsNull.length <= positionCount) { - growCapacity(); - } + ensureCapacity(positionCount + 1); int valueIndex = positionCount * 2; values[valueIndex] = high; @@ -68,12 +67,168 @@ public void writeInt128(long high, long low) } @Override - public BlockBuilder appendNull() + public void append(ValueBlock block, int position) + { + ensureCapacity(positionCount + 1); + + Int128ArrayBlock int128ArrayBlock = (Int128ArrayBlock) block; + if (int128ArrayBlock.isNull(position)) { + valueIsNull[positionCount] = true; + hasNullValue = true; + } + else { + long[] rawValues = int128ArrayBlock.getRawValues(); + int rawValuePosition = (int128ArrayBlock.getRawOffset() + position) * 2; + + int positionIndex = positionCount * 2; + values[positionIndex] = rawValues[rawValuePosition]; + values[positionIndex + 1] = rawValues[rawValuePosition + 1]; + hasNonNullValue = true; + } + positionCount++; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(Int128ArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public void appendRepeated(ValueBlock block, int position, int count) + { + if (count == 0) { + return; + } + if (count == 1) { + append(block, position); + return; + } + + ensureCapacity(positionCount + count); + + Int128ArrayBlock int128ArrayBlock = (Int128ArrayBlock) block; + if (int128ArrayBlock.isNull(position)) { + Arrays.fill(valueIsNull, positionCount, positionCount + count, true); + hasNullValue = true; + } + else { + long[] rawValues = int128ArrayBlock.getRawValues(); + int rawValuePosition = (int128ArrayBlock.getRawOffset() + position) * 2; + long valueHigh = rawValues[rawValuePosition]; + long valueLow = rawValues[rawValuePosition + 1]; + + int positionIndex = positionCount * 2; + for (int i = 0; i < count; i++) { + values[positionIndex] = valueHigh; + values[positionIndex + 1] = valueLow; + positionIndex += 2; + } + hasNonNullValue = true; + } + positionCount += count; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(count * Int128ArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public void appendRange(ValueBlock block, int offset, int length) + { + if (length == 0) { + return; + } + if (length == 1) { + append(block, offset); + return; + } + + ensureCapacity(positionCount + length); + + Int128ArrayBlock int128ArrayBlock = (Int128ArrayBlock) block; + int rawOffset = int128ArrayBlock.getRawOffset(); + + long[] rawValues = int128ArrayBlock.getRawValues(); + System.arraycopy(rawValues, (rawOffset + offset) * 2, values, positionCount * 2, length * 2); + + boolean[] rawValueIsNull = int128ArrayBlock.getRawValueIsNull(); + if (rawValueIsNull != null) { + for (int i = 0; i < length; i++) { + if (rawValueIsNull[rawOffset + offset + i]) { + valueIsNull[positionCount + i] = true; + hasNullValue = true; + } + else { + hasNonNullValue = true; + } + } + } + else { + hasNonNullValue = true; + } + positionCount += length; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(length * LongArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public void appendPositions(ValueBlock block, int[] positions, int offset, int length) { - if (valueIsNull.length <= positionCount) { - growCapacity(); + if (length == 0) { + return; + } + if (length == 1) { + append(block, positions[offset]); + return; } + ensureCapacity(positionCount + length); + + Int128ArrayBlock int128ArrayBlock = (Int128ArrayBlock) block; + int rawOffset = int128ArrayBlock.getRawOffset(); + long[] rawValues = int128ArrayBlock.getRawValues(); + boolean[] rawValueIsNull = int128ArrayBlock.getRawValueIsNull(); + + int positionIndex = positionCount * 2; + if (rawValueIsNull != null) { + for (int i = 0; i < length; i++) { + int rawPosition = positions[offset + i] + rawOffset; + if (rawValueIsNull[rawPosition]) { + valueIsNull[positionCount + i] = true; + hasNullValue = true; + } + else { + int rawValuePosition = rawPosition * 2; + values[positionIndex] = rawValues[rawValuePosition]; + values[positionIndex + 1] = rawValues[rawValuePosition + 1]; + hasNonNullValue = true; + } + positionIndex += 2; + } + } + else { + for (int i = 0; i < length; i++) { + int rawPosition = positions[offset + i] + rawOffset; + int rawValuePosition = rawPosition * 2; + values[positionIndex] = rawValues[rawValuePosition]; + values[positionIndex + 1] = rawValues[rawValuePosition + 1]; + positionIndex += 2; + } + hasNonNullValue = true; + } + positionCount += length; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(length * Int128ArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public BlockBuilder appendNull() + { + ensureCapacity(positionCount + 1); + valueIsNull[positionCount] = true; hasNullValue = true; @@ -105,23 +260,28 @@ public BlockBuilder newBlockBuilderLike(int expectedEntries, BlockBuilderStatus return new Int128ArrayBlockBuilder(blockBuilderStatus, expectedEntries); } - private void growCapacity() + private void ensureCapacity(int capacity) { + if (valueIsNull.length >= capacity) { + return; + } + int newSize; if (initialized) { - newSize = BlockUtil.calculateNewArraySize(valueIsNull.length); + newSize = calculateNewArraySize(capacity); } else { newSize = initialEntryCount; initialized = true; } + newSize = max(newSize, capacity); valueIsNull = Arrays.copyOf(valueIsNull, newSize); values = Arrays.copyOf(values, newSize * 2); - updateDataSize(); + updateRetainedSize(); } - private void updateDataSize() + private void updateRetainedSize() { retainedSizeInBytes = INSTANCE_SIZE + sizeOf(valueIsNull) + sizeOf(values); if (blockBuilderStatus != null) { diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/Int128ArrayBlockEncoding.java b/core/trino-spi/src/main/java/io/trino/spi/block/Int128ArrayBlockEncoding.java index 78e8191202e5..55539440d8cd 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/Int128ArrayBlockEncoding.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/Int128ArrayBlockEncoding.java @@ -40,7 +40,7 @@ public void writeBlock(BlockEncodingSerde blockEncodingSerde, SliceOutput sliceO encodeNullsAsBits(sliceOutput, int128ArrayBlock); if (!int128ArrayBlock.mayHaveNull()) { - sliceOutput.writeLongs(int128ArrayBlock.getRawValues(), int128ArrayBlock.getPositionOffset() * 2, int128ArrayBlock.getPositionCount() * 2); + sliceOutput.writeLongs(int128ArrayBlock.getRawValues(), int128ArrayBlock.getRawOffset() * 2, int128ArrayBlock.getPositionCount() * 2); } else { long[] valuesWithoutNull = new long[positionCount * 2]; diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/IntArrayBlock.java b/core/trino-spi/src/main/java/io/trino/spi/block/IntArrayBlock.java index 93fa86da8456..11d426889fb2 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/IntArrayBlock.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/IntArrayBlock.java @@ -237,6 +237,11 @@ public String toString() return sb.toString(); } + boolean[] getRawValueIsNull() + { + return valueIsNull; + } + @Experimental(eta = "2023-12-31") public int[] getRawValues() { diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/IntArrayBlockBuilder.java b/core/trino-spi/src/main/java/io/trino/spi/block/IntArrayBlockBuilder.java index bf124103418b..50ebd1029e23 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/IntArrayBlockBuilder.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/IntArrayBlockBuilder.java @@ -19,6 +19,7 @@ import static io.airlift.slice.SizeOf.instanceSize; import static io.airlift.slice.SizeOf.sizeOf; +import static io.trino.spi.block.BlockUtil.calculateNewArraySize; import static java.lang.Math.max; public class IntArrayBlockBuilder @@ -47,14 +48,12 @@ public IntArrayBlockBuilder(@Nullable BlockBuilderStatus blockBuilderStatus, int this.blockBuilderStatus = blockBuilderStatus; this.initialEntryCount = max(expectedEntries, 1); - updateDataSize(); + updateRetainedSize(); } public BlockBuilder writeInt(int value) { - if (values.length <= positionCount) { - growCapacity(); - } + ensureCapacity(positionCount + 1); values[positionCount] = value; @@ -67,12 +66,146 @@ public BlockBuilder writeInt(int value) } @Override - public BlockBuilder appendNull() + public void append(ValueBlock block, int position) + { + ensureCapacity(positionCount + 1); + + IntArrayBlock intArrayBlock = (IntArrayBlock) block; + if (intArrayBlock.isNull(position)) { + valueIsNull[positionCount] = true; + hasNullValue = true; + } + else { + values[positionCount] = intArrayBlock.getInt(position); + hasNonNullValue = true; + } + positionCount++; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(IntArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public void appendRepeated(ValueBlock block, int position, int count) { - if (values.length <= positionCount) { - growCapacity(); + if (count == 0) { + return; + } + if (count == 1) { + append(block, position); + return; } + ensureCapacity(positionCount + count); + + IntArrayBlock intArrayBlock = (IntArrayBlock) block; + if (intArrayBlock.isNull(position)) { + Arrays.fill(valueIsNull, positionCount, positionCount + count, true); + hasNullValue = true; + } + else { + int value = intArrayBlock.getInt(position); + Arrays.fill(values, positionCount, positionCount + count, value); + hasNonNullValue = true; + } + positionCount += count; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(count * IntArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public void appendRange(ValueBlock block, int offset, int length) + { + if (length == 0) { + return; + } + if (length == 1) { + append(block, offset); + return; + } + + ensureCapacity(positionCount + length); + + IntArrayBlock intArrayBlock = (IntArrayBlock) block; + int rawOffset = intArrayBlock.getRawValuesOffset(); + + int[] rawValues = intArrayBlock.getRawValues(); + System.arraycopy(rawValues, rawOffset + offset, values, positionCount, length); + + boolean[] rawValueIsNull = intArrayBlock.getRawValueIsNull(); + if (rawValueIsNull != null) { + for (int i = 0; i < length; i++) { + if (rawValueIsNull[rawOffset + offset + i]) { + valueIsNull[positionCount + i] = true; + hasNullValue = true; + } + else { + hasNonNullValue = true; + } + } + } + else { + hasNonNullValue = true; + } + positionCount += length; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(length * IntArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public void appendPositions(ValueBlock block, int[] positions, int offset, int length) + { + if (length == 0) { + return; + } + if (length == 1) { + append(block, positions[offset]); + return; + } + + ensureCapacity(positionCount + length); + + IntArrayBlock intArrayBlock = (IntArrayBlock) block; + int rawOffset = intArrayBlock.getRawValuesOffset(); + int[] rawValues = intArrayBlock.getRawValues(); + boolean[] rawValueIsNull = intArrayBlock.getRawValueIsNull(); + if (rawValueIsNull != null) { + for (int i = 0; i < length; i++) { + int rawPosition = positions[offset + i] + rawOffset; + if (rawValueIsNull[rawPosition]) { + valueIsNull[positionCount + i] = true; + hasNullValue = true; + } + else { + values[positionCount + i] = rawValues[rawPosition]; + hasNonNullValue = true; + } + } + } + else { + for (int i = 0; i < length; i++) { + int rawPosition = positions[offset + i] + rawOffset; + values[positionCount + i] = rawValues[rawPosition]; + } + hasNonNullValue = true; + } + positionCount += length; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(length * IntArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public BlockBuilder appendNull() + { + ensureCapacity(positionCount + 1); + valueIsNull[positionCount] = true; hasNullValue = true; @@ -104,23 +237,28 @@ public BlockBuilder newBlockBuilderLike(int expectedEntries, BlockBuilderStatus return new IntArrayBlockBuilder(blockBuilderStatus, expectedEntries); } - private void growCapacity() + private void ensureCapacity(int capacity) { + if (values.length >= capacity) { + return; + } + int newSize; if (initialized) { - newSize = BlockUtil.calculateNewArraySize(values.length); + newSize = calculateNewArraySize(capacity); } else { newSize = initialEntryCount; initialized = true; } + newSize = max(newSize, capacity); valueIsNull = Arrays.copyOf(valueIsNull, newSize); values = Arrays.copyOf(values, newSize); - updateDataSize(); + updateRetainedSize(); } - private void updateDataSize() + private void updateRetainedSize() { retainedSizeInBytes = INSTANCE_SIZE + sizeOf(valueIsNull) + sizeOf(values); if (blockBuilderStatus != null) { diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/LongArrayBlock.java b/core/trino-spi/src/main/java/io/trino/spi/block/LongArrayBlock.java index 2b9aec633844..b79e5939ba79 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/LongArrayBlock.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/LongArrayBlock.java @@ -283,13 +283,18 @@ public String toString() return sb.toString(); } + int getRawValuesOffset() + { + return arrayOffset; + } + long[] getRawValues() { return values; } - int getRawValuesOffset() + boolean[] getRawValueIsNull() { - return arrayOffset; + return valueIsNull; } } diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/LongArrayBlockBuilder.java b/core/trino-spi/src/main/java/io/trino/spi/block/LongArrayBlockBuilder.java index 09a530971ac1..059709edbb1e 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/LongArrayBlockBuilder.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/LongArrayBlockBuilder.java @@ -19,6 +19,7 @@ import static io.airlift.slice.SizeOf.instanceSize; import static io.airlift.slice.SizeOf.sizeOf; +import static io.trino.spi.block.BlockUtil.calculateNewArraySize; import static java.lang.Math.max; public class LongArrayBlockBuilder @@ -47,14 +48,12 @@ public LongArrayBlockBuilder(@Nullable BlockBuilderStatus blockBuilderStatus, in this.blockBuilderStatus = blockBuilderStatus; this.initialEntryCount = max(expectedEntries, 1); - updateDataSize(); + updateRetainedSize(); } public BlockBuilder writeLong(long value) { - if (values.length <= positionCount) { - growCapacity(); - } + ensureCapacity(positionCount + 1); values[positionCount] = value; @@ -67,12 +66,146 @@ public BlockBuilder writeLong(long value) } @Override - public BlockBuilder appendNull() + public void append(ValueBlock block, int position) + { + ensureCapacity(positionCount + 1); + + LongArrayBlock longArrayBlock = (LongArrayBlock) block; + if (longArrayBlock.isNull(position)) { + valueIsNull[positionCount] = true; + hasNullValue = true; + } + else { + values[positionCount] = longArrayBlock.getLong(position); + hasNonNullValue = true; + } + positionCount++; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(LongArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public void appendRepeated(ValueBlock block, int position, int count) { - if (values.length <= positionCount) { - growCapacity(); + if (count == 0) { + return; + } + if (count == 1) { + append(block, position); + return; } + ensureCapacity(positionCount + count); + + LongArrayBlock longArrayBlock = (LongArrayBlock) block; + if (longArrayBlock.isNull(position)) { + Arrays.fill(valueIsNull, positionCount, positionCount + count, true); + hasNullValue = true; + } + else { + long value = longArrayBlock.getLong(position); + Arrays.fill(values, positionCount, positionCount + count, value); + hasNonNullValue = true; + } + positionCount += count; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(count * LongArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public void appendRange(ValueBlock block, int offset, int length) + { + if (length == 0) { + return; + } + if (length == 1) { + append(block, offset); + return; + } + + ensureCapacity(positionCount + length); + + LongArrayBlock longArrayBlock = (LongArrayBlock) block; + int rawOffset = longArrayBlock.getRawValuesOffset(); + + long[] rawValues = longArrayBlock.getRawValues(); + System.arraycopy(rawValues, rawOffset + offset, values, positionCount, length); + + boolean[] rawValueIsNull = longArrayBlock.getRawValueIsNull(); + if (rawValueIsNull != null) { + for (int i = 0; i < length; i++) { + if (rawValueIsNull[rawOffset + offset + i]) { + valueIsNull[positionCount + i] = true; + hasNullValue = true; + } + else { + hasNonNullValue = true; + } + } + } + else { + hasNonNullValue = true; + } + positionCount += length; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(length * LongArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public void appendPositions(ValueBlock block, int[] positions, int offset, int length) + { + if (length == 0) { + return; + } + if (length == 1) { + append(block, positions[offset]); + return; + } + + ensureCapacity(positionCount + length); + + LongArrayBlock longArrayBlock = (LongArrayBlock) block; + int rawOffset = longArrayBlock.getRawValuesOffset(); + long[] rawValues = longArrayBlock.getRawValues(); + boolean[] rawValueIsNull = longArrayBlock.getRawValueIsNull(); + if (rawValueIsNull != null) { + for (int i = 0; i < length; i++) { + int rawPosition = positions[offset + i] + rawOffset; + if (rawValueIsNull[rawPosition]) { + valueIsNull[positionCount + i] = true; + hasNullValue = true; + } + else { + values[positionCount + i] = rawValues[rawPosition]; + hasNonNullValue = true; + } + } + } + else { + for (int i = 0; i < length; i++) { + int rawPosition = positions[offset + i] + rawOffset; + values[positionCount + i] = rawValues[rawPosition]; + } + hasNonNullValue = true; + } + positionCount += length; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(length * LongArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public BlockBuilder appendNull() + { + ensureCapacity(positionCount + 1); + valueIsNull[positionCount] = true; hasNullValue = true; @@ -104,23 +237,28 @@ public BlockBuilder newBlockBuilderLike(int expectedEntries, BlockBuilderStatus return new LongArrayBlockBuilder(blockBuilderStatus, expectedEntries); } - private void growCapacity() + private void ensureCapacity(int capacity) { + if (values.length >= capacity) { + return; + } + int newSize; if (initialized) { - newSize = BlockUtil.calculateNewArraySize(values.length); + newSize = calculateNewArraySize(capacity); } else { newSize = initialEntryCount; initialized = true; } + newSize = max(newSize, capacity); valueIsNull = Arrays.copyOf(valueIsNull, newSize); values = Arrays.copyOf(values, newSize); - updateDataSize(); + updateRetainedSize(); } - private void updateDataSize() + private void updateRetainedSize() { retainedSizeInBytes = INSTANCE_SIZE + sizeOf(valueIsNull) + sizeOf(values); if (blockBuilderStatus != null) { diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/MapBlockBuilder.java b/core/trino-spi/src/main/java/io/trino/spi/block/MapBlockBuilder.java index 477ae48f4ce1..c1e4af377f2c 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/MapBlockBuilder.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/MapBlockBuilder.java @@ -23,6 +23,7 @@ import static io.airlift.slice.SizeOf.instanceSize; import static io.airlift.slice.SizeOf.sizeOf; +import static io.trino.spi.block.BlockUtil.appendRawBlockRange; import static io.trino.spi.block.BlockUtil.calculateNewArraySize; import static io.trino.spi.block.MapBlock.createMapBlockInternal; import static io.trino.spi.block.MapHashTables.HASH_MULTIPLIER; @@ -132,6 +133,64 @@ public void buildEntry(MapValueBuilder builder) currentEntryOpened = false; } + @Override + public void append(ValueBlock block, int position) + { + if (currentEntryOpened) { + throw new IllegalStateException("Current entry must be closed before a null can be written"); + } + + MapBlock mapBlock = (MapBlock) block; + if (block.isNull(position)) { + entryAdded(true); + return; + } + + int offsetBase = mapBlock.getOffsetBase(); + int[] offsets = mapBlock.getOffsets(); + int startOffset = offsets[offsetBase + position]; + int length = offsets[offsetBase + position + 1] - startOffset; + + appendRawBlockRange(mapBlock.getRawKeyBlock(), startOffset, length, keyBlockBuilder); + appendRawBlockRange(mapBlock.getRawValueBlock(), startOffset, length, valueBlockBuilder); + entryAdded(false); + } + + @Override + public void appendRange(ValueBlock block, int offset, int length) + { + if (currentEntryOpened) { + throw new IllegalStateException("Current entry must be closed before a null can be written"); + } + + // this could be optimized to append all array elements using a single append range call + for (int i = 0; i < length; i++) { + append(block, offset + i); + } + } + + @Override + public void appendRepeated(ValueBlock block, int position, int count) + { + if (currentEntryOpened) { + throw new IllegalStateException("Current entry must be closed before a null can be written"); + } + for (int i = 0; i < count; i++) { + append(block, position); + } + } + + @Override + public void appendPositions(ValueBlock block, int[] positions, int offset, int length) + { + if (currentEntryOpened) { + throw new IllegalStateException("Current entry must be closed before a null can be written"); + } + for (int i = 0; i < length; i++) { + append(block, positions[offset + i]); + } + } + @Override public BlockBuilder appendNull() { @@ -167,6 +226,29 @@ private void entryAdded(boolean isNull) @Override public Block build() { + if (positionCount > 1 && hasNullValue) { + boolean hasNonNull = false; + for (int i = 0; i < positionCount; i++) { + hasNonNull |= !mapIsNull[i]; + } + if (!hasNonNull) { + Block emptyKeyBlock = mapType.getKeyType().createBlockBuilder(null, 0).build(); + Block emptyValueBlock = mapType.getValueType().createBlockBuilder(null, 0).build(); + int[] emptyOffsets = {0, 0}; + boolean[] nulls = {true}; + return RunLengthEncodedBlock.create( + createMapBlockInternal( + mapType, + 0, + 1, + Optional.of(nulls), + emptyOffsets, + emptyKeyBlock, + emptyValueBlock, + MapHashTables.create(hashBuildMode, mapType, 0, emptyKeyBlock, emptyOffsets, nulls)), + positionCount); + } + } return buildValueBlock(); } diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/RowBlock.java b/core/trino-spi/src/main/java/io/trino/spi/block/RowBlock.java index 930f82bbe50a..2e4d98ee2769 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/RowBlock.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/RowBlock.java @@ -152,6 +152,11 @@ public boolean mayHaveNull() return rowIsNull != null; } + boolean[] getRawRowIsNull() + { + return rowIsNull; + } + @Override public int getPositionCount() { diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/RowBlockBuilder.java b/core/trino-spi/src/main/java/io/trino/spi/block/RowBlockBuilder.java index 291c97c1b9ae..ea1a16f4bdf9 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/RowBlockBuilder.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/RowBlockBuilder.java @@ -112,6 +112,205 @@ public void buildEntry(RowValueBuilder builder) currentEntryOpened = false; } + @Override + public void append(ValueBlock block, int position) + { + if (currentEntryOpened) { + throw new IllegalStateException("Current entry must be closed before a null can be written"); + } + + RowBlock rowBlock = (RowBlock) block; + if (block.isNull(position)) { + appendNull(); + return; + } + + List fieldBlocks = rowBlock.getFieldBlocks(); + for (int fieldId = 0; fieldId < fieldBlockBuilders.length; fieldId++) { + appendToField(fieldBlocks.get(fieldId), position, fieldBlockBuilders[fieldId]); + } + entryAdded(false); + } + + private static void appendToField(Block fieldBlock, int position, BlockBuilder fieldBlockBuilder) + { + fieldBlock = fieldBlock.getLoadedBlock(); + if (fieldBlock instanceof RunLengthEncodedBlock rleBlock) { + fieldBlockBuilder.append(rleBlock.getValue(), 0); + } + else if (fieldBlock instanceof DictionaryBlock dictionaryBlock) { + fieldBlockBuilder.append(dictionaryBlock.getDictionary(), dictionaryBlock.getId(position)); + } + else if (fieldBlock instanceof ValueBlock valueBlock) { + fieldBlockBuilder.append(valueBlock, position); + } + else { + throw new IllegalArgumentException("Unsupported block type " + fieldBlock.getClass().getSimpleName()); + } + } + + @Override + public void appendRange(ValueBlock block, int offset, int length) + { + if (currentEntryOpened) { + throw new IllegalStateException("Current entry must be closed before a null can be written"); + } + if (length == 0) { + return; + } + + RowBlock rowBlock = (RowBlock) block; + ensureCapacity(positionCount + length); + + List fieldBlocks = rowBlock.getFieldBlocks(); + for (int fieldId = 0; fieldId < fieldBlockBuilders.length; fieldId++) { + appendRangeToField(fieldBlocks.get(fieldId), offset, length, fieldBlockBuilders[fieldId]); + } + + boolean[] rawRowIsNull = rowBlock.getRawRowIsNull(); + if (rawRowIsNull != null) { + for (int i = 0; i < length; i++) { + if (rawRowIsNull[offset + i]) { + rowIsNull[positionCount + i] = true; + hasNullRow = true; + } + else { + hasNonNullRow = true; + } + } + } + else { + hasNonNullRow = true; + } + positionCount += length; + } + + private static void appendRangeToField(Block fieldBlock, int offset, int length, BlockBuilder fieldBlockBuilder) + { + fieldBlock = fieldBlock.getLoadedBlock(); + if (fieldBlock instanceof RunLengthEncodedBlock rleBlock) { + fieldBlockBuilder.appendRepeated(rleBlock.getValue(), 0, length); + } + else if (fieldBlock instanceof DictionaryBlock dictionaryBlock) { + int[] rawIds = dictionaryBlock.getRawIds(); + int rawIdsOffset = dictionaryBlock.getRawIdsOffset(); + fieldBlockBuilder.appendPositions(dictionaryBlock.getDictionary(), rawIds, rawIdsOffset + offset, length); + } + else if (fieldBlock instanceof ValueBlock valueBlock) { + fieldBlockBuilder.appendRange(valueBlock, offset, length); + } + else { + throw new IllegalArgumentException("Unsupported block type " + fieldBlock.getClass().getSimpleName()); + } + } + + @Override + public void appendRepeated(ValueBlock block, int position, int count) + { + if (currentEntryOpened) { + throw new IllegalStateException("Current entry must be closed before a null can be written"); + } + if (count == 0) { + return; + } + + RowBlock rowBlock = (RowBlock) block; + ensureCapacity(positionCount + count); + + List fieldBlocks = rowBlock.getFieldBlocks(); + for (int fieldId = 0; fieldId < fieldBlockBuilders.length; fieldId++) { + appendRepeatedToField(fieldBlocks.get(fieldId), position, count, fieldBlockBuilders[fieldId]); + } + + if (rowBlock.isNull(position)) { + Arrays.fill(rowIsNull, positionCount, positionCount + count, true); + hasNullRow = true; + } + else { + hasNonNullRow = true; + } + + positionCount += count; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(Integer.BYTES + Byte.BYTES); + } + } + + private static void appendRepeatedToField(Block fieldBlock, int position, int count, BlockBuilder fieldBlockBuilder) + { + fieldBlock = fieldBlock.getLoadedBlock(); + if (fieldBlock instanceof RunLengthEncodedBlock rleBlock) { + fieldBlockBuilder.appendRepeated(rleBlock.getValue(), 0, count); + } + else if (fieldBlock instanceof DictionaryBlock dictionaryBlock) { + fieldBlockBuilder.appendRepeated(dictionaryBlock.getDictionary(), dictionaryBlock.getId(position), count); + } + else if (fieldBlock instanceof ValueBlock valueBlock) { + fieldBlockBuilder.appendRepeated(valueBlock, position, count); + } + else { + throw new IllegalArgumentException("Unsupported block type " + fieldBlock.getClass().getSimpleName()); + } + } + + @Override + public void appendPositions(ValueBlock block, int[] positions, int offset, int length) + { + if (currentEntryOpened) { + throw new IllegalStateException("Current entry must be closed before a null can be written"); + } + if (length == 0) { + return; + } + + RowBlock rowBlock = (RowBlock) block; + ensureCapacity(positionCount + length); + + List fieldBlocks = rowBlock.getFieldBlocks(); + for (int fieldId = 0; fieldId < fieldBlockBuilders.length; fieldId++) { + appendPositionsToField(fieldBlocks.get(fieldId), positions, offset, length, fieldBlockBuilders[fieldId]); + } + + boolean[] rawRowIsNull = rowBlock.getRawRowIsNull(); + if (rawRowIsNull != null) { + for (int i = 0; i < length; i++) { + if (rawRowIsNull[positions[offset + i]]) { + rowIsNull[positionCount + i] = true; + hasNullRow = true; + } + else { + hasNonNullRow = true; + } + } + } + else { + hasNonNullRow = true; + } + positionCount += length; + } + + private static void appendPositionsToField(Block fieldBlock, int[] positions, int offset, int length, BlockBuilder fieldBlockBuilder) + { + fieldBlock = fieldBlock.getLoadedBlock(); + if (fieldBlock instanceof RunLengthEncodedBlock rleBlock) { + fieldBlockBuilder.appendRepeated(rleBlock.getValue(), 0, length); + } + else if (fieldBlock instanceof DictionaryBlock dictionaryBlock) { + int[] newPositions = new int[length]; + for (int i = 0; i < newPositions.length; i++) { + newPositions[i] = dictionaryBlock.getId(positions[offset + i]); + } + fieldBlockBuilder.appendPositions(dictionaryBlock.getDictionary(), newPositions, 0, length); + } + else if (fieldBlock instanceof ValueBlock valueBlock) { + fieldBlockBuilder.appendPositions(valueBlock, positions, offset, length); + } + else { + throw new IllegalArgumentException("Unsupported block type " + fieldBlock.getClass().getSimpleName()); + } + } + @Override public BlockBuilder appendNull() { @@ -129,10 +328,7 @@ public BlockBuilder appendNull() private void entryAdded(boolean isNull) { - if (rowIsNull.length <= positionCount) { - int newSize = BlockUtil.calculateNewArraySize(rowIsNull.length); - rowIsNull = Arrays.copyOf(rowIsNull, newSize); - } + ensureCapacity(positionCount + 1); rowIsNull[positionCount] = isNull; hasNullRow |= isNull; @@ -176,6 +372,17 @@ public RowBlock buildValueBlock() return createRowBlockInternal(positionCount, hasNullRow ? rowIsNull : null, fieldBlocks); } + private void ensureCapacity(int capacity) + { + if (rowIsNull.length >= capacity) { + return; + } + + // todo add lazy initialize + int newSize = BlockUtil.calculateNewArraySize(rowIsNull.length); + rowIsNull = Arrays.copyOf(rowIsNull, newSize); + } + @Override public String toString() { diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/ShortArrayBlock.java b/core/trino-spi/src/main/java/io/trino/spi/block/ShortArrayBlock.java index 336a5b845539..88ed0b470405 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/ShortArrayBlock.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/ShortArrayBlock.java @@ -244,4 +244,9 @@ short[] getRawValues() { return values; } + + boolean[] getRawValueIsNull() + { + return valueIsNull; + } } diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/ShortArrayBlockBuilder.java b/core/trino-spi/src/main/java/io/trino/spi/block/ShortArrayBlockBuilder.java index ee44b44b6dc2..a18d02142b1a 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/ShortArrayBlockBuilder.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/ShortArrayBlockBuilder.java @@ -19,6 +19,7 @@ import static io.airlift.slice.SizeOf.instanceSize; import static io.airlift.slice.SizeOf.sizeOf; +import static io.trino.spi.block.BlockUtil.calculateNewArraySize; import static java.lang.Math.max; public class ShortArrayBlockBuilder @@ -47,14 +48,12 @@ public ShortArrayBlockBuilder(@Nullable BlockBuilderStatus blockBuilderStatus, i this.blockBuilderStatus = blockBuilderStatus; this.initialEntryCount = max(expectedEntries, 1); - updateDataSize(); + updateRetainedSize(); } - public BlockBuilder writeShort(short value) + public ShortArrayBlockBuilder writeShort(short value) { - if (values.length <= positionCount) { - growCapacity(); - } + ensureCapacity(positionCount + 1); values[positionCount] = value; @@ -67,12 +66,146 @@ public BlockBuilder writeShort(short value) } @Override - public BlockBuilder appendNull() + public void append(ValueBlock block, int position) + { + ensureCapacity(positionCount + 1); + + ShortArrayBlock shortArrayBlock = (ShortArrayBlock) block; + if (shortArrayBlock.isNull(position)) { + valueIsNull[positionCount] = true; + hasNullValue = true; + } + else { + values[positionCount] = shortArrayBlock.getShort(position); + hasNonNullValue = true; + } + positionCount++; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(ShortArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public void appendRepeated(ValueBlock block, int position, int count) + { + if (count == 0) { + return; + } + if (count == 1) { + append(block, position); + return; + } + + ensureCapacity(positionCount + count); + + ShortArrayBlock shortArrayBlock = (ShortArrayBlock) block; + if (shortArrayBlock.isNull(position)) { + Arrays.fill(valueIsNull, positionCount, positionCount + count, true); + hasNullValue = true; + } + else { + short value = shortArrayBlock.getShort(position); + Arrays.fill(values, positionCount, positionCount + count, value); + hasNonNullValue = true; + } + positionCount += count; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(count * ShortArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public void appendRange(ValueBlock block, int offset, int length) { - if (values.length <= positionCount) { - growCapacity(); + if (length == 0) { + return; + } + if (length == 1) { + append(block, offset); + return; } + ensureCapacity(positionCount + length); + + ShortArrayBlock shortArrayBlock = (ShortArrayBlock) block; + int rawOffset = shortArrayBlock.getRawValuesOffset(); + + short[] rawValues = shortArrayBlock.getRawValues(); + System.arraycopy(rawValues, rawOffset + offset, values, positionCount, length); + + boolean[] rawValueIsNull = shortArrayBlock.getRawValueIsNull(); + if (rawValueIsNull != null) { + for (int i = 0; i < length; i++) { + if (rawValueIsNull[rawOffset + offset + i]) { + valueIsNull[positionCount + i] = true; + hasNullValue = true; + } + else { + hasNonNullValue = true; + } + } + } + else { + hasNonNullValue = true; + } + positionCount += length; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(length * ShortArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public void appendPositions(ValueBlock block, int[] positions, int offset, int length) + { + if (length == 0) { + return; + } + if (length == 1) { + append(block, positions[offset]); + return; + } + + ensureCapacity(positionCount + length); + + ShortArrayBlock shortArrayBlock = (ShortArrayBlock) block; + int rawOffset = shortArrayBlock.getRawValuesOffset(); + short[] rawValues = shortArrayBlock.getRawValues(); + boolean[] rawValueIsNull = shortArrayBlock.getRawValueIsNull(); + if (rawValueIsNull != null) { + for (int i = 0; i < length; i++) { + int rawPosition = positions[offset + i] + rawOffset; + if (rawValueIsNull[rawPosition]) { + valueIsNull[positionCount + i] = true; + hasNullValue = true; + } + else { + values[positionCount + i] = rawValues[rawPosition]; + hasNonNullValue = true; + } + } + } + else { + for (int i = 0; i < length; i++) { + int rawPosition = positions[offset + i] + rawOffset; + values[positionCount + i] = rawValues[rawPosition]; + } + hasNonNullValue = true; + } + positionCount += length; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(length * ShortArrayBlock.SIZE_IN_BYTES_PER_POSITION); + } + } + + @Override + public ShortArrayBlockBuilder appendNull() + { + ensureCapacity(positionCount + 1); + valueIsNull[positionCount] = true; hasNullValue = true; @@ -104,23 +237,28 @@ public BlockBuilder newBlockBuilderLike(int expectedEntries, BlockBuilderStatus return new ShortArrayBlockBuilder(blockBuilderStatus, expectedEntries); } - private void growCapacity() + private void ensureCapacity(int capacity) { + if (values.length >= capacity) { + return; + } + int newSize; if (initialized) { - newSize = BlockUtil.calculateNewArraySize(values.length); + newSize = calculateNewArraySize(capacity); } else { newSize = initialEntryCount; initialized = true; } + newSize = max(newSize, capacity); valueIsNull = Arrays.copyOf(valueIsNull, newSize); values = Arrays.copyOf(values, newSize); - updateDataSize(); + updateRetainedSize(); } - private void updateDataSize() + private void updateRetainedSize() { retainedSizeInBytes = INSTANCE_SIZE + sizeOf(valueIsNull) + sizeOf(values); if (blockBuilderStatus != null) { diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/VariableWidthBlock.java b/core/trino-spi/src/main/java/io/trino/spi/block/VariableWidthBlock.java index 931d4cae70eb..58bc8535bebb 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/VariableWidthBlock.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/VariableWidthBlock.java @@ -332,6 +332,21 @@ public VariableWidthBlock copyWithAppendedNull() return new VariableWidthBlock(arrayOffset, positionCount + 1, slice, newOffsets, newValueIsNull); } + int getRawArrayBase() + { + return arrayOffset; + } + + int[] getRawOffsets() + { + return offsets; + } + + boolean[] getRawValueIsNull() + { + return valueIsNull; + } + @Override public VariableWidthBlock getUnderlyingValueBlock() { diff --git a/core/trino-spi/src/main/java/io/trino/spi/block/VariableWidthBlockBuilder.java b/core/trino-spi/src/main/java/io/trino/spi/block/VariableWidthBlockBuilder.java index 6ec063828cf1..59aca4f3b550 100644 --- a/core/trino-spi/src/main/java/io/trino/spi/block/VariableWidthBlockBuilder.java +++ b/core/trino-spi/src/main/java/io/trino/spi/block/VariableWidthBlockBuilder.java @@ -13,35 +13,34 @@ */ package io.trino.spi.block; -import io.airlift.slice.DynamicSliceOutput; import io.airlift.slice.Slice; -import io.airlift.slice.SliceOutput; +import io.airlift.slice.Slices; import jakarta.annotation.Nullable; import java.util.Arrays; -import static io.airlift.slice.SizeOf.SIZE_OF_BYTE; -import static io.airlift.slice.SizeOf.SIZE_OF_INT; import static io.airlift.slice.SizeOf.instanceSize; import static io.airlift.slice.SizeOf.sizeOf; import static io.airlift.slice.Slices.EMPTY_SLICE; import static io.trino.spi.block.BlockUtil.MAX_ARRAY_SIZE; import static io.trino.spi.block.BlockUtil.calculateBlockResetBytes; +import static io.trino.spi.block.BlockUtil.calculateNewArraySize; import static java.lang.Math.min; +import static java.lang.Math.toIntExact; public class VariableWidthBlockBuilder implements BlockBuilder { private static final int INSTANCE_SIZE = instanceSize(VariableWidthBlockBuilder.class); - private static final Block NULL_VALUE_BLOCK = new VariableWidthBlock(0, 1, EMPTY_SLICE, new int[] {0, 0}, new boolean[] {true}); + private static final Block NULL_VALUE_BLOCK = new VariableWidthBlock(0, 1, EMPTY_SLICE, new int[]{0, 0}, new boolean[]{true}); + private static final int SIZE_IN_BYTES_PER_POSITION = Integer.BYTES + Byte.BYTES; private final BlockBuilderStatus blockBuilderStatus; - private boolean initialized; private final int initialEntryCount; private final int initialSliceOutputSize; - private SliceOutput sliceOutput = new DynamicSliceOutput(0); + private byte[] bytes = new byte[0]; private boolean hasNullValue; private boolean hasNonNullValue; @@ -49,7 +48,7 @@ public class VariableWidthBlockBuilder private boolean[] valueIsNull = new boolean[0]; private int[] offsets = new int[1]; - private int positions; + private int positionCount; private long arraysRetainedSizeInBytes; @@ -60,26 +59,25 @@ public VariableWidthBlockBuilder(@Nullable BlockBuilderStatus blockBuilderStatus initialEntryCount = expectedEntries; initialSliceOutputSize = min(expectedBytes, MAX_ARRAY_SIZE); - updateArraysDataSize(); + updateRetainedSize(); } @Override public int getPositionCount() { - return positions; + return positionCount; } @Override public long getSizeInBytes() { - long arraysSizeInBytes = (Integer.BYTES + Byte.BYTES) * (long) positions; - return sliceOutput.size() + arraysSizeInBytes; + return offsets[positionCount] + SIZE_IN_BYTES_PER_POSITION * (long) positionCount; } @Override public long getRetainedSizeInBytes() { - long size = INSTANCE_SIZE + sliceOutput.getRetainedSize() + arraysRetainedSizeInBytes; + long size = INSTANCE_SIZE + arraysRetainedSizeInBytes; if (blockBuilderStatus != null) { size += BlockBuilderStatus.INSTANCE_SIZE; } @@ -93,102 +91,269 @@ public VariableWidthBlockBuilder writeEntry(Slice source) public VariableWidthBlockBuilder writeEntry(Slice source, int sourceIndex, int length) { - if (!initialized) { - initializeCapacity(); - } - - sliceOutput.writeBytes(source, sourceIndex, length); + ensureFreeSpace(length); + source.getBytes(sourceIndex, bytes, offsets[positionCount], length); entryAdded(length, false); return this; } public VariableWidthBlockBuilder writeEntry(byte[] source, int sourceIndex, int length) { - if (!initialized) { - initializeCapacity(); - } - - sliceOutput.writeBytes(source, sourceIndex, length); + ensureFreeSpace(length); + System.arraycopy(source, sourceIndex, bytes, offsets[positionCount], length); entryAdded(length, false); return this; } - public void buildEntry(VariableWidthEntryBuilder builder) - throws E + @Override + public void append(ValueBlock block, int position) { - if (!initialized) { - initializeCapacity(); + ensureCapacity(positionCount + 1); + + VariableWidthBlock variableWidthBlock = (VariableWidthBlock) block; + int bytesWritten = 0; + if (variableWidthBlock.isNull(position)) { + valueIsNull[positionCount] = true; + hasNullValue = true; + } + else { + int rawArrayBase = variableWidthBlock.getRawArrayBase(); + int[] rawOffsets = variableWidthBlock.getRawOffsets(); + int startValueOffset = rawOffsets[rawArrayBase + position]; + int endValueOffset = rawOffsets[rawArrayBase + position + 1]; + int length = endValueOffset - startValueOffset; + ensureFreeSpace(length); + + Slice rawSlice = variableWidthBlock.getRawSlice(); + byte[] rawByteArray = rawSlice.byteArray(); + int byteArrayOffset = rawSlice.byteArrayOffset(); + + System.arraycopy(rawByteArray, byteArrayOffset + startValueOffset, bytes, offsets[positionCount], length); + bytesWritten = length; + hasNonNullValue = true; } + offsets[positionCount + 1] = offsets[positionCount] + bytesWritten; + positionCount++; - int start = sliceOutput.size(); - builder.build(sliceOutput); - int length = sliceOutput.size() - start; - entryAdded(length, false); + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(SIZE_IN_BYTES_PER_POSITION + bytesWritten); + } } - public interface VariableWidthEntryBuilder + @Override + public void appendRepeated(ValueBlock block, int position, int count) { - void build(SliceOutput output) - throws E; + if (count == 0) { + return; + } + if (count == 1) { + append(block, position); + return; + } + + ensureCapacity(positionCount + count); + + VariableWidthBlock variableWidthBlock = (VariableWidthBlock) block; + int bytesWritten = 0; + if (variableWidthBlock.isNull(position)) { + Arrays.fill(valueIsNull, positionCount, positionCount + count, true); + Arrays.fill(offsets, positionCount + 1, positionCount + count + 1, offsets[positionCount]); + hasNullValue = true; + } + else { + int rawArrayBase = variableWidthBlock.getRawArrayBase(); + int[] rawOffsets = variableWidthBlock.getRawOffsets(); + int startValueOffset = rawOffsets[rawArrayBase + position]; + int endValueOffset = rawOffsets[rawArrayBase + position + 1]; + int length = endValueOffset - startValueOffset; + + if (length > 0) { + bytesWritten = toIntExact((long) length * count); + ensureFreeSpace(bytesWritten); + + // copy in the value + Slice rawSlice = variableWidthBlock.getRawSlice(); + byte[] rawByteArray = rawSlice.byteArray(); + int byteArrayOffset = rawSlice.byteArrayOffset(); + + int currentOffset = offsets[positionCount]; + System.arraycopy(rawByteArray, byteArrayOffset + startValueOffset, bytes, currentOffset, length); + + // repeatedly duplicate the written vales, doubling the number of values copied each time + int duplicatedBytes = length; + while (duplicatedBytes * 2 <= bytesWritten) { + System.arraycopy(bytes, currentOffset, bytes, currentOffset + duplicatedBytes, duplicatedBytes); + duplicatedBytes = duplicatedBytes * 2; + } + // copy the remaining values + System.arraycopy(bytes, currentOffset, bytes, currentOffset + duplicatedBytes, bytesWritten - duplicatedBytes); + + // set the offsets + int previousOffset = currentOffset; + for (int i = 0; i < count; i++) { + previousOffset += length; + offsets[positionCount + i + 1] = previousOffset; + } + } + else { + // zero length array + Arrays.fill(offsets, positionCount + 1, positionCount + count + 1, offsets[positionCount]); + } + + hasNonNullValue = true; + } + positionCount += count; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(count * SIZE_IN_BYTES_PER_POSITION + bytesWritten); + } } @Override - public BlockBuilder appendNull() + public void appendRange(ValueBlock block, int offset, int length) { - hasNullValue = true; - entryAdded(0, true); - return this; + if (length == 0) { + return; + } + if (length == 1) { + append(block, offset); + return; + } + + ensureCapacity(positionCount + length); + + VariableWidthBlock variableWidthBlock = (VariableWidthBlock) block; + int rawArrayBase = variableWidthBlock.getRawArrayBase(); + int[] rawOffsets = variableWidthBlock.getRawOffsets(); + int startValueOffset = rawOffsets[rawArrayBase + offset]; + int totalSize = rawOffsets[rawArrayBase + offset + length] - startValueOffset; + // grow the buffer for the new data + ensureFreeSpace(totalSize); + + Slice sourceSlice = variableWidthBlock.getRawSlice(); + System.arraycopy(sourceSlice.byteArray(), sourceSlice.byteArrayOffset() + startValueOffset, bytes, offsets[positionCount], totalSize); + + // update offsets for copied data + int offsetDelta = offsets[positionCount] - rawOffsets[rawArrayBase + offset]; + for (int i = 0; i < length; i++) { + offsets[positionCount + i + 1] = rawOffsets[rawArrayBase + offset + i + 1] + offsetDelta; + } + + // update nulls + boolean[] rawValueIsNull = variableWidthBlock.getRawValueIsNull(); + if (rawValueIsNull != null) { + for (int i = 0; i < length; i++) { + if (rawValueIsNull[rawArrayBase + offset + i]) { + valueIsNull[positionCount + i] = true; + hasNullValue = true; + } + else { + hasNonNullValue = true; + } + } + } + else { + hasNonNullValue = true; + } + positionCount += length; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(length * SIZE_IN_BYTES_PER_POSITION); + } } - private void entryAdded(int bytesWritten, boolean isNull) + @Override + public void appendPositions(ValueBlock block, int[] positions, int offset, int length) { - if (!initialized) { - initializeCapacity(); + if (length == 0) { + return; } - if (valueIsNull.length <= positions) { - growCapacity(); + if (length == 1) { + append(block, positions[offset]); + return; } - valueIsNull[positions] = isNull; - offsets[positions + 1] = sliceOutput.size(); + ensureCapacity(positionCount + length); + + VariableWidthBlock variableWidthBlock = (VariableWidthBlock) block; + int rawArrayBase = variableWidthBlock.getRawArrayBase(); + int[] rawOffsets = variableWidthBlock.getRawOffsets(); + + // update the offsets and compute the total size + int initialOffset = offsets[positionCount]; + int totalSize = 0; + for (int i = 0; i < length; i++) { + int position = positions[offset + i]; + totalSize += rawOffsets[rawArrayBase + position + 1] - rawOffsets[rawArrayBase + position]; + offsets[positionCount + i + 1] = initialOffset + totalSize; + } + // grow the buffer for the new data + ensureFreeSpace(totalSize); + + // copy values to buffer + Slice rawSlice = variableWidthBlock.getRawSlice(); + byte[] sourceBytes = rawSlice.byteArray(); + int sourceBytesOffset = rawSlice.byteArrayOffset(); + for (int i = 0; i < length; i++) { + int position = positions[offset + i]; + int sourceStart = rawOffsets[rawArrayBase + position]; + int sourceLength = rawOffsets[rawArrayBase + position + 1] - sourceStart; + System.arraycopy(sourceBytes, sourceBytesOffset + sourceStart, bytes, offsets[positionCount + i], sourceLength); + totalSize += sourceLength; + } + + // update nulls + boolean[] rawValueIsNull = variableWidthBlock.getRawValueIsNull(); + if (rawValueIsNull != null) { + for (int i = 0; i < length; i++) { + int rawPosition = positions[offset + i] + rawArrayBase; + if (rawValueIsNull[rawPosition]) { + valueIsNull[positionCount + i] = true; + hasNullValue = true; + } + else { + hasNonNullValue = true; + } + } + } + else { + hasNonNullValue = true; + } + positionCount += length; - positions++; - hasNonNullValue |= !isNull; if (blockBuilderStatus != null) { - blockBuilderStatus.addBytes(SIZE_OF_BYTE + SIZE_OF_INT + bytesWritten); + blockBuilderStatus.addBytes(length * SIZE_IN_BYTES_PER_POSITION + totalSize); } } - private void growCapacity() + @Override + public BlockBuilder appendNull() { - int newSize = BlockUtil.calculateNewArraySize(valueIsNull.length); - valueIsNull = Arrays.copyOf(valueIsNull, newSize); - offsets = Arrays.copyOf(offsets, newSize + 1); - updateArraysDataSize(); + hasNullValue = true; + entryAdded(0, true); + return this; } - private void initializeCapacity() + private void entryAdded(int bytesWritten, boolean isNull) { - if (positions != 0) { - throw new IllegalStateException(getClass().getSimpleName() + " was used before initialization"); - } - initialized = true; - valueIsNull = new boolean[initialEntryCount]; - offsets = new int[initialEntryCount + 1]; - sliceOutput = new DynamicSliceOutput(initialSliceOutputSize); - updateArraysDataSize(); - } + ensureCapacity(positionCount + 1); - private void updateArraysDataSize() - { - arraysRetainedSizeInBytes = sizeOf(valueIsNull) + sizeOf(offsets); + valueIsNull[positionCount] = isNull; + offsets[positionCount + 1] = offsets[positionCount] + bytesWritten; + + positionCount++; + hasNonNullValue |= !isNull; + + if (blockBuilderStatus != null) { + blockBuilderStatus.addBytes(SIZE_IN_BYTES_PER_POSITION + bytesWritten); + } } @Override public Block build() { if (!hasNonNullValue) { - return RunLengthEncodedBlock.create(NULL_VALUE_BLOCK, positions); + return RunLengthEncodedBlock.create(NULL_VALUE_BLOCK, positionCount); } return buildValueBlock(); } @@ -196,13 +361,13 @@ public Block build() @Override public VariableWidthBlock buildValueBlock() { - return new VariableWidthBlock(0, positions, sliceOutput.slice(), offsets, hasNullValue ? valueIsNull : null); + return new VariableWidthBlock(0, positionCount, Slices.wrappedBuffer(bytes, 0, offsets[positionCount]), offsets, hasNullValue ? valueIsNull : null); } @Override public BlockBuilder newBlockBuilderLike(int expectedEntries, BlockBuilderStatus blockBuilderStatus) { - int currentSizeInBytes = positions == 0 ? positions : (getOffset(positions) - getOffset(0)); + int currentSizeInBytes = positionCount == 0 ? positionCount : (getOffset(positionCount) - getOffset(0)); return new VariableWidthBlockBuilder(blockBuilderStatus, expectedEntries, calculateBlockResetBytes(currentSizeInBytes)); } @@ -211,12 +376,41 @@ private int getOffset(int position) return offsets[position]; } + private void ensureCapacity(int capacity) + { + if (valueIsNull.length >= capacity) { + return; + } + + int newSize = calculateNewArraySize(capacity, initialEntryCount); + valueIsNull = Arrays.copyOf(valueIsNull, newSize); + offsets = Arrays.copyOf(offsets, newSize + 1); + updateRetainedSize(); + } + + private void ensureFreeSpace(int extraBytesCapacity) + { + int requiredSize = offsets[positionCount] + extraBytesCapacity; + if (bytes.length >= requiredSize) { + return; + } + + int newSize = calculateNewArraySize(requiredSize, initialSliceOutputSize); + bytes = Arrays.copyOf(bytes, newSize); + updateRetainedSize(); + } + + private void updateRetainedSize() + { + arraysRetainedSizeInBytes = sizeOf(valueIsNull) + sizeOf(offsets) + sizeOf(bytes); + } + @Override public String toString() { StringBuilder sb = new StringBuilder("VariableWidthBlockBuilder{"); - sb.append("positionCount=").append(positions); - sb.append(", size=").append(sliceOutput.size()); + sb.append("positionCount=").append(positionCount); + sb.append(", size=").append(offsets[positionCount]); sb.append('}'); return sb.toString(); } diff --git a/core/trino-spi/src/test/java/io/trino/spi/block/AbstractTestBlockBuilder.java b/core/trino-spi/src/test/java/io/trino/spi/block/AbstractTestBlockBuilder.java new file mode 100644 index 000000000000..bcc0057aded7 --- /dev/null +++ b/core/trino-spi/src/test/java/io/trino/spi/block/AbstractTestBlockBuilder.java @@ -0,0 +1,236 @@ +/* + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package io.trino.spi.block; + +import com.google.common.collect.Iterables; +import org.junit.jupiter.api.Test; + +import java.util.ArrayList; +import java.util.List; + +import static java.util.Collections.nCopies; +import static org.assertj.core.api.Assertions.assertThat; + +public abstract class AbstractTestBlockBuilder +{ + protected abstract BlockBuilder createBlockBuilder(); + + protected abstract List getTestValues(); + + protected abstract T getUnusedTestValue(); + + protected abstract ValueBlock blockFromValues(Iterable values); + + protected abstract List blockToValues(ValueBlock valueBlock); + + @Test + public void verifyTestData() + { + List values = getTestValues(); + assertThat(values) + .hasSize(5) + .doesNotHaveDuplicates() + .doesNotContainNull() + .doesNotContain(getUnusedTestValue()); + + ValueBlock valueBlock = blockFromValues(values); + assertThat(blockToValues(valueBlock)).isEqualTo(values); + } + + @Test + public void testAppend() + { + List values = getTestValues(); + ValueBlock inputValues = createOffsetBlock(values); + + BlockBuilder blockBuilder = createBlockBuilder(); + blockBuilder.append(inputValues, 1); + blockBuilder.append(inputValues, 3); + blockBuilder.append(inputValues, 1); + blockBuilder.append(inputValues, 3); + ValueBlock valueBlock = blockBuilder.buildValueBlock(); + + assertThat(valueBlock.mayHaveNull()).isFalse(); + + List actualValues = blockToValues(valueBlock); + assertThat(actualValues).containsExactly(values.get(1), values.get(3), values.get(1), values.get(3)); + } + + @Test + public void testAppendWithNulls() + { + List values = getTestValues(); + ValueBlock inputValues = createOffsetBlockWithOddPositionsNull(values); + + BlockBuilder blockBuilder = createBlockBuilder(); + blockBuilder.append(inputValues, 1); + blockBuilder.append(inputValues, 3); + ValueBlock valueBlock = blockBuilder.buildValueBlock(); + + assertThat(valueBlock.mayHaveNull()).isTrue(); + + List actualValues = blockToValues(valueBlock); + assertThat(actualValues).hasSize(2).containsOnlyNulls(); + + // add a non-null value + blockBuilder.append(inputValues, 2); + valueBlock = blockBuilder.buildValueBlock(); + + actualValues = blockToValues(valueBlock); + assertThat(actualValues).containsExactly(null, null, values.get(2)); + } + + @Test + public void testAppendRepeated() + { + List values = getTestValues(); + ValueBlock inputValues = createOffsetBlock(values); + + BlockBuilder blockBuilder = createBlockBuilder(); + blockBuilder.appendRepeated(inputValues, 1, 10); + blockBuilder.appendRepeated(inputValues, 3, 10); + ValueBlock valueBlock = blockBuilder.buildValueBlock(); + + assertThat(valueBlock.mayHaveNull()).isFalse(); + + List actualValues = blockToValues(valueBlock); + assertThat(actualValues).containsExactlyElementsOf(Iterables.concat(nCopies(10, values.get(1)), nCopies(10, values.get(3)))); + } + + @Test + public void testAppendRepeatedWithNulls() + { + List values = getTestValues(); + ValueBlock inputValues = createOffsetBlockWithOddPositionsNull(values); + + BlockBuilder blockBuilder = createBlockBuilder(); + blockBuilder.appendRepeated(inputValues, 1, 10); + blockBuilder.appendRepeated(inputValues, 3, 10); + ValueBlock valueBlock = blockBuilder.buildValueBlock(); + + assertThat(valueBlock.mayHaveNull()).isTrue(); + + List actualValues = blockToValues(valueBlock); + assertThat(actualValues).hasSize(20).containsOnlyNulls(); + + // an all-null block should be converted to a RunLengthEncodedBlock + assertThat(blockBuilder.build()).isInstanceOf(RunLengthEncodedBlock.class); + + // add some non-null values + blockBuilder.appendRepeated(inputValues, 2, 10); + valueBlock = blockBuilder.buildValueBlock(); + + actualValues = blockToValues(valueBlock); + assertThat(actualValues).containsExactlyElementsOf(Iterables.concat(nCopies(20, null), nCopies(10, values.get(2)))); + } + + @Test + public void testAppendRange() + { + List values = getTestValues(); + ValueBlock inputValues = createOffsetBlock(values); + + BlockBuilder blockBuilder = createBlockBuilder(); + blockBuilder.appendRange(inputValues, 1, 3); + blockBuilder.appendRange(inputValues, 2, 3); + ValueBlock valueBlock = blockBuilder.buildValueBlock(); + + assertThat(valueBlock.mayHaveNull()).isFalse(); + + List actualValues = blockToValues(valueBlock); + assertThat(actualValues).containsExactlyElementsOf(Iterables.concat(values.subList(1, 4), values.subList(2, 5))); + } + + @Test + public void testAppendRangeWithNulls() + { + List values = getTestValues(); + ValueBlock inputValues = createOffsetBlockWithOddPositionsNull(values); + + BlockBuilder blockBuilder = createBlockBuilder(); + blockBuilder.appendRange(inputValues, 1, 3); + blockBuilder.appendRange(inputValues, 2, 3); + ValueBlock valueBlock = blockBuilder.buildValueBlock(); + + assertThat(valueBlock.mayHaveNull()).isTrue(); + + List actualValues = blockToValues(valueBlock); + assertThat(actualValues).containsExactly(null, values.get(2), null, values.get(2), null, values.get(4)); + } + + @Test + public void testAppendPositions() + { + List values = getTestValues(); + ValueBlock inputValues = createOffsetBlock(values); + + BlockBuilder blockBuilder = createBlockBuilder(); + blockBuilder.appendPositions(inputValues, new int[] {-100, 1, 3, 2, -100}, 1, 3); + blockBuilder.appendPositions(inputValues, new int[] {-100, 4, 0, -100}, 1, 2); + ValueBlock valueBlock = blockBuilder.buildValueBlock(); + + assertThat(valueBlock.mayHaveNull()).isFalse(); + + List actualValues = blockToValues(valueBlock); + assertThat(actualValues).containsExactly(values.get(1), values.get(3), values.get(2), values.get(4), values.get(0)); + } + + @Test + public void testAppendPositionsWithNull() + { + List values = getTestValues(); + ValueBlock inputValues = createOffsetBlockWithOddPositionsNull(values); + + BlockBuilder blockBuilder = createBlockBuilder(); + blockBuilder.appendPositions(inputValues, new int[] {-100, 1, 3, 2, -100}, 1, 3); + blockBuilder.appendPositions(inputValues, new int[] {-100, 4, 0, -100}, 1, 2); + ValueBlock valueBlock = blockBuilder.buildValueBlock(); + + assertThat(valueBlock.mayHaveNull()).isTrue(); + + List actualValues = blockToValues(valueBlock); + assertThat(actualValues).containsExactly(null, null, values.get(2), values.get(4), values.get(0)); + } + + /** + * Create a block that is offset from the start of the underlying array + */ + private ValueBlock createOffsetBlock(List values) + { + return blockFromValues(Iterables.concat(nCopies(2, getUnusedTestValue()), values, nCopies(2, getUnusedTestValue()))) + .getRegion(2, values.size()); + } + + /** + * Create a block that is offset from the start of the underlying array + */ + private ValueBlock createOffsetBlockWithOddPositionsNull(List values) + { + ArrayList blockValues = new ArrayList<>(); + blockValues.add(getUnusedTestValue()); + blockValues.add(getUnusedTestValue()); + for (int i = 0; i < values.size(); i++) { + T value = values.get(i); + if (i % 2 == 0) { + blockValues.add(value); + } + else { + blockValues.add(null); + } + } + blockValues.add(getUnusedTestValue()); + blockValues.add(getUnusedTestValue()); + return blockFromValues(blockValues).getRegion(2, values.size()); + } +} diff --git a/core/trino-spi/src/test/java/io/trino/spi/block/TestArrayBlockBuilder.java b/core/trino-spi/src/test/java/io/trino/spi/block/TestArrayBlockBuilder.java index 511961cb8260..689504f8f927 100644 --- a/core/trino-spi/src/test/java/io/trino/spi/block/TestArrayBlockBuilder.java +++ b/core/trino-spi/src/test/java/io/trino/spi/block/TestArrayBlockBuilder.java @@ -15,13 +15,19 @@ import org.junit.jupiter.api.Test; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; + import static io.airlift.slice.SizeOf.instanceSize; import static io.trino.spi.type.BigintType.BIGINT; +import static io.trino.spi.type.VarcharType.VARCHAR; import static java.lang.Long.BYTES; import static org.assertj.core.api.Assertions.assertThat; import static org.assertj.core.api.Assertions.assertThatThrownBy; public class TestArrayBlockBuilder + extends AbstractTestBlockBuilder> { // ArrayBlockBuilder: isNull, offset, 3 * value (FixedWidthBlockBuilder: isNull, value) private static final int THREE_INTS_ENTRY_SIZE = Byte.BYTES + Integer.BYTES + 3 * (Byte.BYTES + Long.BYTES); @@ -107,4 +113,77 @@ private static void assertIsAllNulls(Block block, int expectedPositionCount) assertThat(block.isNull(0)).isTrue(); } } + + @Override + protected BlockBuilder createBlockBuilder() + { + return new ArrayBlockBuilder(new VariableWidthBlockBuilder(null, 1, 100), null, 1); + } + + @Override + protected List> getTestValues() + { + return List.of( + List.of("a", "apple", "ape"), + Arrays.asList("b", null, "bear", "break"), + List.of("c", "cherry"), + Arrays.asList("d", "date", "dinosaur", null, "dirt"), + List.of("e", "eggplant", "empty", "")); + } + + @Override + protected List getUnusedTestValue() + { + return List.of("unused", "ignore me"); + } + + @Override + protected ValueBlock blockFromValues(Iterable> values) + { + ArrayBlockBuilder blockBuilder = new ArrayBlockBuilder(new VariableWidthBlockBuilder(null, 1, 100), null, 1); + for (List array : values) { + if (array == null) { + blockBuilder.appendNull(); + } + else { + blockBuilder.buildEntry(elementBuilder -> { + for (String entry : array) { + if (entry == null) { + elementBuilder.appendNull(); + } + else { + VARCHAR.writeString(elementBuilder, entry); + } + } + }); + } + } + return blockBuilder.buildValueBlock(); + } + + @Override + protected List> blockToValues(ValueBlock valueBlock) + { + ArrayBlock block = (ArrayBlock) valueBlock; + List> actualValues = new ArrayList<>(block.getPositionCount()); + for (int i = 0; i < block.getPositionCount(); i++) { + if (block.isNull(i)) { + actualValues.add(null); + } + else { + Block array = block.getArray(i); + ArrayList arrayBuilder = new ArrayList<>(); + for (int j = 0; j < array.getPositionCount(); j++) { + if (array.isNull(j)) { + arrayBuilder.add(null); + } + else { + arrayBuilder.add(VARCHAR.getSlice(array, j).toStringUtf8()); + } + } + actualValues.add(arrayBuilder); + } + } + return actualValues; + } } diff --git a/core/trino-spi/src/test/java/io/trino/spi/block/TestBlockUtil.java b/core/trino-spi/src/test/java/io/trino/spi/block/TestBlockUtil.java index a58582cdb617..496dc2590d53 100644 --- a/core/trino-spi/src/test/java/io/trino/spi/block/TestBlockUtil.java +++ b/core/trino-spi/src/test/java/io/trino/spi/block/TestBlockUtil.java @@ -16,21 +16,30 @@ import org.junit.jupiter.api.Test; import static io.trino.spi.block.BlockUtil.MAX_ARRAY_SIZE; -import static java.lang.String.format; +import static io.trino.spi.block.BlockUtil.calculateNewArraySize; import static org.assertj.core.api.Assertions.assertThat; +import static org.assertj.core.api.Assertions.assertThatThrownBy; public class TestBlockUtil { @Test public void testCalculateNewArraySize() { - assertThat(BlockUtil.calculateNewArraySize(200)).isEqualTo(300); - assertThat(BlockUtil.calculateNewArraySize(Integer.MAX_VALUE)).isEqualTo(MAX_ARRAY_SIZE); - try { - BlockUtil.calculateNewArraySize(MAX_ARRAY_SIZE); - } - catch (IllegalArgumentException e) { - assertThat(e.getMessage()).isEqualTo(format("Cannot grow array beyond '%s'", MAX_ARRAY_SIZE)); - } + assertThat(calculateNewArraySize(200)).isEqualTo(300); + assertThat(calculateNewArraySize(200, 10)).isEqualTo(300); + assertThat(calculateNewArraySize(200, 500)).isEqualTo(500); + + assertThat(calculateNewArraySize(MAX_ARRAY_SIZE - 1)).isEqualTo(MAX_ARRAY_SIZE); + assertThat(calculateNewArraySize(10, MAX_ARRAY_SIZE)).isEqualTo(MAX_ARRAY_SIZE); + + assertThat(calculateNewArraySize(1, 0)).isEqualTo(64); + + assertThatThrownBy(() -> calculateNewArraySize(Integer.MAX_VALUE)) + .isInstanceOf(IllegalArgumentException.class); + assertThatThrownBy(() -> calculateNewArraySize(0, Integer.MAX_VALUE)) + .isInstanceOf(IllegalArgumentException.class); + assertThatThrownBy(() -> calculateNewArraySize(MAX_ARRAY_SIZE)) + .isInstanceOf(IllegalArgumentException.class) + .hasMessage("Cannot grow array beyond size %d".formatted(MAX_ARRAY_SIZE)); } } diff --git a/core/trino-spi/src/test/java/io/trino/spi/block/TestByteArrayBlockBuilder.java b/core/trino-spi/src/test/java/io/trino/spi/block/TestByteArrayBlockBuilder.java new file mode 100644 index 000000000000..91463d2a473f --- /dev/null +++ b/core/trino-spi/src/test/java/io/trino/spi/block/TestByteArrayBlockBuilder.java @@ -0,0 +1,70 @@ +/* + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package io.trino.spi.block; + +import java.util.ArrayList; +import java.util.List; + +public class TestByteArrayBlockBuilder + extends AbstractTestBlockBuilder +{ + @Override + protected BlockBuilder createBlockBuilder() + { + return new ByteArrayBlockBuilder(null, 1); + } + + @Override + protected List getTestValues() + { + return List.of((byte) 10, (byte) 11, (byte) 12, (byte) 13, (byte) 14); + } + + @Override + protected Byte getUnusedTestValue() + { + return (byte) -1; + } + + @Override + protected ValueBlock blockFromValues(Iterable values) + { + ByteArrayBlockBuilder blockBuilder = new ByteArrayBlockBuilder(null, 1); + for (Byte value : values) { + if (value == null) { + blockBuilder.appendNull(); + } + else { + blockBuilder.writeByte(value); + } + } + return blockBuilder.buildValueBlock(); + } + + @Override + protected List blockToValues(ValueBlock valueBlock) + { + ByteArrayBlock byteArrayBlock = (ByteArrayBlock) valueBlock; + List actualValues = new ArrayList<>(byteArrayBlock.getPositionCount()); + for (int i = 0; i < byteArrayBlock.getPositionCount(); i++) { + if (byteArrayBlock.isNull(i)) { + actualValues.add(null); + } + else { + actualValues.add(byteArrayBlock.getByte(i)); + } + } + return actualValues; + } +} diff --git a/core/trino-spi/src/test/java/io/trino/spi/block/TestFixed12BlockBuilder.java b/core/trino-spi/src/test/java/io/trino/spi/block/TestFixed12BlockBuilder.java new file mode 100644 index 000000000000..392125972bfa --- /dev/null +++ b/core/trino-spi/src/test/java/io/trino/spi/block/TestFixed12BlockBuilder.java @@ -0,0 +1,72 @@ +/* + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package io.trino.spi.block; + +import java.util.ArrayList; +import java.util.List; + +public class TestFixed12BlockBuilder + extends AbstractTestBlockBuilder +{ + @Override + protected BlockBuilder createBlockBuilder() + { + return new Fixed12BlockBuilder(null, 1); + } + + @Override + protected List getTestValues() + { + return List.of(new Fixed12(90, 10), new Fixed12(91, 11), new Fixed12(92, 12), new Fixed12(93, 13), new Fixed12(94, 14)); + } + + @Override + protected Fixed12 getUnusedTestValue() + { + return new Fixed12(-1, -2); + } + + @Override + protected ValueBlock blockFromValues(Iterable values) + { + Fixed12BlockBuilder blockBuilder = new Fixed12BlockBuilder(null, 1); + for (Fixed12 value : values) { + if (value == null) { + blockBuilder.appendNull(); + } + else { + blockBuilder.writeFixed12(value.first(), value.second()); + } + } + return blockBuilder.buildValueBlock(); + } + + @Override + protected List blockToValues(ValueBlock valueBlock) + { + Fixed12Block fixed12ArrayBlock = (Fixed12Block) valueBlock; + List actualValues = new ArrayList<>(fixed12ArrayBlock.getPositionCount()); + for (int i = 0; i < fixed12ArrayBlock.getPositionCount(); i++) { + if (fixed12ArrayBlock.isNull(i)) { + actualValues.add(null); + } + else { + actualValues.add(new Fixed12(fixed12ArrayBlock.getFixed12First(i), fixed12ArrayBlock.getFixed12Second(i))); + } + } + return actualValues; + } + + public record Fixed12(long first, int second) {} +} diff --git a/core/trino-spi/src/test/java/io/trino/spi/block/TestInt128ArrayBlockBuilder.java b/core/trino-spi/src/test/java/io/trino/spi/block/TestInt128ArrayBlockBuilder.java new file mode 100644 index 000000000000..e25fe5af37f8 --- /dev/null +++ b/core/trino-spi/src/test/java/io/trino/spi/block/TestInt128ArrayBlockBuilder.java @@ -0,0 +1,72 @@ +/* + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package io.trino.spi.block; + +import io.trino.spi.type.Int128; + +import java.util.ArrayList; +import java.util.List; + +public class TestInt128ArrayBlockBuilder + extends AbstractTestBlockBuilder +{ + @Override + protected BlockBuilder createBlockBuilder() + { + return new Int128ArrayBlockBuilder(null, 1); + } + + @Override + protected List getTestValues() + { + return List.of(Int128.valueOf(90, 10), Int128.valueOf(91, 11), Int128.valueOf(92, 12), Int128.valueOf(93, 13), Int128.valueOf(94, 14)); + } + + @Override + protected Int128 getUnusedTestValue() + { + return Int128.valueOf(-1, -2); + } + + @Override + protected ValueBlock blockFromValues(Iterable values) + { + Int128ArrayBlockBuilder blockBuilder = new Int128ArrayBlockBuilder(null, 1); + for (Int128 value : values) { + if (value == null) { + blockBuilder.appendNull(); + } + else { + blockBuilder.writeInt128(value.getHigh(), value.getLow()); + } + } + return blockBuilder.buildValueBlock(); + } + + @Override + protected List blockToValues(ValueBlock valueBlock) + { + Int128ArrayBlock int128ArrayBlock = (Int128ArrayBlock) valueBlock; + List actualValues = new ArrayList<>(int128ArrayBlock.getPositionCount()); + for (int i = 0; i < int128ArrayBlock.getPositionCount(); i++) { + if (int128ArrayBlock.isNull(i)) { + actualValues.add(null); + } + else { + actualValues.add(int128ArrayBlock.getInt128(i)); + } + } + return actualValues; + } +} diff --git a/core/trino-spi/src/test/java/io/trino/spi/block/TestIntArrayBlockBuilder.java b/core/trino-spi/src/test/java/io/trino/spi/block/TestIntArrayBlockBuilder.java new file mode 100644 index 000000000000..9996772166fd --- /dev/null +++ b/core/trino-spi/src/test/java/io/trino/spi/block/TestIntArrayBlockBuilder.java @@ -0,0 +1,70 @@ +/* + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package io.trino.spi.block; + +import java.util.ArrayList; +import java.util.List; + +public class TestIntArrayBlockBuilder + extends AbstractTestBlockBuilder +{ + @Override + protected BlockBuilder createBlockBuilder() + { + return new IntArrayBlockBuilder(null, 1); + } + + @Override + protected List getTestValues() + { + return List.of(10, 11, 12, 13, 14); + } + + @Override + protected Integer getUnusedTestValue() + { + return -1; + } + + @Override + protected ValueBlock blockFromValues(Iterable values) + { + IntArrayBlockBuilder blockBuilder = new IntArrayBlockBuilder(null, 1); + for (Integer value : values) { + if (value == null) { + blockBuilder.appendNull(); + } + else { + blockBuilder.writeInt(value); + } + } + return blockBuilder.buildValueBlock(); + } + + @Override + protected List blockToValues(ValueBlock valueBlock) + { + IntArrayBlock intArrayBlock = (IntArrayBlock) valueBlock; + List actualValues = new ArrayList<>(intArrayBlock.getPositionCount()); + for (int i = 0; i < intArrayBlock.getPositionCount(); i++) { + if (intArrayBlock.isNull(i)) { + actualValues.add(null); + } + else { + actualValues.add(intArrayBlock.getInt(i)); + } + } + return actualValues; + } +} diff --git a/core/trino-spi/src/test/java/io/trino/spi/block/TestLongArrayBlockBuilder.java b/core/trino-spi/src/test/java/io/trino/spi/block/TestLongArrayBlockBuilder.java new file mode 100644 index 000000000000..88766a0f3e67 --- /dev/null +++ b/core/trino-spi/src/test/java/io/trino/spi/block/TestLongArrayBlockBuilder.java @@ -0,0 +1,70 @@ +/* + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package io.trino.spi.block; + +import java.util.ArrayList; +import java.util.List; + +public class TestLongArrayBlockBuilder + extends AbstractTestBlockBuilder +{ + @Override + protected BlockBuilder createBlockBuilder() + { + return new LongArrayBlockBuilder(null, 1); + } + + @Override + protected List getTestValues() + { + return List.of(10L, 11L, 12L, 13L, 14L); + } + + @Override + protected Long getUnusedTestValue() + { + return -1L; + } + + @Override + protected ValueBlock blockFromValues(Iterable values) + { + LongArrayBlockBuilder blockBuilder = new LongArrayBlockBuilder(null, 1); + for (Long value : values) { + if (value == null) { + blockBuilder.appendNull(); + } + else { + blockBuilder.writeLong(value); + } + } + return blockBuilder.buildValueBlock(); + } + + @Override + protected List blockToValues(ValueBlock valueBlock) + { + LongArrayBlock longArrayBlock = (LongArrayBlock) valueBlock; + List actualValues = new ArrayList<>(longArrayBlock.getPositionCount()); + for (int i = 0; i < longArrayBlock.getPositionCount(); i++) { + if (longArrayBlock.isNull(i)) { + actualValues.add(null); + } + else { + actualValues.add(longArrayBlock.getLong(i)); + } + } + return actualValues; + } +} diff --git a/core/trino-spi/src/test/java/io/trino/spi/block/TestMapBlockBuilder.java b/core/trino-spi/src/test/java/io/trino/spi/block/TestMapBlockBuilder.java new file mode 100644 index 000000000000..137d0cc6c2cb --- /dev/null +++ b/core/trino-spi/src/test/java/io/trino/spi/block/TestMapBlockBuilder.java @@ -0,0 +1,101 @@ +/* + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package io.trino.spi.block; + +import io.trino.spi.type.MapType; +import io.trino.spi.type.TypeOperators; + +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import static io.trino.spi.type.IntegerType.INTEGER; +import static io.trino.spi.type.VarcharType.VARCHAR; + +public class TestMapBlockBuilder + extends AbstractTestBlockBuilder> +{ + private static final TypeOperators TYPE_OPERATORS = new TypeOperators(); + private static final MapType MAP_TYPE = new MapType(VARCHAR, INTEGER, TYPE_OPERATORS); + + @Override + protected BlockBuilder createBlockBuilder() + { + return new MapBlockBuilder(MAP_TYPE, null, 1); + } + + @Override + protected List> getTestValues() + { + return List.of( + Map.of("a", 0, "apple", 1, "ape", 2), + Map.of("b", 3, "banana", 4, "bear", 5, "break", 6), + Map.of("c", 7, "cherry", 8), + Map.of("d", 9, "date", 10, "dinosaur", 11, "dinner", 12, "dirt", 13), + Map.of("e", 14, "eggplant", 15, "empty", 16, "", 17)); + } + + @Override + protected Map getUnusedTestValue() + { + return Map.of("unused", -1, "ignore me", -2); + } + + @Override + protected ValueBlock blockFromValues(Iterable> maps) + { + MapBlockBuilder blockBuilder = new MapBlockBuilder(MAP_TYPE, null, 1); + for (Map map : maps) { + if (map == null) { + blockBuilder.appendNull(); + } + else { + blockBuilder.buildEntry((keyBuilder, valueBuilder) -> { + for (Map.Entry entry : map.entrySet()) { + VARCHAR.writeString(keyBuilder, entry.getKey()); + INTEGER.writeLong(valueBuilder, entry.getValue()); + } + }); + } + } + return blockBuilder.buildValueBlock(); + } + + @Override + protected List> blockToValues(ValueBlock valueBlock) + { + MapBlock block = (MapBlock) valueBlock; + List> actualValues = new ArrayList<>(block.getPositionCount()); + for (int i = 0; i < block.getPositionCount(); i++) { + if (block.isNull(i)) { + actualValues.add(null); + } + else { + SqlMap sqlMap = block.getMap(i); + int rawOffset = sqlMap.getRawOffset(); + Block rawKeyBlock = sqlMap.getRawKeyBlock(); + Block rawValueBlock = sqlMap.getRawValueBlock(); + Map actualMap = new HashMap<>(); + for (int entryIndex = 0; entryIndex < sqlMap.getSize(); entryIndex++) { + actualMap.put( + VARCHAR.getSlice(rawKeyBlock, rawOffset + entryIndex).toStringUtf8(), + INTEGER.getInt(rawValueBlock, rawOffset + entryIndex)); + } + actualValues.add(actualMap); + } + } + return actualValues; + } +} diff --git a/core/trino-spi/src/test/java/io/trino/spi/block/TestRowBlockBuilder.java b/core/trino-spi/src/test/java/io/trino/spi/block/TestRowBlockBuilder.java index c46a57ded70e..6befb3606c99 100644 --- a/core/trino-spi/src/test/java/io/trino/spi/block/TestRowBlockBuilder.java +++ b/core/trino-spi/src/test/java/io/trino/spi/block/TestRowBlockBuilder.java @@ -16,10 +16,17 @@ import com.google.common.collect.ImmutableList; import org.junit.jupiter.api.Test; +import java.util.ArrayList; +import java.util.List; + import static io.trino.spi.type.BigintType.BIGINT; +import static io.trino.spi.type.BooleanType.BOOLEAN; +import static io.trino.spi.type.IntegerType.INTEGER; +import static io.trino.spi.type.VarcharType.VARCHAR; import static org.assertj.core.api.Assertions.assertThat; public class TestRowBlockBuilder + extends AbstractTestBlockBuilder { @Test public void testBuilderProducesNullRleForNullRows() @@ -53,4 +60,73 @@ private static void assertIsAllNulls(Block block, int expectedPositionCount) assertThat(block.isNull(0)).isTrue(); } } + + @Override + protected BlockBuilder createBlockBuilder() + { + return new RowBlockBuilder(List.of(VARCHAR, INTEGER, BOOLEAN), null, 1); + } + + @Override + protected List getTestValues() + { + return List.of( + new TestRow("apple", 2, true), + new TestRow("bear", 5, false), + new TestRow(null, 7, true), + new TestRow("dinosaur", 9, false), + new TestRow("", 22, true)); + } + + @Override + protected TestRow getUnusedTestValue() + { + return new TestRow("unused", -1, false); + } + + @Override + protected ValueBlock blockFromValues(Iterable values) + { + RowBlockBuilder blockBuilder = new RowBlockBuilder(List.of(VARCHAR, INTEGER, BOOLEAN), null, 1); + for (TestRow row : values) { + if (row == null) { + blockBuilder.appendNull(); + } + else { + blockBuilder.buildEntry(fieldBuilders -> { + if (row.name() == null) { + fieldBuilders.get(0).appendNull(); + } + else { + VARCHAR.writeString(fieldBuilders.get(0), row.name()); + } + INTEGER.writeLong(fieldBuilders.get(1), row.number()); + BOOLEAN.writeBoolean(fieldBuilders.get(2), row.flag()); + }); + } + } + return blockBuilder.buildValueBlock(); + } + + @Override + protected List blockToValues(ValueBlock valueBlock) + { + RowBlock block = (RowBlock) valueBlock; + List actualValues = new ArrayList<>(block.getPositionCount()); + for (int i = 0; i < block.getPositionCount(); i++) { + if (block.isNull(i)) { + actualValues.add(null); + } + else { + SqlRow sqlRow = block.getRow(i); + actualValues.add(new TestRow( + (String) VARCHAR.getObjectValue(null, sqlRow.getUnderlyingFieldBlock(0), sqlRow.getUnderlyingFieldPosition(0)), + INTEGER.getInt(sqlRow.getUnderlyingFieldBlock(1), sqlRow.getUnderlyingFieldPosition(1)), + BOOLEAN.getBoolean(sqlRow.getUnderlyingFieldBlock(2), sqlRow.getUnderlyingFieldPosition(2)))); + } + } + return actualValues; + } + + public record TestRow(String name, int number, boolean flag) {} } diff --git a/core/trino-spi/src/test/java/io/trino/spi/block/TestShortArrayBlockBuilder.java b/core/trino-spi/src/test/java/io/trino/spi/block/TestShortArrayBlockBuilder.java new file mode 100644 index 000000000000..031e16b81079 --- /dev/null +++ b/core/trino-spi/src/test/java/io/trino/spi/block/TestShortArrayBlockBuilder.java @@ -0,0 +1,70 @@ +/* + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package io.trino.spi.block; + +import java.util.ArrayList; +import java.util.List; + +public class TestShortArrayBlockBuilder + extends AbstractTestBlockBuilder +{ + @Override + protected BlockBuilder createBlockBuilder() + { + return new ShortArrayBlockBuilder(null, 1); + } + + @Override + protected List getTestValues() + { + return List.of((short) 10, (short) 11, (short) 12, (short) 13, (short) 14); + } + + @Override + protected Short getUnusedTestValue() + { + return (short) -1; + } + + @Override + protected ValueBlock blockFromValues(Iterable values) + { + ShortArrayBlockBuilder blockBuilder = new ShortArrayBlockBuilder(null, 1); + for (Short value : values) { + if (value == null) { + blockBuilder.appendNull(); + } + else { + blockBuilder.writeShort(value); + } + } + return blockBuilder.buildValueBlock(); + } + + @Override + protected List blockToValues(ValueBlock valueBlock) + { + ShortArrayBlock shortArrayBlock = (ShortArrayBlock) valueBlock; + List actualValues = new ArrayList<>(shortArrayBlock.getPositionCount()); + for (int i = 0; i < shortArrayBlock.getPositionCount(); i++) { + if (shortArrayBlock.isNull(i)) { + actualValues.add(null); + } + else { + actualValues.add(shortArrayBlock.getShort(i)); + } + } + return actualValues; + } +} diff --git a/core/trino-spi/src/test/java/io/trino/spi/block/TestVariableWidthBlockBuilder.java b/core/trino-spi/src/test/java/io/trino/spi/block/TestVariableWidthBlockBuilder.java index f41eaf4fc22d..5e86f7d4da6f 100644 --- a/core/trino-spi/src/test/java/io/trino/spi/block/TestVariableWidthBlockBuilder.java +++ b/core/trino-spi/src/test/java/io/trino/spi/block/TestVariableWidthBlockBuilder.java @@ -13,22 +13,26 @@ */ package io.trino.spi.block; -import io.airlift.slice.DynamicSliceOutput; -import io.airlift.slice.Slice; +import com.google.common.collect.ImmutableList; import io.airlift.slice.Slices; import org.junit.jupiter.api.Test; +import java.util.ArrayList; +import java.util.List; + import static io.airlift.slice.SizeOf.SIZE_OF_INT; import static io.airlift.slice.SizeOf.instanceSize; import static io.airlift.slice.SizeOf.sizeOf; +import static io.airlift.slice.Slices.utf8Slice; import static io.trino.spi.type.VarcharType.VARCHAR; import static java.lang.Math.ceil; +import static java.util.Collections.nCopies; import static org.assertj.core.api.Assertions.assertThat; public class TestVariableWidthBlockBuilder + extends AbstractTestBlockBuilder { private static final int BLOCK_BUILDER_INSTANCE_SIZE = instanceSize(VariableWidthBlockBuilder.class); - private static final int SLICE_INSTANCE_SIZE = instanceSize(DynamicSliceOutput.class) + instanceSize(Slice.class); private static final int VARCHAR_VALUE_SIZE = 7; private static final int VARCHAR_ENTRY_SIZE = SIZE_OF_INT + VARCHAR_VALUE_SIZE; private static final int EXPECTED_ENTRY_COUNT = 3; @@ -52,9 +56,9 @@ public void testNewBlockBuilderLike() // force to initialize capacity blockBuilder.writeEntry(Slices.wrappedBuffer((byte) 1)); - long actualArrayBytes = sizeOf(new int[(int) ceil(resetSkew * (entries + 1))]) + sizeOf(new boolean[(int) ceil(resetSkew * entries)]); - long actualSliceBytes = SLICE_INSTANCE_SIZE + sizeOf(new byte[(int) ceil(resetSkew * entries)]); - assertThat(blockBuilder.getRetainedSizeInBytes()).isEqualTo(BLOCK_BUILDER_INSTANCE_SIZE + actualSliceBytes + actualArrayBytes); + long actualArraySize = sizeOf(new int[(int) ceil(resetSkew * (entries + 1))]) + sizeOf(new boolean[(int) ceil(resetSkew * entries)]); + long actualBytesSize = sizeOf(new byte[(int) ceil(resetSkew * entries)]); + assertThat(blockBuilder.getRetainedSizeInBytes()).isEqualTo(BLOCK_BUILDER_INSTANCE_SIZE + actualBytesSize + actualArraySize); } private void testIsFull(PageBuilderStatus pageBuilderStatus) @@ -81,6 +85,116 @@ public void testBuilderProducesNullRleForNullRows() assertIsAllNulls(blockBuilder().appendNull().appendNull().build(), 2); } + @Test + public void testAppendRepeatedEmpty() + { + VariableWidthBlockBuilder blockBuilder = new VariableWidthBlockBuilder(null, 1, 100); + ValueBlock value = VARCHAR.createBlockBuilder(null, 1) + .writeEntry(utf8Slice("ignored")) + .writeEntry(Slices.EMPTY_SLICE) + .writeEntry(utf8Slice("ignored")) + .buildValueBlock(); + blockBuilder.appendRepeated(value, 1, 10); + + List strings = toStrings(blockBuilder.buildValueBlock()); + assertThat(strings).isEqualTo(nCopies(10, "")); + } + + @Test + public void testAppendRepeatedSingle() + { + VariableWidthBlockBuilder blockBuilder = new VariableWidthBlockBuilder(null, 1, 100); + VariableWidthBlock value = VARCHAR.createBlockBuilder(null, 1) + .writeEntry(utf8Slice("ignored")) + .writeEntry(utf8Slice("ab")) + .writeEntry(utf8Slice("ignored")) + .buildValueBlock(); + blockBuilder.appendRepeated(value, 1, 1); + + List strings = toStrings(blockBuilder.buildValueBlock()); + assertThat(strings).isEqualTo(ImmutableList.of("ab")); + } + + @Test + public void testAppendRepeated1Byte() + { + VariableWidthBlockBuilder blockBuilder = new VariableWidthBlockBuilder(null, 1, 100); + VariableWidthBlock value = VARCHAR.createBlockBuilder(null, 1) + .writeEntry(utf8Slice("ignored")) + .writeEntry(utf8Slice("X")) + .writeEntry(utf8Slice("ignored")) + .writeEntry(utf8Slice("Y")) + .writeEntry(utf8Slice("ignored")) + .buildValueBlock(); + blockBuilder.appendRepeated(value, 1, 3); + blockBuilder.appendRepeated(value, 3, 2); + + List strings = toStrings(blockBuilder.buildValueBlock()); + assertThat(strings).isEqualTo(ImmutableList.of("X", "X", "X", "Y", "Y")); + } + + @Test + public void testAppendRepeated2Bytes() + { + VariableWidthBlockBuilder blockBuilder = new VariableWidthBlockBuilder(null, 1, 100); + VariableWidthBlock value = VARCHAR.createBlockBuilder(null, 1) + .writeEntry(utf8Slice("ignored")) + .writeEntry(utf8Slice("ab")) + .writeEntry(utf8Slice("ignored")) + .writeEntry(utf8Slice("Y")) + .writeEntry(utf8Slice("ignored")) + .buildValueBlock(); + blockBuilder.appendRepeated(value, 1, 3); + + List strings = toStrings(blockBuilder.buildValueBlock()); + assertThat(strings).isEqualTo(ImmutableList.of("ab", "ab", "ab")); + } + + @Test + public void testAppendRepeatedMultipleBytesOddNumberOfTimes() + { + VariableWidthBlockBuilder blockBuilder = new VariableWidthBlockBuilder(null, 1, 100); + VariableWidthBlock value = VARCHAR.createBlockBuilder(null, 1) + .writeEntry(utf8Slice("ignored")) + .writeEntry(utf8Slice("abc")) + .writeEntry(utf8Slice("ignored")) + .buildValueBlock(); + blockBuilder.appendRepeated(value, 1, 5); + + List strings = toStrings(blockBuilder.buildValueBlock()); + assertThat(strings).isEqualTo(nCopies(5, "abc")); + } + + @Test + public void testAppendRepeatedMultipleBytesEvenNumberOfTimes() + { + VariableWidthBlockBuilder blockBuilder = new VariableWidthBlockBuilder(null, 1, 100); + VariableWidthBlock value = VARCHAR.createBlockBuilder(null, 1) + .writeEntry(utf8Slice("ignored")) + .writeEntry(utf8Slice("abc")) + .writeEntry(utf8Slice("ignored")) + .buildValueBlock(); + blockBuilder.appendRepeated(value, 1, 6); + + List strings = toStrings(blockBuilder.buildValueBlock()); + assertThat(strings).isEqualTo(nCopies(6, "abc")); + } + + @Test + public void testAppendRepeatedMultipleBytesPowerOf2NumberOfTimes() + { + VariableWidthBlockBuilder blockBuilder = new VariableWidthBlockBuilder(null, 1, 100); + VariableWidthBlock value = VARCHAR.createBlockBuilder(null, 1) + .writeEntry(utf8Slice("ignored")) + .writeEntry(utf8Slice("abc")) + .writeEntry(utf8Slice("ignored")) + .buildValueBlock(); + blockBuilder.appendRepeated(value, 1, 8); + + List strings = toStrings(blockBuilder.buildValueBlock()); + assertThat(strings).isEqualTo(nCopies(8, "abc")); + } + private static BlockBuilder blockBuilder() { return new VariableWidthBlockBuilder(null, 10, 0); @@ -100,4 +214,62 @@ private static void assertIsAllNulls(Block block, int expectedPositionCount) assertThat(block.isNull(0)).isTrue(); } } + + private static List toStrings(VariableWidthBlock block) + { + ImmutableList.Builder list = ImmutableList.builder(); + for (int i = 0; i < block.getPositionCount(); i++) { + list.add(VARCHAR.getSlice(block, i).toStringUtf8()); + } + return list.build(); + } + + @Override + protected BlockBuilder createBlockBuilder() + { + return new VariableWidthBlockBuilder(null, 1, 100); + } + + @Override + protected List getTestValues() + { + return List.of("a", "bb", "cCc", "dddd", "eeEee"); + } + + @Override + protected String getUnusedTestValue() + { + return "unused"; + } + + @Override + protected ValueBlock blockFromValues(Iterable values) + { + VariableWidthBlockBuilder blockBuilder = new VariableWidthBlockBuilder(null, 1, 100); + for (String value : values) { + if (value == null) { + blockBuilder.appendNull(); + } + else { + blockBuilder.writeEntry(utf8Slice(value)); + } + } + return blockBuilder.buildValueBlock(); + } + + @Override + protected List blockToValues(ValueBlock valueBlock) + { + VariableWidthBlock block = (VariableWidthBlock) valueBlock; + List actualValues = new ArrayList<>(block.getPositionCount()); + for (int i = 0; i < block.getPositionCount(); i++) { + if (block.isNull(i)) { + actualValues.add(null); + } + else { + actualValues.add(block.getSlice(i).toStringUtf8()); + } + } + return actualValues; + } }