
io.prestosql.operator.aggregation.AbstractGroupCollectionAggregationState Maven / Gradle / Ivy
/*
* 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.prestosql.operator.aggregation;
import io.prestosql.array.IntBigArray;
import io.prestosql.array.ShortBigArray;
import io.prestosql.operator.aggregation.state.AbstractGroupedAccumulatorState;
import io.prestosql.spi.PageBuilder;
import io.prestosql.spi.block.Block;
import it.unimi.dsi.fastutil.longs.LongArrayList;
import it.unimi.dsi.fastutil.longs.LongList;
import org.openjdk.jol.info.ClassLayout;
import java.util.ArrayList;
import java.util.List;
import static com.google.common.base.Verify.verify;
/**
* Instances of this state use a single PageBuilder for all groups.
*/
public abstract class AbstractGroupCollectionAggregationState
extends AbstractGroupedAccumulatorState
{
private static final int INSTANCE_SIZE = ClassLayout.parseClass(AbstractGroupCollectionAggregationState.class).instanceSize();
private static final int MAX_NUM_BLOCKS = 30000;
private static final short NULL = -1;
private final ShortBigArray headBlockIndex;
private final IntBigArray headPosition;
private final ShortBigArray nextBlockIndex;
private final IntBigArray nextPosition;
private final ShortBigArray tailBlockIndex;
private final IntBigArray tailPosition;
private final List values;
private final LongList sumPositions;
private final IntBigArray groupEntryCount;
private PageBuilder currentPageBuilder;
private long valueBlocksRetainedSizeInBytes;
private long totalPositions;
private long capacity;
protected AbstractGroupCollectionAggregationState(PageBuilder pageBuilder)
{
this.headBlockIndex = new ShortBigArray(NULL);
this.headPosition = new IntBigArray(NULL);
this.nextBlockIndex = new ShortBigArray(NULL);
this.nextPosition = new IntBigArray(NULL);
this.tailBlockIndex = new ShortBigArray(NULL);
this.tailPosition = new IntBigArray(NULL);
this.currentPageBuilder = pageBuilder;
this.values = new ArrayList<>();
this.sumPositions = new LongArrayList();
this.groupEntryCount = new IntBigArray();
values.add(currentPageBuilder);
sumPositions.add(0L);
valueBlocksRetainedSizeInBytes = 0;
totalPositions = 0;
capacity = 1024;
nextBlockIndex.ensureCapacity(capacity);
nextPosition.ensureCapacity(capacity);
groupEntryCount.ensureCapacity(capacity);
}
@Override
public void ensureCapacity(long size)
{
headBlockIndex.ensureCapacity(size);
headPosition.ensureCapacity(size);
tailBlockIndex.ensureCapacity(size);
tailPosition.ensureCapacity(size);
groupEntryCount.ensureCapacity(size);
}
@Override
public long getEstimatedSize()
{
return INSTANCE_SIZE +
headBlockIndex.sizeOf() +
headPosition.sizeOf() +
tailBlockIndex.sizeOf() +
tailPosition.sizeOf() +
nextBlockIndex.sizeOf() +
nextPosition.sizeOf() +
groupEntryCount.sizeOf() +
valueBlocksRetainedSizeInBytes +
// valueBlocksRetainedSizeInBytes doesn't contain the current block builder
currentPageBuilder.getRetainedSizeInBytes();
}
/**
* This method should be called before {@link #appendAtChannel(int, Block, int)} to update the internal linked list, where
* {@link #appendAtChannel(int, Block, int)} is called for each channel that has a new entry to be added.
*/
protected final void prepareAdd()
{
if (currentPageBuilder.isFull()) {
valueBlocksRetainedSizeInBytes += currentPageBuilder.getRetainedSizeInBytes();
sumPositions.add(totalPositions);
currentPageBuilder = currentPageBuilder.newPageBuilderLike();
values.add(currentPageBuilder);
verify(values.size() <= MAX_NUM_BLOCKS);
}
long currentGroupId = getGroupId();
short insertedBlockIndex = (short) (values.size() - 1);
int insertedPosition = currentPageBuilder.getPositionCount();
if (totalPositions == capacity) {
capacity *= 1.5;
nextBlockIndex.ensureCapacity(capacity);
nextPosition.ensureCapacity(capacity);
}
if (isEmpty()) {
// new linked list, set up the header pointer
headBlockIndex.set(currentGroupId, insertedBlockIndex);
headPosition.set(currentGroupId, insertedPosition);
}
else {
// existing linked list, link the new entry to the tail
long absoluteTailAddress = toAbsolutePosition(tailBlockIndex.get(currentGroupId), tailPosition.get(currentGroupId));
nextBlockIndex.set(absoluteTailAddress, insertedBlockIndex);
nextPosition.set(absoluteTailAddress, insertedPosition);
}
tailBlockIndex.set(currentGroupId, insertedBlockIndex);
tailPosition.set(currentGroupId, insertedPosition);
groupEntryCount.increment(currentGroupId);
currentPageBuilder.declarePosition();
totalPositions++;
}
protected final void appendAtChannel(int channel, Block block, int position)
{
currentPageBuilder.getType(channel).appendTo(block, position, currentPageBuilder.getBlockBuilder(channel));
}
public void forEach(T consumer)
{
short currentBlockId = headBlockIndex.get(getGroupId());
int currentPosition = headPosition.get(getGroupId());
while (currentBlockId != NULL) {
accept(consumer, values.get(currentBlockId), currentPosition);
long absoluteCurrentAddress = toAbsolutePosition(currentBlockId, currentPosition);
currentBlockId = nextBlockIndex.get(absoluteCurrentAddress);
currentPosition = nextPosition.get(absoluteCurrentAddress);
}
}
public boolean isEmpty()
{
return headBlockIndex.get(getGroupId()) == NULL;
}
public final int getEntryCount()
{
return groupEntryCount.get(getGroupId());
}
private long toAbsolutePosition(short blockId, int position)
{
return sumPositions.get(blockId) + position;
}
protected abstract void accept(T consumer, PageBuilder pageBuilder, int currentPosition);
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy