/*
* 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 com.facebook.presto.operator.aggregation;
import com.facebook.presto.bytecode.BytecodeBlock;
import com.facebook.presto.bytecode.BytecodeNode;
import com.facebook.presto.bytecode.ClassDefinition;
import com.facebook.presto.bytecode.DynamicClassLoader;
import com.facebook.presto.bytecode.FieldDefinition;
import com.facebook.presto.bytecode.MethodDefinition;
import com.facebook.presto.bytecode.Parameter;
import com.facebook.presto.bytecode.Scope;
import com.facebook.presto.bytecode.Variable;
import com.facebook.presto.bytecode.control.ForLoop;
import com.facebook.presto.bytecode.control.IfStatement;
import com.facebook.presto.bytecode.expression.BytecodeExpression;
import com.facebook.presto.bytecode.expression.BytecodeExpressions;
import com.facebook.presto.operator.GroupByIdBlock;
import com.facebook.presto.spi.Page;
import com.facebook.presto.spi.block.Block;
import com.facebook.presto.spi.block.BlockBuilder;
import com.facebook.presto.spi.function.AccumulatorStateFactory;
import com.facebook.presto.spi.function.AccumulatorStateSerializer;
import com.facebook.presto.spi.function.WindowIndex;
import com.facebook.presto.spi.type.Type;
import com.facebook.presto.sql.gen.Binding;
import com.facebook.presto.sql.gen.CallSiteBinder;
import com.facebook.presto.sql.gen.CompilerOperations;
import com.google.common.collect.ImmutableList;
import io.airlift.slice.Slice;
import java.lang.invoke.MethodHandle;
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;
import java.util.Optional;
import java.util.function.Function;
import static com.facebook.presto.bytecode.Access.FINAL;
import static com.facebook.presto.bytecode.Access.PRIVATE;
import static com.facebook.presto.bytecode.Access.PUBLIC;
import static com.facebook.presto.bytecode.Access.a;
import static com.facebook.presto.bytecode.CompilerUtils.defineClass;
import static com.facebook.presto.bytecode.CompilerUtils.makeClassName;
import static com.facebook.presto.bytecode.Parameter.arg;
import static com.facebook.presto.bytecode.ParameterizedType.type;
import static com.facebook.presto.bytecode.expression.BytecodeExpressions.constantFalse;
import static com.facebook.presto.bytecode.expression.BytecodeExpressions.constantInt;
import static com.facebook.presto.bytecode.expression.BytecodeExpressions.constantString;
import static com.facebook.presto.bytecode.expression.BytecodeExpressions.invokeDynamic;
import static com.facebook.presto.bytecode.expression.BytecodeExpressions.invokeStatic;
import static com.facebook.presto.bytecode.expression.BytecodeExpressions.not;
import static com.facebook.presto.operator.aggregation.AggregationMetadata.ParameterMetadata;
import static com.facebook.presto.operator.aggregation.AggregationMetadata.countInputChannels;
import static com.facebook.presto.sql.gen.Bootstrap.BOOTSTRAP_METHOD;
import static com.facebook.presto.sql.gen.BytecodeUtils.invoke;
import static com.facebook.presto.sql.gen.SqlTypeBytecodeExpression.constantType;
import static com.google.common.base.Preconditions.checkState;
import static java.lang.String.format;
public class AccumulatorCompiler
{
private AccumulatorCompiler()
{
}
public static GenericAccumulatorFactoryBinder generateAccumulatorFactoryBinder(AggregationMetadata metadata, DynamicClassLoader classLoader)
{
Class<? extends Accumulator> accumulatorClass = generateAccumulatorClass(
Accumulator.class,
metadata,
classLoader);
Class<? extends GroupedAccumulator> groupedAccumulatorClass = generateAccumulatorClass(
GroupedAccumulator.class,
metadata,
classLoader);
return new GenericAccumulatorFactoryBinder(
metadata.getStateSerializer(),
metadata.getStateFactory(),
accumulatorClass,
groupedAccumulatorClass);
}
private static <T> Class<? extends T> generateAccumulatorClass(
Class<T> accumulatorInterface,
AggregationMetadata metadata,
DynamicClassLoader classLoader)
{
boolean grouped = accumulatorInterface == GroupedAccumulator.class;
ClassDefinition definition = new ClassDefinition(
a(PUBLIC, FINAL),
makeClassName(metadata.getName() + accumulatorInterface.getSimpleName()),
type(Object.class),
type(accumulatorInterface));
CallSiteBinder callSiteBinder = new CallSiteBinder();
AccumulatorStateSerializer<?> stateSerializer = metadata.getStateSerializer();
AccumulatorStateFactory<?> stateFactory = metadata.getStateFactory();
FieldDefinition stateSerializerField = definition.declareField(a(PRIVATE, FINAL), "stateSerializer", AccumulatorStateSerializer.class);
FieldDefinition stateFactoryField = definition.declareField(a(PRIVATE, FINAL), "stateFactory", AccumulatorStateFactory.class);
FieldDefinition inputChannelsField = definition.declareField(a(PRIVATE, FINAL), "inputChannels", type(List.class, Integer.class));
FieldDefinition maskChannelField = definition.declareField(a(PRIVATE, FINAL), "maskChannel", type(Optional.class, Integer.class));
FieldDefinition stateField = definition.declareField(a(PRIVATE, FINAL), "state", grouped ? stateFactory.getGroupedStateClass() : stateFactory.getSingleStateClass());
// Generate constructor
generateConstructor(
definition,
stateSerializerField,
stateFactoryField,
inputChannelsField,
maskChannelField,
stateField,
grouped);
// Generate methods
generateAddInput(definition, stateField, inputChannelsField, maskChannelField, metadata.getInputMetadata(), metadata.getInputFunction(), callSiteBinder, grouped);
generateAddInputWindowIndex(definition, stateField, metadata.getInputMetadata(), metadata.getInputFunction(), callSiteBinder);
generateGetEstimatedSize(definition, stateField);
generateGetIntermediateType(definition, callSiteBinder, stateSerializer.getSerializedType());
generateGetFinalType(definition, callSiteBinder, metadata.getOutputType());
generateAddIntermediateAsCombine(definition, stateField, stateSerializerField, stateFactoryField, metadata.getCombineFunction(), stateFactory.getSingleStateClass(), callSiteBinder, grouped);
if (grouped) {
generateGroupedEvaluateIntermediate(definition, stateSerializerField, stateField);
}
else {
generateEvaluateIntermediate(definition, stateSerializerField, stateField);
}
if (grouped) {
generateGroupedEvaluateFinal(definition, stateField, metadata.getOutputFunction(), callSiteBinder);
}
else {
generateEvaluateFinal(definition, stateField, metadata.getOutputFunction(), callSiteBinder);
}
return defineClass(definition, accumulatorInterface, callSiteBinder.getBindings(), classLoader);
}
private static MethodDefinition generateGetIntermediateType(ClassDefinition definition, CallSiteBinder callSiteBinder, Type type)
{
MethodDefinition methodDefinition = definition.declareMethod(a(PUBLIC), "getIntermediateType", type(Type.class));
methodDefinition.getBody()
.append(constantType(callSiteBinder, type))
.retObject();
return methodDefinition;
}
private static MethodDefinition generateGetFinalType(ClassDefinition definition, CallSiteBinder callSiteBinder, Type type)
{
MethodDefinition methodDefinition = definition.declareMethod(a(PUBLIC), "getFinalType", type(Type.class));
methodDefinition.getBody()
.append(constantType(callSiteBinder, type))
.retObject();
return methodDefinition;
}
private static void generateGetEstimatedSize(ClassDefinition definition, FieldDefinition stateField)
{
MethodDefinition method = definition.declareMethod(a(PUBLIC), "getEstimatedSize", type(long.class));
BytecodeExpression state = method.getThis().getField(stateField);
method.getBody()
.append(state.invoke("getEstimatedSize", long.class).ret());
}
private static void generateAddInput(
ClassDefinition definition,
FieldDefinition stateField,
FieldDefinition inputChannelsField,
FieldDefinition maskChannelField,
List<ParameterMetadata> parameterMetadatas,
MethodHandle inputFunction,
CallSiteBinder callSiteBinder,
boolean grouped)
{
ImmutableList.Builder<Parameter> parameters = ImmutableList.builder();
if (grouped) {
parameters.add(arg("groupIdsBlock", GroupByIdBlock.class));
}
Parameter page = arg("page", Page.class);
parameters.add(page);
MethodDefinition method = definition.declareMethod(a(PUBLIC), "addInput", type(void.class), parameters.build());
Scope scope = method.getScope();
BytecodeBlock body = method.getBody();
Variable thisVariable = method.getThis();
if (grouped) {
generateEnsureCapacity(scope, stateField, body);
}
List<Variable> parameterVariables = new ArrayList<>();
for (int i = 0; i < countInputChannels(parameterMetadatas); i++) {
parameterVariables.add(scope.declareVariable(Block.class, "block" + i));
}
Variable masksBlock = scope.declareVariable(Block.class, "masksBlock");
body.comment("masksBlock = maskChannel.map(page.blockGetter()).orElse(null);")
.append(thisVariable.getField(maskChannelField))
.append(page)
.invokeStatic(type(AggregationUtils.class), "pageBlockGetter", type(Function.class, Integer.class, Block.class), type(Page.class))
.invokeVirtual(Optional.class, "map", Optional.class, Function.class)
.pushNull()
.invokeVirtual(Optional.class, "orElse", Object.class, Object.class)
.checkCast(Block.class)
.putVariable(masksBlock);
// Get all parameter blocks
for (int i = 0; i < countInputChannels(parameterMetadatas); i++) {
body.comment("%s = page.getBlock(inputChannels.get(%d));", parameterVariables.get(i).getName(), i)
.append(page)
.append(thisVariable.getField(inputChannelsField))
.push(i)
.invokeInterface(List.class, "get", Object.class, int.class)
.checkCast(Integer.class)
.invokeVirtual(Integer.class, "intValue", int.class)
.invokeVirtual(Page.class, "getBlock", Block.class, int.class)
.putVariable(parameterVariables.get(i));
}
BytecodeBlock block = generateInputForLoop(stateField, parameterMetadatas, inputFunction, scope, parameterVariables, masksBlock, callSiteBinder, grouped);
body.append(block);
body.ret();
}
private static void generateAddInputWindowIndex(
ClassDefinition definition,
FieldDefinition stateField,
List<ParameterMetadata> parameterMetadatas,
MethodHandle inputFunction,
CallSiteBinder callSiteBinder)
{
// TODO: implement masking based on maskChannel field once Window Functions support DISTINCT arguments to the functions.
Parameter index = arg("index", WindowIndex.class);
Parameter channels = arg("channels", type(List.class, Integer.class));
Parameter startPosition = arg("startPosition", int.class);
Parameter endPosition = arg("endPosition", int.class);
MethodDefinition method = definition.declareMethod(a(PUBLIC), "addInput", type(void.class), ImmutableList.of(index, channels, startPosition, endPosition));
Scope scope = method.getScope();
Variable position = scope.declareVariable(int.class, "position");
Binding binding = callSiteBinder.bind(inputFunction);
BytecodeExpression invokeInputFunction = invokeDynamic(
BOOTSTRAP_METHOD,
ImmutableList.of(binding.getBindingId()),
"input",
binding.getType(),
getInvokeFunctionOnWindowIndexParameters(
scope,
inputFunction.type().parameterArray(),
parameterMetadatas,
stateField,
index,
channels,
position));
method.getBody()
.append(new ForLoop()
.initialize(position.set(startPosition))
.condition(BytecodeExpressions.lessThanOrEqual(position, endPosition))
.update(position.increment())
.body(new IfStatement()
.condition(anyParametersAreNull(parameterMetadatas, index, channels, position))
.ifFalse(invokeInputFunction)))
.ret();
}
private static BytecodeExpression anyParametersAreNull(
List<ParameterMetadata> parameterMetadatas,
Variable index,
Variable channels,
Variable position)
{
int inputChannel = 0;
BytecodeExpression isNull = constantFalse();
for (ParameterMetadata parameterMetadata : parameterMetadatas) {
switch (parameterMetadata.getParameterType()) {
case BLOCK_INPUT_CHANNEL:
case INPUT_CHANNEL:
BytecodeExpression getChannel = channels.invoke("get", Object.class, constantInt(inputChannel)).cast(int.class);
isNull = BytecodeExpressions.or(isNull, index.invoke("isNull", boolean.class, getChannel, position));
inputChannel++;
break;
case NULLABLE_BLOCK_INPUT_CHANNEL:
inputChannel++;
break;
}
}
return isNull;
}
private static List<BytecodeExpression> getInvokeFunctionOnWindowIndexParameters(
Scope scope,
Class<?>[] parameterTypes,
List<ParameterMetadata> parameterMetadatas,
FieldDefinition stateField,
Variable index,
Variable channels,
Variable position)
{
int inputChannel = 0;
List<BytecodeExpression> expressions = new ArrayList<>();
for (int i = 0; i < parameterTypes.length; i++) {
ParameterMetadata parameterMetadata = parameterMetadatas.get(i);
Class<?> parameterType = parameterTypes[i];
BytecodeExpression getChannel = channels.invoke("get", Object.class, constantInt(inputChannel)).cast(int.class);
switch (parameterMetadata.getParameterType()) {
case STATE:
expressions.add(scope.getThis().getField(stateField));
break;
case BLOCK_INDEX:
expressions.add(constantInt(0)); // index.getSingleValueBlock(channel, position) generates always a page with only one position
break;
case BLOCK_INPUT_CHANNEL:
case NULLABLE_BLOCK_INPUT_CHANNEL:
expressions.add(index.invoke(
"getSingleValueBlock",
Block.class,
getChannel,
position));
inputChannel++;
break;
case INPUT_CHANNEL:
if (parameterType == long.class) {
expressions.add(index.invoke("getLong", long.class, getChannel, position));
}
else if (parameterType == double.class) {
expressions.add(index.invoke("getDouble", double.class, getChannel, position));
}
else if (parameterType == boolean.class) {
expressions.add(index.invoke("getBoolean", boolean.class, getChannel, position));
}
else if (parameterType == Slice.class) {
expressions.add(index.invoke("getSlice", Slice.class, getChannel, position));
}
else if (parameterType == Block.class) {
expressions.add(index.invoke("getObject", Block.class, getChannel, position));
}
else {
throw new IllegalArgumentException(format("Unsupported parameter type: %s", parameterType));
}
inputChannel++;
break;
}
}
return expressions;
}
private static BytecodeBlock generateInputForLoop(
FieldDefinition stateField,
List<ParameterMetadata> parameterMetadatas,
MethodHandle inputFunction,
Scope scope,
List<Variable> parameterVariables,
Variable masksBlock,
CallSiteBinder callSiteBinder,
boolean grouped)
{
// For-loop over rows
Variable page = scope.getVariable("page");
Variable positionVariable = scope.declareVariable(int.class, "position");
Variable rowsVariable = scope.declareVariable(int.class, "rows");
BytecodeBlock block = new BytecodeBlock()
.append(page)
.invokeVirtual(Page.class, "getPositionCount", int.class)
.putVariable(rowsVariable)
.initializeVariable(positionVariable);
BytecodeNode loopBody = generateInvokeInputFunction(scope, stateField, positionVariable, parameterVariables, parameterMetadatas, inputFunction, callSiteBinder, grouped);
// Wrap with null checks
List<Boolean> nullable = new ArrayList<>();
for (ParameterMetadata metadata : parameterMetadatas) {
switch (metadata.getParameterType()) {
case INPUT_CHANNEL:
case BLOCK_INPUT_CHANNEL:
nullable.add(false);
break;
case NULLABLE_BLOCK_INPUT_CHANNEL:
nullable.add(true);
break;
default: // do nothing
}
}
checkState(nullable.size() == parameterVariables.size(), "Number of parameters does not match");
for (int i = 0; i < parameterVariables.size(); i++) {
if (!nullable.get(i)) {
Variable variableDefinition = parameterVariables.get(i);
loopBody = new IfStatement("if(!%s.isNull(position))", variableDefinition.getName())
.condition(new BytecodeBlock()
.getVariable(variableDefinition)
.getVariable(positionVariable)
.invokeInterface(Block.class, "isNull", boolean.class, int.class))
.ifFalse(loopBody);
}
}
loopBody = new IfStatement("if(testMask(%s, position))", masksBlock.getName())
.condition(new BytecodeBlock()
.getVariable(masksBlock)
.getVariable(positionVariable)
.invokeStatic(CompilerOperations.class, "testMask", boolean.class, Block.class, int.class))
.ifTrue(loopBody);
block.append(new ForLoop()
.initialize(new BytecodeBlock().putVariable(positionVariable, 0))
.condition(new BytecodeBlock()
.getVariable(positionVariable)
.getVariable(rowsVariable)
.invokeStatic(CompilerOperations.class, "lessThan", boolean.class, int.class, int.class))
.update(new BytecodeBlock().incrementVariable(positionVariable, (byte) 1))
.body(loopBody));
return block;
}
private static BytecodeBlock generateInvokeInputFunction(
Scope scope,
FieldDefinition stateField,
Variable position,
List<Variable> parameterVariables,
List<ParameterMetadata> parameterMetadatas,
MethodHandle inputFunction,
CallSiteBinder callSiteBinder,
boolean grouped)
{
BytecodeBlock block = new BytecodeBlock();
if (grouped) {
generateSetGroupIdFromGroupIdsBlock(scope, stateField, block);
}
block.comment("Call input function with unpacked Block arguments");
Class<?>[] parameters = inputFunction.type().parameterArray();
int inputChannel = 0;
for (int i = 0; i < parameters.length; i++) {
ParameterMetadata parameterMetadata = parameterMetadatas.get(i);
switch (parameterMetadata.getParameterType()) {
case STATE:
block.append(scope.getThis().getField(stateField));
break;
case BLOCK_INDEX:
block.getVariable(position);
break;
case BLOCK_INPUT_CHANNEL:
case NULLABLE_BLOCK_INPUT_CHANNEL:
block.getVariable(parameterVariables.get(inputChannel));
inputChannel++;
break;
case INPUT_CHANNEL:
BytecodeBlock getBlockBytecode = new BytecodeBlock()
.getVariable(parameterVariables.get(inputChannel));
pushStackType(scope, block, parameterMetadata.getSqlType(), getBlockBytecode, parameters[i], callSiteBinder);
inputChannel++;
break;
default:
throw new IllegalArgumentException("Unsupported parameter type: " + parameterMetadata.getParameterType());
}
}
block.append(invoke(callSiteBinder.bind(inputFunction), "input"));
return block;
}
// Assumes that there is a variable named 'position' in the block, which is the current index
private static void pushStackType(Scope scope, BytecodeBlock block, Type sqlType, BytecodeBlock getBlockBytecode, Class<?> parameter, CallSiteBinder callSiteBinder)
{
Variable position = scope.getVariable("position");
if (parameter == long.class) {
block.comment("%s.getLong(block, position)", sqlType.getTypeSignature())
.append(constantType(callSiteBinder, sqlType))
.append(getBlockBytecode)
.append(position)
.invokeInterface(Type.class, "getLong", long.class, Block.class, int.class);
}
else if (parameter == double.class) {
block.comment("%s.getDouble(block, position)", sqlType.getTypeSignature())
.append(constantType(callSiteBinder, sqlType))
.append(getBlockBytecode)
.append(position)
.invokeInterface(Type.class, "getDouble", double.class, Block.class, int.class);
}
else if (parameter == boolean.class) {
block.comment("%s.getBoolean(block, position)", sqlType.getTypeSignature())
.append(constantType(callSiteBinder, sqlType))
.append(getBlockBytecode)
.append(position)
.invokeInterface(Type.class, "getBoolean", boolean.class, Block.class, int.class);
}
else if (parameter == Slice.class) {
block.comment("%s.getSlice(block, position)", sqlType.getTypeSignature())
.append(constantType(callSiteBinder, sqlType))
.append(getBlockBytecode)
.append(position)
.invokeInterface(Type.class, "getSlice", Slice.class, Block.class, int.class);
}
else {
block.comment("%s.getObject(block, position)", sqlType.getTypeSignature())
.append(constantType(callSiteBinder, sqlType))
.append(getBlockBytecode)
.append(position)
.invokeInterface(Type.class, "getObject", Object.class, Block.class, int.class);
}
}
private static void generateAddIntermediateAsCombine(
ClassDefinition definition,
FieldDefinition stateField,
FieldDefinition stateSerializerField,
FieldDefinition stateFactoryField,
MethodHandle combineFunction,
Class<?> singleStateClass,
CallSiteBinder callSiteBinder,
boolean grouped)
{
MethodDefinition method = declareAddIntermediate(definition, grouped);
Scope scope = method.getScope();
BytecodeBlock body = method.getBody();
Variable thisVariable = method.getThis();
Variable block = scope.getVariable("block");
Variable scratchState = scope.declareVariable(singleStateClass, "scratchState");
Variable position = scope.declareVariable(int.class, "position");
body.comment("scratchState = stateFactory.createSingleState();")
.append(thisVariable.getField(stateFactoryField))
.invokeInterface(AccumulatorStateFactory.class, "createSingleState", Object.class)
.checkCast(scratchState.getType())
.putVariable(scratchState);
if (grouped) {
generateEnsureCapacity(scope, stateField, body);
}
BytecodeBlock loopBody = new BytecodeBlock();
if (grouped) {
Variable groupIdsBlock = scope.getVariable("groupIdsBlock");
loopBody.append(thisVariable.getField(stateField).invoke("setGroupId", void.class, groupIdsBlock.invoke("getGroupId", long.class, position)));
}
loopBody.append(thisVariable.getField(stateSerializerField).invoke("deserialize", void.class, block, position, scratchState.cast(Object.class)));
loopBody.comment("combine(state, scratchState)")
.append(thisVariable.getField(stateField))
.append(scratchState)
.append(invoke(callSiteBinder.bind(combineFunction), "combine"));
if (grouped) {
// skip rows with null group id
IfStatement ifStatement = new IfStatement("if (!groupIdsBlock.isNull(position))")
.condition(not(scope.getVariable("groupIdsBlock").invoke("isNull", boolean.class, position)))
.ifTrue(loopBody);
loopBody = new BytecodeBlock().append(ifStatement);
}
body.append(generateBlockNonNullPositionForLoop(scope, position, loopBody))
.ret();
}
private static void generateSetGroupIdFromGroupIdsBlock(Scope scope, FieldDefinition stateField, BytecodeBlock block)
{
Variable groupIdsBlock = scope.getVariable("groupIdsBlock");
Variable position = scope.getVariable("position");
BytecodeExpression state = scope.getThis().getField(stateField);
block.append(state.invoke("setGroupId", void.class, groupIdsBlock.invoke("getGroupId", long.class, position)));
}
private static void generateEnsureCapacity(Scope scope, FieldDefinition stateField, BytecodeBlock block)
{
Variable groupIdsBlock = scope.getVariable("groupIdsBlock");
BytecodeExpression state = scope.getThis().getField(stateField);
block.append(state.invoke("ensureCapacity", void.class, groupIdsBlock.invoke("getGroupCount", long.class)));
}
private static MethodDefinition declareAddIntermediate(ClassDefinition definition, boolean grouped)
{
ImmutableList.Builder<Parameter> parameters = ImmutableList.builder();
if (grouped) {
parameters.add(arg("groupIdsBlock", GroupByIdBlock.class));
}
parameters.add(arg("block", Block.class));
return definition.declareMethod(
a(PUBLIC),
"addIntermediate",
type(void.class),
parameters.build());
}
// Generates a for-loop with a local variable named "position" defined, with the current position in the block,
// loopBody will only be executed for non-null positions in the Block
private static BytecodeBlock generateBlockNonNullPositionForLoop(Scope scope, Variable positionVariable, BytecodeBlock loopBody)
{
Variable rowsVariable = scope.declareVariable(int.class, "rows");
Variable blockVariable = scope.getVariable("block");
BytecodeBlock block = new BytecodeBlock()
.append(blockVariable)
.invokeInterface(Block.class, "getPositionCount", int.class)
.putVariable(rowsVariable);
IfStatement ifStatement = new IfStatement("if(!block.isNull(position))")
.condition(new BytecodeBlock()
.append(blockVariable)
.append(positionVariable)
.invokeInterface(Block.class, "isNull", boolean.class, int.class))
.ifFalse(loopBody);
block.append(new ForLoop()
.initialize(positionVariable.set(constantInt(0)))
.condition(new BytecodeBlock()
.append(positionVariable)
.append(rowsVariable)
.invokeStatic(CompilerOperations.class, "lessThan", boolean.class, int.class, int.class))
.update(new BytecodeBlock().incrementVariable(positionVariable, (byte) 1))
.body(ifStatement));
return block;
}
private static void generateGroupedEvaluateIntermediate(ClassDefinition definition, FieldDefinition stateSerializerField, FieldDefinition stateField)
{
Parameter groupId = arg("groupId", int.class);
Parameter out = arg("out", BlockBuilder.class);
MethodDefinition method = definition.declareMethod(a(PUBLIC), "evaluateIntermediate", type(void.class), groupId, out);
Variable thisVariable = method.getThis();
BytecodeExpression state = thisVariable.getField(stateField);
BytecodeExpression stateSerializer = thisVariable.getField(stateSerializerField);
method.getBody()
.append(state.invoke("setGroupId", void.class, groupId.cast(long.class)))
.append(stateSerializer.invoke("serialize", void.class, state.cast(Object.class), out))
.ret();
}
private static void generateEvaluateIntermediate(ClassDefinition definition, FieldDefinition stateSerializerField, FieldDefinition stateField)
{
Parameter out = arg("out", BlockBuilder.class);
MethodDefinition method = definition.declareMethod(
a(PUBLIC),
"evaluateIntermediate",
type(void.class),
out);
Variable thisVariable = method.getThis();
BytecodeExpression stateSerializer = thisVariable.getField(stateSerializerField);
BytecodeExpression state = thisVariable.getField(stateField);
method.getBody()
.append(stateSerializer.invoke("serialize", void.class, state.cast(Object.class), out))
.ret();
}
private static void generateGroupedEvaluateFinal(
ClassDefinition definition,
FieldDefinition stateField,
MethodHandle outputFunction,
CallSiteBinder callSiteBinder)
{
Parameter groupId = arg("groupId", int.class);
Parameter out = arg("out", BlockBuilder.class);
MethodDefinition method = definition.declareMethod(a(PUBLIC), "evaluateFinal", type(void.class), groupId, out);
BytecodeBlock body = method.getBody();
Variable thisVariable = method.getThis();
BytecodeExpression state = thisVariable.getField(stateField);
body.append(state.invoke("setGroupId", void.class, groupId.cast(long.class)));
body.comment("output(state, out)");
body.append(state);
body.append(out);
body.append(invoke(callSiteBinder.bind(outputFunction), "output"));
body.ret();
}
private static void generateEvaluateFinal(
ClassDefinition definition,
FieldDefinition stateField,
MethodHandle outputFunction,
CallSiteBinder callSiteBinder)
{
Parameter out = arg("out", BlockBuilder.class);
MethodDefinition method = definition.declareMethod(
a(PUBLIC),
"evaluateFinal",
type(void.class),
out);
BytecodeBlock body = method.getBody();
Variable thisVariable = method.getThis();
BytecodeExpression state = thisVariable.getField(stateField);
body.comment("output(state, out)");
body.append(state);
body.append(out);
body.append(invoke(callSiteBinder.bind(outputFunction), "output"));
body.ret();
}
private static void generateConstructor(
ClassDefinition definition,
FieldDefinition stateSerializerField,
FieldDefinition stateFactoryField,
FieldDefinition inputChannelsField,
FieldDefinition maskChannelField,
FieldDefinition stateField,
boolean grouped)
{
Parameter stateSerializer = arg("stateSerializer", AccumulatorStateSerializer.class);
Parameter stateFactory = arg("stateFactory", AccumulatorStateFactory.class);
Parameter inputChannels = arg("inputChannels", type(List.class, Integer.class));
Parameter maskChannel = arg("maskChannel", type(Optional.class, Integer.class));
MethodDefinition method = definition.declareConstructor(
a(PUBLIC),
stateSerializer,
stateFactory,
inputChannels,
maskChannel);
BytecodeBlock body = method.getBody();
Variable thisVariable = method.getThis();
body.comment("super();")
.append(thisVariable)
.invokeConstructor(Object.class);
body.append(thisVariable.setField(stateSerializerField, generateRequireNotNull(stateSerializer)));
body.append(thisVariable.setField(stateFactoryField, generateRequireNotNull(stateFactory)));
body.append(thisVariable.setField(inputChannelsField, generateRequireNotNull(inputChannels)));
body.append(thisVariable.setField(maskChannelField, generateRequireNotNull(maskChannel)));
String createState;
if (grouped) {
createState = "createGroupedState";
}
else {
createState = "createSingleState";
}
body.append(thisVariable.setField(stateField, stateFactory.invoke(createState, Object.class).cast(stateField.getType())));
body.ret();
}
private static BytecodeExpression generateRequireNotNull(Variable variable)
{
return invokeStatic(Objects.class, "requireNonNull", Object.class, variable.cast(Object.class), constantString(variable.getName() + " is null"))
.cast(variable.getType());
}
}