package net.bytebuddy.implementation.bytecode.assign.primitive; import net.bytebuddy.description.type.TypeDescription; import net.bytebuddy.implementation.Implementation; import net.bytebuddy.implementation.bytecode.StackManipulation; import net.bytebuddy.implementation.bytecode.StackSize; import net.bytebuddy.implementation.bytecode.assign.Assigner; import net.bytebuddy.test.utility.MockitoRule; import org.junit.After; import org.junit.Before; import org.junit.Rule; import org.junit.Test; import org.junit.rules.TestRule; import org.junit.runner.RunWith; import org.junit.runners.Parameterized; import org.mockito.Mock; import org.objectweb.asm.MethodVisitor; import org.objectweb.asm.Opcodes; import java.util.Arrays; import java.util.Collection; import static org.hamcrest.CoreMatchers.is; import static org.hamcrest.MatcherAssert.assertThat; import static org.mockito.Mockito.*; @RunWith(Parameterized.class) public class VoidAwareAssignerNonVoidToVoidTest { private final Class<?> sourceType; private final int opcode; @Rule public TestRule mockitoRule = new MockitoRule(this); @Mock private TypeDescription.Generic source, target; @Mock private TypeDescription rawSource; @Mock private Assigner chainedAssigner; @Mock private MethodVisitor methodVisitor; @Mock private Implementation.Context implementationContext; public VoidAwareAssignerNonVoidToVoidTest(Class<?> sourceType, int opcode) { this.sourceType = sourceType; this.opcode = opcode; } @Parameterized.Parameters public static Collection<Object[]> data() { return Arrays.asList(new Object[][]{ {byte.class, Opcodes.POP}, {short.class, Opcodes.POP}, {char.class, Opcodes.POP}, {int.class, Opcodes.POP}, {long.class, Opcodes.POP2}, {float.class, Opcodes.POP}, {double.class, Opcodes.POP2}, {Object.class, Opcodes.POP} }); } @Before public void setUp() throws Exception { when(source.represents(sourceType)).thenReturn(true); when(source.getStackSize()).thenReturn(StackSize.of(sourceType)); when(source.asErasure()).thenReturn(rawSource); when(rawSource.represents(sourceType)).thenReturn(true); when(rawSource.asErasure()).thenReturn(rawSource); if (sourceType.isPrimitive()) { when(source.isPrimitive()).thenReturn(true); } when(target.represents(void.class)).thenReturn(true); when(target.isPrimitive()).thenReturn(true); } @After public void tearDown() throws Exception { verifyZeroInteractions(chainedAssigner); verifyZeroInteractions(implementationContext); } @Test public void testAssignDefaultValue() throws Exception { testAssignDefaultValue(true); } @Test public void testAssignNoDefaultValue() throws Exception { testAssignDefaultValue(false); } private void testAssignDefaultValue(boolean dynamicallyTyped) throws Exception { Assigner voidAwareAssigner = new VoidAwareAssigner(chainedAssigner); StackManipulation stackManipulation = voidAwareAssigner.assign(source, target, Assigner.Typing.of(dynamicallyTyped)); assertThat(stackManipulation.isValid(), is(true)); StackManipulation.Size size = stackManipulation.apply(methodVisitor, implementationContext); assertThat(size.getSizeImpact(), is(-1 * StackSize.of(sourceType).getSize())); assertThat(size.getMaximalSize(), is(0)); verify(methodVisitor).visitInsn(opcode); verifyNoMoreInteractions(methodVisitor); } }