384 lines
15 KiB
Java
384 lines
15 KiB
Java
|
/*
|
||
|
* 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.
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
* @test
|
||
|
* @modules jdk.incubator.foreign/jdk.internal.foreign
|
||
|
* jdk.incubator.foreign/jdk.internal.foreign.abi
|
||
|
* jdk.incubator.foreign/jdk.internal.foreign.abi.aarch64
|
||
|
* @build CallArrangerTestBase
|
||
|
* @run testng TestAarch64CallArranger
|
||
|
*/
|
||
|
|
||
|
import jdk.incubator.foreign.FunctionDescriptor;
|
||
|
import jdk.incubator.foreign.MemoryAddress;
|
||
|
import jdk.incubator.foreign.MemoryLayout;
|
||
|
import jdk.incubator.foreign.MemorySegment;
|
||
|
import jdk.internal.foreign.abi.Binding;
|
||
|
import jdk.internal.foreign.abi.CallingSequence;
|
||
|
import jdk.internal.foreign.abi.aarch64.CallArranger;
|
||
|
import org.testng.annotations.DataProvider;
|
||
|
import org.testng.annotations.Test;
|
||
|
|
||
|
import java.lang.invoke.MethodType;
|
||
|
|
||
|
import static jdk.internal.foreign.PlatformLayouts.AArch64.*;
|
||
|
import static jdk.internal.foreign.abi.Binding.*;
|
||
|
import static jdk.internal.foreign.abi.aarch64.AArch64Architecture.*;
|
||
|
import static org.testng.Assert.assertEquals;
|
||
|
import static org.testng.Assert.assertFalse;
|
||
|
import static org.testng.Assert.assertTrue;
|
||
|
|
||
|
public class TestAarch64CallArranger extends CallArrangerTestBase {
|
||
|
|
||
|
@Test
|
||
|
public void testEmpty() {
|
||
|
MethodType mt = MethodType.methodType(void.class);
|
||
|
FunctionDescriptor fd = FunctionDescriptor.ofVoid();
|
||
|
CallArranger.Bindings bindings = CallArranger.getBindings(mt, fd, false);
|
||
|
|
||
|
assertFalse(bindings.isInMemoryReturn);
|
||
|
CallingSequence callingSequence = bindings.callingSequence;
|
||
|
assertEquals(callingSequence.methodType(), mt);
|
||
|
assertEquals(callingSequence.functionDesc(), fd);
|
||
|
|
||
|
checkArgumentBindings(callingSequence, new Binding[][]{});
|
||
|
|
||
|
checkReturnBindings(callingSequence, new Binding[]{});
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testInteger() {
|
||
|
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);
|
||
|
FunctionDescriptor fd = FunctionDescriptor.ofVoid(
|
||
|
C_INT, C_INT, C_INT, C_INT,
|
||
|
C_INT, C_INT, C_INT, C_INT,
|
||
|
C_INT, C_INT);
|
||
|
CallArranger.Bindings bindings = CallArranger.getBindings(mt, fd, false);
|
||
|
|
||
|
assertFalse(bindings.isInMemoryReturn);
|
||
|
CallingSequence callingSequence = bindings.callingSequence;
|
||
|
assertEquals(callingSequence.methodType(), mt);
|
||
|
assertEquals(callingSequence.functionDesc(), fd);
|
||
|
|
||
|
checkArgumentBindings(callingSequence, new Binding[][]{
|
||
|
{ 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(0), int.class) },
|
||
|
{ vmStore(stackStorage(1), int.class) },
|
||
|
});
|
||
|
|
||
|
checkReturnBindings(callingSequence, new Binding[]{});
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testTwoIntTwoFloat() {
|
||
|
MethodType mt = MethodType.methodType(void.class,
|
||
|
int.class, int.class, float.class, float.class);
|
||
|
FunctionDescriptor fd = FunctionDescriptor.ofVoid(
|
||
|
C_INT, C_INT, C_FLOAT, C_FLOAT);
|
||
|
CallArranger.Bindings bindings = CallArranger.getBindings(mt, fd, false);
|
||
|
|
||
|
assertFalse(bindings.isInMemoryReturn);
|
||
|
CallingSequence callingSequence = bindings.callingSequence;
|
||
|
assertEquals(callingSequence.methodType(), mt);
|
||
|
assertEquals(callingSequence.functionDesc(), fd);
|
||
|
|
||
|
checkArgumentBindings(callingSequence, new Binding[][]{
|
||
|
{ vmStore(r0, int.class) },
|
||
|
{ vmStore(r1, int.class) },
|
||
|
{ vmStore(v0, float.class) },
|
||
|
{ vmStore(v1, float.class) },
|
||
|
});
|
||
|
|
||
|
checkReturnBindings(callingSequence, new Binding[]{});
|
||
|
}
|
||
|
|
||
|
@Test(dataProvider = "structs")
|
||
|
public void testStruct(MemoryLayout struct, Binding[] expectedBindings) {
|
||
|
MethodType mt = MethodType.methodType(void.class, MemorySegment.class);
|
||
|
FunctionDescriptor fd = FunctionDescriptor.ofVoid(struct);
|
||
|
CallArranger.Bindings bindings = CallArranger.getBindings(mt, fd, false);
|
||
|
|
||
|
assertFalse(bindings.isInMemoryReturn);
|
||
|
CallingSequence callingSequence = bindings.callingSequence;
|
||
|
assertEquals(callingSequence.methodType(), mt);
|
||
|
assertEquals(callingSequence.functionDesc(), fd);
|
||
|
|
||
|
checkArgumentBindings(callingSequence, new Binding[][]{
|
||
|
expectedBindings
|
||
|
});
|
||
|
|
||
|
checkReturnBindings(callingSequence, new Binding[]{});
|
||
|
}
|
||
|
|
||
|
@DataProvider
|
||
|
public static Object[][] structs() {
|
||
|
MemoryLayout struct2 = MemoryLayout.ofStruct(C_INT, C_INT, C_DOUBLE, C_INT);
|
||
|
return new Object[][]{
|
||
|
// struct s { int32_t a, b; double c; };
|
||
|
{ MemoryLayout.ofStruct(C_INT, C_INT, C_DOUBLE), new Binding[] {
|
||
|
dup(),
|
||
|
// s.a & s.b
|
||
|
bufferLoad(0, long.class), vmStore(r0, long.class),
|
||
|
// s.c --> note AArch64 passes this in an *integer* register
|
||
|
bufferLoad(8, long.class), vmStore(r1, long.class),
|
||
|
}},
|
||
|
// struct s { int32_t a, b; double c; int32_t d };
|
||
|
{ struct2, new Binding[] {
|
||
|
copy(struct2),
|
||
|
baseAddress(),
|
||
|
unboxAddress(),
|
||
|
vmStore(r0, long.class)
|
||
|
}},
|
||
|
// struct s { int32_t a[2]; float b[2] };
|
||
|
{ MemoryLayout.ofStruct(C_INT, C_INT, C_FLOAT, C_FLOAT), new Binding[] {
|
||
|
dup(),
|
||
|
// s.a[0] & s.a[1]
|
||
|
bufferLoad(0, long.class), vmStore(r0, long.class),
|
||
|
// s.b[0] & s.b[1]
|
||
|
bufferLoad(8, long.class), vmStore(r1, long.class),
|
||
|
}},
|
||
|
// struct s { float a; /* padding */ double b };
|
||
|
{ MemoryLayout.ofStruct(C_FLOAT, MemoryLayout.ofPaddingBits(32), C_DOUBLE),
|
||
|
new Binding[] {
|
||
|
dup(),
|
||
|
// s.a
|
||
|
bufferLoad(0, long.class), vmStore(r0, long.class),
|
||
|
// s.b
|
||
|
bufferLoad(8, long.class), vmStore(r1, long.class),
|
||
|
}},
|
||
|
};
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testMultipleStructs() {
|
||
|
MemoryLayout struct1 = MemoryLayout.ofStruct(C_INT, C_INT, C_DOUBLE, C_INT);
|
||
|
MemoryLayout struct2 = MemoryLayout.ofStruct(C_LONG, C_LONG, C_LONG);
|
||
|
|
||
|
MethodType mt = MethodType.methodType(void.class, MemorySegment.class, MemorySegment.class, int.class);
|
||
|
FunctionDescriptor fd = FunctionDescriptor.ofVoid(struct1, struct2, C_INT);
|
||
|
CallArranger.Bindings bindings = CallArranger.getBindings(mt, fd, false);
|
||
|
|
||
|
assertFalse(bindings.isInMemoryReturn);
|
||
|
CallingSequence callingSequence = bindings.callingSequence;
|
||
|
assertEquals(callingSequence.methodType(), mt);
|
||
|
assertEquals(callingSequence.functionDesc(), fd);
|
||
|
|
||
|
checkArgumentBindings(callingSequence, new Binding[][]{
|
||
|
{
|
||
|
copy(struct1),
|
||
|
baseAddress(),
|
||
|
unboxAddress(),
|
||
|
vmStore(r0, long.class)
|
||
|
},
|
||
|
{
|
||
|
copy(struct2),
|
||
|
baseAddress(),
|
||
|
unboxAddress(),
|
||
|
vmStore(r1, long.class)
|
||
|
},
|
||
|
{ vmStore(r2, int.class) }
|
||
|
});
|
||
|
|
||
|
checkReturnBindings(callingSequence, new Binding[]{});
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testReturnStruct1() {
|
||
|
MemoryLayout struct = MemoryLayout.ofStruct(C_LONG, C_LONG, C_FLOAT);
|
||
|
|
||
|
MethodType mt = MethodType.methodType(MemorySegment.class);
|
||
|
FunctionDescriptor fd = FunctionDescriptor.of(struct);
|
||
|
CallArranger.Bindings bindings = CallArranger.getBindings(mt, fd, false);
|
||
|
|
||
|
assertTrue(bindings.isInMemoryReturn);
|
||
|
CallingSequence callingSequence = bindings.callingSequence;
|
||
|
assertEquals(callingSequence.methodType(), MethodType.methodType(void.class, MemoryAddress.class));
|
||
|
assertEquals(callingSequence.functionDesc(), FunctionDescriptor.ofVoid(C_POINTER));
|
||
|
|
||
|
checkArgumentBindings(callingSequence, new Binding[][]{
|
||
|
{
|
||
|
unboxAddress(),
|
||
|
vmStore(r8, long.class)
|
||
|
}
|
||
|
});
|
||
|
|
||
|
checkReturnBindings(callingSequence, new Binding[]{});
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testReturnStruct2() {
|
||
|
MemoryLayout struct = MemoryLayout.ofStruct(C_LONG, C_LONG);
|
||
|
|
||
|
MethodType mt = MethodType.methodType(MemorySegment.class);
|
||
|
FunctionDescriptor fd = FunctionDescriptor.of(struct);
|
||
|
CallArranger.Bindings bindings = CallArranger.getBindings(mt, fd, false);
|
||
|
|
||
|
assertFalse(bindings.isInMemoryReturn);
|
||
|
CallingSequence callingSequence = bindings.callingSequence;
|
||
|
assertEquals(callingSequence.methodType(), mt);
|
||
|
assertEquals(callingSequence.functionDesc(), fd);
|
||
|
|
||
|
checkArgumentBindings(callingSequence, new Binding[][]{});
|
||
|
|
||
|
checkReturnBindings(callingSequence, new Binding[]{
|
||
|
allocate(struct),
|
||
|
dup(),
|
||
|
vmLoad(r0, long.class),
|
||
|
bufferStore(0, long.class),
|
||
|
dup(),
|
||
|
vmLoad(r1, long.class),
|
||
|
bufferStore(8, long.class),
|
||
|
});
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testStructHFA1() {
|
||
|
MemoryLayout hfa = MemoryLayout.ofStruct(C_FLOAT, C_FLOAT);
|
||
|
|
||
|
MethodType mt = MethodType.methodType(MemorySegment.class, float.class, int.class, MemorySegment.class);
|
||
|
FunctionDescriptor fd = FunctionDescriptor.of(hfa, C_FLOAT, C_INT, hfa);
|
||
|
CallArranger.Bindings bindings = CallArranger.getBindings(mt, fd, false);
|
||
|
|
||
|
assertFalse(bindings.isInMemoryReturn);
|
||
|
CallingSequence callingSequence = bindings.callingSequence;
|
||
|
assertEquals(callingSequence.methodType(), mt);
|
||
|
assertEquals(callingSequence.functionDesc(), fd);
|
||
|
|
||
|
checkArgumentBindings(callingSequence, new Binding[][]{
|
||
|
{ vmStore(v0, float.class) },
|
||
|
{ vmStore(r0, int.class) },
|
||
|
{
|
||
|
dup(),
|
||
|
bufferLoad(0, int.class),
|
||
|
vmStore(v1, int.class),
|
||
|
bufferLoad(4, int.class),
|
||
|
vmStore(v2, int.class)
|
||
|
}
|
||
|
});
|
||
|
|
||
|
checkReturnBindings(callingSequence, new Binding[]{
|
||
|
allocate(hfa),
|
||
|
dup(),
|
||
|
vmLoad(v0, int.class),
|
||
|
bufferStore(0, int.class),
|
||
|
dup(),
|
||
|
vmLoad(v1, int.class),
|
||
|
bufferStore(4, int.class),
|
||
|
});
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testStructHFA3() {
|
||
|
MemoryLayout struct = MemoryLayout.ofStruct(C_FLOAT, C_FLOAT, C_FLOAT);
|
||
|
|
||
|
MethodType mt = MethodType.methodType(void.class, MemorySegment.class, MemorySegment.class, MemorySegment.class);
|
||
|
FunctionDescriptor fd = FunctionDescriptor.ofVoid(struct, struct, struct);
|
||
|
CallArranger.Bindings bindings = CallArranger.getBindings(mt, fd, false);
|
||
|
|
||
|
assertFalse(bindings.isInMemoryReturn);
|
||
|
CallingSequence callingSequence = bindings.callingSequence;
|
||
|
assertEquals(callingSequence.methodType(), mt);
|
||
|
assertEquals(callingSequence.functionDesc(), fd);
|
||
|
|
||
|
checkArgumentBindings(callingSequence, new Binding[][]{
|
||
|
{
|
||
|
dup(),
|
||
|
bufferLoad(0, int.class),
|
||
|
vmStore(v0, int.class),
|
||
|
dup(),
|
||
|
bufferLoad(4, int.class),
|
||
|
vmStore(v1, int.class),
|
||
|
bufferLoad(8, int.class),
|
||
|
vmStore(v2, int.class)
|
||
|
},
|
||
|
{
|
||
|
dup(),
|
||
|
bufferLoad(0, int.class),
|
||
|
vmStore(v3, int.class),
|
||
|
dup(),
|
||
|
bufferLoad(4, int.class),
|
||
|
vmStore(v4, int.class),
|
||
|
bufferLoad(8, int.class),
|
||
|
vmStore(v5, int.class)
|
||
|
},
|
||
|
{
|
||
|
dup(),
|
||
|
bufferLoad(0, long.class),
|
||
|
vmStore(stackStorage(0), long.class),
|
||
|
bufferLoad(8, int.class),
|
||
|
vmStore(stackStorage(1), int.class),
|
||
|
}
|
||
|
});
|
||
|
|
||
|
checkReturnBindings(callingSequence, new Binding[]{});
|
||
|
}
|
||
|
|
||
|
@Test
|
||
|
public void testStructStackSpill() {
|
||
|
// A large (> 16 byte) struct argument that is spilled to the
|
||
|
// stack should be passed as a pointer to a copy and occupy one
|
||
|
// stack slot.
|
||
|
|
||
|
MemoryLayout struct = MemoryLayout.ofStruct(C_INT, C_INT, C_DOUBLE, C_INT);
|
||
|
|
||
|
MethodType mt = MethodType.methodType(
|
||
|
void.class, MemorySegment.class, MemorySegment.class, int.class, int.class,
|
||
|
int.class, int.class, int.class, int.class, MemorySegment.class, int.class);
|
||
|
FunctionDescriptor fd = FunctionDescriptor.ofVoid(
|
||
|
struct, struct, C_INT, C_INT, C_INT, C_INT, C_INT, C_INT, struct, C_INT);
|
||
|
CallArranger.Bindings bindings = CallArranger.getBindings(mt, fd, false);
|
||
|
|
||
|
assertFalse(bindings.isInMemoryReturn);
|
||
|
CallingSequence callingSequence = bindings.callingSequence;
|
||
|
assertEquals(callingSequence.methodType(), mt);
|
||
|
assertEquals(callingSequence.functionDesc(), fd);
|
||
|
|
||
|
checkArgumentBindings(callingSequence, new Binding[][]{
|
||
|
{ copy(struct), baseAddress(), unboxAddress(), vmStore(r0, long.class) },
|
||
|
{ copy(struct), baseAddress(), unboxAddress(), vmStore(r1, long.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) },
|
||
|
{ copy(struct), baseAddress(), unboxAddress(), vmStore(stackStorage(0), long.class) },
|
||
|
{ vmStore(stackStorage(1), int.class) },
|
||
|
});
|
||
|
|
||
|
checkReturnBindings(callingSequence, new Binding[]{});
|
||
|
}
|
||
|
}
|