1 /*
   2  * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
   3  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
   4  *
   5  * This code is free software; you can redistribute it and/or modify it
   6  * under the terms of the GNU General Public License version 2 only, as
   7  * published by the Free Software Foundation.  Oracle designates this
   8  * particular file as subject to the "Classpath" exception as provided
   9  * by Oracle in the LICENSE file that accompanied this code.
  10  *
  11  * This code is distributed in the hope that it will be useful, but WITHOUT
  12  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  13  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
  14  * version 2 for more details (a copy is included in the LICENSE file that
  15  * accompanied this code).
  16  *
  17  * You should have received a copy of the GNU General Public License version
  18  * 2 along with this work; if not, write to the Free Software Foundation,
  19  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
  20  *
  21  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
  22  * or visit www.oracle.com if you need additional information or have any
  23  * questions.
  24  */
  25 
  26 /*
  27  * @test
  28  * @run testng/othervm -Xint -Xverify:none -XX:+EnableMVT -XX:+ValueArrayFlatten MVTTest
  29  * @run testng/othervm -Xint -Xverify:none -XX:+EnableMVT -XX:-ValueArrayFlatten MVTTest
  30  * @run testng/othervm -Xint -Xverify:none -XX:+EnableMVT -Dvalhalla.enableValueLambdaForms=true MVTTest
  31  * @run testng/othervm -Xint -Xverify:none -XX:+EnableMVT -Dvalhalla.enableValueLambdaForms=true -Dvalhalla.enablePoolPatches=true MVTTest
  32  */
  33 
  34 import jdk.experimental.value.ValueType;
  35 import org.testng.annotations.Test;
  36 import valhalla.shady.MinimalValueTypes_1_0;
  37 
  38 import java.lang.invoke.MethodHandle;
  39 import java.lang.invoke.MethodHandles;
  40 import java.lang.reflect.Field;
  41 
  42 import static java.lang.invoke.MethodType.methodType;
  43 import static org.testng.Assert.assertEquals;
  44 
  45 @Test
  46 public class MVTTest {
  47     static final Class<?> DVT;
  48 
  49     static final ValueType<?> VT = ValueType.forClass(Point.class);
  50 
  51     static final Class<?>[] FIELD_TYPES;
  52 
  53     static final String[] FIELD_NAMES;
  54 
  55     static String TEMPLATE = "Point[x=#x, y=#y, z=#z]";
  56 
  57     static final Object[] FIELD_VALUES = {42, (short) 43, (short) 44};
  58 
  59     static final MethodHandles.Lookup LOOKUP = MethodHandles.lookup();
  60 
  61     static final MethodHandle PRINT_POINT;
  62 
  63     static {
  64         try {
  65             DVT = MinimalValueTypes_1_0.getValueTypeClass(Point.class);
  66         }
  67         catch (ClassNotFoundException e) {
  68             throw new RuntimeException(e);
  69         }
  70 
  71         Field[] fs = Point.class.getFields();
  72 
  73         FIELD_TYPES = new Class<?>[fs.length];
  74         FIELD_NAMES = new String[fs.length];
  75 
  76         for (int i = 0; i < fs.length; i++) {
  77             FIELD_TYPES[i] = fs[i].getType();
  78             FIELD_NAMES[i] = fs[i].getName();
  79         }
  80 
  81         try {
  82             PRINT_POINT = LOOKUP.findStatic(MVTTest.class, "print", methodType(String.class, Point.class))
  83                     .asType(methodType(String.class, DVT));
  84         }
  85         catch (Exception e) {
  86             throw new RuntimeException(e);
  87         }
  88     }
  89 
  90     public void testDefaultValue() throws Throwable {
  91         for (int i = 0; i < FIELD_NAMES.length; i++) {
  92             MethodHandle getter = MethodHandles.collectArguments(
  93                     VT.findGetter(LOOKUP, FIELD_NAMES[i], FIELD_TYPES[i]),
  94                     0,
  95                     VT.defaultValueConstant());
  96 
  97             assertEquals((int) getter.invoke(), 0);
  98         }
  99     }
 100 
 101     public void testWither() throws Throwable {
 102         for (int i = 0; i < FIELD_NAMES.length; i++) {
 103             MethodHandle wither = MethodHandles.collectArguments(
 104                     VT.findWither(LOOKUP, FIELD_NAMES[i], FIELD_TYPES[i]), 0, VT.defaultValueConstant());
 105             String expected = TEMPLATE.replace("#" + FIELD_NAMES[i], String.valueOf(FIELD_VALUES[i]))
 106                     .replaceAll("#[xyz]", "0");
 107 
 108             assertEquals(printReturn(wither).invoke(FIELD_VALUES[i]), expected);
 109         }
 110     }
 111 
 112     public void testSubstitutability() throws Throwable {
 113         Point[] pts = {new Point(1, (short) 6, (short) 3), new Point(1, (short) 2, (short) 3)};
 114 
 115         MethodHandle substTest = VT.substitutabilityTest();
 116         for (Point p1 : pts) {
 117             for (Point p2 : pts) {
 118                 assertEquals((boolean) substTest.invoke(p1, p2), p1.equals(p2));
 119             }
 120         }
 121 
 122         MethodHandle hash = VT.substitutabilityHashCode();
 123         for (Point p1 : pts) {
 124             for (Point p2 : pts) {
 125                 boolean vHashEq = (int) hash.invoke(p1) == (int) hash.invoke(p2);
 126                 boolean rHashEq = p1.hashCode() == p2.hashCode();
 127                 assertEquals(vHashEq, rHashEq);
 128             }
 129         }
 130     }
 131 
 132     public void testIdentity() throws Throwable {
 133         String actual = (String) printReturn(MethodHandles.identity(VT.valueClass()))
 134                 .invoke(new Point(1, (short) 2, (short) 3));
 135         assertEquals(actual, "Point[x=1, y=2, z=3]");
 136     }
 137 
 138     public void testZero() throws Throwable {
 139         String actual = (String) printReturn(MethodHandles.zero(VT.valueClass()))
 140                 .invoke();
 141         assertEquals(actual, "Point[x=0, y=0, z=0]");
 142     }
 143 
 144     public void testEmpty() throws Throwable {
 145         String actual = (String) printReturn(MethodHandles.empty(methodType(VT.valueClass(), int.class, String.class)))
 146                 .invoke(1, "");
 147         assertEquals(actual, "Point[x=0, y=0, z=0]");
 148     }
 149 
 150     public void testArray1D() throws Throwable {
 151         //test monodimensional array
 152         Object arr = MethodHandles.arrayConstructor(VT.arrayValueClass()).invoke(10);
 153         for (int i = 0; i < 10; i++) {
 154             Point p = new Point(i, (short) 9, (short) 9);
 155             MethodHandles.arrayElementSetter(VT.arrayValueClass()).invoke(arr, i, p);
 156         }
 157         for (int i = 0; i < 10; i++) {
 158             String actual = (String) printReturn(MethodHandles.arrayElementGetter(VT.arrayValueClass()))
 159                     .invoke(arr, i);
 160             String expected = TEMPLATE.replace("#x", String.valueOf(i))
 161                     .replaceAll("#[yz]", "9");
 162             assertEquals(actual, expected);
 163         }
 164     }
 165 
 166     public void testArray10D() throws Throwable {
 167         //test multidimensional array
 168         Object[] arr2 = (Object[]) MethodHandles.arrayConstructor(VT.arrayValueClass(2)).invoke(10);
 169         for (int i = 0; i < 10; i++) {
 170             Object innerArr = MethodHandles.arrayConstructor(VT.arrayValueClass()).invoke(10);
 171             MethodHandles.arrayElementSetter(VT.arrayValueClass(2)).invoke(arr2, i, innerArr);
 172             for (int j = 0; i < 10; i++) {
 173                 Point p = new Point(i, (short) j, (short) 9);
 174                 MethodHandles.arrayElementSetter(VT.arrayValueClass()).invoke(innerArr, i, p);
 175             }
 176         }
 177         for (int i = 0; i < 10; i++) {
 178             Object innerArr = MethodHandles.arrayElementGetter(VT.arrayValueClass(2)).invoke(arr2, i);
 179             for (int j = 0; i < 10; i++) {
 180                 String actual = (String) printReturn(MethodHandles.arrayElementGetter(VT.arrayValueClass()))
 181                         .invoke(innerArr, i);
 182                 String expected = TEMPLATE.replace("#x", String.valueOf(i))
 183                         .replace("#y", String.valueOf(j))
 184                         .replace("#z", "9");
 185                 assertEquals(actual, expected);
 186             }
 187         }
 188     }
 189 
 190     public void testMultiArray() throws Throwable {
 191         Object[] arr43 = (Object[]) VT.newMultiArray(2).invoke(4, 3);
 192         for (int i = 0; i < 4; i++) {
 193             Object innerArr = arr43[i];
 194             for (int j = 0; i < 3; i++) {
 195                 Point p = new Point(i, (short) j, (short) 9);
 196                 MethodHandles.arrayElementSetter(VT.arrayValueClass()).invoke(innerArr, i, p);
 197             }
 198         }
 199         for (int i = 0; i < 4; i++) {
 200             Object innerArr = MethodHandles.arrayElementGetter(VT.arrayValueClass(2)).invoke(arr43, i);
 201             for (int j = 0; i < 3; i++) {
 202                 String actual = (String) printReturn(MethodHandles.arrayElementGetter(VT.arrayValueClass()))
 203                         .invoke(innerArr, i);
 204                 String expected = TEMPLATE.replace("#x", String.valueOf(i))
 205                         .replace("#y", String.valueOf(j))
 206                         .replace("#z", "9");
 207                 assertEquals(actual, expected);
 208             }
 209         }
 210     }
 211 
 212     public void testLoop() throws Throwable {
 213         Object arr = MethodHandles.arrayConstructor(VT.arrayValueClass()).invoke(10);
 214         for (int i = 0; i < 10; i++) {
 215             Point p = new Point(i, (short) 9, (short) 9);
 216             MethodHandles.arrayElementSetter(VT.arrayValueClass()).invoke(arr, i, p);
 217         }
 218 
 219         /*
 220           iters -> (Point[] )int
 221 
 222           init  -> (Point[] )int
 223 
 224           sum   -> (int, int, int, int)int
 225           a     -> (int, Point, Point, Point)int
 226           b     -> (int, Point)int
 227           c     -> (int, Point[], int)int
 228           body  -> (int, int, Point[])int
 229          */
 230 
 231         MethodHandle iters = MethodHandles.arrayLength(VT.arrayValueClass());
 232 
 233         MethodHandle init = MethodHandles.dropArguments(MethodHandles.constant(int.class, 0),
 234                                                         0,
 235                                                         VT.arrayValueClass());
 236 
 237         MethodHandle sum = LOOKUP.findStatic(MVTTest.class,
 238                                              "sum",
 239                                              methodType(int.class, int.class, int.class, short.class, short.class));
 240 
 241         MethodHandle a = MethodHandles.filterArguments(sum, 1,
 242                                                        VT.findGetter(LOOKUP, FIELD_NAMES[0], FIELD_TYPES[0]),
 243                                                        VT.findGetter(LOOKUP, FIELD_NAMES[1], FIELD_TYPES[1]),
 244                                                        VT.findGetter(LOOKUP, FIELD_NAMES[2], FIELD_TYPES[2]));
 245 
 246         MethodHandle b = MethodHandles.permuteArguments(a,
 247                                                         methodType(int.class, int.class, VT.valueClass()),
 248                                                         0, 1, 1, 1);
 249 
 250         MethodHandle c = MethodHandles.collectArguments(b,
 251                                                         1,
 252                                                         MethodHandles.arrayElementGetter(VT.arrayValueClass()));
 253 
 254         MethodHandle body = MethodHandles.permuteArguments(c,
 255                                                            methodType(int.class, int.class, int.class, VT.arrayValueClass()),
 256                                                            0, 2, 1);
 257 
 258         MethodHandle loop = MethodHandles.countedLoop(iters, init, body);
 259         int actual = (int) loop.invoke(arr);
 260         int expected = 9 * 10 * 2 + 10 * (0 + 9) / 2;
 261         assertEquals(actual, expected);
 262     }
 263 
 264     static int sum(int v, int x, short y, short z) {
 265         return v + x + y + z;
 266     }
 267 
 268     static MethodHandle printReturn(MethodHandle mh) {
 269         return MethodHandles.filterReturnValue(mh, PRINT_POINT);
 270     }
 271 
 272     static String print(Point p) {
 273         return String.format("Point[x=%d, y=%d, z=%d]", p.x, p.y, p.z);
 274     }
 275 }