/* * Copyright (c) 2020, 2023, 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. * */ /* * @test * @enablePreview * @requires sun.arch.data.model == "64" * @compile platform/PlatformLayouts.java * @modules java.base/jdk.internal.foreign * java.base/jdk.internal.foreign.abi * java.base/jdk.internal.foreign.abi.aarch64 * @build CallArrangerTestBase * @run testng TestMacOsAArch64CallArranger */ import java.lang.foreign.FunctionDescriptor; import java.lang.foreign.MemoryLayout; import java.lang.foreign.StructLayout; import java.lang.foreign.MemorySegment; import jdk.internal.foreign.abi.Binding; import jdk.internal.foreign.abi.CallingSequence; import jdk.internal.foreign.abi.LinkerOptions; import jdk.internal.foreign.abi.StubLocations; import jdk.internal.foreign.abi.VMStorage; import jdk.internal.foreign.abi.aarch64.CallArranger; import org.testng.annotations.DataProvider; import org.testng.annotations.Test; import java.lang.invoke.MethodType; import static java.lang.foreign.Linker.Option.firstVariadicArg; import static java.lang.foreign.ValueLayout.ADDRESS; import static jdk.internal.foreign.abi.Binding.*; import static jdk.internal.foreign.abi.aarch64.AArch64Architecture.*; import static jdk.internal.foreign.abi.aarch64.AArch64Architecture.Regs.*; import static platform.PlatformLayouts.AArch64.*; import static org.testng.Assert.assertEquals; import static org.testng.Assert.assertFalse; import static org.testng.Assert.assertTrue; public class TestMacOsAArch64CallArranger extends CallArrangerTestBase { private static final VMStorage TARGET_ADDRESS_STORAGE = StubLocations.TARGET_ADDRESS.storage(StorageType.PLACEHOLDER); @Test public void testVarArgsOnStack() { MethodType mt = MethodType.methodType(void.class, int.class, int.class, float.class); FunctionDescriptor fd = FunctionDescriptor.ofVoid(C_INT, C_INT, C_FLOAT); FunctionDescriptor fdExpected = FunctionDescriptor.ofVoid(ADDRESS, C_INT, C_INT, C_FLOAT); CallArranger.Bindings bindings = CallArranger.MACOS.getBindings(mt, fd, false, LinkerOptions.forDowncall(fd, firstVariadicArg(1))); assertFalse(bindings.isInMemoryReturn()); CallingSequence callingSequence = bindings.callingSequence(); assertEquals(callingSequence.callerMethodType(), mt.insertParameterTypes(0, MemorySegment.class)); assertEquals(callingSequence.functionDesc(), fdExpected); // The two variadic arguments should be allocated on the stack checkArgumentBindings(callingSequence, new Binding[][]{ { unboxAddress(), vmStore(TARGET_ADDRESS_STORAGE, long.class) }, { vmStore(r0, int.class) }, { vmStore(stackStorage((short) 4, 0), int.class) }, { vmStore(stackStorage((short) 4, 8), float.class) }, }); checkReturnBindings(callingSequence, new Binding[]{}); } @Test public void testMacArgsOnStack() { MethodType mt = MethodType.methodType(void.class, int.class, int.class, int.class, int.class, int.class, int.class, int.class, int.class, int.class, int.class, short.class, byte.class); FunctionDescriptor fd = FunctionDescriptor.ofVoid( C_INT, C_INT, C_INT, C_INT, C_INT, C_INT, C_INT, C_INT, C_INT, C_INT, C_SHORT, C_CHAR); CallArranger.Bindings bindings = CallArranger.MACOS.getBindings(mt, fd, false); assertFalse(bindings.isInMemoryReturn()); CallingSequence callingSequence = bindings.callingSequence(); assertEquals(callingSequence.callerMethodType(), mt.insertParameterTypes(0, MemorySegment.class)); assertEquals(callingSequence.functionDesc(), fd.insertArgumentLayouts(0, ADDRESS)); checkArgumentBindings(callingSequence, new Binding[][]{ { unboxAddress(), vmStore(TARGET_ADDRESS_STORAGE, long.class) }, { vmStore(r0, int.class) }, { vmStore(r1, int.class) }, { vmStore(r2, int.class) }, { vmStore(r3, int.class) }, { vmStore(r4, int.class) }, { vmStore(r5, int.class) }, { vmStore(r6, int.class) }, { vmStore(r7, int.class) }, { vmStore(stackStorage((short) 4, 0), int.class) }, { vmStore(stackStorage((short) 4, 4), int.class) }, { cast(short.class, int.class), vmStore(stackStorage((short) 2, 8), int.class) }, { cast(byte.class, int.class), vmStore(stackStorage((short) 1, 10), int.class) }, }); checkReturnBindings(callingSequence, new Binding[]{}); } @Test public void testMacArgsOnStack2() { StructLayout struct = MemoryLayout.structLayout( C_FLOAT, C_FLOAT ); MethodType mt = MethodType.methodType(void.class, long.class, long.class, long.class, long.class, long.class, long.class, long.class, long.class, double.class, double.class, double.class, double.class, double.class, double.class, double.class, double.class, int.class, MemorySegment.class); FunctionDescriptor fd = FunctionDescriptor.ofVoid( C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_INT, struct); CallArranger.Bindings bindings = CallArranger.MACOS.getBindings(mt, fd, false); assertFalse(bindings.isInMemoryReturn()); CallingSequence callingSequence = bindings.callingSequence(); assertEquals(callingSequence.callerMethodType(), mt.insertParameterTypes(0, MemorySegment.class)); assertEquals(callingSequence.functionDesc(), fd.insertArgumentLayouts(0, ADDRESS)); checkArgumentBindings(callingSequence, new Binding[][]{ { unboxAddress(), vmStore(TARGET_ADDRESS_STORAGE, long.class) }, { vmStore(r0, long.class) }, { vmStore(r1, long.class) }, { vmStore(r2, long.class) }, { vmStore(r3, long.class) }, { vmStore(r4, long.class) }, { vmStore(r5, long.class) }, { vmStore(r6, long.class) }, { vmStore(r7, long.class) }, { vmStore(v0, double.class) }, { vmStore(v1, double.class) }, { vmStore(v2, double.class) }, { vmStore(v3, double.class) }, { vmStore(v4, double.class) }, { vmStore(v5, double.class) }, { vmStore(v6, double.class) }, { vmStore(v7, double.class) }, { vmStore(stackStorage((short) 4, 0), int.class) }, { dup(), bufferLoad(0, int.class), vmStore(stackStorage((short) 4, 4), int.class), bufferLoad(4, int.class), vmStore(stackStorage((short) 4, 8), int.class), } }); checkReturnBindings(callingSequence, new Binding[]{}); } @Test public void testMacArgsOnStack3() { StructLayout struct = MemoryLayout.structLayout( C_POINTER, C_POINTER ); MethodType mt = MethodType.methodType(void.class, long.class, long.class, long.class, long.class, long.class, long.class, long.class, long.class, double.class, double.class, double.class, double.class, double.class, double.class, double.class, double.class, MemorySegment.class, float.class); FunctionDescriptor fd = FunctionDescriptor.ofVoid( C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, struct, C_FLOAT); CallArranger.Bindings bindings = CallArranger.MACOS.getBindings(mt, fd, false); assertFalse(bindings.isInMemoryReturn()); CallingSequence callingSequence = bindings.callingSequence(); assertEquals(callingSequence.callerMethodType(), mt.insertParameterTypes(0, MemorySegment.class)); assertEquals(callingSequence.functionDesc(), fd.insertArgumentLayouts(0, ADDRESS)); checkArgumentBindings(callingSequence, new Binding[][]{ { unboxAddress(), vmStore(TARGET_ADDRESS_STORAGE, long.class) }, { vmStore(r0, long.class) }, { vmStore(r1, long.class) }, { vmStore(r2, long.class) }, { vmStore(r3, long.class) }, { vmStore(r4, long.class) }, { vmStore(r5, long.class) }, { vmStore(r6, long.class) }, { vmStore(r7, long.class) }, { vmStore(v0, double.class) }, { vmStore(v1, double.class) }, { vmStore(v2, double.class) }, { vmStore(v3, double.class) }, { vmStore(v4, double.class) }, { vmStore(v5, double.class) }, { vmStore(v6, double.class) }, { vmStore(v7, double.class) }, { dup(), bufferLoad(0, long.class), vmStore(stackStorage((short) 8, 0), long.class), bufferLoad(8, long.class), vmStore(stackStorage((short) 8, 8), long.class) }, { vmStore(stackStorage((short) 4, 16), float.class) }, }); checkReturnBindings(callingSequence, new Binding[]{}); } @Test public void testMacArgsOnStack4() { StructLayout struct = MemoryLayout.structLayout( C_INT, C_INT, C_POINTER ); MethodType mt = MethodType.methodType(void.class, long.class, long.class, long.class, long.class, long.class, long.class, long.class, long.class, double.class, double.class, double.class, double.class, double.class, double.class, double.class, double.class, float.class, MemorySegment.class); FunctionDescriptor fd = FunctionDescriptor.ofVoid( C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_FLOAT, struct); CallArranger.Bindings bindings = CallArranger.MACOS.getBindings(mt, fd, false); assertFalse(bindings.isInMemoryReturn()); CallingSequence callingSequence = bindings.callingSequence(); assertEquals(callingSequence.callerMethodType(), mt.insertParameterTypes(0, MemorySegment.class)); assertEquals(callingSequence.functionDesc(), fd.insertArgumentLayouts(0, ADDRESS)); checkArgumentBindings(callingSequence, new Binding[][]{ { unboxAddress(), vmStore(TARGET_ADDRESS_STORAGE, long.class) }, { vmStore(r0, long.class) }, { vmStore(r1, long.class) }, { vmStore(r2, long.class) }, { vmStore(r3, long.class) }, { vmStore(r4, long.class) }, { vmStore(r5, long.class) }, { vmStore(r6, long.class) }, { vmStore(r7, long.class) }, { vmStore(v0, double.class) }, { vmStore(v1, double.class) }, { vmStore(v2, double.class) }, { vmStore(v3, double.class) }, { vmStore(v4, double.class) }, { vmStore(v5, double.class) }, { vmStore(v6, double.class) }, { vmStore(v7, double.class) }, { vmStore(stackStorage((short) 4, 0), float.class) }, { dup(), bufferLoad(0, long.class), vmStore(stackStorage((short) 8, 8), long.class), bufferLoad(8, long.class), vmStore(stackStorage((short) 8, 16), long.class) }, }); checkReturnBindings(callingSequence, new Binding[]{}); } // structs that are passed field-wise should not have padding after them @Test public void testMacArgsOnStack5() { StructLayout struct = MemoryLayout.structLayout( C_FLOAT ); MethodType mt = MethodType.methodType(void.class, long.class, long.class, long.class, long.class, long.class, long.class, long.class, long.class, double.class, double.class, double.class, double.class, double.class, double.class, double.class, double.class, MemorySegment.class, int.class, MemorySegment.class); FunctionDescriptor fd = FunctionDescriptor.ofVoid( C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, struct, C_INT, C_POINTER); CallArranger.Bindings bindings = CallArranger.MACOS.getBindings(mt, fd, false); assertFalse(bindings.isInMemoryReturn()); CallingSequence callingSequence = bindings.callingSequence(); assertEquals(callingSequence.callerMethodType(), mt.insertParameterTypes(0, MemorySegment.class)); assertEquals(callingSequence.functionDesc(), fd.insertArgumentLayouts(0, ADDRESS)); checkArgumentBindings(callingSequence, new Binding[][]{ { unboxAddress(), vmStore(TARGET_ADDRESS_STORAGE, long.class) }, { vmStore(r0, long.class) }, { vmStore(r1, long.class) }, { vmStore(r2, long.class) }, { vmStore(r3, long.class) }, { vmStore(r4, long.class) }, { vmStore(r5, long.class) }, { vmStore(r6, long.class) }, { vmStore(r7, long.class) }, { vmStore(v0, double.class) }, { vmStore(v1, double.class) }, { vmStore(v2, double.class) }, { vmStore(v3, double.class) }, { vmStore(v4, double.class) }, { vmStore(v5, double.class) }, { vmStore(v6, double.class) }, { vmStore(v7, double.class) }, { bufferLoad(0, int.class), vmStore(stackStorage((short) 4, 0), int.class), }, { vmStore(stackStorage((short) 4, 4), int.class) }, { unboxAddress(), vmStore(stackStorage((short) 8, 8), long.class) }, }); checkReturnBindings(callingSequence, new Binding[]{}); } // structs that are passed chunk-wise should have padding before them, as well as after @Test public void testMacArgsOnStack6() { StructLayout struct = MemoryLayout.structLayout( C_INT ); MethodType mt = MethodType.methodType(void.class, long.class, long.class, long.class, long.class, long.class, long.class, long.class, long.class, double.class, double.class, double.class, double.class, double.class, double.class, double.class, double.class, int.class, MemorySegment.class, double.class, MemorySegment.class); FunctionDescriptor fd = FunctionDescriptor.ofVoid( C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE, C_INT, struct, C_DOUBLE, C_POINTER); CallArranger.Bindings bindings = CallArranger.MACOS.getBindings(mt, fd, false); assertFalse(bindings.isInMemoryReturn()); CallingSequence callingSequence = bindings.callingSequence(); assertEquals(callingSequence.callerMethodType(), mt.insertParameterTypes(0, MemorySegment.class)); assertEquals(callingSequence.functionDesc(), fd.insertArgumentLayouts(0, ADDRESS)); checkArgumentBindings(callingSequence, new Binding[][]{ { unboxAddress(), vmStore(TARGET_ADDRESS_STORAGE, long.class) }, { vmStore(r0, long.class) }, { vmStore(r1, long.class) }, { vmStore(r2, long.class) }, { vmStore(r3, long.class) }, { vmStore(r4, long.class) }, { vmStore(r5, long.class) }, { vmStore(r6, long.class) }, { vmStore(r7, long.class) }, { vmStore(v0, double.class) }, { vmStore(v1, double.class) }, { vmStore(v2, double.class) }, { vmStore(v3, double.class) }, { vmStore(v4, double.class) }, { vmStore(v5, double.class) }, { vmStore(v6, double.class) }, { vmStore(v7, double.class) }, { vmStore(stackStorage((short) 4, 0), int.class) }, { bufferLoad(0, int.class), vmStore(stackStorage((short) 4, 8), int.class), }, { vmStore(stackStorage((short) 8, 16), double.class) }, { unboxAddress(), vmStore(stackStorage((short) 8, 24), long.class) }, }); checkReturnBindings(callingSequence, new Binding[]{}); } }