/*
 * Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
 *  DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 *  This code is free software; you can redistribute it and/or modify it
 *  under the terms of the GNU General Public License version 2 only, as
 *  published by the Free Software Foundation.
 *
 *  This code is distributed in the hope that it will be useful, but WITHOUT
 *  ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 *  FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 *  version 2 for more details (a copy is included in the LICENSE file that
 *  accompanied this code).
 *
 *  You should have received a copy of the GNU General Public License version
 *  2 along with this work; if not, write to the Free Software Foundation,
 *  Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 *   Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 *  or visit www.oracle.com if you need additional information or have any
 *  questions.
 *
 */

import jdk.incubator.foreign.MemoryAddress;
import jdk.incubator.foreign.MemoryHandles;
import jdk.incubator.foreign.MemoryLayout;
import jdk.incubator.foreign.MemoryLayout.PathElement;
import jdk.incubator.foreign.MemoryLayouts;
import jdk.incubator.foreign.MemorySegment;
import java.lang.invoke.VarHandle;
import java.nio.ByteOrder;
import java.util.Arrays;
import java.util.stream.IntStream;
import java.util.stream.LongStream;
import org.testng.annotations.*;
import static java.nio.ByteOrder.BIG_ENDIAN;
import static org.testng.Assert.*;

/*
 * @test
 * @run testng TestMemoryHandleAsUnsigned
 */

public class TestMemoryHandleAsUnsigned {

    @DataProvider(name = "unsignedIntToByteData")
    public Object[][] unsignedIntToByteData() {
        return IntStream.range(0, 256)
                .mapToObj(v -> new Object[] { v }).toArray(Object[][]::new);
    }

    @Test(dataProvider = "unsignedIntToByteData")
    public void testUnsignedIntToByte(int intValue) {
        byte byteValue = (byte) (intValue & 0xFF);

        MemoryLayout layout = MemoryLayouts.BITS_8_BE;
        VarHandle byteHandle = layout.varHandle(byte.class);
        VarHandle intHandle = MemoryHandles.asUnsigned(byteHandle, int.class);

        try (MemorySegment segment = MemorySegment.allocateNative(layout)) {
            intHandle.set(segment.baseAddress(), intValue);
            int expectedIntValue = Byte.toUnsignedInt(byteValue);
            assertEquals((int) intHandle.get(segment.baseAddress()), expectedIntValue);
            assertEquals((byte) byteHandle.get(segment.baseAddress()), byteValue);
        }
    }

    @DataProvider(name = "unsignedLongToByteData")
    public Object[][] unsignedLongToByteData() {
        return LongStream.range(0L, 256L)
                .mapToObj(v -> new Object[] { v }).toArray(Object[][]::new);
    }

    @Test(dataProvider = "unsignedLongToByteData")
    public void testUnsignedLongToByte(long longValue) {
        byte byteValue = (byte) (longValue & 0xFFL);

        MemoryLayout layout = MemoryLayouts.BITS_8_BE;
        VarHandle byteHandle = layout.varHandle(byte.class);
        VarHandle longHandle = MemoryHandles.asUnsigned(byteHandle, long.class);

        try (MemorySegment segment = MemorySegment.allocateNative(layout)) {
            longHandle.set(segment.baseAddress(), longValue);
            long expectedLongValue = Byte.toUnsignedLong(byteValue);
            assertEquals((long) longHandle.get(segment.baseAddress()), expectedLongValue);
            assertEquals((byte) byteHandle.get(segment.baseAddress()), byteValue);
        }
    }

    @DataProvider(name = "unsignedIntToShortData")
    public Object[][] unsignedIntToShortData() {
        return IntStream.range(0, 65_536).filter(i -> i % 99 == 0)
                .mapToObj(v -> new Object[] { v }).toArray(Object[][]::new);
    }

