2021-11-24 11:51:16 +00:00
|
|
|
/*
|
2023-10-12 19:50:08 +00:00
|
|
|
* Copyright (c) 2022, 2023, Oracle and/or its affiliates. All rights reserved.
|
2021-11-24 11:51:16 +00:00
|
|
|
* 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
|
2022-12-05 13:49:53 +00:00
|
|
|
* published by the Free Software Foundation.
|
2021-11-24 11:51:16 +00:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2022-12-05 13:49:53 +00:00
|
|
|
/*
|
|
|
|
* @test
|
2023-05-02 13:56:32 +00:00
|
|
|
* @modules java.base/jdk.internal.foreign
|
2022-12-05 13:49:53 +00:00
|
|
|
* @run testng TestLinker
|
2023-09-07 11:52:14 +00:00
|
|
|
* @run testng/othervm/policy=security.policy
|
|
|
|
* -Djava.security.manager=default TestLinker
|
2022-12-05 13:49:53 +00:00
|
|
|
*/
|
|
|
|
|
2023-05-02 13:56:32 +00:00
|
|
|
import jdk.internal.foreign.CABI;
|
2022-12-05 14:47:12 +00:00
|
|
|
import org.testng.annotations.DataProvider;
|
2022-12-05 13:49:53 +00:00
|
|
|
import org.testng.annotations.Test;
|
2021-11-24 11:51:16 +00:00
|
|
|
|
2022-12-05 13:49:53 +00:00
|
|
|
import java.lang.foreign.FunctionDescriptor;
|
|
|
|
import java.lang.foreign.Linker;
|
2023-10-12 19:50:08 +00:00
|
|
|
import java.lang.foreign.MemoryLayout;
|
|
|
|
import java.lang.foreign.ValueLayout;
|
2022-12-05 13:49:53 +00:00
|
|
|
import java.lang.invoke.MethodHandle;
|
2023-05-02 13:56:32 +00:00
|
|
|
import java.util.ArrayList;
|
|
|
|
import java.util.Arrays;
|
|
|
|
import java.util.List;
|
2022-07-11 14:30:19 +00:00
|
|
|
|
2023-05-01 13:41:00 +00:00
|
|
|
import static java.lang.foreign.MemoryLayout.*;
|
|
|
|
import static java.lang.foreign.ValueLayout.JAVA_CHAR;
|
|
|
|
import static java.lang.foreign.ValueLayout.JAVA_SHORT;
|
2023-10-12 19:50:08 +00:00
|
|
|
import static org.testng.Assert.assertNotNull;
|
2023-05-01 13:41:00 +00:00
|
|
|
import static org.testng.Assert.assertSame;
|
2022-12-05 13:49:53 +00:00
|
|
|
import static org.testng.Assert.assertNotSame;
|
2023-10-12 19:50:08 +00:00
|
|
|
import static org.testng.Assert.assertTrue;
|
2022-07-11 14:30:19 +00:00
|
|
|
|
2022-12-05 13:49:53 +00:00
|
|
|
public class TestLinker extends NativeTestHelper {
|
|
|
|
|
2023-05-02 13:56:32 +00:00
|
|
|
static final boolean IS_FALLBACK_LINKER = CABI.current() == CABI.FALLBACK;
|
|
|
|
|
2023-05-01 13:41:00 +00:00
|
|
|
record LinkRequest(FunctionDescriptor descriptor, Linker.Option... options) {}
|
|
|
|
|
|
|
|
@Test(dataProvider = "notSameCases")
|
|
|
|
public void testLinkerOptionsCache(LinkRequest l1, LinkRequest l2) {
|
2022-12-05 13:49:53 +00:00
|
|
|
Linker linker = Linker.nativeLinker();
|
2023-05-01 13:41:00 +00:00
|
|
|
MethodHandle mh1 = linker.downcallHandle(l1.descriptor(), l1.options());
|
|
|
|
MethodHandle mh2 = linker.downcallHandle(l2.descriptor(), l2.options());
|
2022-12-05 13:49:53 +00:00
|
|
|
// assert that these are 2 distinct link request. No caching allowed
|
|
|
|
assertNotSame(mh1, mh2);
|
2022-07-11 14:30:19 +00:00
|
|
|
}
|
2022-12-05 13:49:53 +00:00
|
|
|
|
2023-05-01 13:41:00 +00:00
|
|
|
@DataProvider
|
|
|
|
public static Object[][] notSameCases() {
|
|
|
|
FunctionDescriptor fd_II_V = FunctionDescriptor.ofVoid(C_INT, C_INT);
|
|
|
|
return new Object[][]{
|
|
|
|
{new LinkRequest(fd_II_V), new LinkRequest(fd_II_V, Linker.Option.firstVariadicArg(1))},
|
|
|
|
{new LinkRequest(FunctionDescriptor.ofVoid(JAVA_SHORT)), new LinkRequest(FunctionDescriptor.ofVoid(JAVA_CHAR))},
|
|
|
|
{new LinkRequest(FunctionDescriptor.ofVoid(JAVA_SHORT)), new LinkRequest(FunctionDescriptor.ofVoid(JAVA_CHAR))},
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test(dataProvider = "namedDescriptors")
|
|
|
|
public void testNamedLinkerCache(FunctionDescriptor f1, FunctionDescriptor f2) {
|
|
|
|
Linker linker = Linker.nativeLinker();
|
|
|
|
MethodHandle mh1 = linker.downcallHandle(f1);
|
|
|
|
MethodHandle mh2 = linker.downcallHandle(f2);
|
|
|
|
// assert that these are the same link request, even though layout names differ
|
|
|
|
assertSame(mh1, mh2);
|
|
|
|
}
|
|
|
|
|
|
|
|
@DataProvider
|
|
|
|
public static Object[][] namedDescriptors() {
|
2023-05-02 13:56:32 +00:00
|
|
|
List<Object[]> cases = new ArrayList<>(Arrays.asList(new Object[][]{
|
2023-05-01 13:41:00 +00:00
|
|
|
{ FunctionDescriptor.ofVoid(C_INT),
|
|
|
|
FunctionDescriptor.ofVoid(C_INT.withName("x")) },
|
|
|
|
{ FunctionDescriptor.ofVoid(structLayout(C_INT)),
|
|
|
|
FunctionDescriptor.ofVoid(structLayout(C_INT).withName("x")) },
|
|
|
|
{ FunctionDescriptor.ofVoid(structLayout(C_INT)),
|
|
|
|
FunctionDescriptor.ofVoid(structLayout(C_INT.withName("x"))) },
|
|
|
|
{ FunctionDescriptor.ofVoid(structLayout(sequenceLayout(1, C_INT))),
|
|
|
|
FunctionDescriptor.ofVoid(structLayout(sequenceLayout(1, C_INT).withName("x"))) },
|
|
|
|
{ FunctionDescriptor.ofVoid(structLayout(sequenceLayout(1, C_INT))),
|
|
|
|
FunctionDescriptor.ofVoid(structLayout(sequenceLayout(1, C_INT.withName("x")))) },
|
|
|
|
{ FunctionDescriptor.ofVoid(C_POINTER),
|
|
|
|
FunctionDescriptor.ofVoid(C_POINTER.withName("x")) },
|
|
|
|
{ FunctionDescriptor.ofVoid(C_POINTER.withTargetLayout(C_INT)),
|
|
|
|
FunctionDescriptor.ofVoid(C_POINTER.withTargetLayout(C_INT.withName("x"))) },
|
|
|
|
{ FunctionDescriptor.ofVoid(C_POINTER.withTargetLayout(C_INT)),
|
|
|
|
FunctionDescriptor.ofVoid(C_POINTER.withName("x").withTargetLayout(C_INT.withName("x"))) },
|
2023-05-02 13:56:32 +00:00
|
|
|
}));
|
|
|
|
|
|
|
|
if (!IS_FALLBACK_LINKER) {
|
|
|
|
cases.add(new Object[]{ FunctionDescriptor.ofVoid(unionLayout(C_INT)),
|
|
|
|
FunctionDescriptor.ofVoid(unionLayout(C_INT).withName("x")) });
|
|
|
|
cases.add(new Object[]{ FunctionDescriptor.ofVoid(unionLayout(C_INT)),
|
|
|
|
FunctionDescriptor.ofVoid(unionLayout(C_INT.withName("x"))) });
|
|
|
|
}
|
2023-10-12 19:50:08 +00:00
|
|
|
if (C_LONG_LONG.byteAlignment() == 8) {
|
|
|
|
cases.add(new Object[]{ FunctionDescriptor.ofVoid(structLayout(C_INT, paddingLayout(4), C_LONG_LONG)),
|
|
|
|
FunctionDescriptor.ofVoid(structLayout(C_INT, paddingLayout(4), C_LONG_LONG.withName("x"))) });
|
|
|
|
cases.add(new Object[]{ FunctionDescriptor.ofVoid(structLayout(C_INT, paddingLayout(4), C_LONG_LONG)),
|
|
|
|
FunctionDescriptor.ofVoid(structLayout(C_INT, paddingLayout(4).withName("x"), C_LONG_LONG)) });
|
|
|
|
}
|
2023-05-02 13:56:32 +00:00
|
|
|
|
|
|
|
return cases.toArray(Object[][]::new);
|
2023-05-01 13:41:00 +00:00
|
|
|
}
|
|
|
|
|
2022-12-05 14:47:12 +00:00
|
|
|
@DataProvider
|
|
|
|
public static Object[][] invalidIndexCases() {
|
|
|
|
return new Object[][]{
|
|
|
|
{ -1, },
|
|
|
|
{ 42, },
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test(dataProvider = "invalidIndexCases",
|
|
|
|
expectedExceptions = IllegalArgumentException.class,
|
|
|
|
expectedExceptionsMessageRegExp = ".*not in bounds for descriptor.*")
|
|
|
|
public void testInvalidOption(int invalidIndex) {
|
|
|
|
Linker.Option option = Linker.Option.firstVariadicArg(invalidIndex);
|
|
|
|
FunctionDescriptor desc = FunctionDescriptor.ofVoid();
|
|
|
|
Linker.nativeLinker().downcallHandle(desc, option); // throws
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class,
|
|
|
|
expectedExceptionsMessageRegExp = ".*Unknown name.*")
|
|
|
|
public void testInvalidPreservedValueName() {
|
|
|
|
Linker.Option.captureCallState("foo"); // throws
|
|
|
|
}
|
|
|
|
|
2023-10-12 19:50:08 +00:00
|
|
|
@Test(dataProvider = "canonicalTypeNames")
|
|
|
|
public void testCanonicalLayouts(String typeName) {
|
|
|
|
MemoryLayout layout = LINKER.canonicalLayouts().get(typeName);
|
|
|
|
assertNotNull(layout);
|
|
|
|
assertTrue(layout instanceof ValueLayout);
|
|
|
|
}
|
|
|
|
|
|
|
|
@DataProvider
|
|
|
|
public static Object[][] canonicalTypeNames() {
|
|
|
|
return new Object[][]{
|
|
|
|
{ "bool" },
|
|
|
|
{ "char" },
|
|
|
|
{ "short" },
|
|
|
|
{ "int" },
|
|
|
|
{ "long" },
|
|
|
|
{ "long long" },
|
|
|
|
{ "float" },
|
|
|
|
{ "double" },
|
|
|
|
{ "void*" },
|
|
|
|
{ "size_t" },
|
|
|
|
{ "wchar_t" },
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test(expectedExceptions=UnsupportedOperationException.class)
|
|
|
|
public void testCanonicalLayoutsUnmodifiable() {
|
|
|
|
LINKER.canonicalLayouts().put("asdf", C_INT);
|
|
|
|
}
|
2021-11-24 11:51:16 +00:00
|
|
|
}
|