2021-06-02 10:53:06 +00:00
|
|
|
/*
|
2023-08-07 10:58:11 +00:00
|
|
|
* Copyright (c) 2021, 2023, Oracle and/or its affiliates. All rights reserved.
|
|
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
2021-06-02 10:53:06 +00:00
|
|
|
*
|
2023-08-07 10:58:11 +00:00
|
|
|
* 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.
|
2021-06-02 10:53:06 +00:00
|
|
|
*
|
2023-08-07 10:58:11 +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).
|
2021-06-02 10:53:06 +00:00
|
|
|
*
|
2023-08-07 10:58:11 +00:00
|
|
|
* 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.
|
2021-06-02 10:53:06 +00:00
|
|
|
*
|
2023-08-07 10:58:11 +00:00
|
|
|
* 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.
|
2021-06-02 10:53:06 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @test
|
|
|
|
* @run testng/othervm --enable-native-access=ALL-UNNAMED TestScopedOperations
|
|
|
|
*/
|
|
|
|
|
2022-12-05 13:49:53 +00:00
|
|
|
import java.lang.foreign.Arena;
|
2024-01-08 08:20:07 +00:00
|
|
|
import java.lang.foreign.MemoryLayout;
|
2022-05-12 16:17:45 +00:00
|
|
|
import java.lang.foreign.MemorySegment;
|
|
|
|
import java.lang.foreign.ValueLayout;
|
2022-12-05 13:49:53 +00:00
|
|
|
|
2021-06-02 10:53:06 +00:00
|
|
|
import org.testng.annotations.DataProvider;
|
|
|
|
import org.testng.annotations.Test;
|
|
|
|
|
|
|
|
import java.io.File;
|
|
|
|
import java.io.IOException;
|
|
|
|
import java.nio.channels.FileChannel;
|
|
|
|
import java.nio.file.Path;
|
2022-05-12 16:17:45 +00:00
|
|
|
import java.nio.file.StandardOpenOption;
|
2021-06-02 10:53:06 +00:00
|
|
|
import java.util.ArrayList;
|
|
|
|
import java.util.List;
|
|
|
|
import java.util.concurrent.atomic.AtomicReference;
|
|
|
|
import java.util.function.Consumer;
|
|
|
|
import java.util.function.Function;
|
|
|
|
|
2022-05-12 16:17:45 +00:00
|
|
|
import static java.lang.foreign.ValueLayout.JAVA_BYTE;
|
2021-06-02 10:53:06 +00:00
|
|
|
import static org.testng.Assert.assertEquals;
|
|
|
|
import static org.testng.Assert.assertNotNull;
|
|
|
|
import static org.testng.Assert.assertTrue;
|
|
|
|
import static org.testng.Assert.fail;
|
|
|
|
|
|
|
|
public class TestScopedOperations {
|
|
|
|
|
|
|
|
static Path tempPath;
|
|
|
|
|
|
|
|
static {
|
|
|
|
try {
|
|
|
|
File file = File.createTempFile("scopedBuffer", "txt");
|
|
|
|
file.deleteOnExit();
|
|
|
|
tempPath = file.toPath();
|
|
|
|
} catch (IOException ex) {
|
|
|
|
throw new ExceptionInInitializerError(ex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test(dataProvider = "scopedOperations")
|
2021-11-24 11:51:16 +00:00
|
|
|
public <Z> void testOpAfterClose(String name, ScopedOperation<Z> scopedOperation) {
|
2023-04-27 09:00:58 +00:00
|
|
|
Arena arena = Arena.ofConfined();
|
|
|
|
Z obj = scopedOperation.apply(arena);
|
2022-12-05 13:49:53 +00:00
|
|
|
arena.close();
|
2021-06-02 10:53:06 +00:00
|
|
|
try {
|
2021-11-24 11:51:16 +00:00
|
|
|
scopedOperation.accept(obj);
|
2021-06-02 10:53:06 +00:00
|
|
|
fail();
|
|
|
|
} catch (IllegalStateException ex) {
|
|
|
|
assertTrue(ex.getMessage().contains("closed"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test(dataProvider = "scopedOperations")
|
2021-11-24 11:51:16 +00:00
|
|
|
public <Z> void testOpOutsideConfinement(String name, ScopedOperation<Z> scopedOperation) {
|
2023-04-27 09:00:58 +00:00
|
|
|
try (Arena arena = Arena.ofConfined()) {
|
|
|
|
Z obj = scopedOperation.apply(arena);
|
2021-06-02 10:53:06 +00:00
|
|
|
AtomicReference<Throwable> failed = new AtomicReference<>();
|
|
|
|
Thread t = new Thread(() -> {
|
|
|
|
try {
|
2021-11-24 11:51:16 +00:00
|
|
|
scopedOperation.accept(obj);
|
2021-06-02 10:53:06 +00:00
|
|
|
} catch (Throwable ex) {
|
|
|
|
failed.set(ex);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
t.start();
|
|
|
|
t.join();
|
|
|
|
assertNotNull(failed.get());
|
2022-05-25 09:50:55 +00:00
|
|
|
assertEquals(failed.get().getClass(), WrongThreadException.class);
|
2021-06-02 10:53:06 +00:00
|
|
|
assertTrue(failed.get().getMessage().contains("outside"));
|
|
|
|
} catch (InterruptedException ex) {
|
|
|
|
throw new AssertionError(ex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static List<ScopedOperation> scopedOperations = new ArrayList<>();
|
|
|
|
|
|
|
|
static {
|
2022-05-12 16:17:45 +00:00
|
|
|
// session operations
|
2023-04-27 09:00:58 +00:00
|
|
|
ScopedOperation.ofScope(session -> session.allocate(100, 1), "MemorySession::allocate");
|
2022-05-12 16:17:45 +00:00
|
|
|
ScopedOperation.ofScope(session -> {
|
|
|
|
try (FileChannel fileChannel = FileChannel.open(tempPath, StandardOpenOption.READ, StandardOpenOption.WRITE)) {
|
|
|
|
fileChannel.map(FileChannel.MapMode.READ_WRITE, 0L, 10L, session);
|
2021-06-02 10:53:06 +00:00
|
|
|
} catch (IOException ex) {
|
|
|
|
fail();
|
|
|
|
}
|
2022-05-12 16:17:45 +00:00
|
|
|
}, "FileChannel::map");
|
2021-06-02 10:53:06 +00:00
|
|
|
// segment operations
|
2021-11-24 11:51:16 +00:00
|
|
|
ScopedOperation.ofSegment(s -> s.toArray(JAVA_BYTE), "MemorySegment::toArray(BYTE)");
|
2021-06-02 10:53:06 +00:00
|
|
|
ScopedOperation.ofSegment(s -> s.copyFrom(s), "MemorySegment::copyFrom");
|
|
|
|
ScopedOperation.ofSegment(s -> s.mismatch(s), "MemorySegment::mismatch");
|
|
|
|
ScopedOperation.ofSegment(s -> s.fill((byte) 0), "MemorySegment::fill");
|
|
|
|
// allocator operations
|
2023-04-27 09:00:58 +00:00
|
|
|
ScopedOperation.ofScope(a -> a.allocate(1), "Arena::allocate/size");
|
|
|
|
ScopedOperation.ofScope(a -> a.allocate(1, 1), "Arena::allocate/size/align");
|
|
|
|
ScopedOperation.ofScope(a -> a.allocate(JAVA_BYTE), "Arena::allocate/layout");
|
2023-10-12 19:50:08 +00:00
|
|
|
ScopedOperation.ofScope(a -> a.allocateFrom(JAVA_BYTE, (byte) 0), "Arena::allocate/byte");
|
|
|
|
ScopedOperation.ofScope(a -> a.allocateFrom(ValueLayout.JAVA_CHAR, (char) 0), "Arena::allocate/char");
|
|
|
|
ScopedOperation.ofScope(a -> a.allocateFrom(ValueLayout.JAVA_SHORT, (short) 0), "Arena::allocate/short");
|
|
|
|
ScopedOperation.ofScope(a -> a.allocateFrom(ValueLayout.JAVA_INT, 0), "Arena::allocate/int");
|
|
|
|
ScopedOperation.ofScope(a -> a.allocateFrom(ValueLayout.JAVA_FLOAT, 0f), "Arena::allocate/float");
|
|
|
|
ScopedOperation.ofScope(a -> a.allocateFrom(ValueLayout.JAVA_LONG, 0L), "Arena::allocate/long");
|
|
|
|
ScopedOperation.ofScope(a -> a.allocateFrom(ValueLayout.JAVA_DOUBLE, 0d), "Arena::allocate/double");
|
|
|
|
ScopedOperation.ofScope(a -> a.allocate(JAVA_BYTE, 1L), "Arena::allocate/size");
|
|
|
|
ScopedOperation.ofScope(a -> a.allocateFrom(JAVA_BYTE, new byte[]{0}), "Arena::allocateFrom/byte");
|
|
|
|
ScopedOperation.ofScope(a -> a.allocateFrom(ValueLayout.JAVA_CHAR, new char[]{0}), "Arena::allocateFrom/char");
|
|
|
|
ScopedOperation.ofScope(a -> a.allocateFrom(ValueLayout.JAVA_SHORT, new short[]{0}), "Arena::allocateFrom/short");
|
|
|
|
ScopedOperation.ofScope(a -> a.allocateFrom(ValueLayout.JAVA_INT, new int[]{0}), "Arena::allocateFrom/int");
|
|
|
|
ScopedOperation.ofScope(a -> a.allocateFrom(ValueLayout.JAVA_FLOAT, new float[]{0}), "Arena::allocateFrom/float");
|
|
|
|
ScopedOperation.ofScope(a -> a.allocateFrom(ValueLayout.JAVA_LONG, new long[]{0}), "Arena::allocateFrom/long");
|
|
|
|
ScopedOperation.ofScope(a -> a.allocateFrom(ValueLayout.JAVA_DOUBLE, new double[]{0}), "Arena::allocateFrom/double");
|
2024-01-08 08:20:07 +00:00
|
|
|
var source = MemorySegment.ofArray(new byte[]{});
|
|
|
|
ScopedOperation.ofScope(a -> a.allocateFrom(ValueLayout.JAVA_INT, source, JAVA_BYTE, 0, 1), "Arena::allocateFrom/5arg");
|
2021-06-02 10:53:06 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
@DataProvider(name = "scopedOperations")
|
|
|
|
static Object[][] scopedOperations() {
|
|
|
|
return scopedOperations.stream().map(op -> new Object[] { op.name, op }).toArray(Object[][]::new);
|
|
|
|
}
|
|
|
|
|
2023-04-27 09:00:58 +00:00
|
|
|
static class ScopedOperation<X> implements Consumer<X>, Function<Arena, X> {
|
2021-06-02 10:53:06 +00:00
|
|
|
|
2023-04-27 09:00:58 +00:00
|
|
|
final Function<Arena, X> factory;
|
2021-11-24 11:51:16 +00:00
|
|
|
final Consumer<X> operation;
|
2021-06-02 10:53:06 +00:00
|
|
|
final String name;
|
|
|
|
|
2023-04-27 09:00:58 +00:00
|
|
|
private ScopedOperation(Function<Arena, X> factory, Consumer<X> operation, String name) {
|
2021-11-24 11:51:16 +00:00
|
|
|
this.factory = factory;
|
|
|
|
this.operation = operation;
|
2021-06-02 10:53:06 +00:00
|
|
|
this.name = name;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2021-11-24 11:51:16 +00:00
|
|
|
public void accept(X obj) {
|
|
|
|
operation.accept(obj);
|
2021-06-02 10:53:06 +00:00
|
|
|
}
|
|
|
|
|
2021-11-24 11:51:16 +00:00
|
|
|
@Override
|
2023-04-27 09:00:58 +00:00
|
|
|
public X apply(Arena session) {
|
2022-05-12 16:17:45 +00:00
|
|
|
return factory.apply(session);
|
2021-06-02 10:53:06 +00:00
|
|
|
}
|
|
|
|
|
2023-04-27 09:00:58 +00:00
|
|
|
static <Z> void of(Function<Arena, Z> factory, Consumer<Z> consumer, String name) {
|
2021-11-24 11:51:16 +00:00
|
|
|
scopedOperations.add(new ScopedOperation<>(factory, consumer, name));
|
2021-06-02 10:53:06 +00:00
|
|
|
}
|
|
|
|
|
2023-04-27 09:00:58 +00:00
|
|
|
static void ofScope(Consumer<Arena> scopeConsumer, String name) {
|
2021-11-24 11:51:16 +00:00
|
|
|
scopedOperations.add(new ScopedOperation<>(Function.identity(), scopeConsumer, name));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ofSegment(Consumer<MemorySegment> segmentConsumer, String name) {
|
2021-06-02 10:53:06 +00:00
|
|
|
for (SegmentFactory segmentFactory : SegmentFactory.values()) {
|
2021-11-24 11:51:16 +00:00
|
|
|
scopedOperations.add(new ScopedOperation<>(
|
|
|
|
segmentFactory.segmentFactory,
|
|
|
|
segmentConsumer,
|
|
|
|
segmentFactory.name() + "/" + name));
|
2021-06-02 10:53:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
enum SegmentFactory {
|
|
|
|
|
2023-04-27 09:00:58 +00:00
|
|
|
NATIVE(session -> session.allocate(10, 1)),
|
2022-05-12 16:17:45 +00:00
|
|
|
MAPPED(session -> {
|
|
|
|
try (FileChannel fileChannel = FileChannel.open(Path.of("foo.txt"), StandardOpenOption.READ, StandardOpenOption.WRITE)) {
|
|
|
|
return fileChannel.map(FileChannel.MapMode.READ_WRITE, 0L, 10L, session);
|
2021-06-02 10:53:06 +00:00
|
|
|
} catch (IOException ex) {
|
|
|
|
throw new AssertionError(ex);
|
|
|
|
}
|
|
|
|
}),
|
2023-04-27 09:00:58 +00:00
|
|
|
UNSAFE(session -> MemorySegment.NULL.reinterpret(10, session, null));
|
2021-06-02 10:53:06 +00:00
|
|
|
|
|
|
|
static {
|
|
|
|
try {
|
|
|
|
File f = new File("foo.txt");
|
|
|
|
f.createNewFile();
|
|
|
|
f.deleteOnExit();
|
|
|
|
} catch (IOException ex) {
|
|
|
|
throw new ExceptionInInitializerError(ex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-27 09:00:58 +00:00
|
|
|
final Function<Arena, MemorySegment> segmentFactory;
|
2021-06-02 10:53:06 +00:00
|
|
|
|
2023-04-27 09:00:58 +00:00
|
|
|
SegmentFactory(Function<Arena, MemorySegment> segmentFactory) {
|
2021-06-02 10:53:06 +00:00
|
|
|
this.segmentFactory = segmentFactory;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|