    @Test(dataProvider = "unsignedIntToShortData")
    public void testUnsignedIntToShort(int intValue) {
        short shortValue = (short) (intValue & 0xFFFF);

        MemoryLayout layout = MemoryLayouts.BITS_16_BE;
        VarHandle shortHandle = layout.varHandle(short.class);
        VarHandle intHandle = MemoryHandles.asUnsigned(shortHandle, int.class);

        try (MemorySegment segment = MemorySegment.allocateNative(layout)) {
            intHandle.set(segment.baseAddress(), intValue);
            int expectedIntValue = Short.toUnsignedInt(shortValue);
            assertEquals((int) intHandle.get(segment.baseAddress()), expectedIntValue);
            assertEquals((short) shortHandle.get(segment.baseAddress()), shortValue);
        }
    }

    @DataProvider(name = "unsignedLongToShortData")
    public Object[][] unsignedLongToShortData() {
        return LongStream.range(0L, 65_536L).filter(i -> i % 99 == 0)
                .mapToObj(v -> new Object[] { v }).toArray(Object[][]::new);
    }

    @Test(dataProvider = "unsignedLongToShortData")
    public void testUnsignedLongToShort(long longValue) {
        short shortValue = (short) (longValue & 0xFFFFL);

        MemoryLayout layout = MemoryLayouts.BITS_16_BE;
        VarHandle shortHandle = layout.varHandle(short.class);
        VarHandle longHandle = MemoryHandles.asUnsigned(shortHandle, long.class);

        try (MemorySegment segment = MemorySegment.allocateNative(layout)) {
            longHandle.set(segment.baseAddress(), longValue);
            long expectedLongValue = Short.toUnsignedLong(shortValue);
            assertEquals((long) longHandle.get(segment.baseAddress()), expectedLongValue);
            assertEquals((short) shortHandle.get(segment.baseAddress()), shortValue);
        }
    }

    @DataProvider(name = "unsignedLongToIntData")
    public Object[][] unsignedLongToIntData() {
        // some boundary values
        long[] l = new long[] { Long.MAX_VALUE, Long.MIN_VALUE,
                Short.MAX_VALUE - 1L, Short.MAX_VALUE, Short.MAX_VALUE + 1L,
                Short.MIN_VALUE - 1L, Short.MIN_VALUE, Short.MIN_VALUE + 1L, };
        return LongStream.concat(LongStream.range(-256L, 256L), Arrays.stream(l))
                .mapToObj(v -> new Object[] { v }).toArray(Object[][]::new);
    }

    @Test(dataProvider = "unsignedLongToIntData")
    public void testUnsignedLongToInt(long longValue) {
        int intValue = (int) (longValue & 0xFFFF_FFFFL);

        MemoryLayout layout = MemoryLayouts.BITS_32_BE;
        VarHandle intHandle = layout.varHandle(int.class);
        VarHandle longHandle = MemoryHandles.asUnsigned(intHandle, long.class);

        try (MemorySegment segment = MemorySegment.allocateNative(layout)) {
            longHandle.set(segment.baseAddress(), longValue);
            long expectedLongValue = Integer.toUnsignedLong(intValue);
            assertEquals((long) longHandle.get(segment.baseAddress()), expectedLongValue);
            assertEquals((int) intHandle.get(segment.baseAddress()), intValue);
        }
    }

    @Test
    public void testCoordinatesSequenceLayout() {
        MemoryLayout layout = MemoryLayout.ofSequence(2, MemoryLayouts.BITS_8_BE);
        VarHandle byteHandle = layout.varHandle(byte.class, PathElement.sequenceElement());
        VarHandle intHandle = MemoryHandles.asUnsigned(byteHandle, int.class);

        try (MemorySegment segment = MemorySegment.allocateNative(layout)) {
            intHandle.set(segment.baseAddress(), 0L, (int) -1);
            assertEquals((int) intHandle.get(segment.baseAddress(), 0L), 255);
            intHandle.set(segment.baseAddress(), 1L, (int) 200);
            assertEquals((int) intHandle.get(segment.baseAddress(), 1L), 200);
        }
    }

    @Test
    public void testCoordinatesStride() {
        byte[] arr = { 0, 0, (byte) 129, 0 };
        MemorySegment segment = MemorySegment.ofArray(arr);
        MemoryAddress addr = segment.baseAddress();

        {
            VarHandle byteHandle = MemoryHandles.varHandle(byte.class, ByteOrder.nativeOrder());
            VarHandle intHandle = MemoryHandles.asUnsigned(byteHandle, int.class);
            VarHandle strideHandle = MemoryHandles.withStride(intHandle, 1);
            assertEquals((int) strideHandle.get(addr, 2L), 129);
        }
        {
            VarHandle byteHandle = MemoryHandles.varHandle(byte.class, ByteOrder.nativeOrder());
            VarHandle strideHandle = MemoryHandles.withStride(byteHandle, 1);
            VarHandle intHandle = MemoryHandles.asUnsigned(strideHandle, int.class);
            assertEquals((int) intHandle.get(addr, 2L), 129);
        }
    }

    static final Class<NullPointerException> NPE = NullPointerException.class;

    @Test
    public void testNull() {
        VarHandle handle = MemoryHandles.varHandle(byte.class, BIG_ENDIAN);
        assertThrows(NPE, () -> MemoryHandles.asUnsigned(handle, null));
        assertThrows(NPE, () -> MemoryHandles.asUnsigned(null, short.class));
        assertThrows(NPE, () -> MemoryHandles.asUnsigned(null, null));
    }

    static final Class<IllegalArgumentException> IAE = IllegalArgumentException.class;

    static void assertIllegalArgumentExceptionIllegalCarrier(Class<?> carrier, Class<?> adaptedType) {
        var vh = MemoryHandles.varHandle(carrier, BIG_ENDIAN);
        var exception = expectThrows(IAE, () -> MemoryHandles.asUnsigned(vh, adaptedType));
        var msg = exception.getMessage();
        assertTrue(msg.contains("illegal carrier"), "Expected \"illegal carrier\" in:[" + msg +"]");
    }

    static void assertIllegalArgumentExceptionIllegalAdapter(Class<?> carrier, Class<?> adaptedType) {
        var vh = MemoryHandles.varHandle(carrier, BIG_ENDIAN);
        var exception = expectThrows(IAE, () -> MemoryHandles.asUnsigned(vh, adaptedType));
        var msg = exception.getMessage();
        assertTrue(msg.contains("illegal adapter type"), "Expected \"illegal adapter type\" in:[" + msg +"]");
    }

    static void assertIllegalArgumentExceptionIsNotWiderThan(Class<?> carrier, Class<?> adaptedType) {
        var vh = MemoryHandles.varHandle(carrier, BIG_ENDIAN);
        var exception = expectThrows(IAE, () -> MemoryHandles.asUnsigned(vh, adaptedType));
        var msg = exception.getMessage();
        assertTrue(msg.contains("is not wider than"), "Expected \"is not wider than\" in:[" + msg +"]");
    }

    @Test
    public void testIllegalArgumentException() {
        assertIllegalArgumentExceptionIllegalCarrier(char.class,   long.class);
        assertIllegalArgumentExceptionIllegalCarrier(double.class, long.class);
        assertIllegalArgumentExceptionIllegalCarrier(float.class,  long.class);
        assertIllegalArgumentExceptionIllegalCarrier(long.class,   long.class);

        assertIllegalArgumentExceptionIllegalAdapter(byte.class, void.class);
        assertIllegalArgumentExceptionIllegalAdapter(byte.class, byte.class);
        assertIllegalArgumentExceptionIllegalAdapter(byte.class, short.class);
        assertIllegalArgumentExceptionIllegalAdapter(byte.class, char.class);
        assertIllegalArgumentExceptionIllegalAdapter(byte.class, double.class);
        assertIllegalArgumentExceptionIllegalAdapter(byte.class, float.class);
        assertIllegalArgumentExceptionIllegalAdapter(byte.class, Object.class);
        assertIllegalArgumentExceptionIllegalAdapter(byte.class, Integer.class);
        assertIllegalArgumentExceptionIllegalAdapter(byte.class, Long.class);
        assertIllegalArgumentExceptionIllegalAdapter(byte.class, long[].class);
        assertIllegalArgumentExceptionIllegalAdapter(byte.class, int[].class);
        assertIllegalArgumentExceptionIllegalAdapter(byte.class, Integer[].class);
        assertIllegalArgumentExceptionIllegalAdapter(byte.class, Long[].class);

        assertIllegalArgumentExceptionIsNotWiderThan(int.class, int.class);
    }
}